WorkspaceListBuilder.php

Same filename in other branches
  1. 9 core/modules/workspaces/src/WorkspaceListBuilder.php
  2. 8.9.x core/modules/workspaces/src/WorkspaceListBuilder.php
  3. 10 core/modules/workspaces/src/WorkspaceListBuilder.php

Namespace

Drupal\workspaces

File

core/modules/workspaces/src/WorkspaceListBuilder.php

View source
<?php

namespace Drupal\workspaces;

use Drupal\Component\Serialization\Json;
use Drupal\Core\Ajax\AjaxHelperTrait;
use Drupal\Core\Entity\EntityInterface;
use Drupal\Core\Entity\EntityListBuilder;
use Drupal\Core\Entity\EntityStorageInterface;
use Drupal\Core\Entity\EntityTypeInterface;
use Drupal\Core\Render\RendererInterface;
use Drupal\Core\Url;
use Drupal\user\UserInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;

/**
 * Defines a class to build a listing of workspace entities.
 *
 * @see \Drupal\workspaces\Entity\Workspace
 */
class WorkspaceListBuilder extends EntityListBuilder {
    use AjaxHelperTrait;
    
    /**
     * The workspace manager service.
     *
     * @var \Drupal\workspaces\WorkspaceManagerInterface
     */
    protected $workspaceManager;
    
    /**
     * The workspace repository service.
     *
     * @var \Drupal\workspaces\WorkspaceRepositoryInterface
     */
    protected $workspaceRepository;
    
    /**
     * The renderer service.
     *
     * @var \Drupal\Core\Render\RendererInterface
     */
    protected $renderer;
    
    /**
     * Constructs a new EntityListBuilder object.
     *
     * @param \Drupal\Core\Entity\EntityTypeInterface $entity_type
     *   The entity type definition.
     * @param \Drupal\Core\Entity\EntityStorageInterface $storage
     *   The entity storage class.
     * @param \Drupal\workspaces\WorkspaceManagerInterface $workspace_manager
     *   The workspace manager service.
     * @param \Drupal\workspaces\WorkspaceRepositoryInterface $workspace_repository
     *   The workspace repository service.
     * @param \Drupal\Core\Render\RendererInterface $renderer
     *   The renderer service.
     */
    public function __construct(EntityTypeInterface $entity_type, EntityStorageInterface $storage, WorkspaceManagerInterface $workspace_manager, WorkspaceRepositoryInterface $workspace_repository, RendererInterface $renderer) {
        parent::__construct($entity_type, $storage);
        $this->workspaceManager = $workspace_manager;
        $this->workspaceRepository = $workspace_repository;
        $this->renderer = $renderer;
    }
    
    /**
     * {@inheritdoc}
     */
    public static function createInstance(ContainerInterface $container, EntityTypeInterface $entity_type) {
        return new static($entity_type, $container->get('entity_type.manager')
            ->getStorage($entity_type->id()), $container->get('workspaces.manager'), $container->get('workspaces.repository'), $container->get('renderer'));
    }
    
    /**
     * {@inheritdoc}
     */
    public function load() {
        // Get all the workspace entities and sort them in tree order.
        $workspace_tree = $this->workspaceRepository
            ->loadTree();
        $entities = array_replace($workspace_tree, $this->storage
            ->loadMultiple());
        foreach ($entities as $id => $entity) {
            $entity->_depth = $workspace_tree[$id]['depth'];
        }
        return $entities;
    }
    
    /**
     * {@inheritdoc}
     */
    public function buildHeader() {
        $header['label'] = $this->t('Workspace');
        $header['uid'] = $this->t('Owner');
        return $header + parent::buildHeader();
    }
    
    /**
     * {@inheritdoc}
     */
    public function buildRow(EntityInterface $entity) {
        
        /** @var \Drupal\workspaces\WorkspaceInterface $entity */
        if (isset($entity->_depth) && $entity->_depth > 0) {
            $indentation = [
                '#theme' => 'indentation',
                '#size' => $entity->_depth,
            ];
        }
        $row['data'] = [
            'label' => [
                'data' => [
                    '#prefix' => isset($indentation) ? $this->renderer
                        ->render($indentation) : '',
                    '#type' => 'link',
                    '#title' => $entity->label(),
                    '#url' => $entity->toUrl(),
                ],
            ],
            'owner' => ($owner = $entity->getOwner()) && $owner instanceof UserInterface ? $owner->getDisplayName() : $this->t('N/A'),
        ];
        $row['data'] = $row['data'] + parent::buildRow($entity);
        $active_workspace = $this->workspaceManager
            ->getActiveWorkspace();
        if ($active_workspace && $entity->id() === $active_workspace->id()) {
            $row['class'] = [
                'active-workspace',
                'active-workspace--not-default',
            ];
        }
        return $row;
    }
    
    /**
     * {@inheritdoc}
     */
    public function getDefaultOperations(EntityInterface $entity) {
        
        /** @var \Drupal\workspaces\WorkspaceInterface $entity */
        $operations = parent::getDefaultOperations($entity);
        if (isset($operations['edit'])) {
            $operations['edit']['query']['destination'] = $entity->toUrl('collection')
                ->toString();
        }
        $active_workspace = $this->workspaceManager
            ->getActiveWorkspace();
        if (!$active_workspace || $entity->id() != $active_workspace->id()) {
            $operations['activate'] = [
                'title' => $this->t('Switch to @workspace', [
                    '@workspace' => $entity->label(),
                ]),
                // Use a weight lower than the one of the 'Edit' operation because we
                // want the 'Activate' operation to be the primary operation.
'weight' => 0,
                'url' => $entity->toUrl('activate-form', [
                    'query' => [
                        'destination' => $entity->toUrl('collection')
                            ->toString(),
                    ],
                ]),
            ];
        }
        if (!$entity->hasParent()) {
            $operations['publish'] = [
                'title' => $this->t('Publish content'),
                // The 'Publish' operation should be the default one for the currently
                // active workspace.
'weight' => $active_workspace && $entity->id() == $active_workspace->id() ? 0 : 20,
                'url' => Url::fromRoute('entity.workspace.publish_form', [
                    'workspace' => $entity->id(),
                ], [
                    'query' => [
                        'destination' => $entity->toUrl('collection')
                            ->toString(),
                    ],
                ]),
            ];
        }
        else {
            
            /** @var \Drupal\workspaces\WorkspaceInterface $parent */
            $parent = $entity->parent->entity;
            $operations['merge'] = [
                'title' => $this->t('Merge into @target_label', [
                    '@target_label' => $parent->label(),
                ]),
                'weight' => 5,
                'url' => Url::fromRoute('entity.workspace.merge_form', [
                    'source_workspace' => $entity->id(),
                    'target_workspace' => $parent->id(),
                ], [
                    'query' => [
                        'destination' => $entity->toUrl('collection')
                            ->toString(),
                    ],
                ]),
            ];
        }
        $operations['manage'] = [
            'title' => $this->t('Manage'),
            'weight' => 5,
            'url' => $entity->toUrl(),
        ];
        return $operations;
    }
    
    /**
     * {@inheritdoc}
     */
    public function render() {
        $build = parent::render();
        if ($this->isAjax()) {
            $this->offCanvasRender($build);
        }
        else {
            // Add a row for switching to Live.
            $has_active_workspace = $this->workspaceManager
                ->hasActiveWorkspace();
            $row_live = [
                'data' => [
                    'label' => [
                        'data' => [
                            '#markup' => $this->t('Live'),
                        ],
                    ],
                    'owner' => '',
                    'operations' => [
                        'data' => [
                            '#type' => 'operations',
                            '#links' => [
                                'activate' => [
                                    'title' => 'Switch to Live',
                                    'weight' => 0,
                                    'url' => Url::fromRoute('workspaces.switch_to_live', [], [
                                        'query' => $this->getDestinationArray(),
                                    ]),
                                ],
                            ],
                            '#access' => $has_active_workspace,
                        ],
                    ],
                ],
            ];
            if (!$has_active_workspace) {
                $row_live['class'] = [
                    'active-workspace',
                    'active-workspace--default',
                ];
            }
            array_unshift($build['table']['#rows'], $row_live);
            $build['#attached'] = [
                'library' => [
                    'workspaces/drupal.workspaces.overview',
                ],
            ];
        }
        return $build;
    }
    
    /**
     * Renders the off canvas elements.
     *
     * @param array $build
     *   A render array.
     */
    protected function offCanvasRender(array &$build) {
        $active_workspace = $this->workspaceManager
            ->getActiveWorkspace();
        if ($active_workspace) {
            $active_workspace_classes = [
                'active-workspace--not-default',
                'active-workspace--' . $active_workspace->id(),
            ];
        }
        else {
            $active_workspace_classes = [
                'active-workspace--default',
            ];
        }
        $build['active_workspace'] = [
            '#type' => 'container',
            '#weight' => -20,
            '#attributes' => [
                'class' => array_merge([
                    'active-workspace',
                ], $active_workspace_classes),
            ],
            'title' => [
                '#type' => 'html_tag',
                '#tag' => 'div',
                '#value' => $this->t('Current workspace:'),
                '#attributes' => [
                    'class' => 'active-workspace__title',
                ],
            ],
            'label' => [
                '#type' => 'container',
                '#attributes' => [
                    'class' => 'active-workspace__label',
                ],
                'value' => [
                    '#type' => 'html_tag',
                    '#tag' => 'span',
                    '#value' => $active_workspace ? $active_workspace->label() : $this->t('Live'),
                ],
            ],
        ];
        if ($active_workspace) {
            $build['active_workspace']['label']['manage'] = [
                '#type' => 'link',
                '#title' => $this->t('Manage workspace'),
                '#url' => $active_workspace->toUrl('canonical'),
                '#attributes' => [
                    'class' => [
                        'active-workspace__manage',
                    ],
                ],
            ];
            $build['active_workspace']['actions'] = [
                '#type' => 'container',
                '#weight' => 20,
                '#attributes' => [
                    'class' => [
                        'active-workspace__actions',
                    ],
                ],
            ];
            if (!$active_workspace->hasParent()) {
                $build['active_workspace']['actions']['publish'] = [
                    '#type' => 'link',
                    '#title' => $this->t('Publish content'),
                    '#url' => Url::fromRoute('entity.workspace.publish_form', [
                        'workspace' => $active_workspace->id(),
                    ], [
                        'query' => [
                            'destination' => $active_workspace->toUrl('collection')
                                ->toString(),
                        ],
                    ]),
                    '#attributes' => [
                        'class' => [
                            'button',
                            'button--primary',
                            'active-workspace__button',
                        ],
                    ],
                ];
            }
            else {
                $build['active_workspace']['actions']['merge'] = [
                    '#type' => 'link',
                    '#title' => $this->t('Merge content'),
                    '#url' => Url::fromRoute('entity.workspace.merge_form', [
                        'source_workspace' => $active_workspace->id(),
                        'target_workspace' => $active_workspace->parent->target_id,
                    ], [
                        'query' => [
                            'destination' => $active_workspace->toUrl('collection')
                                ->toString(),
                        ],
                    ]),
                    '#attributes' => [
                        'class' => [
                            'button',
                            'button--primary',
                            'active-workspace__button',
                        ],
                    ],
                ];
            }
        }
        $items = [];
        $rows = array_slice($build['table']['#rows'], 0, 5, TRUE);
        foreach ($rows as $id => $row) {
            if (!$active_workspace || $active_workspace->id() !== $id) {
                $url = Url::fromRoute('entity.workspace.activate_form', [
                    'workspace' => $id,
                ], [
                    'query' => $this->getDestinationArray(),
                ]);
                $items[] = [
                    '#type' => 'link',
                    '#title' => ltrim($row['data']['label']['data']['#title']),
                    '#url' => $url,
                    '#attributes' => [
                        'class' => [
                            'use-ajax',
                            'workspaces__item',
                            'workspaces__item--not-default',
                        ],
                        'data-dialog-type' => 'modal',
                        'data-dialog-options' => Json::encode([
                            'width' => 500,
                        ]),
                    ],
                ];
            }
        }
        // Add an item for switching to Live.
        if ($active_workspace) {
            $items[] = [
                '#type' => 'link',
                '#title' => $this->t('Live'),
                '#url' => Url::fromRoute('workspaces.switch_to_live', [], [
                    'query' => $this->getDestinationArray(),
                ]),
                '#attributes' => [
                    'class' => [
                        'use-ajax',
                        'workspaces__item',
                        'workspaces__item--default',
                    ],
                    'data-dialog-type' => 'modal',
                    'data-dialog-options' => Json::encode([
                        'width' => 500,
                    ]),
                ],
            ];
        }
        $build['workspaces_list'] = [
            '#type' => 'container',
            '#attributes' => [
                'class' => 'workspaces',
            ],
        ];
        $build['workspaces_list']['workspaces'] = [
            '#theme' => 'item_list',
            '#title' => $this->t('Other workspaces:'),
            '#items' => $items,
            '#wrapper_attributes' => [
                'class' => [
                    'workspaces__list',
                ],
            ],
            '#cache' => [
                'contexts' => $this->entityType
                    ->getListCacheContexts(),
                'tags' => $this->entityType
                    ->getListCacheTags(),
            ],
        ];
        $build['workspaces_list']['all_workspaces'] = [
            '#type' => 'link',
            '#title' => $this->t('View all workspaces'),
            '#url' => Url::fromRoute('entity.workspace.collection'),
            '#attributes' => [
                'class' => [
                    'all-workspaces',
                ],
            ],
        ];
        unset($build['table']);
        unset($build['pager']);
    }

}

Classes

Title Deprecated Summary
WorkspaceListBuilder Defines a class to build a listing of workspace entities.

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