ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilDclBaseFieldModel.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 {
23  protected string $id = "";
24  protected int $table_id = 0;
25  protected string $title = "";
26  protected string $description = "";
27  protected int $datatype_id = 0;
28  protected ?int $order = null;
29  protected bool $unique = false;
31  protected array $property = [];
32  protected bool $exportable = false;
37  protected ?int $storage_location_override = null;
41  public const PROP_LENGTH = "lenght";
42  public const PROP_REGEX = "regex";
43  public const PROP_REFERENCE = "table_id";
44  public const PROP_URL = "url";
45  public const PROP_TEXTAREA = "text_area";
46  public const PROP_REFERENCE_LINK = "reference_link";
47  public const PROP_WIDTH = "width";
48  public const PROP_HEIGHT = "height";
49  public const PROP_LEARNING_PROGRESS = "learning_progress";
50  public const PROP_ILIAS_REFERENCE_LINK = "ILIAS_reference_link";
51  public const PROP_N_REFERENCE = "multiple_selection";
52  public const PROP_FORMULA_EXPRESSION = "expression";
53  public const PROP_DISPLAY_COPY_LINK_ACTION_MENU = "display_action_menu";
54  public const PROP_LINK_DETAIL_PAGE_TEXT = "link_detail_page_text";
55  public const PROP_LINK_DETAIL_PAGE_MOB = "link_detail_page_mob";
56  public const PROP_SUPPORTED_FILE_TYPES = "supported_file_types";
57  public const PROP_PLUGIN_HOOK_NAME = "plugin_hook_name";
58  // type of table il_dcl_view
59  public const EDIT_VIEW = 2;
60  public const EXPORTABLE_VIEW = 4;
61 
62  protected ilDBInterface $db;
63  protected ilLanguage $lng;
64 
65  public function __construct(int $a_id = 0)
66  {
67  global $DIC;
68  $this->db = $DIC->database();
69  $this->lng = $DIC->language();
70 
71  if ($a_id != 0) {
72  $this->id = (string) $a_id;
73  $this->doRead();
74  }
75  }
76 
80  public static function _getTitleInvalidChars(bool $a_as_regex = true): string
81  {
82  if ($a_as_regex) {
83  return '/^[^<>\\\\":]*$/i';
84  } else {
85  return '\ < > " :';
86  }
87  }
88 
89  public static function _getFieldIdByTitle(string $title, int $table_id): int
90  {
91  global $DIC;
92  $ilDB = $DIC->database();
93 
94  $result = $ilDB->query(
95  'SELECT id FROM il_dcl_field WHERE title = ' . $ilDB->quote($title, 'text') . ' AND table_id = '
96  . $ilDB->quote($table_id, 'integer')
97  );
98  $id = 0;
99  while ($rec = $ilDB->fetchAssoc($result)) {
100  $id = $rec['id'];
101  }
102 
103  return $id;
104  }
105 
110  public function setId($a_id): void
111  {
112  $this->id = (string) $a_id;
113  }
114 
118  public function getId(): string
119  {
120  return $this->id;
121  }
122 
126  public function setTableId(int $a_id): void
127  {
128  $this->table_id = $a_id;
129  }
130 
134  public function getTableId(): int
135  {
136  return $this->table_id;
137  }
138 
142  public function setTitle(string $a_title): void
143  {
144  //title cannot begin with _ as this is saved for other purposes. make __ instead.
145  if (substr($a_title, 0, 1) == "_" && substr($a_title, 0, 2) != "__") {
146  $a_title = "_" . $a_title;
147  }
148  $this->title = $a_title;
149  }
150 
154  public function getTitle(): string
155  {
156  return $this->title;
157  }
158 
162  public function setDescription(string $a_desc): void
163  {
164  $this->description = $a_desc;
165  }
166 
170  public function getDescription(): string
171  {
172  return $this->description;
173  }
174 
178  public function setDatatypeId(int $a_id): void
179  {
180  //unset the cached datatype.
181  $this->datatype = null;
182  $this->datatype_id = $a_id;
183  }
184 
188  public function getDatatypeId(): ?int
189  {
190  if ($this->isStandardField()) {
192  }
193 
194  return $this->datatype_id;
195  }
196 
197  public function isUnique(): bool
198  {
199  return $this->unique;
200  }
201 
202  public function setUnique(?bool $unique): void
203  {
204  $this->unique = (bool) $unique;
205  }
206 
207  public function getDatatype(): ilDclDatatype
208  {
209  $this->loadDatatype();
210 
211  return $this->datatype;
212  }
213 
214  public function getDatatypeTitle(): string
215  {
216  $this->loadDatatype();
217  return $this->datatype->getTitle();
218  }
219 
220  public function getPresentationTitle(): string
221  {
222  return $this->lng->txt('dcl_' . $this->getDatatypeTitle());
223  }
224 
225  public function getPresentationDescription(): string
226  {
227  return $this->lng->txt('dcl_' . $this->getDatatypeTitle() . '_desc');
228  }
229 
233  public function getStorageLocation(): ?int
234  {
235  if ($this->getStorageLocationOverride() !== null) {
236  return $this->getStorageLocationOverride();
237  }
238 
239  $this->loadDatatype();
240 
241  return $this->datatype->getStorageLocation();
242  }
243 
247  protected function loadDatatype(): void
248  {
249  if ($this->datatype == null) {
250  $this->datatype = ilDclCache::getDatatype($this->datatype_id);
251  }
252  }
253 
257  protected function loadTableFieldSetting(): void
258  {
259  $tablefield_setting = ilDclTableFieldSetting::getInstance($this->getTableId(), $this->getId());
260  $this->exportable = $tablefield_setting->isExportable();
261  $this->order = $tablefield_setting->getFieldOrder();
262  }
263 
267  public function getExportable(): bool
268  {
269  if (!isset($this->exportable)) {
270  $this->loadExportability();
271  }
272 
273  return $this->exportable;
274  }
275 
279  private function loadExportability(): void
280  {
281  if ($this->exportable == null) {
282  $this->loadTableFieldSetting();
283  }
284  }
285 
286  public function toArray(): array
287  {
288  return (array) $this;
289  }
290 
291  public function isStandardField(): bool
292  {
293  return false;
294  }
295 
296  public function doRead(): void
297  {
298  //THEN 1 ELSE 0 END AS has_options FROM il_dcl_field f WHERE id = ".$ilDB->quote($this->getId(),"integer");
299  $query = "SELECT * FROM il_dcl_field WHERE id = " . $this->db->quote($this->getId(), "integer");
300  $set = $this->db->query($query);
301  $rec = $this->db->fetchAssoc($set);
302 
303  if ($rec) {
304  $this->setTableId($rec["table_id"]);
305  if (null !== $rec["title"]) {
306  $this->setTitle($rec["title"]);
307  }
308  if (null !== $rec["description"]) {
309  $this->setDescription($rec["description"]);
310  }
311  $this->setDatatypeId($rec["datatype_id"]);
312  $this->setUnique((bool) $rec["is_unique"]);
313  }
314 
315  $this->loadProperties();
316  $this->loadTableFieldSetting();
317  }
318 
322  public function buildFromDBRecord(array $rec): void
323  {
324  $this->setId($rec["id"]);
325  $this->setTableId($rec["table_id"]);
326  $this->setTitle($rec["title"]);
327  $this->setDescription($rec["description"]);
328  $this->setDatatypeId($rec["datatype_id"]);
329  $this->setUnique($rec["is_unique"] ?? null);
330  }
331 
332  public function doCreate(): void
333  {
334  if (!ilDclTable::_tableExists($this->getTableId())) {
335  throw new ilException("The field does not have a related table!");
336  }
337 
338  $id = $this->db->nextId("il_dcl_field");
339  $this->setId($id);
340  $query = "INSERT INTO il_dcl_field (" . "id" . ", table_id" . ", datatype_id" . ", title" . ", description" . ", is_unique"
341  . " ) VALUES (" . $this->db->quote($this->getId(), "integer") . "," . $this->db->quote(
342  $this->getTableId(),
343  "integer"
344  ) . ","
345  . $this->db->quote($this->getDatatypeId(), "integer") . "," . $this->db->quote($this->getTitle(), "text") . ","
346  . $this->db->quote($this->getDescription(), "text") . "," . $this->db->quote($this->isUnique(), "integer") . ")";
347  $this->db->manipulate($query);
348 
349  $this->updateTableFieldSetting();
350 
351  $this->addToTableViews();
352  }
353 
357  protected function addToTableViews(): void
358  {
359  foreach (ilDclTableView::getAllForTableId($this->table_id) as $tableview) {
360  $tableview->createFieldSetting($this->id);
361  }
362  }
363 
364  public function doUpdate(): void
365  {
366  $this->db->update(
367  "il_dcl_field",
368  [
369  "table_id" => [
370  "integer",
371  $this->getTableId(),
372  ],
373  "datatype_id" => [
374  "text",
375  $this->getDatatypeId(),
376  ],
377  "title" => [
378  "text",
379  $this->getTitle(),
380  ],
381  "description" => [
382  "text",
383  $this->getDescription(),
384  ],
385  "is_unique" => [
386  "integer",
387  $this->isUnique(),
388  ],
389  ],
390  [
391  "id" => [
392  "integer",
393  $this->getId(),
394  ],
395  ]
396  );
397  $this->updateTableFieldSetting();
398  $this->updateProperties();
399  }
400 
404  public function updateProperties(): void
405  {
406  foreach ($this->property as $prop) {
407  $prop->store();
408  }
409  }
410 
414  protected function updateTableFieldSetting(): void
415  {
416  $tablefield_setting = ilDclTableFieldSetting::getInstance($this->getTableId(), $this->getId());
417  $tablefield_setting->setExportable($this->exportable);
418  $tablefield_setting->setFieldOrder($this->getOrder());
419  $tablefield_setting->store();
420  }
421 
425  public function doDelete(): void
426  {
427  // delete tablefield setting.
428  ilDclTableFieldSetting::getInstance($this->getTableId(), $this->getId())->delete();
429 
430  $query = "DELETE FROM il_dcl_field_prop WHERE field_id = " . $this->db->quote($this->getId(), "text");
431  $this->db->manipulate($query);
432 
433  $query = "DELETE FROM il_dcl_field WHERE id = " . $this->db->quote($this->getId(), "text");
434  $this->db->manipulate($query);
435 
436  foreach ($this->getViewSettings() as $field_setting) {
437  $field_setting->delete();
438  }
439  }
440 
444  public function getViewSettings(): array
445  {
446  return ilDclTableViewFieldSetting::where(['field' => $this->getId()])->get();
447  }
448 
449  public function getViewSetting(int $tableview_id): ilDclTableViewFieldSetting
450  {
451  return ilDclTableViewFieldSetting::getTableViewFieldSetting($this->getId(), $tableview_id);
452  }
453 
454  public function getOrder(): int
455  {
456  if ($this->order == null) {
457  $this->loadTableFieldSetting();
458  }
459 
460  return !$this->order ? 0 : $this->order;
461  }
462 
463  public function setOrder(int $order): void
464  {
465  $this->order = $order;
466  }
467 
471  protected function loadProperties(): void
472  {
473  $this->property = ilDclCache::getFieldProperties($this->getId());
474  }
475 
479  public function hasProperty(string $key): bool
480  {
481  $this->loadProperties();
482 
483  return (isset($this->property[$key]) && $this->property[$key]->getValue() != null);
484  }
485 
486  public function getProperty(string $key): mixed
487  {
488  $instance = $this->getPropertyInstance($key);
489 
490  return ($instance !== null) ? $instance->getValue() : null;
491  }
492 
493  public function getPropertyInstance(string $key): ?ilDclFieldProperty
494  {
495  $this->loadProperties();
496  if ($this->hasProperty($key)) {
497  return $this->property[$key];
498  }
499 
500  return null;
501  }
502 
503  public function setProperty(string $key, $value): ?ilDclFieldProperty
504  {
505  $this->loadProperties();
506  if (isset($this->property[$key])) {
507  $this->property[$key]->setValue($value);
508  } else {
509  $property = new ilDclFieldProperty();
510  $property->setName($key);
511  $property->setFieldId((int) $this->getId());
512  $property->setValue($value);
513 
514  $this->property[$key] = $property;
515  }
516 
517  return $this->property[$key];
518  }
519 
523  public function getValidFieldProperties(): array
524  {
525  return [];
526  }
527 
528  public function checkValidityFromForm(ilPropertyFormGUI &$form, ?int $record_id = null): void
529  {
530  $value = $form->getInput('field_' . $this->getId());
531  $this->checkValidity($value, $record_id);
532  }
533 
539  public function checkValidity($value, ?int $record_id = null): bool
540  {
541  //Don't check empty values
542  if (!isset($value)) {
543  return true;
544  }
545 
546  if ($this->isUnique()) {
547  $table = ilDclCache::getTableCache($this->getTableId());
548  foreach ($table->getRecords() as $record) {
549  if ($record->getId() !== $record_id || $record_id === 0) {
550  if ($this->normalizeValue($record->getRecordFieldValue($this->getId())) === $this->normalizeValue($value)) {
552  }
553  }
554  }
555  }
556 
557  return true;
558  }
559 
560  protected function normalizeValue(mixed $value)
561  {
562  if (is_string($value)) {
563  $value = trim(preg_replace("/\\s+/uism", " ", $value));
564  }
565 
566  return $value;
567  }
568 
572  public function cloneStructure(int $original_id): void
573  {
574  $original = ilDclCache::getFieldCache($original_id);
575  $this->setTitle($original->getTitle());
576  $this->setDatatypeId($original->getDatatypeId());
577  $this->setDescription($original->getDescription());
578  $this->setOrder($original->getOrder());
579  $this->setUnique($original->isUnique());
580  $this->setExportable($original->getExportable());
581  $this->doCreate();
582  $this->cloneProperties($original);
583 
584  // mandatory for all cloning functions
585  ilDclCache::setCloneOf($original_id, (int) $this->getId(), ilDclCache::TYPE_FIELD);
586  }
587 
588  public function afterClone(array $records)
589  {
590  foreach ($records as $rec) {
591  ilDclCache::getRecordFieldCache($rec, $this)->afterClone();
592  }
593  }
594 
595  public function cloneProperties(ilDclBaseFieldModel $originalField): void
596  {
597  $orgProps = $originalField->getValidFieldProperties();
598  if (count($orgProps) == 0) {
599  return;
600  }
601  foreach ($orgProps as $prop_name) {
602  $fieldprop_obj = new ilDclFieldProperty();
603  $fieldprop_obj->setFieldId((int) $this->getId());
604  $fieldprop_obj->setName($prop_name);
605 
606  $value = $originalField->getProperty($prop_name);
607 
608  // If reference field, we must reset the referenced field, otherwise it will point to the old ID
610  $value = null;
611  }
612 
613  if ($value) {
614  $fieldprop_obj->setValue($value);
615  $fieldprop_obj->create();
616  }
617  }
618  }
619 
620  public function setExportable(bool $exportable): void
621  {
622  $this->exportable = $exportable;
623  }
624 
625  public function allowFilterInListView(): bool
626  {
627  return true;
628  }
629 
634  public function getRecordQuerySortObject(
635  string $direction = "asc",
636  bool $sort_by_status = false
638  $sql_obj = new ilDclRecordQueryObject();
639 
640  $select_str = "sort_stloc_{$this->getId()}.value AS field_{$this->getId()}";
641  $join_str
642  = "LEFT JOIN il_dcl_record_field AS sort_record_field_{$this->getId()} ON (sort_record_field_{$this->getId()}.record_id = record.id AND sort_record_field_{$this->getId()}.field_id = "
643  . $this->db->quote($this->getId(), 'integer') . ") ";
644  $join_str .= "LEFT JOIN il_dcl_stloc{$this->getStorageLocation()}_value AS sort_stloc_{$this->getId()} ON (sort_stloc_{$this->getId()}.record_field_id = sort_record_field_{$this->getId()}.id)";
645 
646  $sql_obj->setSelectStatement($select_str);
647  $sql_obj->setJoinStatement($join_str);
648  $sql_obj->setOrderStatement("field_{$this->getId()} $direction, ID ASC");
649 
650  return $sql_obj;
651  }
652 
657  public function getRecordQueryFilterObject(
658  $filter_value = "",
659  ?ilDclBaseFieldModel $sort_field = null
661  return null;
662  }
663 
667  public function getSortField(): string
668  {
669  return $this->getTitle();
670  }
671 
675  public function hasNumericSorting(): bool
676  {
678  return true;
679  }
680 
681  return false;
682  }
683 
689  public function checkFieldCreationInput(ilPropertyFormGUI $form): bool
690  {
691  return true;
692  }
693 
697  public function getStorageLocationOverride(): ?int
698  {
700  }
701 
705  public function setStorageLocationOverride(?int $storage_location_override): void
706  {
707  $this->storage_location_override = $storage_location_override;
708  }
709 
710  public function fillHeaderExcel(ilExcel $worksheet, int &$row, int &$col): void
711  {
712  $worksheet->setCell($row, $col, $this->getTitle());
713  $col++;
714  }
715 
716  public function checkTitlesForImport(array &$titles, array &$import_fields): void
717  {
718  foreach ($titles as $k => $title) {
719  if (!mb_detect_encoding($title, "UTF-8", true) == "UTF-8") {
720  $title = mb_convert_encoding($title, 'UTF-8', 'ISO-8859-1');
721  }
722  if ($title == $this->getTitle()) {
723  $import_fields[$k] = $this;
724  }
725  }
726  }
727 
731  public function storePropertiesFromForm(ilPropertyFormGUI $form): void
732  {
733  $field_props = $this->getValidFieldProperties();
734  $representation = ilDclFieldFactory::getFieldRepresentationInstance($this);
735 
736  foreach ($field_props as $property) {
737  $value = $form->getInput($representation->getPropertyInputFieldId($property));
738 
739  // save non empty values and set them to null, when they already exist. Do not override plugin-hook when already set.
740  if (!empty($value) || ($this->getPropertyInstance($property) != null && $property != self::PROP_PLUGIN_HOOK_NAME)) {
741  $this->setProperty($property, $value)->store();
742  }
743  }
744  }
745 
749  public function fillPropertiesForm(ilPropertyFormGUI &$form): bool
750  {
751  $values = [
752  'table_id' => $this->getTableId(),
753  'field_id' => $this->getId(),
754  'title' => $this->getTitle(),
755  'datatype' => $this->getDatatypeId(),
756  'description' => $this->getDescription(),
757  'unique' => $this->isUnique(),
758  ];
759 
760  $properties = $this->getValidFieldProperties();
761  foreach ($properties as $prop) {
762  $values['prop_' . $prop] = $this->getProperty($prop);
763  }
764 
765  $form->setValuesByArray($values);
766 
767  return true;
768  }
769 
774  public function isConfirmationRequired(ilPropertyFormGUI $form): bool
775  {
776  return false;
777  }
778 
783  {
784  $ilConfirmationGUI = new ilConfirmationGUI();
785  $ilConfirmationGUI->setFormAction($form->getFormAction());
786  $ilConfirmationGUI->addHiddenItem('confirmed', "1");
787  $ilConfirmationGUI->addHiddenItem('field_id', $form->getInput('field_id'));
788  $ilConfirmationGUI->addHiddenItem('title', $form->getInput('title'));
789  $ilConfirmationGUI->addHiddenItem('description', $form->getInput('description'));
790  $ilConfirmationGUI->addHiddenItem('datatype', $form->getInput('datatype'));
791  $ilConfirmationGUI->addHiddenItem('unique', $form->getInput('unique'));
792  $ilConfirmationGUI->setConfirm($this->lng->txt('dcl_update_field'), 'update');
793  $ilConfirmationGUI->setCancel($this->lng->txt('cancel'), 'edit');
794 
795  return $ilConfirmationGUI;
796  }
797 }
hasProperty(string $key)
Checks if a certain property for a field is set.
checkTitlesForImport(array &$titles, array &$import_fields)
getConfirmationGUI(ilPropertyFormGUI $form)
called by ilDclFieldEditGUI if isConfirmationRequired returns true
checkFieldCreationInput(ilPropertyFormGUI $form)
Checks input of specific fields befor saving.
getRecordQueryFilterObject( $filter_value="", ?ilDclBaseFieldModel $sort_field=null)
Returns a query-object for building the record-loader-sql-query.
checkValidityFromForm(ilPropertyFormGUI &$form, ?int $record_id=null)
getValue()
Get the value that is displayed in the input client side.
Definition: Group.php:49
static getFieldRepresentationInstance(ilDclBaseFieldModel $field)
isConfirmationRequired(ilPropertyFormGUI $form)
called by ilDclFieldEditGUI when updating field properties if you overwrite this method, remember to also overwrite getConfirmationGUI
doDelete()
Remove field and properties.
storePropertiesFromForm(ilPropertyFormGUI $form)
called when saving the &#39;edit field&#39; form
buildFromDBRecord(array $rec)
Builds model from db record.
loadTableFieldSetting()
loadTableFieldSetting
static getFieldCache(int $field_id=0)
updateProperties()
Update properties of this field in Database.
getSortField()
Returns the sort-field id.
static _getDatatypeForId(string $id)
gives you the datatype id of a specified standard field.
fillPropertiesForm(ilPropertyFormGUI &$form)
called to fill the &#39;edit field&#39; form
static _getFieldIdByTitle(string $title, int $table_id)
getStorageLocation()
Get storage location for the model.
static where($where, $operator=null)
checkValidity($value, ?int $record_id=null)
Check if input is valid.
getInput(string $a_post_var, bool $ensureValidation=true)
Returns the input of an item, if item provides getInput method and as fallback the value of the HTTP-...
addToTableViews()
create ilDclTableViewFieldSettings for this field in each tableview
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
loadExportability()
Load exportability.
static getDatatype(int $datatyp_id)
Get cached datatypes.
setCell(int $a_row, int $col, $value, ?string $datatype=null, bool $disable_strip_tags_for_strings=false)
Set cell value.
loadProperties()
Get all properties of a field.
setDatatypeId(int $a_id)
Set datatype id.
static getTableViewFieldSetting(string $id, int $tableview_id)
cloneProperties(ilDclBaseFieldModel $originalField)
const PROP_LENGTH
General properties.
global $DIC
Definition: shib_login.php:22
static getTableCache(?int $table_id=null)
getViewSetting(int $tableview_id)
setValuesByArray(array $a_values, bool $a_restrict_to_value_keys=false)
static setCloneOf(int $old, int $new, string $type)
static getInstance(int $table_id, string $field)
getDescription()
Get description.
loadDatatype()
Load datatype for model.
updateTableFieldSetting()
update exportable and fieldorder
int $storage_location_override
With this property the datatype-storage-location can be overwritten.
hasNumericSorting()
Set to true, when the sorting should be handled numerical.
fillHeaderExcel(ilExcel $worksheet, int &$row, int &$col)
getRecordQuerySortObject(string $direction="asc", bool $sort_by_status=false)
Returns a query-object for building the record-loader-sql-query.
setStorageLocationOverride(?int $storage_location_override)
setTitle(string $a_title)
Set title.
setDescription(string $a_desc)
Set description.
getValidFieldProperties()
Returns all valid properties for a field-type.
static getRecordFieldCache(object $record, object $field)
static getAllForTableId(int $table_id)
setTableId(int $a_id)
Set table id.
static _tableExists(int $table_id)
setProperty(string $key, $value)
static getFieldProperties(string $field_id)
Cache Field properties.
getDatatypeId()
Get datatype_id.
static _getTitleInvalidChars(bool $a_as_regex=true)
All valid chars for filed titles.