ILIAS  trunk Revision v11.0_alpha-1715-g7fc467680fb
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilDclRecordListGUI.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 {
23  public const GET_TABLE_ID = 'table_id';
24  public const GET_TABLEVIEW_ID = 'tableview_id';
25  public const GET_MODE = "mode";
26  public const MODE_VIEW = "view";
27  public const MODE_MANAGE = "manage";
28  public const CMD_LIST_RECORDS = 'listRecords';
29  public const CMD_SHOW = 'show';
30  public const CMD_CONFIRM_DELETE_RECORDS = 'confirmDeleteRecords';
31  public const CMD_CANCEL_DELETE = 'cancelDelete';
32  public const CMD_DELETE_RECORDS = 'deleteRecords';
33  public const CMD_SHOW_IMPORT_EXCEL = 'showImportExcel';
34 
37  protected \ILIAS\UI\Factory $ui_factory;
38  protected \ILIAS\UI\Renderer $renderer;
39 
43  protected string $mode = self::MODE_VIEW;
45  protected ?int $table_id;
46  protected int $obj_id;
48  protected ?int $tableview_id;
49  protected static array $available_modes = [self::MODE_VIEW, self::MODE_MANAGE];
50 
51  protected ilCtrl $ctrl;
53  protected ilLanguage $lng;
54  protected ilTabsGUI $tabs;
57  protected \ILIAS\ResourceStorage\Services $irss;
58  protected bool $filter_changed = false;
59 
63  public function __construct(ilObjDataCollectionGUI $a_parent_obj, int $table_id, int $tableview_id)
64  {
65  global $DIC;
66 
67  $this->ctrl = $DIC->ctrl();
68  $this->toolbar = $DIC->toolbar();
69  $this->lng = $DIC->language();
70  $this->tabs = $DIC->tabs();
71  $this->http = $DIC->http();
72  $this->refinery = $DIC->refinery();
73  $this->irss = $DIC->resourceStorage();
74  $this->access = $DIC->access();
75  $this->tpl = $DIC->ui()->mainTemplate();
76  $this->ui_factory = $DIC->ui()->factory();
77  $this->renderer = $DIC->ui()->renderer();
78 
79  $this->table_id = $table_id;
80  $this->tableview_id = $tableview_id;
81 
82  $this->obj_id = $a_parent_obj->getObject()->getId();
83  $this->parent_obj = $a_parent_obj;
84  $this->table_obj = ilDclCache::getTableCache($table_id);
85 
86  $this->ctrl->setParameterByClass(ilDclRecordEditGUI::class, self::GET_TABLE_ID, $this->table_id);
87  $this->ctrl->setParameterByClass(ilDclRecordEditGUI::class, self::GET_TABLEVIEW_ID, $this->tableview_id);
88  $this->ctrl->setParameterByClass(ilDclDetailedViewGUI::class, self::GET_TABLEVIEW_ID, $this->tableview_id);
89 
90  $this->mode = self::MODE_VIEW;
91 
92  if ($this->http->wrapper()->query()->has(self::GET_MODE)) {
93  $mode = $this->http->wrapper()->query()->retrieve(self::GET_MODE, $this->refinery->kindlyTo()->string());
94  if (in_array($mode, self::$available_modes, true)) {
95  $this->mode = $mode;
96  }
97  }
98  }
99 
100  public function getRefId(): int
101  {
102  return $this->parent_obj->getRefId();
103  }
104 
105  public function getObjId(): int
106  {
107  return $this->parent_obj->getObject()->getId();
108  }
109 
114  public function executeCommand(): void
115  {
116  if (!$this->checkAccess()) {
117  $this->tpl->setOnScreenMessage('failure', $this->lng->txt('permission_denied'), true);
118  return;
119  }
120 
121  $this->ctrl->saveParameter($this, self::GET_MODE);
122  $cmd = $this->ctrl->getCmd(self::CMD_SHOW);
123 
124  // 'show' fills all filters with the predefined values from the tableview,
125  // whereas 'listRecords' handels the filters "normally", filling them from the POST-variable
126  switch ($cmd) {
127  case self::CMD_SHOW:
128  $this->setSubTabs($this->mode);
129  $this->listRecords(true);
130  break;
131  case self::CMD_CANCEL_DELETE:
132  case self::CMD_LIST_RECORDS:
133  $this->setSubTabs($this->mode);
134  $this->listRecords();
135  break;
136  case self::CMD_CONFIRM_DELETE_RECORDS:
137  $this->confirmDeleteRecords();
138  break;
139  case self::CMD_DELETE_RECORDS:
140  $this->deleteRecords();
141  break;
142  case self::CMD_SHOW_IMPORT_EXCEL:
143  $this->tabs->setBackTarget($this->lng->txt('back'), $this->ctrl->getLinkTarget($this));
144  $this->$cmd();
145  break;
146 
147  default:
148  $this->$cmd();
149  break;
150  }
151  }
152 
153  public function listRecords(bool $use_tableview_filter = false): void
154  {
155  $list = $this->getRecordListTableGUI($use_tableview_filter);
156 
157  $this->createSwitchers();
158 
159  $permission_to_add_or_import = ilObjDataCollectionAccess::hasPermissionToAddRecord(
160  $this->parent_obj->getRefId(),
162  ) && $this->table_obj->hasCustomFields();
163  if ($permission_to_add_or_import) {
164  $this->ctrl->setParameterByClass(ilDclRecordEditGUI::class, "record_id", null);
165 
166  $add_new = $this->ui_factory->button()->primary(
167  $this->lng->txt("dcl_add_new_record"),
168  $this->ctrl->getFormActionByClass(ilDclRecordEditGUI::class, "create")
169  );
170  $this->toolbar->addStickyItem($add_new);
171  }
172 
173  if ($permission_to_add_or_import && $this->table_obj->getImportEnabled()) {
174  $this->ctrl->setParameterByClass(ilDclRecordEditGUI::class, "record_id", null);
175 
176  $import_button = $this->ui_factory->button()->standard(
177  $this->lng->txt("dcl_import_records .xls"),
178  $this->ctrl->getFormActionByClass(ilDclRecordListGUI::class, self::CMD_SHOW_IMPORT_EXCEL)
179  );
180  $this->toolbar->addComponent($import_button);
181  }
182 
183  if (count($this->table_obj->getRecordFields()) == 0) {
184  $message = $this->lng->txt("dcl_no_fields_yet") . " "
186  $this->parent_obj->getRefId(),
188  ) ? $this->lng->txt("dcl_create_fields") : "");
189  $this->tpl->setOnScreenMessage('info', $message, true);
190  }
191 
192  $target = '';
193  if ($this->http->wrapper()->query()->has('table_id')) {
194  $target .= $this->http->wrapper()->query()->retrieve('table_id', $this->refinery->to()->string());
195  }
196  if ($this->http->wrapper()->query()->has('tableview_id')) {
197  $target .= '_' . $this->http->wrapper()->query()->retrieve('tableview_id', $this->refinery->to()->string());
198  }
199 
200  $this->tpl->setPermanentLink("dcl", $this->parent_obj->getRefId(), $target);
201 
202  if ($desc = $this->table_obj->getDescription()) {
203  $desc = $this->refinery->string()->markdown()->toHTML()->transform($desc);
204  $desc = '<div class="ilDclTableDescription">' . $desc . '</div>';
205  }
206  $this->tpl->setContent($desc . $list->getHTML());
207  }
208 
209  public function showImportExcel(?ilPropertyFormGUI $form = null): void
210  {
211  if (!$form) {
212  $form = $this->initImportForm();
213  }
214  $this->tpl->setContent($form->getHTML());
215  }
216 
221  {
222  $form = new ilPropertyFormGUI();
223 
224  $item = new ilCustomInputGUI();
225  $item->setHtml($this->lng->txt('dcl_file_format_description'));
226  $item->setTitle("Info");
227  $form->addItem($item);
228 
229  $file = new ilFileInputGUI($this->lng->txt("import_file"), "import_file");
230  $file->setRequired(true);
231  $form->addItem($file);
232 
233  $cb = new ilCheckboxInputGUI($this->lng->txt("dcl_simulate_import"), "simulate");
234  $cb->setInfo($this->lng->txt("dcl_simulate_info"));
235  $form->addItem($cb);
236 
237  $form->addCommandButton("importExcel", $this->lng->txt("import"));
238 
239  return $form;
240  }
241 
245  public function importExcel(): void
246  {
248  $this->parent_obj->getRefId(),
250  )) || !$this->table_obj->getImportEnabled()) {
251  throw new ilDclException($this->lng->txt("access_denied"));
252  }
253  $form = $this->initImportForm();
254  if ($form->checkInput()) {
255  $file = $form->getInput("import_file");
256  $file_location = $file["tmp_name"];
257  $simulate = $form->getInput("simulate");
258  $this->importRecords($file_location, (bool) $simulate);
259  } else {
260  $this->showImportExcel($form);
261  }
262  }
263 
267  private function importRecords(string $file, bool $simulate = false): void
268  {
269  $importer = new ilDclContentImporter($this->parent_obj->object->getRefId(), $this->table_id);
270  $result = $importer->import($file, $simulate);
271 
272  $this->endImport($result['line'], $result['warnings']);
273  }
274 
279  public function endImport(int $i, array $warnings): void
280  {
281  $output = new ilTemplate("tpl.dcl_import_terminated.html", true, true, "components/ILIAS/DataCollection");
282  $output->setVariable("IMPORT_TERMINATED", $this->lng->txt("dcl_import_terminated") . ": " . $i);
283  foreach ($warnings as $warning) {
284  $output->setCurrentBlock("warnings");
285  $output->setVariable("WARNING", $warning);
286  $output->parseCurrentBlock();
287  }
288  if (!count($warnings)) {
289  $output->setCurrentBlock("warnings");
290  $output->setVariable("WARNING", $this->lng->txt("dcl_no_warnings"));
291  $output->parseCurrentBlock();
292  }
293  $output->setVariable("BACK_LINK", $this->ctrl->getLinkTargetByClass(ilDclRecordListGUI::class, "listRecords"));
294  $output->setVariable("BACK", $this->lng->txt("back"));
295  $this->tpl->setContent($output->get());
296  }
297 
301  protected function applyFilter(): void
302  {
303  $table = new ilDclRecordListTableGUI($this, "listRecords", $this->table_obj, $this->tableview_id);
304  $table->initFilter();
305  $table->resetOffset();
306  $table->writeFilterToSession();
307  $this->filter_changed = true;
308  $this->listRecords();
309  }
310 
314  protected function resetFilter(): void
315  {
316  $table = new ilDclRecordListTableGUI($this, "show", $this->table_obj, $this->tableview_id);
317  $table->initFilter();
318  $table->resetOffset();
319  $table->resetFilter();
320  $this->filter_changed = true;
321  $this->listRecords();
322  }
323 
327  public function sendFile(): void
328  {
329  $hasIlFileHash = $this->http->wrapper()->query()->has('ilfilehash');
330  //need read access to receive file
331  if ($this->access->checkAccess('read', "", $this->parent_obj->getRefId())) {
332  // deliver temp-files
333  if ($hasIlFileHash) {
334  $filehash = $this->http->wrapper()->query()->retrieve(
335  'ilfilehash',
336  $this->refinery->kindlyTo()->string()
337  );
338  $field_id = $this->http->wrapper()->query()->retrieve('field_id', $this->refinery->kindlyTo()->int());
340 
341  $filepath = $_FILES["field_" . $field_id]['tmp_name'];
342  $filetitle = $_FILES["field_" . $field_id]['name'];
343 
344  ilFileDelivery::deliverFileLegacy($filepath, $filetitle);
345  } else {
346  $rec_id = $this->http->wrapper()
347  ->query()
348  ->retrieve('record_id', $this->refinery->kindlyTo()->int());
349 
350  $record = ilDclCache::getRecordCache($rec_id);
351  if (!$this->recordBelongsToCollection($record)) {
352  return;
353  }
354 
355  $field_id = $this->http->wrapper()
356  ->query()
357  ->retrieve('field_id', $this->refinery->kindlyTo()->string());
358 
359 
360 
361  // Find the current revision
362  $rid_string = $record->getRecordFieldValue($field_id);
363  $identification = $this->irss->manage()->find($rid_string);
364  if ($identification === null) {
365  return;
366  }
367  $current_revision = $this->irss->manage()->getCurrentRevision($identification);
368 
369  // Download the File
370  $this->irss->consume()
371  ->download($identification)
372  ->overrideFileName($current_revision->getTitle())
373  ->run();
374  }
375  }
376  }
377 
381  public function confirmDeleteRecords(): void
382  {
383  $this->tabs->clearTargets();
384 
385  $conf = new ilConfirmationGUI();
386  $conf->setFormAction($this->ctrl->getFormAction($this));
387  $conf->setHeaderText($this->lng->txt('dcl_confirm_delete_records'));
388 
389  $has_record_ids = $this->http->wrapper()->post()->has('record_ids');
390  $record_ids = [];
391  if ($has_record_ids) {
392  $record_ids = $this->http->wrapper()->post()->retrieve(
393  'record_ids',
394  $this->refinery->kindlyTo()->listOf($this->refinery->kindlyTo()->int())
395  );
396  }
397  $all_fields = $this->table_obj->getRecordFields();
398  foreach ($record_ids as $record_id) {
400  $record = ilDclCache::getRecordCache($record_id);
401  $record_data = "";
402  foreach ($all_fields as $field) {
403  $field_record = ilDclCache::getRecordFieldCache($record, $field);
404 
405  $record_representation = ilDclCache::getRecordRepresentation($field_record);
406  if ($record_representation->getConfirmationHTML() != false) {
407  $record_data .= $field->getTitle() . ": " . $record_representation->getConfirmationHTML() . "<br />";
408  }
409  }
410  $conf->addItem('record_ids[]', (string) $record->getId(), $record_data);
411  }
412  $conf->addHiddenItem('table_id', (string) $this->table_id);
413  $conf->setConfirm($this->lng->txt('dcl_delete_records'), self::CMD_DELETE_RECORDS);
414  $conf->setCancel($this->lng->txt('cancel'), self::CMD_CANCEL_DELETE);
415  $this->tpl->setContent($conf->getHTML());
416  }
417 
421  public function deleteRecords(): void
422  {
423  $has_record_ids = $this->http->wrapper()->post()->has('record_ids');
424  $record_ids = [];
425  if ($has_record_ids) {
426  $record_ids = $this->http->wrapper()->post()->retrieve(
427  'record_ids',
428  $this->refinery->kindlyTo()->listOf($this->refinery->kindlyTo()->int())
429  );
430  }
431 
432  // Invoke deletion
433  $n_skipped = 0;
434  foreach ($record_ids as $record_id) {
436  $record = ilDclCache::getRecordCache($record_id);
437  $ref_id = $this->parent_obj->getRefId();
438 
439  if ($record->hasPermissionToDelete($ref_id)) {
440  $record->doDelete();
441  } else {
442  $n_skipped++;
443  }
444  }
445 
446  $n_deleted = (count($record_ids) - $n_skipped);
447  if ($n_deleted) {
448  $message = sprintf(
449  $this->lng->txt('dcl_deleted_records'),
450  $n_deleted
451  );
452  $this->tpl->setOnScreenMessage('success', $message, true);
453  }
454  if ($n_skipped) {
455  $message = sprintf(
456  $this->lng->txt('dcl_skipped_delete_records'),
457  $n_skipped
458  );
459  $this->tpl->setOnScreenMessage('info', $message, true);
460  }
461  $this->ctrl->redirect($this, self::CMD_LIST_RECORDS);
462  }
463 
464  private function recordBelongsToCollection(ilDclBaseRecordModel $record): bool
465  {
466  $table = $record->getTable();
467  $obj_id = $this->parent_obj->object->getId();
468  $obj_id_rec = $table->getCollectionObject()->getId();
469 
470  return $obj_id == $obj_id_rec;
471  }
472 
473  protected function setSubTabs(string $active_mode = self::GET_MODE): void
474  {
475  $this->ctrl->setParameter($this, self::GET_MODE, self::MODE_VIEW);
476  $this->tabs->addSubTab(
477  self::MODE_VIEW,
478  $this->lng->txt('view'),
479  $this->ctrl->getLinkTarget($this, self::CMD_LIST_RECORDS)
480  );
481 
482  $ref_id = $this->http->wrapper()->query()->retrieve('ref_id', $this->refinery->kindlyTo()->int());
483  if ($this->table_obj->hasPermissionToDeleteRecords($ref_id)) {
484  $this->ctrl->setParameter($this, self::GET_MODE, self::MODE_MANAGE);
485  $this->tabs->addSubTab(
486  self::MODE_MANAGE,
487  $this->lng->txt('dcl_manage'),
488  $this->ctrl->getLinkTarget($this, self::CMD_LIST_RECORDS)
489  );
490  }
491  $this->tabs->activateSubTab($active_mode);
492  $this->ctrl->clearParameters($this);
493  }
494 
495  protected function getRecordListTableGUI(bool $use_tableview_filter): ilDclRecordListTableGUI
496  {
497  $table_obj = $this->table_obj;
498 
499  $list = new ilDclRecordListTableGUI($this, "listRecords", $table_obj, $this->tableview_id, $this->mode);
500  $list->initFilter();
501  if ($use_tableview_filter) {
502  $list->initFilter();
503  $list->resetOffset();
504  $list->resetFilter();
505  $list->initFilterFromTableView();
506  }
507 
508  $list->setExternalSegmentation(true);
509  $list->setExternalSorting(true);
510  $list->determineOffsetAndOrder();
511 
512  if ($this->filter_changed) {
513  $list->resetOffset();
514  }
515 
516  $limit = $list->getLimit();
517  $offset = $list->getOffset();
518 
519  $data = $table_obj->getPartialRecords(
520  (string) $this->getRefId(),
521  $list->getOrderField(),
522  $list->getOrderDirection(),
523  $limit,
524  $offset,
525  $list->getFilter()
526  );
527  $records = $data['records'];
528  $total = $data['total'];
529 
530  $list->setMaxCount($total);
531  $list->setRecordData($records);
532 
533  $list->determineOffsetAndOrder();
534  $list->determineLimit();
535 
536  return $list;
537  }
538 
539  protected function createSwitchers(): void
540  {
541  if (ilObjDataCollectionAccess::hasWriteAccess($this->parent_obj->getRefId())) {
542  $tables = $this->parent_obj->object->getTables();
543  } else {
544  $tables = $this->parent_obj->object->getVisibleTables();
545  }
546 
547  $switcher = new ilDclSwitcher($this->toolbar, $this->ui_factory, $this->ctrl, $this->lng);
548  $switcher->addTableSwitcherToToolbar(
549  $tables,
550  self::class,
551  self::CMD_SHOW
552  );
553 
554  $switcher->addViewSwitcherToToolbar(
555  $this->table_obj->getVisibleTableViews($this->parent_obj->getRefId()),
556  $this->getTableId(),
557  self::class,
558  self::CMD_SHOW
559  );
560  }
561 
562  protected function checkAccess(): bool
563  {
564  if (null === $this->table_id || null === $this->tableview_id) {
565  return false;
566  }
567 
568  return ilObjDataCollectionAccess::hasAccessTo(
569  $this->parent_obj->getRefId(),
572  );
573  }
574 
575  public function getTableId(): int
576  {
577  return $this->table_id;
578  }
579 
580  public function getTableviewId(): int
581  {
582  return $this->tableview_id;
583  }
584 }
setSubTabs(string $active_mode=self::GET_MODE)
ilObjDataCollectionGUI $parent_obj
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
This class represents a file property in a property form.
ilGlobalTemplateInterface $tpl
recordBelongsToCollection(ilDclBaseRecordModel $record)
endImport(int $i, array $warnings)
End import.
renderer()
import(string $file, bool $simulate=false)
static rebuildTempFileByHash(string $hash)
executeCommand()
execute command
sendFile()
send File to User
showImportExcel(?ilPropertyFormGUI $form=null)
__construct(ilObjDataCollectionGUI $a_parent_obj, int $table_id, int $tableview_id)
ILIAS HTTP Services $http
getRecordListTableGUI(bool $use_tableview_filter)
static deliverFileLegacy(string $a_file, ?string $a_filename=null, ?string $a_mime=null, ?bool $isInline=false, ?bool $removeAfterDelivery=false, ?bool $a_exit_after=true)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static hasAccessToFields(int $ref_id, int $table_id)
ilObjDataCollectionGUI: ilInfoScreenGUI, ilNoteGUI, ilCommonActionDispatcherGUI ilObjDataCollectionG...
$ref_id
Definition: ltiauth.php:65
static http()
Fetches the global http state from ILIAS.
ILIAS Refinery Factory $refinery
getPartialRecords(string $ref_id, string $sort, string $direction, ?int $limit, int $offset, array $filter=[])
Return only the needed subset of record objects for the table, according to sorting, paging and filters.
global $DIC
Definition: shib_login.php:22
static getTableCache(?int $table_id=null)
static getRecordRepresentation(ilDclBaseRecordFieldModel $record_field)
Returns a record representation.
static hasWriteAccess(int $ref, ?int $user_id=0)
setRequired(bool $a_required)
string $mode
Stores current mode active.
ILIAS ResourceStorage Services $irss
static getRecordCache(?int $record_id)
static hasPermissionToAddRecord(int $ref_id, int $table_id)
$message
Definition: xapiexit.php:31
static getRecordFieldCache(object $record, object $field)
importExcel()
Import Data from Excel sheet.
listRecords(bool $use_tableview_filter=false)
importRecords(string $file, bool $simulate=false)
Import records from Excel file.