Magento2; Расширенный поиск (SmartSearch)
Я пытаюсь создать расширение "SmartSearch" для Magento 2.0, но я действительно не могу понять, каким должен быть следующий шаг. Что у меня есть до сих пор:
- Рабочее расширение 2.0 (registration.php, модуль включен и распознан)
- Я переопределяю функцию автозаполнения по умолчанию с помощью этого кода:
С помощью этого кода я могу манипулировать текстом автозаполнения по умолчанию. Это мой поисковый провайдер:
class SearchDataProvider extends \Magento\CatalogSearch\Model\Autocomplete\DataProvider {
/**
* {@inheritdoc}
*/
public function getItems()
{
$collection = $this->getSuggestCollection();
$query = $this->queryFactory->get()->getQueryText();
$result = [];
foreach ($collection as $item) {
$resultItem = $this->itemFactory->create([
'title' => $item->getQueryText(),
'data' => $item->getData(),
'num_results' => $item->getNumResults(),
]);
if ($resultItem->getTitle() == $query) {
array_unshift($result, $resultItem);
} else {
$result[] = $resultItem;
}
}
return $result;
}
/**
* Retrieve suggest collection for query
*
* @return Collection
*/
private function getSuggestCollection()
{
return $this->queryFactory->get()->getSuggestCollection();
}
}
Если я добавлю что-то после "названия" отображается в раскрывающемся списке автозаполнения.
Что я хочу сделать сейчас, так это загрузить другого поставщика, который вернет коллекцию продуктов, содержащих текст поискового запроса.
Я уже нашел модель полнотекстового поиска, но не могу понять, как правильно ее использовать.
Есть ли у кого-нибудь опыт создания коллекции поиска в Magento 2.0?
1 answers
Для выполнения полнотекстового поиска следует использовать Magento\Framework\Api\Search\SearchInterface
. Пожалуйста, обратите внимание, что концепция сервисных контрактов была введена в Magento 2, несмотря на то, что API сервиса доступен еще не для всех основных функций, его следует использовать, когда это возможно, сторонними модулями вместо прямого доступа к моделям и коллекциям. Чтобы избежать критических изменений в будущем (после выпусков 2.1, 2.2 и т.д.), следует использовать только классы/интерфейсы и методы, помеченные аннотацией @api
.
Ниже полностью рабочий пример, в котором будет отображаться до 5 подходящих товаров в предложении (название и цена). Для тестирования просто замените исходное содержимое \Magento\CatalogSearch\Model\Autocomplete\DataProvider
на это:
<?php
namespace Magento\CatalogSearch\Model\Autocomplete;
use Magento\Search\Model\QueryFactory;
use Magento\Search\Model\Autocomplete\DataProviderInterface;
use Magento\Search\Model\Autocomplete\ItemFactory;
use Magento\Framework\Api\Search\SearchCriteriaFactory as FullTextSearchCriteriaFactory;
use Magento\Framework\Api\Search\SearchInterface as FullTextSearchApi;
use Magento\Framework\Api\Search\FilterGroupBuilder;
use Magento\Framework\Api\FilterBuilder;
use Magento\Framework\Api\SearchCriteriaBuilder;
use Magento\Catalog\Api\ProductRepositoryInterface;
/**
* Full text search implementation of autocomplete.
*/
class DataProvider implements DataProviderInterface
{
const PRODUCTS_NUMBER_IN_SUGGEST = 5;
/** @var QueryFactory */
protected $queryFactory;
/** @var ItemFactory */
protected $itemFactory;
/** @var \Magento\Framework\Api\Search\SearchInterface */
protected $fullTextSearchApi;
/** @var FullTextSearchCriteriaFactory */
protected $fullTextSearchCriteriaFactory;
/** @var FilterGroupBuilder */
protected $searchFilterGroupBuilder;
/** @var FilterBuilder */
protected $filterBuilder;
/** @var ProductRepositoryInterface */
protected $productRepository;
/** @var SearchCriteriaBuilder */
protected $searchCriteriaBuilder;
/**
* Initialize dependencies.
*
* @param QueryFactory $queryFactory
* @param ItemFactory $itemFactory
* @param FullTextSearchApi $search
* @param FullTextSearchCriteriaFactory $searchCriteriaFactory
* @param FilterGroupBuilder $searchFilterGroupBuilder
* @param FilterBuilder $filterBuilder
* @param ProductRepositoryInterface $productRepository
* @param SearchCriteriaBuilder $searchCriteriaBuilder
*/
public function __construct(
QueryFactory $queryFactory,
ItemFactory $itemFactory,
FullTextSearchApi $search,
FullTextSearchCriteriaFactory $searchCriteriaFactory,
FilterGroupBuilder $searchFilterGroupBuilder,
FilterBuilder $filterBuilder,
ProductRepositoryInterface $productRepository,
SearchCriteriaBuilder $searchCriteriaBuilder
) {
$this->queryFactory = $queryFactory;
$this->itemFactory = $itemFactory;
$this->fullTextSearchApi = $search;
$this->fullTextSearchCriteriaFactory = $searchCriteriaFactory;
$this->filterBuilder = $filterBuilder;
$this->searchFilterGroupBuilder = $searchFilterGroupBuilder;
$this->productRepository = $productRepository;
$this->searchCriteriaBuilder = $searchCriteriaBuilder;
}
/**
* {@inheritdoc}
*/
public function getItems()
{
$result = [];
$query = $this->queryFactory->get()->getQueryText();
$productIds = $this->searchProductsFullText($query);
if ($productIds) {
$searchCriteria = $this->searchCriteriaBuilder->addFilter('entity_id', $productIds, 'in')->create();
$products = $this->productRepository->getList($searchCriteria);
foreach ($products->getItems() as $product) {
$resultItem = $this->itemFactory->create([
/** Feel free to add here necessary product data and then render in template */
'title' => $product->getName() . " " . $product->getPrice()
]);
$result[] = $resultItem;
}
}
return $result;
}
/**
* Perform full text search and find IDs of matching products.
*
* @param string
* @return int[]
*/
private function searchProductsFullText($query)
{
$searchCriteria = $this->fullTextSearchCriteriaFactory->create();
/** To get list of available request names see Magento/CatalogSearch/etc/search_request.xml */
$searchCriteria->setRequestName('quick_search_container');
$filter = $this->filterBuilder->setField('search_term')->setValue($query)->setConditionType('like')->create();
$filterGroup = $this->searchFilterGroupBuilder->addFilter($filter)->create();
$currentPage = 1;
$searchCriteria->setFilterGroups([$filterGroup])
->setCurrentPage($currentPage)
->setPageSize(self::PRODUCTS_NUMBER_IN_SUGGEST);
$searchResults = $this->fullTextSearchApi->search($searchCriteria);
$productIds = [];
/**
* Full text search returns document IDs (in this case product IDs),
* so to get products information we need to load them using filtration by these IDs
*/
foreach ($searchResults->getItems() as $searchDocument) {
$productIds[] = $searchDocument->getId();
}
return $productIds;
}
}