- Published: 2019-12-22
- Author: Nickolas Burr
Table of Contents
Several Magento UI components depend on data from data providers. Often times, we need to conditionally modify data provided by a data provider, so data modifiers were introduced to solve this problem. Traditionally, we would need to override a class in order to make those modifications, but Magento provides an improved approach with data modifiers, which accepts the respective metadata/data as an argument, and returns the modified version to the data provider.
In the example below, we've created a trait called ModifierPoolTrait
, which can be used
by a data provider class to access the respective data modifier pool.
<?php
/**
* DataProvider.php
*/
declare(strict_types=1);
namespace Vendor\Package\Ui\DataProvider\Listing\Entity;
use Countable;
use Magento\Framework\View\Element\UiComponent\DataProvider\DataProviderInterface;
use Magento\Ui\{
DataProvider\AbstractDataProvider,
DataProvider\Modifier\ModifierInterface,
DataProvider\Modifier\PoolInterface
};
use Vendor\Package\Component\Ui\DataModifier\ModifierPoolTrait;
class DataProvider extends AbstractDataProvider implements
Countable,
DataProviderInterface
{
/**
* @property PoolInterface $modifierPool
* @method PoolInterface getModifierPool()
* @method ModifierInterface[] getModifiers()
*/
use ModifierPoolTrait;
/** @var array $loadedData */
protected $loadedData = [];
/**
* @param string $name
* @param string $primaryFieldName
* @param string $requestFieldName
* @param array $meta
* @param array $data
* @param PoolInterface $modifierPool
* @return void
*/
public function __construct(
$name,
$primaryFieldName,
$requestFieldName,
array $meta = [],
array $data = [],
PoolInterface $modifierPool
) {
parent::__construct(
$name,
$primaryFieldName,
$requestFieldName,
$meta,
$data
);
$this->modifierPool = $modifierPool;
}
/**
* @return array
*/
public function getMeta(): array
{
/** @var array $meta */
$meta = parent::getMeta();
/** @var ModifierInterface[] $modifiers */
$modifiers = $this->getModifiers();
/** @var ModifierInterface $modifier */
foreach ($modifiers as $modifier) {
$meta = $modifier->modifyMeta($meta);
}
return $meta;
}
/**
* @return array
*/
public function getData(): array
{
if (!empty($this->loadedData)) {
return $this->loadedData;
}
/** @var EntityInterface[] $items */
$items = $this->getCollection()->getItems();
/** @var EntityInterface $entity */
foreach ($items as $entity) {
$this->loadedData[$entity->getId()] = $entity->getData();
}
/** @var ModifierInterface[] $modifiers */
$modifiers = $this->getModifiers();
/** @var ModifierInterface $modifier */
foreach ($modifiers as $modifier) {
$this->loadedData = $modifier->modifyData($this->loadedData);
}
return $this->loadedData;
}
}
<?php
/**
* DataModifier.php
*/
declare(strict_types=1);
namespace Vendor\Package\Ui\DataModifier\Entity;
use Magento\Ui\DataProvider\Modifier\ModifierInterface;
class DataModifier implements ModifierInterface
{
/**
* @param array $meta
* @return array
*/
public function modifyMeta(array $meta)
{
/* Perform whatever modifications you need to make to the metadata. */
return $meta;
}
/**
* @param array $data
* @return array
*/
public function modifyData(array $data)
{
/* Perform whatever modifications you need to make to the data. */
return $data;
}
}
<?xml version="1.0"?>
<!--
/**
* di.xml
*/
-->
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="urn:magento:framework:ObjectManager/etc/config.xsd">
<virtualType name="Vendor\Package\Ui\DataModifier\Entity\Pool" type="Magento\Ui\DataProvider\Modifier\Pool">
<arguments>
<argument name="modifiers" xsi:type="array">
<item name="vendor_package_entity_data_modifier" xsi:type="array">
<item name="class" xsi:type="string">Vendor\Package\Ui\DataModifier\Entity\DataModifier</item>
<item name="sortOrder" xsi:type="number">10</item>
</item>
</argument>
</arguments>
</virtualType>
<type name="Vendor\Package\Ui\DataProvider\Listing\Entity\DataProvider">
<arguments>
<argument name="modifierPool" xsi:type="object">Vendor\Package\Ui\DataModifier\Entity\Pool</argument>
</arguments>
</type>
</config>
```
<?php
/**
* ModifierPoolTrait.php
*/
declare(strict_types=1);
namespace Vendor\Package\Component\Ui\DataModifier;
use Magento\Ui\{
DataProvider\Modifier\ModifierInterface,
DataProvider\Modifier\PoolInterface
};
trait ModifierPoolTrait
{
/** @property PoolInterface $modifierPool */
private $modifierPool;
/**
* @return PoolInterface
*/
public function getModifierPool(): PoolInterface
{
return $this->modifierPool;
}
/**
* @return ModifierInterface[]
*/
public function getModifiers(): array
{
return $this->getModifierPool()
->getModifiersInstances();
}
}