facets-8.x-1.x-dev/src/Plugin/facets/processor/TranslateEntityProcessor.php
src/Plugin/facets/processor/TranslateEntityProcessor.php
<?php
namespace Drupal\facets\Plugin\facets\processor;
use Drupal\Core\Cache\Cache;
use Drupal\Core\Entity\EntityTypeManagerInterface;
use Drupal\Core\Language\LanguageManagerInterface;
use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
use Drupal\Core\TypedData\ComplexDataDefinitionInterface;
use Drupal\Core\TypedData\DataReferenceDefinitionInterface;
use Drupal\Core\TypedData\TranslatableInterface;
use Drupal\facets\Exception\InvalidProcessorException;
use Drupal\facets\FacetInterface;
use Drupal\facets\Processor\BuildProcessorInterface;
use Drupal\facets\Processor\ProcessorPluginBase;
use Symfony\Component\DependencyInjection\ContainerInterface;
/**
* Transforms the results to show the translated entity label.
*
* @FacetsProcessor(
* id = "translate_entity",
* label = @Translation("Transform entity ID to label"),
* description = @Translation("Display the entity label instead of its ID (for example the term name instead of the taxonomy term ID). This only works when an actual entity is indexed, not for the entity id or aggregated fields."),
* stages = {
* "build" = 5
* }
* )
*/
class TranslateEntityProcessor extends ProcessorPluginBase implements BuildProcessorInterface, ContainerFactoryPluginInterface {
/**
* The language manager.
*
* @var \Drupal\Core\Language\LanguageManagerInterface
*/
protected $languageManager;
/**
* The entity type manager.
*
* @var \Drupal\Core\Entity\EntityTypeManagerInterface
*/
protected $entityTypeManager;
/**
* Constructs a new object.
*
* @param array $configuration
* A configuration array containing information about the plugin instance.
* @param string $plugin_id
* The plugin_id for the plugin instance.
* @param mixed $plugin_definition
* The plugin implementation definition.
* @param \Drupal\Core\Language\LanguageManagerInterface $language_manager
* The language manager.
* @param \Drupal\Core\Entity\EntityTypeManagerInterface $entity_type_manager
* The entity type manager.
*/
public function __construct(array $configuration, $plugin_id, $plugin_definition, LanguageManagerInterface $language_manager, EntityTypeManagerInterface $entity_type_manager) {
parent::__construct($configuration, $plugin_id, $plugin_definition);
$this->languageManager = $language_manager;
$this->entityTypeManager = $entity_type_manager;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
return new static(
$configuration,
$plugin_id,
$plugin_definition,
$container->get('language_manager'),
$container->get('entity_type.manager')
);
}
/**
* {@inheritdoc}
*/
public function build(FacetInterface $facet, array $results) {
$language_interface = $this->languageManager->getCurrentLanguage();
/** @var \Drupal\Core\TypedData\DataDefinitionInterface $data_definition */
$data_definition = $facet->getDataDefinition();
$property = NULL;
foreach ($data_definition->getPropertyDefinitions() as $k => $definition) {
if ($definition instanceof DataReferenceDefinitionInterface) {
$property = $k;
break;
}
}
if ($property === NULL) {
throw new InvalidProcessorException("Field doesn't have an entity definition, so this processor doesn't work.");
}
$entity_type = $data_definition
->getPropertyDefinition($property)
->getTargetDefinition()
->getEntityTypeId();
/** @var \Drupal\facets\Result\ResultInterface $result */
$ids = [];
foreach ($results as $delta => $result) {
$ids[$delta] = $result->getRawValue();
}
// Load all indexed entities of this type.
$entities = $this->entityTypeManager
->getStorage($entity_type)
->loadMultiple($ids);
$access = $this->entityTypeManager->getAccessControlHandler($entity_type);
$this->checkEntitiesAccess($entities, $facet, $access);
// Loop over all results.
foreach ($results as $i => $result) {
if (!isset($entities[$ids[$i]])) {
if ($result->isMissing()) {
$results[$i]->setDisplayValue($facet->getMissingLabel());
}
else {
unset($results[$i]);
}
continue;
}
/** @var \Drupal\Core\Entity\ContentEntityBase $entity */
$entity = $entities[$ids[$i]];
// Check for a translation of the entity and load that instead if one's
// found.
if ($entity instanceof TranslatableInterface && $entity->hasTranslation($language_interface->getId())) {
$entity = $entity->getTranslation($language_interface->getId());
}
$facet->addCacheableDependency($entity);
// Overwrite the result's display value.
$results[$i]->setDisplayValue($entity->label());
}
// Return the results with the new display values.
return $results;
}
/**
* {@inheritdoc}
*/
public function supportsFacet(FacetInterface $facet) {
$data_definition = $facet->getDataDefinition();
if ($data_definition->getDataType() === 'entity_reference') {
return TRUE;
}
if (!($data_definition instanceof ComplexDataDefinitionInterface)) {
return FALSE;
}
$property_definitions = $data_definition->getPropertyDefinitions();
foreach ($property_definitions as $definition) {
if ($definition instanceof DataReferenceDefinitionInterface) {
return TRUE;
}
}
return FALSE;
}
/**
* {@inheritdoc}
*/
public function getCacheContexts() {
return Cache::mergeContexts(parent::getCacheContexts(), ['languages:language_interface']);
}
/**
* {@inheritdoc}
*/
public function getCacheMaxAge() {
// This will work unless the Search API Query uses "wrong" caching. Ideally
// we would set a cache tag to invalidate the cache whenever a translatable
// entity is added or changed. But there's no tag in drupal yet.
return Cache::PERMANENT;
}
}
