ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ItemBlockSequenceGenerator.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
30 
37 {
38  protected ?string $lang = null;
39  protected ResourcesManager $copage_resources;
40  protected bool $include_empty_blocks;
41  protected Content\ModeManager $mode_manager;
42  protected \ilAccessHandler $access;
43  protected int $block_limit;
48  protected \ilContainer $container;
50  protected bool $has_other_block = false;
52  protected array $accumulated_ref_ids = [];
54  protected static array $item_group_ref_ids = [];
56  protected array $all_item_group_item_ref_ids = [];
57  protected array $all_ref_ids = [];
58 
59  public function __construct(
60  DataService $data_service,
61  InternalDomainService $domain_service,
62  ResourcesManager $copage_resources,
63  \ilContainer $container,
64  BlockSequence $block_sequence,
65  ItemSetManager $item_set_manager,
66  bool $include_empty_blocks = true,
67  ?string $lang = null
68  ) {
69  $this->access = $domain_service->access();
70  $this->data_service = $data_service;
71  $this->domain_service = $domain_service;
72  $this->copage_resources = $copage_resources;
73  $this->block_sequence = $block_sequence;
74  $this->item_set_manager = $item_set_manager;
75  $this->container = $container;
76  $this->mode_manager = $this->domain_service->content()->mode($container);
77  $this->include_empty_blocks = $include_empty_blocks;
78  $this->block_limit = 0;
79  $this->lang = $lang;
80  if (!$this->mode_manager->isActiveItemOrdering()) {
81  $this->block_limit = (int) \ilContainer::_lookupContainerSetting($container->getId(), "block_limit");
82  }
83  }
84 
85  public function getSequence(): ItemBlockSequence
86  {
87  if (is_null($this->sequence)) {
89  $item_blocks = [];
90  $sorted_blocks = [];
91 
92  // get blocks from block sequence parts (item groups, by type, objective)
93  foreach ($this->block_sequence->getParts() as $part) {
94  foreach ($this->getBlocksForPart($part) as $block) {
95  $item_blocks[$block->getId()] = $block;
96  }
97  }
98 
99  // get blocks of page, put them to the start
100  $embedded_ids = $this->getPageEmbeddedBlockIds();
101  $other_is_page_embedded = false;
102  foreach ($embedded_ids as $id) {
103  if (isset($item_blocks[$id])) {
104  $item_blocks[$id]->setPageEmbedded(true);
105  $sorted_blocks[] = $item_blocks[$id];
106  unset($item_blocks[$id]);
107  } elseif (!is_numeric($id)) {
108  // add item blocks of page, even if originally not in the block set
109  if ($id === "_other") {
110  $this->has_other_block = true;
111  $other_is_page_embedded = true;
112  } else {
113  $ref_ids = $this->item_set_manager->getRefIdsOfType($id);
114  $this->accumulateRefIds($ref_ids);
115  $block_items = $this->determineBlockItems($ref_ids, true);
116  // we remove this check to prevent [list-cat] stuff from appearing in the list
117  // this will output a message (is empty) in editing mode and
118  // remove the block (empty string) in presentation mode
119  if ($this->include_empty_blocks || count($block_items->getRefIds()) > 0) {
120  $block = $this->data_service->itemBlock(
121  $id,
122  $this->data_service->typeBlock($id),
123  $block_items->getRefIds(),
124  $block_items->getLimitExhausted()
125  );
126  $block->setPageEmbedded(true);
127  $sorted_blocks[] = $block;
128  }
129  }
130  } else {
131  // e.g. deleted item group
132  //throw new \ilException("Missing item group data.");
133  }
134  }
135 
136  // get other block
137  $other_block = $this->getOtherBlock();
138  if (!is_null($other_block)) {
139  if ($other_is_page_embedded) {
140  $other_block->setPageEmbedded(true);
141  }
142  $item_blocks["_other"] = $other_block;
143  }
144 
145  // manual sorting
146  $pos = 10;
147  $sorting = \ilContainerSorting::_getInstance($this->container->getId());
148  foreach ($sorting->getBlockPositions() as $id) {
149  if (isset($item_blocks[$id])) {
150  $item_blocks[$id]->setPosition($pos);
151  $sorted_blocks[] = $item_blocks[$id];
152  unset($item_blocks[$id]);
153  $pos += 10;
154  }
155  }
156 
157  // rest in order of base block sequence
158  foreach ($item_blocks as $block) {
159  $block->setPosition($pos);
160  $sorted_blocks[] = $block;
161  $pos += 10;
162  }
163  $this->sequence = new ItemBlockSequence($sorted_blocks);
164  }
165  return $this->sequence;
166  }
167 
168  protected function preloadSessionandItemGroupItemData(): void
169  {
170  foreach ($this->block_sequence->getParts() as $part) {
171  // SessionBlock
172  if ($part instanceof Content\SessionBlock) {
173  /*
174  $ref_ids = $this->item_set_manager->getRefIdsOfType("sess");
175  $block_items = $this->determineBlockItems($ref_ids);
176  if (count($block_items->getRefIds()) > 0) {
177  yield $this->data_service->itemBlock(
178  "sess",
179  $this->data_service->sessionBlock(),
180  $block_items->getRefIds(),
181  $block_items->getLimitExhausted()
182  );
183  }*/
184  }
185  // ItemGroupBlock
186  if ($part instanceof Content\ItemGroupBlock) {
187  $item_ref_ids = $this->getItemGroupItemRefIds($part->getRefId());
188  $this->all_item_group_item_ref_ids = array_unique(array_merge($this->all_item_group_item_ref_ids, $item_ref_ids));
189  }
190  // ItemGroupBlocks
191  if ($part instanceof Content\ItemGroupBlocks) {
192  foreach ($this->item_set_manager->getRefIdsOfType("itgr") as $item_group_ref_id) {
193  $item_ref_ids = $this->getItemGroupItemRefIds($item_group_ref_id);
194  $this->all_item_group_item_ref_ids = array_unique(array_merge($this->all_item_group_item_ref_ids, $item_ref_ids));
195  }
196  }
197  }
198  }
199 
203  protected function getBlocksForPart(BlockSequencePart $part): \Iterator
204  {
205  // TypeBlocks
206  if ($part instanceof Content\TypeBlocks) {
207  foreach ($this->getGroupedObjTypes() as $type) {
208  $ref_ids = $this->item_set_manager->getRefIdsOfType($type);
209  $this->accumulateRefIds($ref_ids);
210  $block_items = $this->determineBlockItems($ref_ids, true);
211  if ($type !== "itgr" && count($block_items->getRefIds()) > 0) {
212  yield $this->data_service->itemBlock(
213  $type,
214  $this->data_service->typeBlock($type),
215  $block_items->getRefIds(),
216  $block_items->getLimitExhausted()
217  );
218  }
219  }
220  }
221  // TypeBlock
222  if ($part instanceof Content\TypeBlock) {
223  $ref_ids = $this->item_set_manager->getRefIdsOfType($part->getType());
224  $this->accumulateRefIds($ref_ids);
225  $block_items = $this->determineBlockItems($ref_ids, true);
226  if (count($block_items->getRefIds()) > 0) {
227  yield $this->data_service->itemBlock(
228  $part->getType(),
229  $this->data_service->typeBlock($part->getType()),
230  $block_items->getRefIds(),
231  $block_items->getLimitExhausted()
232  );
233  }
234  }
235  // SessionBlock
236  if ($part instanceof Content\SessionBlock) {
237  $ref_ids = $this->item_set_manager->getRefIdsOfType("sess");
238  $this->accumulateRefIds($ref_ids);
239  $block_items = $this->determineBlockItems($ref_ids);
240  if (count($block_items->getRefIds()) > 0) {
241  yield $this->data_service->itemBlock(
242  "sess",
243  $this->data_service->sessionBlock(),
244  $block_items->getRefIds(),
245  $block_items->getLimitExhausted()
246  );
247  }
248  }
249  // ItemGroupBlock
250  if ($part instanceof Content\ItemGroupBlock) {
251  $block = $this->getItemGroupBlock($part->getRefId());
252  if (!is_null($block)) {
253  yield $block;
254  }
255  }
256  // ObjectivesBlock
257  if ($part instanceof Content\ObjectivesBlock) {
258  // in admin mode, we do not include the objectives block
259  // -> all items will be presented in item group/other block
260  if (!$this->mode_manager->isAdminMode()) {
261  $objective_ids = \ilCourseObjective::_getObjectiveIds($this->container->getId(), true);
262  $ref_ids = [];
263  foreach ($objective_ids as $objective_id) {
264  foreach (\ilObjectActivation::getItemsByObjective((int) $objective_id) as $data) {
265  $ref_ids[] = (int) $data["ref_id"];
266  }
267  }
268  $this->accumulateRefIds($ref_ids);
269  yield $this->data_service->itemBlock(
270  "_lobj",
271  $part,
272  $ref_ids,
273  false,
275  );
276  }
277  }
278  // ItemGroupBlocks
279  if ($part instanceof Content\ItemGroupBlocks) {
280  foreach ($this->item_set_manager->getRefIdsOfType("itgr") as $item_group_ref_id) {
281  $block = $this->getItemGroupBlock($item_group_ref_id);
282  if (!is_null($block)) {
283  yield $block;
284  }
285  }
286  }
287  // ItemGroupBlocks
288  if ($part instanceof Content\OtherBlock) {
289  $this->has_other_block = true;
290  }
291  }
292 
293  protected function determineBlockItems(
294  array $ref_ids,
295  $filter_session_and_item_group_items = false,
296  bool $prevent_duplicats = false
297  ): BlockItemsInfo {
298  $exhausted = false;
299  $accessible_ref_ids = [];
300  foreach ($ref_ids as $ref_id) {
301  if ($prevent_duplicats && in_array($ref_id, $this->all_ref_ids)) {
302  continue;
303  }
304  if (\ilObject::_lookupType(\ilObject::_lookupObjId($ref_id)) === "itgr") {
305  continue;
306  }
307  $this->all_ref_ids[] = $ref_id;
308  if ($exhausted) {
309  break;
310  }
311  if ($this->access->checkAccess('visible', '', $ref_id)) {
312  if ($this->block_limit > 0 && count($accessible_ref_ids) >= $this->block_limit) {
313  $exhausted = true;
314  } elseif (!$filter_session_and_item_group_items || !in_array($ref_id, $this->all_item_group_item_ref_ids)) {
315  $accessible_ref_ids[] = $ref_id;
316  }
317  }
318  }
319  return $this->data_service->blockItemsInfo(
320  $accessible_ref_ids,
321  $exhausted
322  );
323  }
324 
325  protected function accumulateRefIds(array $ref_ids): void
326  {
327  foreach ($ref_ids as $ref_id) {
328  $this->accumulated_ref_ids[$ref_id] = $ref_id;
329  }
330  }
331 
332  protected function getOtherBlock(): ?ItemBlock
333  {
334  if (!$this->has_other_block) {
335  return null;
336  }
337 
338  $remaining_ref_ids = array_filter(
339  $this->item_set_manager->getAllRefIds(),
340  fn($i) => (!isset($this->accumulated_ref_ids[$i]) && !$this->item_set_manager->isSideBlockItem($i))
341  );
342  $block_items = $this->determineBlockItems($remaining_ref_ids, true, true);
343  // we remove this check to prevent [list-_other] stuff from appearing in the list
344  // this will output a message (is empty) in editing mode and
345  // remove the block (empty string) in presentation mode
346  if ($this->include_empty_blocks || count($block_items->getRefIds()) > 0) {
347  return $this->data_service->itemBlock(
348  "_other",
349  $this->data_service->otherBlock(),
350  $block_items->getRefIds(),
351  $block_items->getLimitExhausted()
352  );
353  }
354  return null;
355  }
356 
361  protected function getGroupedObjTypes(): \Iterator
362  {
363  foreach (\ilObjectDefinition::getGroupedRepositoryObjectTypes($this->container->getType())
364  as $key => $type) {
365  yield $key;
366  }
367  }
368 
369  protected function getItemGroupItemRefIds(int $item_group_ref_id): array
370  {
371  if (!isset(self::$item_group_ref_ids[$item_group_ref_id])) {
373  $this->container->getId()
374  )->sortSubItems('itgr', \ilObject::_lookupObjId($item_group_ref_id), \ilObjectActivation::getItemsByItemGroup($item_group_ref_id));
375 
376  self::$item_group_ref_ids[$item_group_ref_id] = array_map(static function ($i) {
377  return (int) $i["child"];
378  }, $items);
379  }
380  return self::$item_group_ref_ids[$item_group_ref_id];
381  }
382 
383  protected function getItemGroupBlock(int $item_group_ref_id): ?ItemBlock
384  {
385  $ref_ids = $this->getItemGroupItemRefIds($item_group_ref_id);
386  $this->accumulateRefIds($ref_ids);
387  $block_items = $this->determineBlockItems($ref_ids);
388  // #16493
389  if (!$this->access->checkAccess("visible", "", $item_group_ref_id) ||
390  !$this->access->checkAccess("read", "", $item_group_ref_id)) {
391  return null;
392  }
393  // otherwise empty item groups will simply "vanish" from the repository
394  if (count($block_items->getRefIds()) > 0 || $this->access->checkAccess('write', '', $item_group_ref_id)) {
395  return $this->data_service->itemBlock(
396  (string) $item_group_ref_id,
397  $this->data_service->itemGroupBlock($item_group_ref_id),
398  $block_items->getRefIds(),
399  $block_items->getLimitExhausted()
400  );
401  }
402  return null;
403  }
404 
408  public function getPageEmbeddedBlockIds(): array
409  {
410  $page = $this->domain_service->page(
411  $this->container,
412  $this->lang
413  );
414  $dom = $page->getDom();
415 
416  $ids = [];
417 
418  if ($dom) {
419  $ids = $this->copage_resources->getResourceIds($dom);
420  }
421 
422  return $ids;
423  }
424 }
A single block containing items of a (grouped) repo type.
determineBlockItems(array $ref_ids, $filter_session_and_item_group_items=false, bool $prevent_duplicats=false)
static _getObjectiveIds(int $course_id, bool $a_activated_only=false)
A block that holds session in a special presentation.
static _lookupObjId(int $ref_id)
Represents all "by type" blocks of a container.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
A block that holds a single item group.
$ref_id
Definition: ltiauth.php:65
A block that holds all objectives and their items.
Represents all item groups of a container.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static getItemsByItemGroup(int $item_group_ref_id)
Get materials of item group.
A block that holds all other "remaining" items.
static getGroupedRepositoryObjectTypes($parent_obj_type)
static getItemsByObjective(int $objective_id)
Get objective items.
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
static _getInstance(int $a_obj_id)
static _lookupContainerSetting(int $a_id, string $a_keyword, ?string $a_default_value=null)
__construct(DataService $data_service, InternalDomainService $domain_service, ResourcesManager $copage_resources, \ilContainer $container, BlockSequence $block_sequence, ItemSetManager $item_set_manager, bool $include_empty_blocks=true, ?string $lang=null)
static _lookupType(int $id, bool $reference=false)