ILIAS  trunk Revision v12.0_alpha-377-g3641b37b9db
class.ilObjectBadgeTableGUI.php
Go to the documentation of this file.
1<?php
2
19declare(strict_types=1);
20
21namespace ILIAS\Badge;
22
27use ilLanguage;
30use Psr\Http\Message\ServerRequestInterface;
32use Psr\Http\Message\RequestInterface;
34use Generator;
37use ilBadge;
41use ilObject;
42use ilLink;
48
50{
51 private const RECORD_RAW = '__raw__';
52
53 private readonly Factory $factory;
54 private readonly Renderer $renderer;
55 private readonly \ILIAS\Refinery\Factory $refinery;
56 private readonly ServerRequestInterface|RequestInterface $request;
57 private readonly Services $http;
58 private readonly ilLanguage $lng;
61 private readonly ilAccessHandler $access;
62 private readonly \ILIAS\ResourceStorage\Services $irss;
76 private ?array $cached_records = null;
78 private array $has_access_by_parent_cache = [];
81
82 public function __construct(
84 protected bool $has_write = false
85 ) {
86 global $DIC;
87
88 $this->lng = $DIC->language();
89 $this->tpl = $DIC->ui()->mainTemplate();
90 $this->factory = $DIC->ui()->factory();
91 $this->renderer = $DIC->ui()->renderer();
92 $this->refinery = $DIC->refinery();
93 $this->request = $DIC->http()->request();
94 $this->http = $DIC->http();
95 $this->access = $DIC->access();
96 $this->parent_obj = $parentObj;
97 $this->irss = $DIC->resourceStorage();
98 $this->badge_image_service = new ilBadgeImage(
99 $this->irss,
100 $DIC->upload(),
101 $DIC->ui()->mainTemplate()
102 );
103 }
104
105 public function getRows(
106 DataRowBuilder $row_builder,
107 array $visible_column_ids,
109 Order $order,
110 mixed $additional_viewcontrol_data,
111 mixed $filter_data,
112 mixed $additional_parameters
113 ): Generator {
114 $records = $this->getRecords();
115
116 if ($order) {
117 [$order_field, $order_direction] = $order->join(
118 [],
119 fn($ret, $key, $value) => [$key, $value]
120 );
121
122 usort($records, static function (array $left, array $right) use ($order_field): int {
123 if (\in_array($order_field, ['container', 'title', 'type'], true)) {
124 if (\in_array($order_field, ['container', 'title'], true)) {
125 $order_field .= '_sortable';
126 }
127
128 return \ilStr::strCmp(
129 $left[$order_field],
130 $right[$order_field]
131 );
132 }
133
134 if ($order_field === 'active') {
135 return $right[$order_field] <=> $left[$order_field];
136 }
137
138 return $left[$order_field] <=> $right[$order_field];
139 });
140
141 if ($order_direction === Order::DESC) {
142 $records = array_reverse($records);
143 }
144 }
145
146 if ($range) {
147 $records = \array_slice($records, $range->getStart(), $range->getLength());
148 }
149
150 $identifications = [];
151 foreach ($records as $record) {
152 if (isset($record[self::RECORD_RAW]['image_rid']) && $record[self::RECORD_RAW]['image_rid'] !== '') {
153 $identifications[] = $record[self::RECORD_RAW]['image_rid'];
154 }
155 }
156
157 $this->irss->preload($identifications);
158
159 $modal_container = new ModalBuilder();
160 $container_deleted_title_part = '<span class="il_ItemAlertProperty">' . $this->lng->txt('deleted') . '</span>';
161 foreach ($records as $record) {
162 yield $row_builder->buildDataRow(
163 (string) $record['id'],
164 $this->enrichRecord($modal_container, $container_deleted_title_part, $record)
165 );
166 }
167 }
168
169 public function getTotalRowCount(
170 mixed $additional_viewcontrol_data,
171 mixed $filter_data,
172 mixed $additional_parameters
173 ): ?int {
174 return \count($this->getRecords());
175 }
176
212 private function enrichRecord(
213 ModalBuilder $modal_builder,
214 string $container_deleted_title_part,
215 array $record
216 ): array {
217 $badge_item = $record[self::RECORD_RAW];
218
219 $badge = new ilBadge(0);
220 $badge->setId($badge_item['id']);
221 $badge->setImageRid($badge_item['image_rid']);
222 $badge->setImage($badge_item['image']);
223
224 $images = [
225 'rendered' => null,
226 'large' => null,
227 ];
228 $image_src = $this->badge_image_service->getImageFromResourceId($badge);
229 if ($image_src !== '') {
230 $images['rendered'] = $this->renderer->render(
231 $this->factory->image()->responsive(
232 $image_src,
233 $badge_item['title']
234 )
235 );
236
237 $image_src_large = $this->badge_image_service->getImageFromResourceId(
238 $badge,
239 ilBadgeImage::IMAGE_SIZE_XL
240 );
241 if ($image_src_large !== '') {
242 $images['large'] = $this->factory->image()->responsive(
243 $image_src_large,
244 $badge_item['title']
245 );
246 }
247 }
248
249 $container_title_parts = [
250 'icon' => $this->renderer->render(
251 $this->factory->symbol()->icon()->custom(
252 ilObject::_getIcon($badge_item['parent_id'], 'big', $badge_item['parent_type'] ?? ''),
253 $this->lng->txt('obj_' . ($badge_item['parent_type'] ?? ''))
254 )
255 ),
256 'title' => $badge_item['parent_title'] ?? '',
257 ];
258
259 $sortable_container_title_parts = [
260 'title' => $badge_item['parent_title'] ?? ''
261 ];
262 if ($badge_item['deleted']) {
263 $container_title_parts['suffix'] = $container_deleted_title_part;
264 $sortable_container_title_parts['suffix'] = $container_deleted_title_part;
265 } else {
266 if (isset($this->has_access_by_parent_cache[$badge_item['parent_id']])) {
267 $has_access = $this->has_access_by_parent_cache[$badge_item['parent_id']] ?? false;
268 $ref_id = $this->first_ref_id_for_parent_cache[$badge_item['parent_id']] ?? null;
269 } else {
270 $ref_ids = ilObject::_getAllReferences($badge_item['parent_id']);
271 $ref_id = array_shift($ref_ids);
272 $this->first_ref_id_for_parent_cache[$badge_item['parent_id']] = $ref_id;
273 $has_access = $ref_id && $this->access->checkAccess('read', '', $ref_id);
274 $this->has_access_by_parent_cache[$badge_item['parent_id']] = $has_access;
275 }
276
277 if ($has_access) {
278 $container_title_parts['title'] = $this->renderer->render(
279 new Standard(
280 $container_title_parts['title'],
281 (string) new URI(
282 ilLink::_getLink(
283 $ref_id,
284 $badge_item['parent_type'] ?? ''
285 )
286 )
287 )
288 );
289 } else {
290 $container_title_parts['suffix'] = $container_deleted_title_part;
291 $sortable_container_title_parts['suffix'] = $container_deleted_title_part;
292 }
293 }
294
295 $modal = $modal_builder->constructModal(
296 $images['large'],
297 $badge_item['title'],
298 [
299 'active' => $badge_item['active'] ? $this->lng->txt('yes') : $this->lng->txt('no'),
300 'type' => $record['type'],
301 'container' => implode(' ', \array_slice($container_title_parts, 1, null, true)),
302 ]
303 );
304
305 return [
306 'id' => $badge_item['id'],
307 'active' => (bool) $badge_item['active'],
308 'type' => $record['type'],
309 'image' => $images['rendered'] ? ($modal_builder->renderShyButton(
310 $images['rendered'],
311 $modal
312 ) . ' ') : '',
313 'title' => implode('', [
314 $modal_builder->renderShyButton($badge_item['title'], $modal),
315 $modal_builder->renderModal($modal)
316 ]),
317 'title_sortable' => $badge_item['title'],
318 'container' => implode(' ', $container_title_parts),
319 'container_sortable' => implode(' ', $sortable_container_title_parts),
320 ];
321 }
322
348 private function getRecords(): array
349 {
350 if ($this->cached_records !== null) {
351 return $this->cached_records;
352 }
353
354 // A filter is not implemented, yet
355 $filter = [
356 'type' => '',
357 'title' => '',
358 'object' => ''
359 ];
360
361 $types = ilBadgeHandler::getInstance()->getAvailableTypes(false);
362 $raw_records = ilBadge::getObjectInstances($filter);
363
364 $sortable_rows = array_map(function (array $badge_item) use ($types) {
365 return [
366 'id' => $badge_item['id'],
367 'active' => (bool) $badge_item['active'],
368 'type' => ilBadge::getExtendedTypeCaption($types[$badge_item['type_id']]),
369 'title_sortable' => $badge_item['title'],
370 'container_sortable' => ($badge_item['parent_title'] ?? '') .
371 ($badge_item['deleted'] ? ' ' . $this->lng->txt('deleted') : ''),
372 self::RECORD_RAW => $badge_item
373 ];
374 }, $raw_records);
375
376 $this->cached_records = $sortable_rows;
377
378 return $this->cached_records;
379 }
380
384 public function getColumns(): array
385 {
386 return [
387 'image' => $this->factory->table()->column()->text($this->lng->txt('image'))->withIsSortable(false),
388 'title' => $this->factory->table()->column()->text($this->lng->txt('title')),
389 'type' => $this->factory->table()->column()->text($this->lng->txt('type')),
390 'container' => $this->factory->table()->column()->text($this->lng->txt('object')),
391 'active' => $this->factory->table()->column()->boolean(
392 $this->lng->txt('active'),
393 $this->lng->txt('yes'),
394 $this->lng->txt('no')
395 )->withOrderingLabels(
396 $this->lng->txt('badge_sort_active_badges_first'),
397 $this->lng->txt('badge_sort_active_badges_last')
398 )
399 ];
400 }
401
405 private function getActions(
406 URLBuilder $url_builder,
407 URLBuilderToken $action_parameter_token,
408 URLBuilderToken $row_id_token
409 ): array {
410 return $this->has_write ? [
411 'obj_badge_activate' => $this->factory->table()->action()->multi(
412 $this->lng->txt('activate'),
413 $url_builder->withParameter($action_parameter_token, 'obj_badge_activate'),
414 $row_id_token
415 ),
416 'obj_badge_deactivate' =>
417 $this->factory->table()->action()->multi(
418 $this->lng->txt('deactivate'),
419 $url_builder->withParameter($action_parameter_token, 'obj_badge_deactivate'),
420 $row_id_token
421 ),
422 'obj_badge_delete' =>
423 $this->factory->table()->action()->multi(
424 $this->lng->txt('delete'),
425 $url_builder->withParameter($action_parameter_token, 'obj_badge_delete'),
426 $row_id_token
427 ),
428 'obj_badge_show_users' =>
429 $this->factory->table()->action()->single(
430 $this->lng->txt('user'),
431 $url_builder->withParameter($action_parameter_token, 'obj_badge_show_users'),
432 $row_id_token
433 )
434 ] : [];
435 }
436
437 public function renderTable(string $url): void
438 {
439 $df = new \ILIAS\Data\Factory();
440
441 $table_uri = $df->uri($url);
442 $url_builder = new URLBuilder($table_uri);
443 $query_params_namespace = ['tid'];
444
445 [$url_builder, $action_parameter_token, $row_id_token] = $url_builder->acquireParameters(
446 $query_params_namespace,
447 'table_action',
448 'id',
449 );
450
451 $table = $this->factory
452 ->table()
453 ->data($this, $this->lng->txt('badge_object_badges'), $this->getColumns())
454 ->withId(self::class)
455 ->withOrder(new Order('title', Order::ASC))
456 ->withActions($this->getActions($url_builder, $action_parameter_token, $row_id_token))
457 ->withRequest($this->request);
458
459 $out = [$table];
460
461 $query = $this->http->wrapper()->query();
462 if ($query->has($action_parameter_token->getName())) {
463 $action = $query->retrieve($action_parameter_token->getName(), $this->refinery->kindlyTo()->string());
464 $ids = $query->retrieve($row_id_token->getName(), $this->refinery->custom()->transformation(fn($v) => $v));
465
466 if ($action === 'obj_badge_delete') {
467 $items = [];
468 if (\is_array($ids) && \count($ids) > 0) {
469 if ($ids === ['ALL_OBJECTS']) {
470 $filter = [
471 'type' => '',
472 'title' => '',
473 'object' => ''
474 ];
475 $ids = [];
476 foreach (ilBadge::getObjectInstances($filter) as $badge_item) {
477 $ids[] = $badge_item['id'];
478 }
479 }
480
481 foreach ($ids as $id) {
482 $badge = new ilBadge((int) $id);
483 $items[] = $this->factory->modal()->interruptiveItem()->keyValue(
484 (string) $id,
485 (string) $badge->getId(),
486 $badge->getTitle()
487 );
488 }
489
490 $this->http->saveResponse(
491 $this->http
492 ->response()
493 ->withBody(
494 Streams::ofString($this->renderer->renderAsync([
495 $this->factory->modal()->interruptive(
496 $this->lng->txt('badge_deletion'),
497 $this->lng->txt('badge_deletion_confirmation'),
498 '#'
499 )->withAffectedItems($items)
500 ]))
501 )
502 );
503 $this->http->sendResponse();
504 $this->http->close();
505 }
506 }
507 }
508
509 $this->tpl->setContent($this->renderer->render($out));
510 }
511}
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
renderer()
factory()
$out
Definition: buildRTE.php:24
renderShyButton(string $label, Modal $modal)
constructModal(?Image $badge_image, string $badge_title, array $badge_properties=[])
renderModal(Modal $modal)
readonly ilGlobalTemplateInterface $tpl
readonly ilObjBadgeAdministrationGUI $parent_obj
readonly ServerRequestInterface RequestInterface $request
getTotalRowCount(mixed $additional_viewcontrol_data, mixed $filter_data, mixed $additional_parameters)
Mainly for the purpose of pagination-support, it is important to know about the total number of recor...
getRows(DataRowBuilder $row_builder, array $visible_column_ids, Range $range, Order $order, mixed $additional_viewcontrol_data, mixed $filter_data, mixed $additional_parameters)
This is called by the table to retrieve rows; map data-records to rows using the $row_builder e....
getActions(URLBuilder $url_builder, URLBuilderToken $action_parameter_token, URLBuilderToken $row_id_token)
readonly ILIAS ResourceStorage Services $irss
enrichRecord(ModalBuilder $modal_builder, string $container_deleted_title_part, array $record)
readonly ILIAS Refinery Factory $refinery
__construct(ilObjBadgeAdministrationGUI $parentObj, protected bool $has_write=false)
Builds a Color from either hex- or rgb values.
Definition: Factory.php:31
Both the subject and the direction need to be specified when expressing an order.
Definition: Order.php:29
join($init, callable $fn)
Definition: Order.php:75
const DESC
Definition: Order.php:31
A simple class to express a naive range of whole positive numbers.
Definition: Range.php:29
The scope of this class is split ilias-conform URI's into components.
Definition: URI.php:35
Stream factory which enables the user to create streams without the knowledge of the concrete class.
Definition: Streams.php:32
Class Services.
Definition: Services.php:38
Class ilBadgeHandler.
static getExtendedTypeCaption(ilBadgeType $a_type)
static getObjectInstances(?array $filter=null)
language handling
@ilCtrl_Calls ilObjBadgeAdministrationGUI: ilPermissionGUI, ilBadgeManagementGUI @ilCtrl_IsCalledBy i...
Class ilObject Basic functions for all objects.
static _getIcon(int $obj_id=0, string $size="big", string $type="", bool $offline=false)
Get icon for repository item.
static _getAllReferences(int $id)
get all reference ids for object ID
return['delivery_method'=> 'php',]
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
A Column describes the form of presentation for a certain aspect of data, i.e.
Definition: Column.php:28
buildDataRow(string $id, array $record)
This is how the factory for UI elements looks.
Definition: Factory.php:38
An entity that renders components to a string output.
Definition: Renderer.php:31
Interface ilAccessHandler This interface combines all available interfaces which can be called via gl...
$ref_id
Definition: ltiauth.php:66
static http()
Fetches the global http state from ILIAS.
global $DIC
Definition: shib_login.php:26
$url
Definition: shib_logout.php:68