ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilDclSelectionFieldModel.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 {
23  public const SELECTION_TYPE_SINGLE = 'selection_type_single';
24  public const SELECTION_TYPE_MULTI = 'selection_type_multi';
25  public const SELECTION_TYPE_COMBOBOX = 'selection_type_combobox';
26  // those should be overwritten by subclasses
27  public const PROP_SELECTION_TYPE = '';
28  public const PROP_SELECTION_OPTIONS = '';
29 
30  public function getValidFieldProperties(): array
31  {
32  return [static::PROP_SELECTION_OPTIONS, static::PROP_SELECTION_TYPE];
33  }
34 
39  public function getRecordQueryFilterObject(
40  $filter_value = "",
41  ?ilDclBaseFieldModel $sort_field = null
43  global $DIC;
44  $ilDB = $DIC['ilDB'];
45 
46  $join_str
47  = " LEFT JOIN il_dcl_record_field AS filter_record_field_{$this->getId()} ON (filter_record_field_{$this->getId()}.record_id = record.id AND filter_record_field_{$this->getId()}.field_id = "
48  . $ilDB->quote($this->getId(), 'integer') . ") ";
49 
50  $join_str .= " LEFT JOIN il_dcl_stloc{$this->getStorageLocation()}_value AS filter_stloc_{$this->getId()} ON (filter_stloc_{$this->getId()}.record_field_id = filter_record_field_{$this->getId()}.id";
51 
52  $where_str = " AND ";
53  if ($filter_value == 'none') {
54  $where_str .= "("
55  . "filter_stloc_{$this->getId()}.value IS NULL "
56  . " OR filter_stloc_{$this->getId()}.value = " . $ilDB->quote("", 'text')
57  . " OR filter_stloc_{$this->getId()}.value = " . $ilDB->quote("[]", 'text')
58  . ") ";
59  } else {
60  if ($this->isMulti()) {
61  $where_str .= " (" .
62  "filter_stloc_{$this->getId()}.value = " . $ilDB->quote("[$filter_value]", 'text') . " OR " .
63  "filter_stloc_{$this->getId()}.value LIKE " . $ilDB->quote("%\"$filter_value\"%", 'text') . " OR " .
64  "filter_stloc_{$this->getId()}.value LIKE " . $ilDB->quote("%,$filter_value,%", 'text') . " OR " .
65  "filter_stloc_{$this->getId()}.value LIKE " . $ilDB->quote("%[$filter_value,%", 'text') . " OR " .
66  "filter_stloc_{$this->getId()}.value LIKE " . $ilDB->quote("%,$filter_value]%", 'text') .
67  ") ";
68  } else {
69  $where_str .= "filter_stloc_{$this->getId()}.value = "
70  . $ilDB->quote($filter_value, 'integer');
71  }
72  }
73 
74  $join_str .= ") ";
75 
76  $sql_obj = new ilDclRecordQueryObject();
77  $sql_obj->setJoinStatement($join_str);
78  $sql_obj->setWhereStatement($where_str);
79 
80  return $sql_obj;
81  }
82 
83  public function isMulti(): bool
84  {
85  return ($this->getProperty(static::PROP_SELECTION_TYPE) == self::SELECTION_TYPE_MULTI);
86  }
87 
88  public function checkFieldCreationInput(ilPropertyFormGUI $form): bool
89  {
90  $options_post_var = "prop_" . static::PROP_SELECTION_OPTIONS;
91  foreach ($form->getInput($options_post_var) as $value) {
92  if ($value["selection_value"] == "") {
93  $inputObj = $form->getItemByPostVar($options_post_var);
94  $inputObj->setAlert($this->lng->txt("msg_input_is_required"));
95  return false;
96  }
97  }
98 
99  return parent::checkFieldCreationInput($form);
100  }
101 
106  public function storePropertiesFromForm(ilPropertyFormGUI $form): void
107  {
108  $representation = ilDclFieldFactory::getFieldRepresentationInstance($this);
109 
110  $field_props = $this->getValidFieldProperties();
111  foreach ($field_props as $property) {
112  $value = $form->getInput($representation->getPropertyInputFieldId($property));
113 
114  // break down the multidimensional array from the multi input
115  // e.g.: { [0] => { [0] => 'x' }, [1] => { [1] => 'y' } } TO { [0] => 'x', [1] => 'y' }
116  if (is_array($value)) {
117  foreach ($value as $k => $v) {
118  if (is_array($v)) {
119  $value[$k] = array_shift($v);
120  }
121  }
122  }
123 
124  // save non empty values and set them to null, when they already exist. Do not override plugin-hook when already set.
125  if (!empty($value) || ($this->getPropertyInstance($property) != null && $property != self::PROP_PLUGIN_HOOK_NAME)) {
126  $this->setProperty($property, $value);
127  }
128  }
129  }
130 
135  public function fillPropertiesForm(ilPropertyFormGUI &$form): bool
136  {
137  $values = [
138  'table_id' => $this->getTableId(),
139  'field_id' => $this->getId(),
140  'title' => $this->getTitle(),
141  'datatype' => $this->getDatatypeId(),
142  'description' => $this->getDescription(),
143  'unique' => $this->isUnique(),
144  ];
145 
146  $properties = $this->getValidFieldProperties();
147  foreach ($properties as $prop) {
148  if ($prop == static::PROP_SELECTION_OPTIONS) {
149  $options = ilDclSelectionOption::getAllForField((int) $this->getId());
150  $prop_values = [];
151  foreach ($options as $option) {
152  // the 'selection_value' is for a correct input
153  $prop_values[$option->getOptId()] = ['selection_value' => $option->getValue()];
154  }
155 
156  $values['prop_' . $prop] = $prop_values;
157  } else {
158  $values['prop_' . $prop] = $this->getProperty($prop);
159  }
160  }
161 
162  $form->setValuesByArray($values);
163 
164  return true;
165  }
166 
170  public function setProperty(string $key, $value): ?ilDclFieldProperty
171  {
172  $is_update = $this->getProperty($key);
173  switch ($key) {
174  case static::PROP_SELECTION_OPTIONS:
175 
177  $sorting = 1;
178  foreach ($value as $id => $val) {
179  ilDclSelectionOption::storeOption((int) $this->getId(), $id, $sorting, $val);
180  $sorting++;
181  }
182  // if the field is not being created reorder the options in the existing record fields
183  if ($is_update) {
184  $this->reorderExistingValues();
185  }
186  break;
187  case static::PROP_SELECTION_TYPE:
188  $will_be_multi = ($value == self::SELECTION_TYPE_MULTI);
189  // if the "Multi" property has changed, adjust the record field values
190  if ($is_update && ($this->isMulti() && !$will_be_multi || !$this->isMulti() && $will_be_multi)) {
191  $this->multiPropertyChanged($will_be_multi);
192  }
193  parent::setProperty($key, $value)->store();
194  break;
195  default:
196  parent::setProperty($key, $value)->store();
197  }
198 
199  return null;
200  }
201 
205  public function reorderExistingValues(): void
206  {
207  $options = ilDclSelectionOption::getAllForField((int) $this->getId());
208  // loop each record(-field)
209  foreach (ilDclCache::getTableCache($this->getTableId())->getRecords() as $record) {
210  $record_field = $record->getRecordField((int) $this->getId());
211  $record_field_value = $record_field->getValue();
212 
213  if (is_array($record_field_value) && count($record_field_value) > 1) {
214  $sorted_array = [];
215  // $options has the right order, so loop those
216  foreach ($options as $option) {
217  if (in_array($option->getOptId(), $record_field_value)) {
218  $sorted_array[] = $option->getOptId();
219  }
220  }
221  $record_field->setValue($sorted_array);
222  $record_field->doUpdate();
223  }
224  }
225  }
226 
230  protected function multiPropertyChanged(bool $is_multi_now): void
231  {
232  foreach (ilDclCache::getTableCache($this->getTableId())->getRecords() as $record) {
233  $record_field = $record->getRecordField((int) $this->getId());
234  $record_field_value = $record_field->getValue();
235 
236  if ($record_field_value && !is_array($record_field_value) && $is_multi_now) {
237  $record_field->setValue([$record_field_value]);
238  $record_field->doUpdate();
239  } else {
240  if (is_array($record_field_value) && !$is_multi_now) {
241  $record_field->setValue(array_shift($record_field_value));
242  $record_field->doUpdate();
243  }
244  }
245  }
246  }
247 
252  public function getProperty(string $key): mixed
253  {
254  switch ($key) {
255  case static::PROP_SELECTION_OPTIONS:
256  return ilDclSelectionOption::getAllForField((int) $this->getId());
257  default:
258  return parent::getProperty($key);
259  }
260  }
261 
262  public function getRecordQuerySortObject(
263  string $direction = "asc",
264  bool $sort_by_status = false
266  if ($this->isMulti()) {
267  return null;
268  }
269 
270  $sql_obj = new ilDclRecordQueryObject();
271 
272  $select_str = "sel_opts_{$this->getId()}.value AS field_{$this->getId()}";
273  $join_str
274  = "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 = "
275  . $this->db->quote($this->getId(), 'integer') . ") ";
276  $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) ";
277  $join_str .= "LEFT JOIN il_dcl_sel_opts as sel_opts_{$this->getId()} ON (sel_opts_{$this->getId()}.opt_id = sort_stloc_{$this->getId()}.value AND sel_opts_{$this->getId()}.field_id = "
278  . $this->db->quote($this->getId(), 'integer') . ") ";
279 
280  $sql_obj->setSelectStatement($select_str);
281  $sql_obj->setJoinStatement($join_str);
282  $sql_obj->setOrderStatement("field_{$this->getId()} $direction , ID ASC");
283 
284  return $sql_obj;
285  }
286 
287  public function cloneProperties(ilDclBaseFieldModel $originalField): void
288  {
289  parent::cloneProperties($originalField);
290  $options = ilDclSelectionOption::getAllForField((int) $originalField->getId());
291  foreach ($options as $opt) {
292  $new_opt = new ilDclSelectionOption();
293  $new_opt->cloneOption($opt);
294  $new_opt->setFieldId((int) $this->getId());
295  $new_opt->store();
296  }
297  }
298 
299  public function doDelete(): void
300  {
301  foreach (ilDclSelectionOption::getAllForField((int) $this->getId()) as $option) {
302  $option->delete();
303  }
304  parent::doDelete();
305  }
306 
307  public function isConfirmationRequired(ilPropertyFormGUI $form): bool
308  {
309  $will_be_multi = ($form->getInput('prop_' . static::PROP_SELECTION_TYPE) == self::SELECTION_TYPE_MULTI);
310 
311  return $this->isMulti() && !$will_be_multi;
312  }
313 
315  {
316  $representation = ilDclFieldFactory::getFieldRepresentationInstance($this);
317  $prop_selection_options = $representation->getPropertyInputFieldId(static::PROP_SELECTION_OPTIONS);
318  $prop_selection_type = $representation->getPropertyInputFieldId(static::PROP_SELECTION_TYPE);
319 
320  $ilConfirmationGUI = parent::getConfirmationGUI($form);
321  $ilConfirmationGUI->setHeaderText($this->lng->txt('dcl_msg_mc_to_sc_confirmation'));
322  $ilConfirmationGUI->addHiddenItem($prop_selection_type, $form->getInput($prop_selection_type));
323  foreach ($form->getInput($prop_selection_options) as $key => $option) {
324  $ilConfirmationGUI->addHiddenItem(
325  $prop_selection_options . "[$key][selection_value]",
326  $option['selection_value']
327  );
328  }
329 
330  return $ilConfirmationGUI;
331  }
332 }
getRecordQuerySortObject(string $direction="asc", bool $sort_by_status=false)
getItemByPostVar(string $a_post_var)
static getFieldRepresentationInstance(ilDclBaseFieldModel $field)
cloneProperties(ilDclBaseFieldModel $originalField)
checkFieldCreationInput(ilPropertyFormGUI $form)
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-...
getRecordQueryFilterObject( $filter_value="", ?ilDclBaseFieldModel $sort_field=null)
Returns a query-object for building the record-loader-sql-query.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
fillPropertiesForm(ilPropertyFormGUI &$form)
static flushOptions(int $field_id)
global $DIC
Definition: shib_login.php:22
static getAllForField(int $field_id)
static getTableCache(?int $table_id=null)
setValuesByArray(array $a_values, bool $a_restrict_to_value_keys=false)
getDescription()
Get description.
storePropertiesFromForm(ilPropertyFormGUI $form)
called when saving the &#39;edit field&#39; form
isConfirmationRequired(ilPropertyFormGUI $form)
multiPropertyChanged(bool $is_multi_now)
changes the values of all record fields, since the property "multi" has changed
reorderExistingValues()
sorts record field values by the new order
getDatatypeId()
Get datatype_id.
getConfirmationGUI(ilPropertyFormGUI $form)