comparison core/modules/layout_builder/src/Section.php @ 14:1fec387a4317

Update Drupal core to 8.5.2 via Composer
author Chris Cannam
date Mon, 23 Apr 2018 09:46:53 +0100
parents
children c2387f117808
comparison
equal deleted inserted replaced
13:5fb285c0d0e3 14:1fec387a4317
1 <?php
2
3 namespace Drupal\layout_builder;
4
5 /**
6 * Provides a domain object for layout sections.
7 *
8 * A section consists of three parts:
9 * - The layout plugin ID for the layout applied to the section (for example,
10 * 'layout_onecol').
11 * - An array of settings for the layout plugin.
12 * - An array of components that can be rendered in the section.
13 *
14 * @internal
15 * Layout Builder is currently experimental and should only be leveraged by
16 * experimental modules and development releases of contributed modules.
17 * See https://www.drupal.org/core/experimental for more information.
18 *
19 * @see \Drupal\Core\Layout\LayoutDefinition
20 * @see \Drupal\layout_builder\SectionComponent
21 *
22 * @todo Determine whether an interface will be provided for this in
23 * https://www.drupal.org/project/drupal/issues/2930334.
24 */
25 class Section {
26
27 /**
28 * The layout plugin ID.
29 *
30 * @var string
31 */
32 protected $layoutId;
33
34 /**
35 * The layout plugin settings.
36 *
37 * @var array
38 */
39 protected $layoutSettings = [];
40
41 /**
42 * An array of components, keyed by UUID.
43 *
44 * @var \Drupal\layout_builder\SectionComponent[]
45 */
46 protected $components = [];
47
48 /**
49 * Constructs a new Section.
50 *
51 * @param string $layout_id
52 * The layout plugin ID.
53 * @param array $layout_settings
54 * (optional) The layout plugin settings.
55 * @param \Drupal\layout_builder\SectionComponent[] $components
56 * (optional) The components.
57 */
58 public function __construct($layout_id, array $layout_settings = [], array $components = []) {
59 $this->layoutId = $layout_id;
60 $this->layoutSettings = $layout_settings;
61 foreach ($components as $component) {
62 $this->setComponent($component);
63 }
64 }
65
66 /**
67 * Returns the renderable array for this section.
68 *
69 * @param \Drupal\Core\Plugin\Context\ContextInterface[] $contexts
70 * An array of available contexts.
71 * @param bool $in_preview
72 * TRUE if the section is being previewed, FALSE otherwise.
73 *
74 * @return array
75 * A renderable array representing the content of the section.
76 */
77 public function toRenderArray(array $contexts = [], $in_preview = FALSE) {
78 $regions = [];
79 foreach ($this->getComponents() as $component) {
80 if ($output = $component->toRenderArray($contexts, $in_preview)) {
81 $regions[$component->getRegion()][$component->getUuid()] = $output;
82 }
83 }
84
85 return $this->getLayout()->build($regions);
86 }
87
88 /**
89 * Gets the layout plugin for this section.
90 *
91 * @return \Drupal\Core\Layout\LayoutInterface
92 * The layout plugin.
93 */
94 public function getLayout() {
95 return $this->layoutPluginManager()->createInstance($this->getLayoutId(), $this->getLayoutSettings());
96 }
97
98 /**
99 * Gets the layout plugin ID for this section.
100 *
101 * @return string
102 * The layout plugin ID.
103 *
104 * @internal
105 * This method should only be used by code responsible for storing the data.
106 */
107 public function getLayoutId() {
108 return $this->layoutId;
109 }
110
111 /**
112 * Gets the layout plugin settings for this section.
113 *
114 * @return mixed[]
115 * The layout plugin settings.
116 *
117 * @internal
118 * This method should only be used by code responsible for storing the data.
119 */
120 public function getLayoutSettings() {
121 return $this->layoutSettings;
122 }
123
124 /**
125 * Sets the layout plugin settings for this section.
126 *
127 * @param mixed[] $layout_settings
128 * The layout plugin settings.
129 *
130 * @return $this
131 */
132 public function setLayoutSettings(array $layout_settings) {
133 $this->layoutSettings = $layout_settings;
134 return $this;
135 }
136
137 /**
138 * Gets the default region.
139 *
140 * @return string
141 * The machine-readable name of the default region.
142 */
143 public function getDefaultRegion() {
144 return $this->layoutPluginManager()->getDefinition($this->getLayoutId())->getDefaultRegion();
145 }
146
147 /**
148 * Returns the components of the section.
149 *
150 * @return \Drupal\layout_builder\SectionComponent[]
151 * The components.
152 */
153 public function getComponents() {
154 return $this->components;
155 }
156
157 /**
158 * Gets the component for a given UUID.
159 *
160 * @param string $uuid
161 * The UUID of the component to retrieve.
162 *
163 * @return \Drupal\layout_builder\SectionComponent
164 * The component.
165 *
166 * @throws \InvalidArgumentException
167 * Thrown when the expected UUID does not exist.
168 */
169 public function getComponent($uuid) {
170 if (!isset($this->components[$uuid])) {
171 throw new \InvalidArgumentException(sprintf('Invalid UUID "%s"', $uuid));
172 }
173
174 return $this->components[$uuid];
175 }
176
177 /**
178 * Helper method to set a component.
179 *
180 * @param \Drupal\layout_builder\SectionComponent $component
181 * The component.
182 *
183 * @return $this
184 */
185 protected function setComponent(SectionComponent $component) {
186 $this->components[$component->getUuid()] = $component;
187 return $this;
188 }
189
190 /**
191 * Removes a given component from a region.
192 *
193 * @param string $uuid
194 * The UUID of the component to remove.
195 *
196 * @return $this
197 */
198 public function removeComponent($uuid) {
199 unset($this->components[$uuid]);
200 return $this;
201 }
202
203 /**
204 * Appends a component to the end of a region.
205 *
206 * @param \Drupal\layout_builder\SectionComponent $component
207 * The component being appended.
208 *
209 * @return $this
210 */
211 public function appendComponent(SectionComponent $component) {
212 $component->setWeight($this->getNextHighestWeight($component->getRegion()));
213 $this->setComponent($component);
214 return $this;
215 }
216
217 /**
218 * Returns the next highest weight of the component in a region.
219 *
220 * @param string $region
221 * The region name.
222 *
223 * @return int
224 * A number higher than the highest weight of the component in the region.
225 */
226 protected function getNextHighestWeight($region) {
227 $components = $this->getComponentsByRegion($region);
228 $weights = array_map(function (SectionComponent $component) {
229 return $component->getWeight();
230 }, $components);
231 return $weights ? max($weights) + 1 : 0;
232 }
233
234 /**
235 * Gets the components for a specific region.
236 *
237 * @param string $region
238 * The region name.
239 *
240 * @return \Drupal\layout_builder\SectionComponent[]
241 * An array of components in the specified region, sorted by weight.
242 */
243 protected function getComponentsByRegion($region) {
244 $components = array_filter($this->getComponents(), function (SectionComponent $component) use ($region) {
245 return $component->getRegion() === $region;
246 });
247 uasort($components, function (SectionComponent $a, SectionComponent $b) {
248 return $a->getWeight() > $b->getWeight() ? 1 : -1;
249 });
250 return $components;
251 }
252
253 /**
254 * Inserts a component after a specified existing component.
255 *
256 * @param string $preceding_uuid
257 * The UUID of the existing component to insert after.
258 * @param \Drupal\layout_builder\SectionComponent $component
259 * The component being inserted.
260 *
261 * @return $this
262 *
263 * @throws \InvalidArgumentException
264 * Thrown when the expected UUID does not exist.
265 */
266 public function insertAfterComponent($preceding_uuid, SectionComponent $component) {
267 // Find the delta of the specified UUID.
268 $uuids = array_keys($this->getComponentsByRegion($component->getRegion()));
269 $delta = array_search($preceding_uuid, $uuids, TRUE);
270 if ($delta === FALSE) {
271 throw new \InvalidArgumentException(sprintf('Invalid preceding UUID "%s"', $preceding_uuid));
272 }
273 return $this->insertComponent($delta + 1, $component);
274 }
275
276 /**
277 * Inserts a component at a specified delta.
278 *
279 * @param int $delta
280 * The zero-based delta in which to insert the component.
281 * @param \Drupal\layout_builder\SectionComponent $new_component
282 * The component being inserted.
283 *
284 * @return $this
285 *
286 * @throws \OutOfBoundsException
287 * Thrown when the specified delta is invalid.
288 */
289 public function insertComponent($delta, SectionComponent $new_component) {
290 $components = $this->getComponentsByRegion($new_component->getRegion());
291 $count = count($components);
292 if ($delta > $count) {
293 throw new \OutOfBoundsException(sprintf('Invalid delta "%s" for the "%s" component', $delta, $new_component->getUuid()));
294 }
295
296 // If the delta is the end of the list, append the component instead.
297 if ($delta === $count) {
298 return $this->appendComponent($new_component);
299 }
300
301 // Find the weight of the component that exists at the specified delta.
302 $weight = array_values($components)[$delta]->getWeight();
303 $this->setComponent($new_component->setWeight($weight++));
304
305 // Increase the weight of every subsequent component.
306 foreach (array_slice($components, $delta) as $component) {
307 $component->setWeight($weight++);
308 }
309 return $this;
310 }
311
312 /**
313 * Wraps the layout plugin manager.
314 *
315 * @return \Drupal\Core\Layout\LayoutPluginManagerInterface
316 * The layout plugin manager.
317 */
318 protected function layoutPluginManager() {
319 return \Drupal::service('plugin.manager.core.layout');
320 }
321
322 /**
323 * Returns an array representation of the section.
324 *
325 * @internal
326 * This is intended for use by a storage mechanism for sections.
327 *
328 * @return array
329 * An array representation of the section component.
330 */
331 public function toArray() {
332 return [
333 'layout_id' => $this->getLayoutId(),
334 'layout_settings' => $this->getLayoutSettings(),
335 'components' => array_map(function (SectionComponent $component) {
336 return $component->toArray();
337 }, $this->getComponents()),
338 ];
339 }
340
341 }