35

What is the right(official) way to programmatically add product attribute option in M2? E.g. for manufacturer product attribute. Obviously existing option would be matched by "Admin" title value.

asked Feb 29, 2016 at 13:00

7 Answers 7

74

Here's the approach I've come up with for handling attribute options. Helper class:

<?php
namespace My\Module\Helper;
class Data extends \Magento\Framework\App\Helper\AbstractHelper
{
 /**
 * @var \Magento\Catalog\Api\ProductAttributeRepositoryInterface
 */
 protected $attributeRepository;
 /**
 * @var array
 */
 protected $attributeValues;
 /**
 * @var \Magento\Eav\Model\Entity\Attribute\Source\TableFactory
 */
 protected $tableFactory;
 /**
 * @var \Magento\Eav\Api\AttributeOptionManagementInterface
 */
 protected $attributeOptionManagement;
 /**
 * @var \Magento\Eav\Api\Data\AttributeOptionLabelInterfaceFactory
 */
 protected $optionLabelFactory;
 /**
 * @var \Magento\Eav\Api\Data\AttributeOptionInterfaceFactory
 */
 protected $optionFactory;
 /**
 * Data constructor.
 *
 * @param \Magento\Framework\App\Helper\Context $context
 * @param \Magento\Catalog\Api\ProductAttributeRepositoryInterface $attributeRepository
 * @param \Magento\Eav\Model\Entity\Attribute\Source\TableFactory $tableFactory
 * @param \Magento\Eav\Api\AttributeOptionManagementInterface $attributeOptionManagement
 * @param \Magento\Eav\Api\Data\AttributeOptionLabelInterfaceFactory $optionLabelFactory
 * @param \Magento\Eav\Api\Data\AttributeOptionInterfaceFactory $optionFactory
 */
 public function __construct(
 \Magento\Framework\App\Helper\Context $context,
 \Magento\Catalog\Api\ProductAttributeRepositoryInterface $attributeRepository,
 \Magento\Eav\Model\Entity\Attribute\Source\TableFactory $tableFactory,
 \Magento\Eav\Api\AttributeOptionManagementInterface $attributeOptionManagement,
 \Magento\Eav\Api\Data\AttributeOptionLabelInterfaceFactory $optionLabelFactory,
 \Magento\Eav\Api\Data\AttributeOptionInterfaceFactory $optionFactory
 ) {
 parent::__construct($context);
 $this->attributeRepository = $attributeRepository;
 $this->tableFactory = $tableFactory;
 $this->attributeOptionManagement = $attributeOptionManagement;
 $this->optionLabelFactory = $optionLabelFactory;
 $this->optionFactory = $optionFactory;
 }
 /**
 * Get attribute by code.
 *
 * @param string $attributeCode
 * @return \Magento\Catalog\Api\Data\ProductAttributeInterface
 */
 public function getAttribute($attributeCode)
 {
 return $this->attributeRepository->get($attributeCode);
 }
 /**
 * Find or create a matching attribute option
 *
 * @param string $attributeCode Attribute the option should exist in
 * @param string $label Label to find or add
 * @return int
 * @throws \Magento\Framework\Exception\LocalizedException
 */
 public function createOrGetId($attributeCode, $label)
 {
 if (strlen($label) < 1) {
 throw new \Magento\Framework\Exception\LocalizedException(
 __('Label for %1 must not be empty.', $attributeCode)
 );
 }
 // Does it already exist?
 $optionId = $this->getOptionId($attributeCode, $label);
 if (!$optionId) {
 // If no, add it.
 /** @var \Magento\Eav\Model\Entity\Attribute\OptionLabel $optionLabel */
 $optionLabel = $this->optionLabelFactory->create();
 $optionLabel->setStoreId(0);
 $optionLabel->setLabel($label);
 $option = $this->optionFactory->create();
 $option->setLabel($optionLabel->getLabel());
 $option->setStoreLabels([$optionLabel]);
 $option->setSortOrder(0);
 $option->setIsDefault(false);
 $this->attributeOptionManagement->add(
 \Magento\Catalog\Model\Product::ENTITY,
 $this->getAttribute($attributeCode)->getAttributeId(),
 $option
 );
 // Get the inserted ID. Should be returned from the installer, but it isn't.
 $optionId = $this->getOptionId($attributeCode, $label, true);
 }
 return $optionId;
 }
 /**
 * Find the ID of an option matching $label, if any.
 *
 * @param string $attributeCode Attribute code
 * @param string $label Label to find
 * @param bool $force If true, will fetch the options even if they're already cached.
 * @return int|false
 */
 public function getOptionId($attributeCode, $label, $force = false)
 {
 /** @var \Magento\Catalog\Model\ResourceModel\Eav\Attribute $attribute */
 $attribute = $this->getAttribute($attributeCode);
 // Build option array if necessary
 if ($force === true || !isset($this->attributeValues[ $attribute->getAttributeId() ])) {
 $this->attributeValues[ $attribute->getAttributeId() ] = [];
 // We have to generate a new sourceModel instance each time through to prevent it from
 // referencing its _options cache. No other way to get it to pick up newly-added values.
 /** @var \Magento\Eav\Model\Entity\Attribute\Source\Table $sourceModel */
 $sourceModel = $this->tableFactory->create();
 $sourceModel->setAttribute($attribute);
 foreach ($sourceModel->getAllOptions() as $option) {
 $this->attributeValues[ $attribute->getAttributeId() ][ $option['label'] ] = $option['value'];
 }
 }
 // Return option ID if exists
 if (isset($this->attributeValues[ $attribute->getAttributeId() ][ $label ])) {
 return $this->attributeValues[ $attribute->getAttributeId() ][ $label ];
 }
 // Return false if does not exist
 return false;
 }
}

Then, either in the same class or including it via dependency injection, you can add or get your option ID by calling createOrGetId($attributeCode, $label).

For example, if you inject My\Module\Helper\Data as $this->moduleHelper, then you can call:

$manufacturerId = $this->moduleHelper->createOrGetId('manufacturer', 'ABC Corp');

If 'ABC Corp' is an existing manufacturer, it will pull the ID. If not, it will add it.

UPDATED 2016年09月09日: Per Ruud N., the original solution used CatalogSetup, which resulted in a bug starting in Magento 2.1. This revised solution bypasses that model, creating the option and label explicitly. It should work on 2.0+.

answered Feb 29, 2016 at 14:54
16
  • 3
    It's as official as you're going to get. All of the lookups and option adding go through core Magento. My class is just a wrapper for those core methods that makes them easier to use. Commented Feb 29, 2016 at 15:35
  • 1
    Hi Ryan, you shouldn't set the value on the option, this is the internal id magento uses and I found out the hard way that if you set the value to a string value with a leading number like '123 abc corp' it causes some serious problems due to the implementation of Magento\Eav\Model\ResourceModel\Entity\Attribute::_processAttributeOptions. See for yourself, if you remove the $option->setValue($label); statement from your code, it will save the option, then when you fetch it Magento will return the value from an auto-increment on the eav_attribute_option table. Commented Sep 12, 2016 at 19:37
  • 2
    if I add this in a foreach function, in the second iteration I will get the error "Magento\Eav\Model\Entity\Attribute\OptionManagement::setOptionValue() must be of the type string, object given" Commented Dec 12, 2018 at 14:57
  • 1
    Yes this code not working Commented Mar 23, 2019 at 11:45
  • 2
    @JELLEJ If you are getting issue Uncaught TypeError: Argument 3 passed to Magento\Eav\Model\Entity\Attribute\OptionManagement::setOptionValue() must be of the type string, object given in foreach function then change $option->setLabel($optionLabel); to $option->setLabel($label); at line 102 Commented Apr 17, 2019 at 7:21
15

tested on Magento 2.1.3.

I didn't find any workable way how to create attribute with options at once. So initially we need to create an attribute and then add options for it.

Inject following class \Magento\Eav\Setup\EavSetupFactory

 $setup->startSetup();
 /** @var \Magento\Eav\Setup\EavSetup $eavSetup */
 $eavSetup = $this->eavSetupFactory->create(['setup' => $setup]);

Create new attribute:

$eavSetup->addAttribute(
 'catalog_product',
 $attributeCode,
 [
 'type' => 'varchar',
 'input' => 'select',
 'required' => false,
 ...
 ],
);

Add custom options.

Function addAttribute doesn't return anything useful which can be used in future. So after attribute creation we need to retrieve attribute object by ourself. !!!Important We need it because function expects only attribute_id, but don't want to work with attribute_code.

In that case we need to get attribute_id and pass it to attribute creation function.

$attributeId = $eavSetup->getAttributeId('catalog_product', 'attribute_code');

Then we need to generate options array in the way magento expects:

$options = [
 'values' => [
 'sort_order1' => 'title1',
 'sort_order2' => 'title2',
 'sort_order3' => 'title3',
 ],
 'attribute_id' => 'some_id',
];

As example:

$options = [
 'values' => [
 '1' => 'Red',
 '2' => 'Yellow',
 '3' => 'Green',
 ],
 'attribute_id' => '32',
];

And pass it to function:

$eavSetup->addAttributeOption($options);
answered Feb 23, 2017 at 7:46
1
  • 3rd param of addAttribute can take the array parameter ['option'] Commented May 7, 2019 at 19:38
10

Using the Magento\Eav\Setup\EavSetupFactory or even the \Magento\Catalog\Setup\CategorySetupFactory class may lead to the following problem: https://github.com/magento/magento2/issues/4896.

The classes you should use:

protected $_logger;
protected $_attributeRepository;
protected $_attributeOptionManagement;
protected $_option;
protected $_attributeOptionLabel;
 public function __construct(
 \Psr\Log\LoggerInterface $logger,
 \Magento\Eav\Model\AttributeRepository $attributeRepository,
 \Magento\Eav\Api\AttributeOptionManagementInterface $attributeOptionManagement,
 \Magento\Eav\Api\Data\AttributeOptionLabelInterface $attributeOptionLabel,
 \Magento\Eav\Model\Entity\Attribute\Option $option
 ){
 $this->_logger = $logger;
 $this->_attributeRepository = $attributeRepository;
 $this->_attributeOptionManagement = $attributeOptionManagement;
 $this->_option = $option;
 $this->_attributeOptionLabel = $attributeOptionLabel;
 }

Then in your function do something like this:

 $attribute_id = $this->_attributeRepository->get('catalog_product', 'your_attribute')->getAttributeId();
$options = $this->_attributeOptionManagement->getItems('catalog_product', $attribute_id);
/* if attribute option already exists, remove it */
foreach($options as $option) {
 if ($option->getLabel() == $oldname) {
 $this->_attributeOptionManagement->delete('catalog_product', $attribute_id, $option->getValue());
 }
}
/* new attribute option */
 $this->_option->setValue($name);
 $this->_attributeOptionLabel->setStoreId(0);
 $this->_attributeOptionLabel->setLabel($name);
 $this->_option->setLabel($this->_attributeOptionLabel);
 $this->_option->setStoreLabels([$this->_attributeOptionLabel]);
 $this->_option->setSortOrder(0);
 $this->_option->setIsDefault(false);
 $this->_attributeOptionManagement->add('catalog_product', $attribute_id, $this->_option);
answered Aug 1, 2016 at 14:55
4
  • 1
    Thanks, you are correct. I've updated my answer accordingly. Note that $attributeOptionLabel and $option are ORM classes; you should not inject them directly. The proper approach is to inject their factory class, then create an instance as needed. Also note you aren't using the API data interfaces consistently. Commented Sep 9, 2016 at 14:55
  • 3
    Hi @Rudd, see my comment on Ryan's answer. You don't want to call $option->setValue() as that is for an internal magento option_id field on the eav_attribute_option table. Commented Sep 12, 2016 at 19:42
  • Thank you. That's what I found out too. Will edit my answer accordingly. Commented Sep 14, 2016 at 5:49
  • 1
    @RuudN.,not working for me, \OptionManagement::setOptionValue() must be of the type string, object given, called in Commented Mar 23, 2020 at 7:39
2

Here's updated code based on Ryan Hoerr. Since magento 2.3.x it's a little bit different.

Update createOrGetId function.

public function createOrGetId($attributeCode, $label) {
 if (strlen($label) < 1) {
 throw new \Magento\Framework\Exception\LocalizedException(
 __('Label for %1 must not be empty.', $attributeCode)
 );
 }
 try {
 $optionId = $this->getOptionId($attributeCode, $label);
 if (!$optionId) {
 // If no, add it.
 /** @var \Magento\Eav\Model\Entity\Attribute\OptionLabel $optionLabel */
 $optionLabel = $this->optionLabelFactory->create();
 $optionLabel->setStoreId(0);
 $optionLabel->setLabel($label);
 $option = $this->optionFactory->create();
 $option->setLabel($optionLabel->getLabel());
 // $option->setLabel($label);
 $option->setStoreLabels([$optionLabel]);
 $option->setSortOrder(0);
 $option->setIsDefault(false);
 $this->attributeOptionManagement->add(
 \Magento\Catalog\Model\Product::ENTITY,
 $this->getAttribute($attributeCode)->getAttributeId(),
 $option
 );
 // Get the inserted ID. Should be returned from the installer, but it isn't.
 $optionId = $this->getOptionId($attributeCode, $label, true);
 }
 } catch (\Exception $e) {
 throw new \Exception($e->getMessage());
 }
 // Does it already exist?
 return $optionId;
 }
answered Sep 21, 2020 at 13:43
1

For Magento 2.3.3 I found that you can take Magento DevTeam approach.

  • Add Patch
bin/magento setup:db-declaration:generate-patch Vendor_Module PatchName
  • Add CategorySetupFactory to constructor
public function __construct(
 ModuleDataSetupInterface $moduleDataSetup,
 Factory $configFactory
 CategorySetupFactory $categorySetupFactory
 ) {
 $this->moduleDataSetup = $moduleDataSetup;
 $this->configFactory = $configFactory;
 $this->categorySetupFactory = $categorySetupFactory;
}
  • Add attribute in apply() function

    public function apply()
    {
     $categorySetup = $this->categorySetupFactory->create(['setup' => $this->moduleDataSetup]);
     $categorySetup->addAttribute(
     \Magento\Catalog\Model\Product::ENTITY,
     'custom_layout',
     [
     'type' => 'varchar',
     'label' => 'New Layout',
     'input' => 'select',
     'source' => \Magento\Catalog\Model\Product\Attribute\Source\Layout::class,
     'required' => false,
     'sort_order' => 50,
     'global' => \Magento\Eav\Model\Entity\Attribute\ScopedAttributeInterface::SCOPE_STORE,
     'group' => 'Schedule Design Update',
     'is_used_in_grid' => true,
     'is_visible_in_grid' => false,
     'is_filterable_in_grid' => false
     ]
     );
    }
    
answered Nov 4, 2019 at 13:57
2
  • uhmm i just find out that i wanted to add this answer to diffrent question. I will just live it here and add reference to this answer there. I hope that it is ok. This is partiaclly answer for this question as well :) Commented Nov 4, 2019 at 13:59
  • Can you please add the complete solution if possible ? Thanks in advance :) Commented Feb 11, 2020 at 7:16
1

For those who are receiving error after implementing accepted answer of @reyan-hoeer

PHP Fatal error: Uncaught TypeError: Argument 3 passed to Magento\\Eav\\Model\\Entity\\Attribute\\OptionManagement::setOptionValue() must be of the type string, object given, called in ....

To fix this issue, update the function createOrGetId with following

 /**
 * Find or create a matching attribute option
 *
 * @param string $attributeCode Attribute the option should exist in
 * @param string $label Label to find or add
 * @return int
 * @throws \Magento\Framework\Exception\LocalizedException
 */
 public function createOrGetId($attributeCode, $label)
 {
 if (strlen($label) < 1) {
 throw new \Magento\Framework\Exception\LocalizedException(
 __('Label for %1 must not be empty.', $attributeCode)
 );
 }
 // Does it already exist?
 $optionId = $this->getOptionId($attributeCode, $label);
 if (!$optionId) {
 // If no, add it.
 try{
 /** @var \Magento\Eav\Model\Entity\Attribute\OptionLabel $optionLabel */
 $optionLabel = $this->optionLabelFactory->create();
 $optionLabel->setStoreId(0);
 $optionLabel->setLabel($label);
 $option = $this->optionFactory->create();
 $option->setLabel($label); // this line is changed to fix the error.
 $option->setStoreLabels([$optionLabel]);
 $option->setSortOrder(0);
 $option->setIsDefault(false);
 $this->attributeOptionManagement->add(
 \Magento\Catalog\Model\Product::ENTITY,
 $this->getAttribute($attributeCode)->getAttributeId(),
 $option
 );
 }catch(\Exception $e){
 $message = $e->getMessage();
 echo $message;
 }
 // Get the inserted ID. Should be returned from the installer, but it isn't.
 $optionId = $this->getOptionId($attributeCode, $label, true);
 }
 return $optionId;
 }
answered Feb 14, 2020 at 7:54
1

My two cents here (similar to some of the answers provided):

<?php
declare(strict_types=1);
namespace Acme\Setup\Setup\Patch\Data;
use Magento\Customer\Model\Indexer\Address\AttributeProvider;
use Magento\Customer\Setup\CustomerSetupFactory;
use Magento\Framework\Exception\LocalizedException;
use Magento\Framework\Setup\ModuleDataSetupInterface;
use Magento\Framework\Setup\Patch\DataPatchInterface;
class CustomerAddressTitleAttributeExtraOptions implements DataPatchInterface
{
 private ModuleDataSetupInterface $moduleDataSetup;
 private CustomerSetupFactory $customerSetupFactory;
 /**
 * @param ModuleDataSetupInterface $moduleDataSetup
 * @param CustomerSetupFactory $customerSetupFactory
 */
 public function __construct(
 ModuleDataSetupInterface $moduleDataSetup,
 CustomerSetupFactory $customerSetupFactory,
 ) {
 $this->moduleDataSetup = $moduleDataSetup;
 $this->customerSetupFactory = $customerSetupFactory;
 }
 /**
 * @return void
 * @throws LocalizedException
 */
 public function apply(): void
 {
 $customerSetup = $this->customerSetupFactory->create(['setup' => $this->moduleDataSetup]);
 $attributeCode = 'title';
 $attributeId = $customerSetup->getAttributeId(AttributeProvider::ENTITY, $attributeCode);
 $options = [
 "Baron",
 "Baroness",
 "Countess",
 "Dame",
 "Dr",
 "Earl",
 "Lady",
 "Lord",
 "Miss",
 "Mr",
 "Mrs",
 "Ms",
 "Prince",
 "Princess",
 "Professor",
 "Sir",
 "The Duchess of",
 "The Duchess of",
 "The Duke of",
 "The Earl of",
 "The Marchioness of",
 "The Marquess",
 "The Revd",
 "Viscount",
 "Viscountess",
 "The Hon",
 "The Hon Mrs",
 "None",
 ];
 $customerSetup->addAttributeOption([
 'values' => $options,
 'attribute_id' => $attributeId
 ]);
 }
 /**
 * {@inheritdoc}
 */
 public static function getDependencies(): array
 {
 return [];
 }
 /**
 * {@inheritdoc}
 */
 public function getAliases(): array
 {
 return [];
 }
}
answered Nov 4, 2021 at 16:29

Your Answer

Draft saved
Draft discarded

Sign up or log in

Sign up using Google
Sign up using Email and Password

Post as a guest

Required, but never shown

Post as a guest

Required, but never shown

By clicking "Post Your Answer", you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.