Magento 2 добавить пользовательскую проверку атрибута продукта из сценария установки

17
[
    'type' => 'int',
    'backend' => '',
    'frontend' => '',
    'label' => 'XXXX',
    'input' => 'text',
    'frontend_class' => 'validate-больше-чем-ноль',
    'source' => '',
    'global' => \ Magento \ Eav \ Model \ Entity \ Attribute \ ScopedAttributeInterface :: SCOPE_GLOBAL,
    'visible' => true,
    'required' => true,
    'user_defined' => false,
    'default' => 0,
    'searchable' => false,
    'фильтруемый' => правда,
    «сопоставимый» => ложный,
    'visible_on_front' => false,
    'used_in_product_listing' => true,
    'unique' => false
]

Я добавляю пользовательский атрибут продукта, который работает нормально, но не может добавить validate-greater-than-zeroпроверку.

Если мы посмотрим на какие-либо свойства атрибута Input Validation for Store Owner, количество вариантов проверки ограничено.

validate-number, validate-digits, validate-email, validate-url, validate-alpha,validate-alphanum

Это единственные проверки, примененные в разделе атрибутов продукта.

Амит Сингх
источник
Пожалуйста, посмотрите мой ответ, он поможет вам проверить значение вашего атрибута.
Маттео Джеффри

Ответы:

13

Одним из решений является добавление backend modelк вашему атрибуту атрибута, который используется для форматирования / проверки значения вашего атрибута перед сохранением и / или после загрузки.

Добавьте бэкэнд-класс:

[
    'type' => 'int',
    'backend' => '\Foo\Bar\Model\Attribute\Backend\YourAttribute',
    'frontend' => '',
    'label' => 'XXXX',
    'input' => 'text',
    'frontend_class' => 'validate-greater-than-zero',
    'source' => '',
    'global' => \Magento\Eav\Model\Entity\Attribute\ScopedAttributeInterface::SCOPE_GLOBAL,
    'visible' => true,
    'required' => true,
    'user_defined' => false,
    'default' => 0,
    'searchable' => false,
    'filterable' => true,
    'comparable' => false,
    'visible_on_front' => false,
    'used_in_product_listing' => true,
    'unique' => false
]

Вот пример вашего пользовательского класса \Foo\Bar\Model\Attribute\Backend\YourAttribute

<?php

namespace Foo\Bar\Model\Attribute\Backend;

/**
 * Class YourAttribute
 */
class YourAttribute extends \Magento\Eav\Model\Entity\Attribute\Backend\AbstractBackend
{

    /**
     * @var int $minimumValueLength
     */
    protected $minimumValueLength = 0;

    /**
     * @param \Magento\Framework\DataObject $object
     *
     * @return $this
     */
    public function afterLoad($object)
    {
        // your after load logic

        return parent::afterLoad($object);
    }

    /**
     * @param \Magento\Framework\DataObject $object
     *
     * @return $this
     */
    public function beforeSave($object)
    {
        $this->validateLength($object);

        return parent::beforeSave($object);
    }

    /**
     * Validate length
     *
     * @param \Magento\Framework\DataObject $object
     *
     * @return bool
     * @throws \Magento\Framework\Exception\LocalizedException
     */
    public function validateLength($object)
    {
        /** @var string $attributeCode */
        $attributeCode = $this->getAttribute()->getAttributeCode();
        /** @var int $value */
        $value = (int)$object->getData($attributeCode);
        /** @var int $minimumValueLength */
        $minimumValueLength = $this->getMinimumValueLength();

        if ($this->getAttribute()->getIsRequired() && $value <= $minimumValueLength) {
            throw new \Magento\Framework\Exception\LocalizedException(
                __('The value of attribute "%1" must be greater than %2', $attributeCode, $minimumValueLength)
            );
        }

        return true;
    }

    /**
     * Get minimum attribute value length
     * 
     * @return int
     */
    public function getMinimumValueLength()
    {
        return $this->minimumValueLength;
    }
}

Если вы хотите простой пример такого класса, вы можете проверить

  • \Magento\Customer\Model\Customer\Attribute\Backend\Website
  • все классы, которые расширяются \Magento\Eav\Model\Entity\Attribute\Backend\AbstractBackend
  • классы в backend_modelстолбец в eav_attributeтаблице


РЕДАКТИРОВАТЬ
Если вы хотите, чтобы класс, который делает почти то же самое, что и вы, вы можете взглянуть на SKUпроверку атрибута, \Magento\Catalog\Model\Product\Attribute\Backend\Sku
я также добавил метод в примере класса


РЕДАКТИРОВАТЬ
Другое решение (возможно, не лучшее) состоит в том, чтобы создать плагин для функции \Magento\Eav\Helper\Data::getFrontendClassesи добавить свой класс веб-интерфейса, который может быть проверен вначале.

Маттео Джеффри
источник
Спасибо за ваш ответ, но можно ли применить проверку интерфейса.
Амит Сингх
Если вы посмотрите на свою строку атрибута в eav_attributeтаблице в столбце, frontend_classбудет ли это значение validate-greater-than-zero?
Маттео Джеффри
Да, но это не работает. Это единственные классы, работа validate-number, validate-digits, validate-email, validate-url, validate-alpha, validate-alphanum.
Амит Сингх
1
Можете ли вы попробовать мое второе редактирование , чтобы добавить свои собственные классы внешнего интерфейса?
Маттео Джеффри
Я сделал это с помощью плагина, спасибо за подсказку
Амит Сингх
12

С помощью Matthéo Geoffrayэтого я и применил проверку внешнего интерфейса для пользовательских атрибутов.

[
    'type' => 'int',
    'backend' => '',
    'frontend' => '',
    'label' => 'XXXX',
    'input' => 'text',
    'frontend_class' => 'validate-greater-than-zero',
    'source' => '',
    'global' => \Magento\Eav\Model\Entity\Attribute\ScopedAttributeInterface::SCOPE_GLOBAL,
    'visible' => true,
    'required' => true,
    'user_defined' => false,
    'default' => 0,
    'searchable' => false,
    'filterable' => true,
    'comparable' => false,
    'visible_on_front' => false,
    'used_in_product_listing' => true,
    'unique' => false
]

Это пользовательский атрибут в скрипте установки.

Я добавил плагин в di.xml

<type name="Magento\Catalog\Ui\DataProvider\CatalogEavValidationRules">
      <plugin name="namespace_custom_validation_for_product_attribute" type="Namespace\Module\Model\Plugin\Product\ValidationRules"/>
</type>

Вот код плагина.

<?php

namespace Namespace\Module\Model\Plugin\Product;

use Closure;

class ValidationRules
{

    /**
     * @param \Magento\Catalog\Ui\DataProvider\CatalogEavValidationRules $rulesObject
     * @param callable $proceed
     * @param \Magento\Catalog\Api\Data\ProductAttributeInterface $attribute,
     * @param array $data
     * @return array
     * @SuppressWarnings(PHPMD.UnusedFormalParameter)
     */
    public function aroundBuild(
        \Magento\Catalog\Ui\DataProvider\CatalogEavValidationRules $rulesObject,
        Closure $proceed,
        \Magento\Catalog\Api\Data\ProductAttributeInterface $attribute,
        array $data
    ){
        $rules = $proceed($attribute,$data);
        if($attribute->getAttributeCode() == 'xyz'){ //custom filter
            $validationClasses = explode(' ', $attribute->getFrontendClass());
            foreach ($validationClasses as $class) {
                $rules[$class] = true;
            }
        }
        return $rules;
    }
}

По сути \Magento\Catalog\Ui\DataProvider\CatalogEavValidationRules, вызываемый метод mapRulesсопоставляет только класс внешнего интерфейса с ограниченным числом правил проверки. Чтобы применить больше правил проверки, нам нужно добавить правила с помощью плагина.

Для проверки на стороне сервера, пожалуйста, обратитесь к Matthéo Geoffrayответу.

Амит Сингх
источник
3

Я не уверен, что это возможно из установочного скрипта. Но я уверен, что это возможно, если вы создадите плагин «перед прослушиванием» с функцией beforeSave()и проверите там значение.

Фред Ороско Диас
источник