Odpowiedzi:
Możesz to zrobić za pomocą kolekcji:
Najpierw musisz wstrzyknąć CategoryFactory
konstruktor klasy.
Magento 2.0 i 2.1:
public function __construct(
...
\Magento\Catalog\Model\CategoryFactory $categoryFactory
) {
$this->_categoryFactory = $categoryFactory;
parent::__construct(...);
}
Następnie w dowolnym miejscu w klasie możesz:
$collection = $this->_categoryFactory->create()->getCollection()->addAttributeToFilter('name',$categoryTitle)->setPageSize(1);
if ($collection->getSize()) {
$categoryId = $collection->getFirstItem()->getId();
}
Magento 2.2:
public function __construct(
...
\Magento\Catalog\Model\ResourceModel\Category\CollectionFactory $collecionFactory
) {
$this->_collectionFactory = $collecionFactory;
parent::__construct(...);
}
Następnie w dowolnym miejscu w klasie możesz:
$collection = $this->collecionFactory
->create()
->addAttributeToFilter('name',$categoryTitle)
->setPageSize(1);
if ($collection->getSize()) {
$categoryId = $collection->getFirstItem()->getId();
}
Można tego dokonać za pomocą umów o świadczenie usług, które są uważane za najlepszą praktykę.
protected $categoryList;
/**
* @var SearchCriteriaBuilder
*/
protected $searchCriteriaBuilder;
/**
* @var FilterBuilder
*/
protected $filterBuilder;
public function __construct(
------------
CategoryListInterface $categoryList,
SearchCriteriaBuilder $searchCriteriaBuilder,
FilterBuilder $filterBuilder,
-----------------
)
{
$this->categoryList = $categoryList;
$this->searchCriteriaBuilder = $searchCriteriaBuilder;
$this->filterBuilder = $filterBuilder;
parent::__construct(----------);
}
public function getNameCategory()
{
$enableFilter[] = $this->filterBuilder
->setField(\Magento\Catalog\Model\Category::KEY_NAME)
->setConditionType('like')
->setValue(self::CATEGORY_NAME_HELP) // name of the categroy on const
->create();
$searchCriteria = $this->searchCriteriaBuilder
->addFilters($enableFilter)
->create();
$items = $this->categoryList->getList($searchCriteria)->getItems();
if(count($items) == 0)
{
return FALSE;
}
foreach ($items as $helpCategory)
{
$CategoryId = $helpCategory->getId()
}
return $CategoryId;
}
Możesz to po prostu zrobić za pomocą name
,
$title = 'womens';
$objectManager = \Magento\Framework\App\ObjectManager::getInstance();
$collection = $_categoryFactory->create()->getCollection()->addFieldToFilter('name',$title);
echo "<pre>";
print_r($collection->getData());
exit;
Wypróbuj poniższy kod pliku phtml:
$objectManager = \Magento\Framework\App\ObjectManager::getInstance();
$_categoryFactory = $objectManager->get('Magento\Catalog\Model\CategoryFactory');
$categoryTitle = 'Outdoor'; // Category Name
$collection = $_categoryFactory->create()->getCollection()->addFieldToFilter('name', ['in' => $categoryTitle]);
if ($collection->getSize()) {
$categoryId = $collection->getFirstItem()->getId();
}
Mam go z pomocą mojego kolażu
$this->_objectManager->get('Magento\Catalog\Model\CategoryFactory')->create()->getCollection()
->addFieldToSelect('name')
->addFieldToFilter('name', ['in' => $categoryTitle]);
:) Ponieważ kolekcja zwróci tylko rekord, który chcesz, możesz pobrać jedyny wynik ->getFirstItem()
na powyższym kodzie
Aby zmienić to w działającym skrypcie, sugeruję użycie następujących
$obj = $bootstrap->getObjectManager();
$_categoryFactory = $obj->get('Magento\Catalog\Model\CategoryFactory');
$collection = $_categoryFactory->create()->getCollection()->addAttributeToFilter('title',$categoryTitle)->setPageSize(1);
if ($collection->getSize()) {
$categoryId = $collection->getFirstItem()->getCategoryId();
}
Edycja: Stworzyłem i przetestowałem skrypt. Utworzyłem plik w /scripts/file.php
<?php
use Magento\Framework\App\Bootstrap;
require __DIR__ . '/../app/bootstrap.php';
$bootstrap = Bootstrap::create(BP, $_SERVER);
$obj = $bootstrap->getObjectManager();
// Set the state (not sure if this is neccessary)
$obj = $bootstrap->getObjectManager();
$_categoryFactory = $obj->get('Magento\Catalog\Model\CategoryFactory');
$categoryTitle = 'Test';
$collection = $_categoryFactory->create()->getCollection()->addAttributeToFilter('name',$categoryTitle)->setPageSize(1);
if ($collection->getSize()) {
$categoryId = $collection->getFirstItem()->getId();
echo $categoryId;
}
Udało mi się napisać własną (bardziej wydajną) metodę:
$entityTypeId = \Magento\Catalog\Setup\CategorySetup::CATEGORY_ENTITY_TYPE_ID;
$row = $this->queryF("SELECT * FROM `eav_attribute` WHERE `entity_type_id` = $entityTypeId AND `attribute_code` = 'name'", 1);
$nameAttributeId = $row['attribute_id'];
$categoryNames = $this->queryF("SELECT * FROM `catalog_category_entity_varchar` WHERE `attribute_id` = '$nameAttributeId'");
$this->categoryNameIdMap = [];
foreach ($categoryNames as $item) {
$id = $item['entity_id'];
$title = $item['value'];
$this->categoryNameIdMap[$title] = $id;
}
Ten kod buforuje wszystkie tablice title: id do tablicy i wysyła zapytanie tylko 2 razy.
Pracował dla mnie. Łatwiejszy w użyciu!
Najpierw musisz wstrzyknąć klasę fabryczną kolekcji
public function __construct(
...
\Magento\Catalog\Model\ResourceModel\Category\CollectionFactory $collecionFactory ) {
$this->_collectionFactory = $collecionFactory;
parent::__construct(...); }
Następnie możesz to zrobić w swojej metodzie,
$categoryTitle = 'Men';
$collection = $this->_categoryCollectionFactory->create()->addAttributeToFilter('name',$categoryTitle)->setPageSize(1);
if ($collection->getSize()) {
$categoryId = $collection->getFirstItem()->getId();
}