Jak dodać pole niestandardowe do opcji w rozszerzeniu Zaawansowane opcje produktu

Opublikowany: 2020-09-08

Z tego artykułu dowiesz się, jak utworzyć pole „GTIN” dla niestandardowych opcji produktu, wyświetlić je w interfejsie strony produktu i wyświetlić w zamówieniu.

Bez zbędnych ceregieli przejdźmy do szczegółowych wskazówek.

Spis treści

  • Krok 1. Utwórz nowy moduł
  • Krok 2. Dodaj nasze nowe pole do bazy danych
  • Krok 3. Dodaj logikę do pracy z backendem
  • Krok 4. Wyświetl nasze pole na stronie produktu Front-End
  • Krok #5. Dodaj nasze dane atrybutów do szczegółów zamówienia w bazie danych
  • Krok #6. Wyświetl dane na stronie zamówień w panelu administracyjnym

Krok 1. Utwórz nowy moduł

Szczegółowo opisaliśmy jak stworzyć moduł w tym artykule. Dlatego pomińmy tę część i przejdźmy od razu do kodu potrzebnego do stworzenia dodatku:

1.kompozytor.json

 { "name": "mageworx/module-optiongtin", "description": "N/A", "require": { "magento/framework" : ">=100.1.0 <101", "magento/module-catalog": ">=101.0.0 <104" }, "type": "magento2-module", "version": "1.0.0", "license": [ "OSL-3.0", "AFL-3.0" ], "autoload": { "files": [ "registration.php" ], "psr-4": { "VendorName\\OptionGtin\\": "" } } }

2.etc/moduł.xml

 <?xml version="1.0"?> <config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Module/etc/module.xsd"> <module name="VendorName_OptionGtin" setup_version="1.0.0"> <sequence> <module name="Magento_Catalog"/> <module name="MageWorx_OptionBase"/> </sequence> </module> </config>

3.rejestracja.php

 <?php \Magento\Framework\Component\ComponentRegistrar::register( \Magento\Framework\Component\ComponentRegistrar::MODULE, 'VendorName_OptionGtin', __DIR__ );

Krok 2. Dodaj nasze nowe pole do bazy danych

Po zbudowaniu pustego modułu nadszedł czas, aby utworzyć nowe pole „GTIN” i dodać je do bazy danych w odpowiedniej tabeli. Gdy dodamy pole na wartości opcji, będziemy potrzebować tabeli „catalog_product_option”.

Stwórzmy następujący plik:

app/code/VendorName/OptionGtin/Setup/InstallSchema.php

 <?php namespace VendorName\OptionGtin\Setup; use Magento\Framework\Setup\InstallSchemaInterface; use Magento\Framework\Setup\ModuleContextInterface; use Magento\Framework\Setup\SchemaSetupInterface; use Magento\Framework\DB\Ddl\Table; class InstallSchema implements InstallSchemaInterface { public function install(SchemaSetupInterface $setup, ModuleContextInterface $context) { $setup->startSetup(); $setup->getConnection()->addColumn( $setup->getTable('catalog_product_option'), 'gtin', [ 'type' => Table::TYPE_TEXT, 'nullable' => true, 'default' => null, 'comment' => 'Gtin (added by VendorName Option Gtin)', ] ); $setup->endSetup(); } }

Krok 3. Dodaj logikę do pracy z backendem

Użyjemy mechanizmu modyfikatora puli, aby dodać nasze nowe pole.

Teraz dodaj następujący plik:

app/code/VendorName/OptionGtin/etc/adminhtml/di.xml

 <?xml version="1.0"?> <config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:ObjectManager/etc/config.xsd"> <virtualType name="MageWorx\OptionBase\Ui\DataProvider\Product\Form\Modifier\Pool"> <arguments> <argument name="modifiers" xsi:type="array"> <item name="mageworx-option-gtin" xsi:type="array"> <item name="class" xsi:type="string">VendorName\OptionGtin\Ui\DataProvider\Product\Form\Modifier\OptionGtin</item> <item name="sortOrder" xsi:type="number">72</item> </item> </argument> </arguments> </virtualType> </config>

Dodajmy tutaj nasz modyfikator do wspólnej puli rozszerzenia Zaawansowane opcje produktu ―”MageWorx\OptionBase\Ui\DataProvider\Product\Form\Modifier\Pool”. „NazwaDostawcy\OptionGtin\Ui\DataProvider\Product\Form\Modifier\OptionGtin” to nasz modyfikator klasy.

Poniżej znajduje się kod umożliwiający dodanie naszego pola do formularza app/code/VendorName/OptionGtin/Ui/DataProvider/Product/Form/Modifier/OptionGtin.php :

 <?php namespace VendorName\OptionGtin\Ui\DataProvider\Product\Form\Modifier; use Magento\Catalog\Ui\DataProvider\Product\Form\Modifier\AbstractModifier; use Magento\Catalog\Ui\DataProvider\Product\Form\Modifier\CustomOptions; use Magento\Ui\Component\Form\Element\Input; use Magento\Ui\Component\Form\Element\DataType\Number; use Magento\Ui\Component\Form\Field; use MageWorx\OptionBase\Ui\DataProvider\Product\Form\Modifier\ModifierInterface; class OptionGtin extends AbstractModifier implements ModifierInterface { /** * @var array */ protected $meta = []; /** * {@inheritdoc} */ public function modifyData(array $data) { return $data; } /** * {@inheritdoc} */ public function modifyMeta(array $meta) { $this->meta = $meta; $this->addFields(); return $this->meta; } /** * Adds fields to the meta-data */ protected function addFields() { $groupCustomOptionsName = CustomOptions::GROUP_CUSTOM_OPTIONS_NAME; $optionContainerName = CustomOptions::CONTAINER_OPTION; $commonOptionContainerName = CustomOptions::CONTAINER_COMMON_NAME; // Add fields to the option $optionFeaturesFields = $this->getOptionGtinFieldsConfig(); $this->meta[$groupCustomOptionsName]['children']['options']['children']['record']['children'] [$optionContainerName]['children'][$commonOptionContainerName]['children'] = array_replace_recursive( $this->meta[$groupCustomOptionsName]['children']['options']['children']['record']['children'] [$optionContainerName]['children'][$commonOptionContainerName]['children'], $optionFeaturesFields ); } /** * The custom option fields config * * @return array */ protected function getOptionGtinFieldsConfig() { $fields['gtin'] = $this->getGtinFieldConfig(); return $fields; } /** * Get gtin field config * * @return array */ protected function getGtinFieldConfig() { return [ 'arguments' => [ 'data' => [ 'config' => [ 'label' => __('GTIN'), 'componentType' => Field::NAME, 'formElement' => Input::NAME, 'dataType' => Number::NAME, 'dataScope' => 'gtin', 'sortOrder' => 65 ], ], ], ]; } /** * Check is current modifier for the product only * * @return bool */ public function isProductScopeOnly() { return false; } /** * Get sort order of modifier to load modifiers in the right order * * @return int */ public function getSortOrder() { return 32; } }

Teraz spróbujmy zainstalować rozszerzenie i sprawdzić, czy wszystko się wyświetla:

  • php bin/magento module: włącz VendorName_OptionGtin
  • Konfiguracja bin/magento php: aktualizacja
  • php bin/magento cache: opróżnianie

Nasze nowe pole zostało pomyślnie dodane:

Jak dodać pole niestandardowe do zaawansowanych opcji produktu | Blog Mageworx

Krok 4. Wyświetl nasze pole na stronie produktu Front-End

Rozszerzenie Mageworx Advanced Product Options ma już wszystko do wyświetlania i pracy z atrybutami dodanymi przez nasz moduł. Wszystko, co musimy zrobić, to dodać nowy atrybut do udostępnionego zbioru danych.

Nasz moduł MageWorx_OptionBase używa już metody getExtendedOptionsConfig() . Zbiera i wyświetla wszystkie atrybuty niestandardowe w bloku na interfejsie użytkownika. Otwórz app/code/MageWorx/OptionBase/Block/Product/View/Options.php , aby zobaczyć, jak zostanie zaimplementowana.

Zacznijmy od stworzenia modelu z naszym atrybutem:

app/code/VendorName/OptionGtin/Model/Attriburte/Option/Gtin.php

 <?php namespace VendorName\OptionGtin\Model\Attribute\Option; use MageWorx\OptionBase\Model\Product\Option\AbstractAttribute; class Gtin extends AbstractAttribute { /** * @return string */ public function getName() { return 'gtin'; } }

Teraz użyj mechanizmu „dependency injection” i dodaj nasz atrybut do zbioru danych atrybutów współdzielonych rozszerzenia Zaawansowane opcje produktu.

app/code/VendorName/OptionGtin/etc/di.xml

 <?xml version="1.0"?> <config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:ObjectManager/etc/config.xsd"> <!-- Data --> <type name="MageWorx\OptionBase\Model\Product\Option\Attributes"> <arguments> <argument name="data" xsi:type="array"> <item name="gtin" xsi:type="object">VendorName\OptionGtin\Model\Attribute\Option\Gtin</item> </argument> </arguments> </type> </config>

Innymi słowy, otwierając MageWorx\OptionBase\Model\Product\Option\Attributes , zobaczysz, że po prostu zbiera ona wszystkie obiekty atrybutów do udostępnionego zbioru danych.

Aby wyświetlić dane naszego nowego atrybutu „GTIN”, zdecydowaliśmy się użyć funkcji firstrun() z app/code/MageWorx/OptionFeatures/view/base/web/js/catalog/product/features.js . Ma już wszystkie wymagane implementacje, które najlepiej pasują do naszego przykładu. Aby uniknąć nadpisania całego pliku, zastosujemy mechanizm „domieszek JavaScript”, który pomoże nam zmienić tylko niezbędną funkcję.

Utwórz następujący plik i zdefiniuj tam nasz mixin: app/code/VendorName/OptionGtin/view/frontend/requirejs-config.js

 var config = { config: { mixins: { 'MageWorx_OptionFeatures/js/catalog/product/features': { 'VendorName_OptionGtin/js/catalog/product/features-gtin-mixin' : true } } } };

Tutaj MageWorx_OptionFeatures/js/catalog/product/features jest katalogiem głównym naszego pliku, a tę metodę musimy przepisać. VendorName_OptionGtin/js/catalog/product/features-gtin-mixin to plik, w którym przepiszemy metodę.

Stwórzmy go więc: app/code/VendorName/OptionGtin/view/frontend/web/js/catalog/product/features-gtin-mixin.js

 define([ 'jquery', 'jquery/ui', 'mage/utils/wrapper' ], function ($, wrapper) { 'use strict'; return function (widget) { $.widget('mageworx.optionFeatures', widget, { /** * Triggers one time at first run (from base.js) * @param optionConfig * @param productConfig * @param base * @param self */ firstRun: function firstRun(optionConfig, productConfig, base, self) { //shareable link $('#mageworx_shareable_hint_icon').qtip({ content: { text: this.options.shareable_link_hint_text }, style: { classes: 'qtip-light' }, position: { target: false } }); $('#mageworx_shareable_link').on('click', function () { try { self.copyTextToClipboard(self.getShareableLink(base)); $('.mageworx-shareable-link-container').hide(); $('.mageworx-shareable-link-success-container').show(); setTimeout(function () { $('.mageworx-shareable-link-container').show(); $('.mageworx-shareable-link-success-container').hide(); }, 2000); } catch (error) { console.log('Something goes wrong. Unable to copy'); } }); setTimeout(function () { // Qty input $('.mageworx-option-qty').each(function () { $(this).on('change', function () { var optionInput = $("[data-selector='" + $(this).attr('data-parent-selector') + "']"); optionInput.trigger('change'); }); }); }, 500); // Option\Value Description & tooltip var extendedOptionsConfig = typeof base.options.extendedOptionsConfig != 'undefined' ? base.options.extendedOptionsConfig : {}; for (var option_id in optionConfig) { if (!optionConfig.hasOwnProperty(option_id)) { continue; } var description = extendedOptionsConfig[option_id]['description'], gtin = extendedOptionsConfig[option_id]['gtin'], gtinTitle = "Global Trade Item Number: ", $option = base.getOptionHtmlById(option_id); if (1 > $option.length) { console.log('Empty option container for option with id: ' + option_id); continue; } var $label = $option.find('label'); if(gtin != null && gtin.length > 0) { if ($label.length > 0) { $label .first() .after($('<p class="option-gtin-text"><span>' + gtinTitle + '</span>' + gtin + '</p>')); } else { $label = $option.find('span'); $label .first() .parent() .after($('<p class="option-gtin-text"><span>' + gtinTitle + '</span>' + gtin + '</p>')); } } if (this.options.option_description_enabled && !_.isEmpty(extendedOptionsConfig[option_id]['description'])) { if (this.options.option_description_mode == this.options.option_description_modes.tooltip) { var $element = $option.find('label span') .first(); if ($element.length == 0) { $element = $option.find('fieldset legend span') .first(); } $element.css('border-bottom', '1px dotted black'); $element.qtip({ content: { text: description }, style: { classes: 'qtip-light' }, position: { target: false } }); } else if (this.options.option_description_mode == this.options.option_description_modes.text) { if ($label.length > 0) { $label .first() .after($('<p class="option-description-text">' + description + '</p>')); } else { $label = $option.find('span'); $label .first() .parent() .after($('<p class="option-description-text">' + description + '</p>')); } } else { console.log('Unknown option mode'); } } if (this.options.value_description_enabled) { this._addValueDescription($option, optionConfig, extendedOptionsConfig); } } } }); return $.mageworx.optionFeatures; }; });

Generalnie możemy teraz uruchomić następujące polecenia:

  • php bin/magento cache: opróżnianie
  • php bin/magento setup:static-content:deploy (tylko w trybie produkcyjnym)

i zobacz, co mamy. Ale najpierw dodaj kilka stylów do naszego nowego atrybutu i spraw, aby wyglądał ładnie na interfejsie użytkownika.

Utwórz układ i zdefiniuj tam nasz nowy plik stylów: app/code/VendorName/OptionGtin/view/frontend/layout/catalog_product_view.xml

 <?xml version="1.0"?> <page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd"> <head> <css src="VendorName_OptionGtin::css/gtin.css"/> </head> </page>

Czas stworzyć plik stylów: app/code/VendorName/OptionGtin/view/frontend/web/css/gtin.css

 .option-gtin-text span { color: #6cc308; font-weight: 700; }

Teraz uruchommy opisane wcześniej polecenia i sprawdźmy wyniki:

Jak dodać pole niestandardowe do zaawansowanych opcji produktu | Blog Mageworx

Krok #5. Dodaj nasze dane atrybutów do szczegółów zamówienia w bazie danych

Gdy klient dokona zakupu, tworzone jest zamówienie. Szczegóły dotyczące dodanych pozycji są zawarte w tabeli sales_order_item . Ta tabela zawiera pole product_options , które zawiera informacje o wybranych parametrach dodawanego towaru. W tym miejscu powinniśmy dodać dane naszego nowego atrybutu.

Podczas tworzenia zamówienia wyzwalane jest zdarzenie sales_quote_address_collect_totals_before . Wykorzystamy go, aby dodać nasze dane do opcji produktu.

Zdefiniujmy zdarzenie tworząc: app/code/VendorName/OptionGtin/etc/events.xml

 <?xml version="1.0"?> <config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Event/etc/events.xsd"> <event name="sales_quote_address_collect_totals_before"> <observer name="mageworx_optiongtin_add_gtin_to_order" instance="VendorName\OptionGtin\Observer\AddGtinToOrder" /> </event> </config>

Następnie utwórz naszego obserwatora: app/code/VendorName/OptionGtin/Observer/AddGtinToOrder.php

 <?php namespace VendorName\OptionGtin\Observer; use Magento\Framework\Event\Observer; use Magento\Framework\Event\ObserverInterface; use Magento\Catalog\Model\ProductRepository as ProductRepository; use MageWorx\OptionBase\Helper\Data as BaseHelper; class AddGtinToOrder implements ObserverInterface { /** * @var BaseHelper */ protected $baseHelper; protected $productRepository; /** * AddGtinToOrder constructor. * @param BaseHelper $baseHelper * @param ProductRepository $productRepository */ public function __construct( BaseHelper $baseHelper, ProductRepository $productRepository ) { $this->baseHelper = $baseHelper; $this->productRepository = $productRepository; } /** * Add product to quote action * Processing: gtin * * @param Observer $observer * @return $this */ public function execute(Observer $observer) { $quoteItems = $observer->getQuote()->getAllItems(); /** @var \Magento\Quote\Model\Quote\Item $quoteItem */ foreach ($quoteItems as $quoteItem) { $buyRequest = $quoteItem->getBuyRequest(); $optionIds = array_keys($buyRequest->getOptions()); $productOptions = $this->productRepository->getById($buyRequest->getProduct())->getOptions(); $quoteItemOptionGtins = []; $optionGtins = []; foreach ($productOptions as $option) { if ($option->getGtin()) { $quoteItemOptionGtins[$option->getOptionId()] = $option->getGtin(); } } foreach ($optionIds as $optionId) { $optionGtins[$optionId] = $optionId; } $optionGtins = array_intersect_key($quoteItemOptionGtins, $optionGtins); $infoBuyRequest = $quoteItem->getOptionByCode('info_buyRequest'); $buyRequest->setData('gtin', $optionGtins); $infoBuyRequest->setValue($this->baseHelper->encodeBuyRequestValue($buyRequest->getData())); $quoteItem->addOption($infoBuyRequest); } } }

Tutaj z pomocą obserwatora otrzymujemy listę wszystkich pozycji w zamówieniu i dodajemy dane naszego atrybutu „GTIN” do tzw. $infoBuyRequest .

Aby sprawdzić, czy wszystko zostało wykonane poprawnie, utwórz zamówienie z produktem, którego opcje mają dane „GTIN”. Możesz sprawdzić, czy dane zostały dodane w sales_order_item table -> pole product_options :

Jak dodać pole niestandardowe do zaawansowanych opcji produktu | Blog Mageworx

Krok #6. Wyświetl dane na stronie zamówień w panelu administracyjnym

Istnieją różne sposoby wyświetlania wymaganych informacji w gotowym szablonie. Na przykład używając „js”. W tym artykule pracowaliśmy z „js”. Popracujmy nad samymi szablonami dla odmiany i spróbujmy je przepisać!

Zmień wcześniej utworzoną app/code/VendorName/OptionGtin/etc/adminhtml/di.xml dodając tam wtyczkę:

 <?xml version="1.0"?> <config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:ObjectManager/etc/config.xsd"> <virtualType name="MageWorx\OptionBase\Ui\DataProvider\Product\Form\Modifier\Pool"> <arguments> <argument name="modifiers" xsi:type="array"> <item name="mageworx-option-gtin" xsi:type="array"> <item name="class" xsi:type="string">VendorName\OptionGtin\Ui\DataProvider\Product\Form\Modifier\OptionGtin</item> <item name="sortOrder" xsi:type="number">72</item> </item> </argument> </arguments> </virtualType> <!-- Plugins--> <type name="Magento\Sales\Block\Adminhtml\Items\Column\DefaultColumn"> <plugin name="mageworx-optiongtin-add-default-column" type="VendorName\OptionGtin\Plugin\AddDefaultColumn" sortOrder="5" disabled="false" /> </type> </config>

Utwórz samą wtyczkę:

app/code/VendorName/OptionGtin/Plugin/AddDefaultColumn.php

 <?php namespace VendorName\OptionGtin\Plugin; class AddDefaultColumn { /** * @param \Magento\Sales\Block\Adminhtml\Items\Column\DefaultColumn $subject * @param $result * @return array */ public function afterGetOrderOptions(\Magento\Sales\Block\Adminhtml\Items\Column\DefaultColumn $subject, $result) { if ($options = $subject->getItem()->getProductOptions()) { if (isset($result)) { foreach ($result as &$option) { if (array_key_exists($option['option_id'], $options['info_buyRequest']['gtin'])) { $option['gtin'] = $options['info_buyRequest']['gtin'][$option['option_id']]; } } } } return $result; } }

Wtyczka ta dodaje informacje o naszym nowym atrybucie opcji zamówień, dla których te dane istnieją.

vendor/magento/module-sales/view/adminhtml/templates/items/column/name.phtml odpowiada za wyświetlanie informacji o opcjach produktu na stronie zamówienia w panelu administracyjnym.

Przepiszmy go, aby wyświetlał nasz „GTIN”. W tym celu musimy przepisać blok „nazwa_kolumny”, a raczej jego szablon. Utwórz układ i szablon:

app/code/VendorName/OptionGtin/view/adminhtml/layout/sales_order_view.xml

 <?xml version="1.0"?> <page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd"> <body> <referenceBlock name="column_name"> <action method="setTemplate"> <argument name="template" xsi:type="string">VendorName_OptionGtin::items/column/name.phtml</argument> </action> </referenceBlock> </body> </page>

app/code/VendorName/OptionGtin/view/adminhtml/templates/items/column/name.phtml

 <?php /* @var $block \Magento\Sales\Block\Adminhtml\Items\Column\Name */ ?> <?php if ($_item = $block->getItem()) : ?> <div class="product-title"> <?= $block->escapeHtml($_item->getName()) ?> </div> <div class="product-sku-block"> <span><?= $block->escapeHtml(__('SKU'))?>:</span> <?= /* @noEscape */ implode('<br />', $this->helper(\Magento\Catalog\Helper\Data::class)->splitSku($block->escapeHtml($block->getSku()))) ?> </div> <?php if ($block->getOrderOptions()) : ?> <dl class="item-options"> <?php foreach ($block->getOrderOptions() as $_option) : ?> <dt><?= $block->escapeHtml($_option['label']) ?>:</dt> <dd> <?php if (isset($_option['custom_view']) && $_option['custom_view']) : ?> <?= /* @noEscape */ $block->getCustomizedOptionValue($_option) ?> <?php else : ?> <?php $optionValue = $block->getFormattedOption($_option['value']); ?> <?php $dots = 'dots' . uniqid(); ?> <?php $ . uniqid(); ?> <?= $block->escapeHtml($optionValue['value'], ['a', 'br']) ?><?php if (isset($optionValue['remainder']) && $optionValue['remainder']) : ?> <span> ...</span> <span><?= $block->escapeHtml($optionValue['remainder'], ['a']) ?></span> <script> require(['prototype'], function() { $('<?= /* @noEscape */ $id; ?>').hide(); $('<?= /* @noEscape */ $id; ?>').up().observe('mouseover', function(){$('<?= /* @noEscape */ $id; ?>').show();}); $('<?= /* @noEscape */ $id; ?>').up().observe('mouseover', function(){$('<?= /* @noEscape */ $dots; ?>').hide();}); $('<?= /* @noEscape */ $id; ?>').up().observe('mouseout', function(){$('<?= /* @noEscape */ $id; ?>').hide();}); $('<?= /* @noEscape */ $id; ?>').up().observe('mouseout', function(){$('<?= /* @noEscape */ $dots; ?>').show();}); }); </script> <?php endif; ?> <?php endif; ?> </dd> <dt> <?php if (isset($_option['gtin']) && $_option['gtin']) : ?> <span>GTIN:</span> <?php endif; ?> </dt> <dd> <?php if (isset($_option['gtin']) && $_option['gtin']) : ?> <span> <?= $block->escapeHtml($_option['gtin']) ?></span> <?php endif; ?> </dd> <?php endforeach; ?> </dl> <?php endif; ?> <?= $block->escapeHtml($_item->getDescription()) ?> <?php endif; ?>

Jeśli wszystko zostało wykonane poprawnie, wyczyszczone i skompilowane, zobaczysz następujący wynik:

Jak dodać pole niestandardowe do zaawansowanych opcji produktu | Blog Mageworx

Mamy nadzieję, że ten artykuł okaże się pomocny. Jeśli masz jakiekolwiek trudności lub problemy, daj nam znać w polu komentarzy poniżej.

Zarezerwuj demo na żywo z Mageworx