annotate core/modules/layout_builder/src/Element/LayoutBuilder.php @ 5:12f9dff5fda9 tip

Update to Drupal core 8.7.1
author Chris Cannam
date Thu, 09 May 2019 15:34:47 +0100
parents
children
rev   line source
Chris@5 1 <?php
Chris@5 2
Chris@5 3 namespace Drupal\layout_builder\Element;
Chris@5 4
Chris@5 5 use Drupal\Core\Ajax\AjaxHelperTrait;
Chris@5 6 use Drupal\Core\Messenger\MessengerInterface;
Chris@5 7 use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
Chris@5 8 use Drupal\Core\Plugin\PluginFormInterface;
Chris@5 9 use Drupal\Core\Render\Element;
Chris@5 10 use Drupal\Core\Render\Element\RenderElement;
Chris@5 11 use Drupal\Core\Url;
Chris@5 12 use Drupal\layout_builder\Context\LayoutBuilderContextTrait;
Chris@5 13 use Drupal\layout_builder\LayoutBuilderHighlightTrait;
Chris@5 14 use Drupal\layout_builder\LayoutTempstoreRepositoryInterface;
Chris@5 15 use Drupal\layout_builder\OverridesSectionStorageInterface;
Chris@5 16 use Drupal\layout_builder\SectionStorageInterface;
Chris@5 17 use Symfony\Component\DependencyInjection\ContainerInterface;
Chris@5 18
Chris@5 19 /**
Chris@5 20 * Defines a render element for building the Layout Builder UI.
Chris@5 21 *
Chris@5 22 * @RenderElement("layout_builder")
Chris@5 23 *
Chris@5 24 * @internal
Chris@5 25 * Plugin classes are internal.
Chris@5 26 */
Chris@5 27 class LayoutBuilder extends RenderElement implements ContainerFactoryPluginInterface {
Chris@5 28
Chris@5 29 use AjaxHelperTrait;
Chris@5 30 use LayoutBuilderContextTrait;
Chris@5 31 use LayoutBuilderHighlightTrait;
Chris@5 32
Chris@5 33 /**
Chris@5 34 * The layout tempstore repository.
Chris@5 35 *
Chris@5 36 * @var \Drupal\layout_builder\LayoutTempstoreRepositoryInterface
Chris@5 37 */
Chris@5 38 protected $layoutTempstoreRepository;
Chris@5 39
Chris@5 40 /**
Chris@5 41 * The messenger service.
Chris@5 42 *
Chris@5 43 * @var \Drupal\Core\Messenger\MessengerInterface
Chris@5 44 */
Chris@5 45 protected $messenger;
Chris@5 46
Chris@5 47 /**
Chris@5 48 * Constructs a new LayoutBuilder.
Chris@5 49 *
Chris@5 50 * @param array $configuration
Chris@5 51 * A configuration array containing information about the plugin instance.
Chris@5 52 * @param string $plugin_id
Chris@5 53 * The plugin ID for the plugin instance.
Chris@5 54 * @param mixed $plugin_definition
Chris@5 55 * The plugin implementation definition.
Chris@5 56 * @param \Drupal\layout_builder\LayoutTempstoreRepositoryInterface $layout_tempstore_repository
Chris@5 57 * The layout tempstore repository.
Chris@5 58 * @param \Drupal\Core\Messenger\MessengerInterface $messenger
Chris@5 59 * The messenger service.
Chris@5 60 */
Chris@5 61 public function __construct(array $configuration, $plugin_id, $plugin_definition, LayoutTempstoreRepositoryInterface $layout_tempstore_repository, MessengerInterface $messenger) {
Chris@5 62 parent::__construct($configuration, $plugin_id, $plugin_definition);
Chris@5 63 $this->layoutTempstoreRepository = $layout_tempstore_repository;
Chris@5 64 $this->messenger = $messenger;
Chris@5 65 }
Chris@5 66
Chris@5 67 /**
Chris@5 68 * {@inheritdoc}
Chris@5 69 */
Chris@5 70 public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
Chris@5 71 return new static(
Chris@5 72 $configuration,
Chris@5 73 $plugin_id,
Chris@5 74 $plugin_definition,
Chris@5 75 $container->get('layout_builder.tempstore_repository'),
Chris@5 76 $container->get('messenger')
Chris@5 77 );
Chris@5 78 }
Chris@5 79
Chris@5 80 /**
Chris@5 81 * {@inheritdoc}
Chris@5 82 */
Chris@5 83 public function getInfo() {
Chris@5 84 return [
Chris@5 85 '#section_storage' => NULL,
Chris@5 86 '#pre_render' => [
Chris@5 87 [$this, 'preRender'],
Chris@5 88 ],
Chris@5 89 ];
Chris@5 90 }
Chris@5 91
Chris@5 92 /**
Chris@5 93 * Pre-render callback: Renders the Layout Builder UI.
Chris@5 94 */
Chris@5 95 public function preRender($element) {
Chris@5 96 if ($element['#section_storage'] instanceof SectionStorageInterface) {
Chris@5 97 $element['layout_builder'] = $this->layout($element['#section_storage']);
Chris@5 98 }
Chris@5 99 return $element;
Chris@5 100 }
Chris@5 101
Chris@5 102 /**
Chris@5 103 * Renders the Layout UI.
Chris@5 104 *
Chris@5 105 * @param \Drupal\layout_builder\SectionStorageInterface $section_storage
Chris@5 106 * The section storage.
Chris@5 107 *
Chris@5 108 * @return array
Chris@5 109 * A render array.
Chris@5 110 */
Chris@5 111 protected function layout(SectionStorageInterface $section_storage) {
Chris@5 112 $this->prepareLayout($section_storage);
Chris@5 113
Chris@5 114 $output = [];
Chris@5 115 if ($this->isAjax()) {
Chris@5 116 $output['status_messages'] = [
Chris@5 117 '#type' => 'status_messages',
Chris@5 118 ];
Chris@5 119 }
Chris@5 120 $count = 0;
Chris@5 121 for ($i = 0; $i < $section_storage->count(); $i++) {
Chris@5 122 $output[] = $this->buildAddSectionLink($section_storage, $count);
Chris@5 123 $output[] = $this->buildAdministrativeSection($section_storage, $count);
Chris@5 124 $count++;
Chris@5 125 }
Chris@5 126 $output[] = $this->buildAddSectionLink($section_storage, $count);
Chris@5 127 $output['#attached']['library'][] = 'layout_builder/drupal.layout_builder';
Chris@5 128 // As the Layout Builder UI is typically displayed using the frontend theme,
Chris@5 129 // it is not marked as an administrative page at the route level even though
Chris@5 130 // it performs an administrative task. Mark this as an administrative page
Chris@5 131 // for JavaScript.
Chris@5 132 $output['#attached']['drupalSettings']['path']['currentPathIsAdmin'] = TRUE;
Chris@5 133 $output['#type'] = 'container';
Chris@5 134 $output['#attributes']['id'] = 'layout-builder';
Chris@5 135 $output['#attributes']['class'][] = 'layout-builder';
Chris@5 136 // Mark this UI as uncacheable.
Chris@5 137 $output['#cache']['max-age'] = 0;
Chris@5 138 return $output;
Chris@5 139 }
Chris@5 140
Chris@5 141 /**
Chris@5 142 * Prepares a layout for use in the UI.
Chris@5 143 *
Chris@5 144 * @param \Drupal\layout_builder\SectionStorageInterface $section_storage
Chris@5 145 * The section storage.
Chris@5 146 */
Chris@5 147 protected function prepareLayout(SectionStorageInterface $section_storage) {
Chris@5 148 // If the layout has pending changes, add a warning.
Chris@5 149 if ($this->layoutTempstoreRepository->has($section_storage)) {
Chris@5 150 $this->messenger->addWarning($this->t('You have unsaved changes.'));
Chris@5 151 }
Chris@5 152 // If the layout is an override that has not yet been overridden, copy the
Chris@5 153 // sections from the corresponding default.
Chris@5 154 elseif ($section_storage instanceof OverridesSectionStorageInterface && !$section_storage->isOverridden()) {
Chris@5 155 $sections = $section_storage->getDefaultSectionStorage()->getSections();
Chris@5 156 foreach ($sections as $section) {
Chris@5 157 $section_storage->appendSection($section);
Chris@5 158 }
Chris@5 159 $this->layoutTempstoreRepository->set($section_storage);
Chris@5 160 }
Chris@5 161 }
Chris@5 162
Chris@5 163 /**
Chris@5 164 * Builds a link to add a new section at a given delta.
Chris@5 165 *
Chris@5 166 * @param \Drupal\layout_builder\SectionStorageInterface $section_storage
Chris@5 167 * The section storage.
Chris@5 168 * @param int $delta
Chris@5 169 * The delta of the section to splice.
Chris@5 170 *
Chris@5 171 * @return array
Chris@5 172 * A render array for a link.
Chris@5 173 */
Chris@5 174 protected function buildAddSectionLink(SectionStorageInterface $section_storage, $delta) {
Chris@5 175 $storage_type = $section_storage->getStorageType();
Chris@5 176 $storage_id = $section_storage->getStorageId();
Chris@5 177
Chris@5 178 // If the delta and the count are the same, it is either the end of the
Chris@5 179 // layout or an empty layout.
Chris@5 180 if ($delta === count($section_storage)) {
Chris@5 181 if ($delta === 0) {
Chris@5 182 $title = $this->t('Add Section');
Chris@5 183 }
Chris@5 184 else {
Chris@5 185 $title = $this->t('Add Section <span class="visually-hidden">at end of layout</span>');
Chris@5 186 }
Chris@5 187 }
Chris@5 188 // If the delta and the count are different, it is either the beginning of
Chris@5 189 // the layout or in between two sections.
Chris@5 190 else {
Chris@5 191 if ($delta === 0) {
Chris@5 192 $title = $this->t('Add Section <span class="visually-hidden">at start of layout</span>');
Chris@5 193 }
Chris@5 194 else {
Chris@5 195 $title = $this->t('Add Section <span class="visually-hidden">between @first and @second</span>', ['@first' => $delta, '@second' => $delta + 1]);
Chris@5 196 }
Chris@5 197 }
Chris@5 198
Chris@5 199 return [
Chris@5 200 'link' => [
Chris@5 201 '#type' => 'link',
Chris@5 202 '#title' => $title,
Chris@5 203 '#url' => Url::fromRoute('layout_builder.choose_section',
Chris@5 204 [
Chris@5 205 'section_storage_type' => $storage_type,
Chris@5 206 'section_storage' => $storage_id,
Chris@5 207 'delta' => $delta,
Chris@5 208 ],
Chris@5 209 [
Chris@5 210 'attributes' => [
Chris@5 211 'class' => [
Chris@5 212 'use-ajax',
Chris@5 213 'layout-builder__link',
Chris@5 214 'layout-builder__link--add',
Chris@5 215 ],
Chris@5 216 'data-dialog-type' => 'dialog',
Chris@5 217 'data-dialog-renderer' => 'off_canvas',
Chris@5 218 ],
Chris@5 219 ]
Chris@5 220 ),
Chris@5 221 ],
Chris@5 222 '#type' => 'container',
Chris@5 223 '#attributes' => [
Chris@5 224 'class' => ['layout-builder__add-section'],
Chris@5 225 'data-layout-builder-highlight-id' => $this->sectionAddHighlightId($delta),
Chris@5 226 ],
Chris@5 227 ];
Chris@5 228 }
Chris@5 229
Chris@5 230 /**
Chris@5 231 * Builds the render array for the layout section while editing.
Chris@5 232 *
Chris@5 233 * @param \Drupal\layout_builder\SectionStorageInterface $section_storage
Chris@5 234 * The section storage.
Chris@5 235 * @param int $delta
Chris@5 236 * The delta of the section.
Chris@5 237 *
Chris@5 238 * @return array
Chris@5 239 * The render array for a given section.
Chris@5 240 */
Chris@5 241 protected function buildAdministrativeSection(SectionStorageInterface $section_storage, $delta) {
Chris@5 242 $storage_type = $section_storage->getStorageType();
Chris@5 243 $storage_id = $section_storage->getStorageId();
Chris@5 244 $section = $section_storage->getSection($delta);
Chris@5 245
Chris@5 246 $layout = $section->getLayout();
Chris@5 247 $build = $section->toRenderArray($this->getAvailableContexts($section_storage), TRUE);
Chris@5 248 $layout_definition = $layout->getPluginDefinition();
Chris@5 249
Chris@5 250 $region_labels = $layout_definition->getRegionLabels();
Chris@5 251 foreach ($layout_definition->getRegions() as $region => $info) {
Chris@5 252 if (!empty($build[$region])) {
Chris@5 253 foreach (Element::children($build[$region]) as $uuid) {
Chris@5 254 $build[$region][$uuid]['#attributes']['class'][] = 'js-layout-builder-block';
Chris@5 255 $build[$region][$uuid]['#attributes']['class'][] = 'layout-builder-block';
Chris@5 256 $build[$region][$uuid]['#attributes']['data-layout-block-uuid'] = $uuid;
Chris@5 257 $build[$region][$uuid]['#attributes']['data-layout-builder-highlight-id'] = $this->blockUpdateHighlightId($uuid);
Chris@5 258 $build[$region][$uuid]['#contextual_links'] = [
Chris@5 259 'layout_builder_block' => [
Chris@5 260 'route_parameters' => [
Chris@5 261 'section_storage_type' => $storage_type,
Chris@5 262 'section_storage' => $storage_id,
Chris@5 263 'delta' => $delta,
Chris@5 264 'region' => $region,
Chris@5 265 'uuid' => $uuid,
Chris@5 266 ],
Chris@5 267 // Add metadata about the current operations available in
Chris@5 268 // contextual links. This will invalidate the client-side cache of
Chris@5 269 // links that were cached before the 'move' link was added.
Chris@5 270 // @see layout_builder.links.contextual.yml
Chris@5 271 'metadata' => [
Chris@5 272 'operations' => 'move:update:remove',
Chris@5 273 ],
Chris@5 274 ],
Chris@5 275 ];
Chris@5 276 }
Chris@5 277 }
Chris@5 278
Chris@5 279 $build[$region]['layout_builder_add_block']['link'] = [
Chris@5 280 '#type' => 'link',
Chris@5 281 // Add one to the current delta since it is zero-indexed.
Chris@5 282 '#title' => $this->t('Add Block <span class="visually-hidden">in section @section, @region region</span>', ['@section' => $delta + 1, '@region' => $region_labels[$region]]),
Chris@5 283 '#url' => Url::fromRoute('layout_builder.choose_block',
Chris@5 284 [
Chris@5 285 'section_storage_type' => $storage_type,
Chris@5 286 'section_storage' => $storage_id,
Chris@5 287 'delta' => $delta,
Chris@5 288 'region' => $region,
Chris@5 289 ],
Chris@5 290 [
Chris@5 291 'attributes' => [
Chris@5 292 'class' => [
Chris@5 293 'use-ajax',
Chris@5 294 'layout-builder__link',
Chris@5 295 'layout-builder__link--add',
Chris@5 296 ],
Chris@5 297 'data-dialog-type' => 'dialog',
Chris@5 298 'data-dialog-renderer' => 'off_canvas',
Chris@5 299 ],
Chris@5 300 ]
Chris@5 301 ),
Chris@5 302 ];
Chris@5 303 $build[$region]['layout_builder_add_block']['#type'] = 'container';
Chris@5 304 $build[$region]['layout_builder_add_block']['#attributes'] = [
Chris@5 305 'class' => ['layout-builder__add-block'],
Chris@5 306 'data-layout-builder-highlight-id' => $this->blockAddHighlightId($delta, $region),
Chris@5 307 ];
Chris@5 308 $build[$region]['layout_builder_add_block']['#weight'] = 1000;
Chris@5 309 $build[$region]['#attributes']['data-region'] = $region;
Chris@5 310 $build[$region]['#attributes']['class'][] = 'layout-builder__region';
Chris@5 311 $build[$region]['#attributes']['class'][] = 'js-layout-builder-region';
Chris@5 312 $build[$region]['#attributes']['role'] = 'group';
Chris@5 313 $build[$region]['#attributes']['aria-label'] = $this->t('@region region in section @section', [
Chris@5 314 '@region' => $info['label'],
Chris@5 315 '@section' => $delta + 1,
Chris@5 316 ]);
Chris@5 317
Chris@5 318 // Get weights of all children for use by the region label.
Chris@5 319 $weights = array_map(function ($a) {
Chris@5 320 return isset($a['#weight']) ? $a['#weight'] : 0;
Chris@5 321 }, $build[$region]);
Chris@5 322
Chris@5 323 // The region label is made visible when the move block dialog is open.
Chris@5 324 $build[$region]['region_label'] = [
Chris@5 325 '#type' => 'container',
Chris@5 326 '#attributes' => [
Chris@5 327 'class' => ['layout__region-info', 'layout-builder__region-label'],
Chris@5 328 // A more detailed version of this information is already read by
Chris@5 329 // screen readers, so this label can be hidden from them.
Chris@5 330 'aria-hidden' => TRUE,
Chris@5 331 ],
Chris@5 332 '#markup' => $this->t('Region: @region', ['@region' => $info['label']]),
Chris@5 333 // Ensures the region label is displayed first.
Chris@5 334 '#weight' => min($weights) - 1,
Chris@5 335 ];
Chris@5 336 }
Chris@5 337
Chris@5 338 $build['#attributes']['data-layout-update-url'] = Url::fromRoute('layout_builder.move_block', [
Chris@5 339 'section_storage_type' => $storage_type,
Chris@5 340 'section_storage' => $storage_id,
Chris@5 341 ])->toString();
Chris@5 342
Chris@5 343 $build['#attributes']['data-layout-delta'] = $delta;
Chris@5 344 $build['#attributes']['class'][] = 'layout-builder__layout';
Chris@5 345 $build['#attributes']['data-layout-builder-highlight-id'] = $this->sectionUpdateHighlightId($delta);
Chris@5 346
Chris@5 347 return [
Chris@5 348 '#type' => 'container',
Chris@5 349 '#attributes' => [
Chris@5 350 'class' => ['layout-builder__section'],
Chris@5 351 'role' => 'group',
Chris@5 352 'aria-label' => $this->t('Section @section', ['@section' => $delta + 1]),
Chris@5 353 ],
Chris@5 354 'remove' => [
Chris@5 355 '#type' => 'link',
Chris@5 356 '#title' => $this->t('Remove section <span class="visually-hidden">@section</span>', ['@section' => $delta + 1]),
Chris@5 357 '#url' => Url::fromRoute('layout_builder.remove_section', [
Chris@5 358 'section_storage_type' => $storage_type,
Chris@5 359 'section_storage' => $storage_id,
Chris@5 360 'delta' => $delta,
Chris@5 361 ]),
Chris@5 362 '#attributes' => [
Chris@5 363 'class' => [
Chris@5 364 'use-ajax',
Chris@5 365 'layout-builder__link',
Chris@5 366 'layout-builder__link--remove',
Chris@5 367 ],
Chris@5 368 'data-dialog-type' => 'dialog',
Chris@5 369 'data-dialog-renderer' => 'off_canvas',
Chris@5 370 ],
Chris@5 371 ],
Chris@5 372 // The section label is added to sections without a "Configure Section"
Chris@5 373 // link, and is only visible when the move block dialog is open.
Chris@5 374 'section_label' => [
Chris@5 375 '#markup' => $this->t('<span class="layout-builder__section-label" aria-hidden="true">Section @section</span>', ['@section' => $delta + 1]),
Chris@5 376 '#access' => !$layout instanceof PluginFormInterface,
Chris@5 377 ],
Chris@5 378 'configure' => [
Chris@5 379 '#type' => 'link',
Chris@5 380 // There are two instances of @section, the one wrapped in
Chris@5 381 // .visually-hidden is for screen readers. The one wrapped in
Chris@5 382 // .layout-builder__section-label is only visible when the
Chris@5 383 // move block dialog is open and it is not seen by screen readers.
Chris@5 384 '#title' => $this->t('Configure section <span class="visually-hidden">@section</span><span aria-hidden="true" class="layout-builder__section-label">@section</span>', ['@section' => $delta + 1]),
Chris@5 385 '#access' => $layout instanceof PluginFormInterface,
Chris@5 386 '#url' => Url::fromRoute('layout_builder.configure_section', [
Chris@5 387 'section_storage_type' => $storage_type,
Chris@5 388 'section_storage' => $storage_id,
Chris@5 389 'delta' => $delta,
Chris@5 390 ]),
Chris@5 391 '#attributes' => [
Chris@5 392 'class' => [
Chris@5 393 'use-ajax',
Chris@5 394 'layout-builder__link',
Chris@5 395 'layout-builder__link--configure',
Chris@5 396 ],
Chris@5 397 'data-dialog-type' => 'dialog',
Chris@5 398 'data-dialog-renderer' => 'off_canvas',
Chris@5 399 ],
Chris@5 400 ],
Chris@5 401 'layout-builder__section' => $build,
Chris@5 402 ];
Chris@5 403 }
Chris@5 404
Chris@5 405 }