EntityOperations.php

Namespace

Drupal\workspaces\Hook

File

core/modules/workspaces/src/Hook/EntityOperations.php

View source
<?php

declare (strict_types=1);
namespace Drupal\workspaces\Hook;

use Drupal\Core\Entity\EntityFormInterface;
use Drupal\Core\Entity\EntityInterface;
use Drupal\Core\Entity\EntityTypeManagerInterface;
use Drupal\Core\Entity\Query\QueryInterface;
use Drupal\Core\Entity\RevisionableInterface;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Hook\Attribute\Hook;
use Drupal\Core\Hook\Attribute\ReorderHook;
use Drupal\Core\Hook\Order\Order;
use Drupal\Core\Hook\Order\OrderBefore;
use Drupal\content_moderation\Hook\ContentModerationHooks;
use Drupal\workspaces\WorkspaceTrackerInterface;
use Drupal\workspaces\WorkspaceInformationInterface;
use Drupal\workspaces\WorkspaceManagerInterface;
use Drupal\workspaces\WorkspaceRepositoryInterface;

/**
 * Defines a class for reacting to entity runtime hooks.
 */
class EntityOperations {
  public function __construct(protected EntityTypeManagerInterface $entityTypeManager, protected WorkspaceManagerInterface $workspaceManager, protected WorkspaceTrackerInterface $workspaceTracker, protected WorkspaceInformationInterface $workspaceInfo, protected WorkspaceRepositoryInterface $workspaceRepository) {
  }
  
  /**
   * Implements hook_entity_preload().
   */
  public function entityPreload(array $ids, string $entity_type_id) : array {
    $entity_type = $this->entityTypeManager
      ->getDefinition($entity_type_id);
    if (!$this->workspaceInfo
      ->isEntityTypeSupported($entity_type)) {
      return [];
    }
    return $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityPreload($ids, $entity_type_id) ?? [];
  }
  
  /**
   * Implements hook_entity_presave().
   */
  public function entityPresave(EntityInterface $entity) : void {
    if ($this->workspaceInfo
      ->isEntityIgnored($entity)) {
      return;
    }
    $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityPresave($entity);
  }
  
  /**
   * Implements hook_entity_insert().
   */
  public function entityInsert(EntityInterface $entity) : void {
    if ($this->workspaceInfo
      ->isEntityIgnored($entity) || !$this->workspaceInfo
      ->isEntitySupported($entity)) {
      return;
    }
    $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityInsert($entity);
  }
  
  /**
   * Implements hook_entity_update().
   */
  public function entityUpdate(EntityInterface $entity) : void {
    if ($this->workspaceInfo
      ->isEntityIgnored($entity) || !$this->workspaceInfo
      ->isEntitySupported($entity)) {
      return;
    }
    $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityUpdate($entity);
  }
  
  /**
   * Implements hook_entity_translation_insert().
   */
  public function entityTranslationInsert(EntityInterface $translation) : void {
    if ($this->workspaceInfo
      ->isEntityIgnored($translation) || !$this->workspaceInfo
      ->isEntitySupported($translation) || $translation->isSyncing()) {
      return;
    }
    $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityTranslationInsert($translation);
  }
  
  /**
   * Implements hook_entity_predelete().
   */
  public function entityPredelete(EntityInterface $entity) : void {
    if ($entity->getEntityTypeId() === 'workspace') {
      $this->workspaceRepository
        ->resetCache();
    }
    if ($this->workspaceInfo
      ->isEntityIgnored($entity)) {
      return;
    }
    $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityPredelete($entity);
  }
  
  /**
   * Implements hook_entity_delete().
   */
  public function entityDelete(EntityInterface $entity) : void {
    if (!$this->workspaceInfo
      ->isEntityTypeSupported($entity->getEntityType())) {
      return;
    }
    $this->workspaceTracker
      ->deleteTrackedEntities(NULL, $entity->getEntityTypeId(), [
      $entity->id(),
    ]);
    $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityDelete($entity);
  }
  
  /**
   * Implements hook_entity_revision_delete().
   */
  public function entityRevisionDelete(EntityInterface $entity) : void {
    if (!$this->workspaceInfo
      ->isEntityTypeSupported($entity->getEntityType())) {
      return;
    }
    $this->workspaceTracker
      ->deleteTrackedEntities(NULL, $entity->getEntityTypeId(), [
      $entity->id(),
    ], [
      $entity->getRevisionId(),
    ]);
    $this->workspaceManager
      ->getActiveWorkspace()?->getProvider()
      ->entityRevisionDelete($entity);
  }
  
  /**
   * Implements hook_entity_query_tag__TAG_alter() for 'latest_translated_affected_revision'.
   */
  public function entityQueryTagLatestTranslatedAffectedRevisionAlter(QueryInterface $query) : void {
    $entity_type = $this->entityTypeManager
      ->getDefinition($query->getEntityTypeId());
    if (!$this->workspaceInfo
      ->isEntityTypeSupported($entity_type) || !$this->workspaceManager
      ->hasActiveWorkspace()) {
      return;
    }
    $active_workspace = $this->workspaceManager
      ->getActiveWorkspace();
    $tracked_entities = $this->workspaceTracker
      ->getTrackedEntities($active_workspace->id());
    if (!isset($tracked_entities[$entity_type->id()])) {
      return;
    }
    if ($revision_id = array_search($query->getMetaData('entity_id'), $tracked_entities[$entity_type->id()])) {
      $query->condition($entity_type->getKey('revision'), $revision_id, '<=');
      $conditions = $query->orConditionGroup();
      $conditions->condition($entity_type->getRevisionMetadataKey('workspace'), $active_workspace->id());
      $conditions->condition($entity_type->getRevisionMetadataKey('revision_default'), TRUE);
      $query->condition($conditions);
    }
  }
  
  /**
   * Implements hook_form_alter().
   *
   * Alters entity forms to disallow concurrent editing in multiple workspaces.
   */
  public function entityFormAlter(array &$form, FormStateInterface $form_state, string $form_id) : void {
    if (!$form_state->getFormObject() instanceof EntityFormInterface) {
      return;
    }
    $entity = $form_state->getFormObject()
      ->getEntity();
    if (!$this->workspaceInfo
      ->isEntitySupported($entity) && !$this->workspaceInfo
      ->isEntityIgnored($entity)) {
      return;
    }
    // For supported and ignored entity types, signal the fact that this form is
    // safe to use in a workspace.
    // @see \Drupal\workspaces\Hook\FormOperations::formAlter()
    $form_state->set('workspace_safe', TRUE);
    // There is nothing more to do for ignored entity types.
    if ($this->workspaceInfo
      ->isEntityIgnored($entity)) {
      return;
    }
    // Add an entity builder to the form which marks the edited entity object as
    // a pending revision. This is needed so validation constraints like
    // \Drupal\path\Plugin\Validation\Constraint\PathAliasConstraintValidator
    // know in advance (before hook_entity_presave()) that the new revision will
    // be a pending one.
    if ($this->workspaceManager
      ->hasActiveWorkspace()) {
      $form['#entity_builders'][] = [
        static::class,
        'entityFormEntityBuild',
      ];
    }
  }
  
  /**
   * Entity builder that marks all supported entities as pending revisions.
   */
  public static function entityFormEntityBuild(string $entity_type_id, RevisionableInterface $entity, array &$form, FormStateInterface &$form_state) : void {
    // Ensure that all entity forms are signaling that a new revision will be
    // created.
    $entity->setNewRevision(TRUE);
    // Set the non-default revision flag so that validation constraints are also
    // aware that a pending revision is about to be created.
    $entity->isDefaultRevision(FALSE);
  }

}

Classes

Title Deprecated Summary
EntityOperations Defines a class for reacting to entity runtime hooks.

Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.