ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilPropertyFormGUI.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
21 use ILIAS\HTTP;
22 use ILIAS\Refinery;
23 
31 {
32  private array $kept_uploads = [];
33  protected bool $required_text = false;
34  protected ilLanguage $lng;
35  protected ilCtrl $ctrl;
36  protected ilTemplate $tpl;
37  protected ?ilObjUser $user = null;
38  protected ?ilSetting $settings = null;
39  private array $buttons = array();
40  private array $items = array();
41  protected string $mode = "std";
42  protected bool $check_input_called = false;
43  protected bool $disable_standard_message = false;
44  protected string $top_anchor = "il_form_top";
45  protected string $title = '';
46  protected string $titleicon = "";
47  protected string $description = "";
48  protected string $tbl_width = "";
49  protected bool $show_top_buttons = true;
50  protected bool $hide_labels = false;
51  protected bool $force_top_buttons = false;
52  protected HTTP\Services $http;
53  protected ?Refinery\Factory $refinery = null;
54 
56  protected $onload_code = [];
57 
58  public function __construct()
59  {
60  global $DIC;
61 
62  $this->lng = $DIC->language();
63  $this->ctrl = $DIC->ctrl();
64 
65  $this->user = null;
66  if (isset($DIC["ilUser"])) {
67  $this->user = $DIC["ilUser"];
68  }
69 
70  $this->settings = null;
71  if (isset($DIC["ilSetting"])) {
72  $this->settings = $DIC["ilSetting"];
73  }
74 
75  $lng = $DIC->language();
76 
77  $lng->loadLanguageModule("form");
78 
79  // avoid double submission
80  $this->setPreventDoubleSubmission(true);
81 
82  // do it as early as possible
83  if (isset($DIC["http"])) {
84  $this->http = $DIC->http();
85  }
86  if (isset($DIC["refinery"])) {
87  $this->refinery = $DIC->refinery();
88  }
89  $this->rebuildUploadedFiles();
90  if (isset($DIC["tpl"])) { // some unit tests will fail otherwise
91  $this->global_tpl = $DIC['tpl'];
92  }
93  }
94 
99  public function executeCommand()
100  {
101  $ilCtrl = $this->ctrl;
102 
103  $next_class = $ilCtrl->getNextClass($this);
104 
105  switch ($next_class) {
106  case 'ilformpropertydispatchgui':
107  $ilCtrl->saveParameter($this, 'postvar');
108  $form_prop_dispatch = new ilFormPropertyDispatchGUI();
109  $item = $this->getItemByPostVar($this->getRequestedPostVar());
110  $form_prop_dispatch->setItem($item);
111  return $ilCtrl->forwardCommand($form_prop_dispatch);
112  }
113  return false;
114  }
115 
116  protected function getRequestedPostVar(): ?string
117  {
118  $t = $this->refinery->kindlyTo()->string();
119  $w = $this->http->wrapper();
120  if ($w->post()->has("postvar")) {
121  return $w->post()->retrieve("postvar", $t);
122  }
123  if ($w->query()->has("postvar")) {
124  return $w->query()->retrieve("postvar", $t);
125  }
126  return null;
127  }
128 
129  final public function setTableWidth(string $a_width): void
130  {
131  $this->tbl_width = $a_width;
132  }
133 
134  final public function getTableWidth(): string
135  {
136  return $this->tbl_width;
137  }
138 
139  // Set Mode ('std', 'subform').
140  public function setMode(string $a_mode): void
141  {
142  $this->mode = $a_mode;
143  }
144 
145  public function getMode(): string
146  {
147  return $this->mode;
148  }
149 
150  public function setTitle(string $a_title): void
151  {
152  $this->title = $a_title;
153  }
154 
155  public function getTitle(): string
156  {
157  return $this->title;
158  }
159 
160  public function setTitleIcon(string $a_titleicon): void
161  {
162  $this->titleicon = $a_titleicon;
163  }
164 
165  public function getTitleIcon(): string
166  {
167  return $this->titleicon;
168  }
169 
170  public function setDescription(string $a_val): void
171  {
172  $this->description = $a_val;
173  }
174 
175  public function getDescription(): string
176  {
177  return $this->description;
178  }
179 
180  public function setTopAnchor(string $a_val): void
181  {
182  $this->top_anchor = $a_val;
183  }
184 
185  public function getTopAnchor(): string
186  {
187  return $this->top_anchor;
188  }
189 
190  public function setShowTopButtons(bool $a_val): void
191  {
192  $this->show_top_buttons = $a_val;
193  }
194 
195  public function getShowTopButtons(): bool
196  {
198  }
199 
200  public function setForceTopButtons(bool $a_val): void
201  {
202  $this->force_top_buttons = $a_val;
203  }
204 
205  public function getForceTopButtons(): bool
206  {
208  }
209 
213  public function addItem($a_item): void
214  {
215  $a_item->setParentForm($this);
216  $this->items[] = $a_item;
217  }
218 
219  public function removeItemByPostVar(
220  string $a_post_var,
221  bool $a_remove_unused_headers = false
222  ): void {
223  foreach ($this->items as $key => $item) {
224  if (method_exists($item, "getPostVar") && $item->getPostVar() == $a_post_var) {
225  unset($this->items[$key]);
226  }
227  }
228 
229  // remove section headers if they do not contain any items anymore
230  if ($a_remove_unused_headers) {
231  $unset_keys = array();
232  $last_item = null;
233  $last_key = null;
234  foreach ($this->items as $key => $item) {
235  if ($item instanceof ilFormSectionHeaderGUI && $last_item instanceof ilFormSectionHeaderGUI) {
236  $unset_keys[] = $last_key;
237  }
238  $last_item = $item;
239  $last_key = $key;
240  }
241  if ($last_item instanceof ilFormSectionHeaderGUI) {
242  $unset_keys[] = $last_key;
243  }
244  foreach ($unset_keys as $key) {
245  unset($this->items[$key]);
246  }
247  }
248  }
249 
250  public function getItemByPostVar(string $a_post_var): ?ilFormPropertyGUI
251  {
252  foreach ($this->items as $item) {
253  if ($item->getType() != "section_header") {
254  //if ($item->getPostVar() == $a_post_var)
255  $ret = $item->getItemByPostVar($a_post_var);
256  if (is_object($ret)) {
257  return $ret;
258  }
259  }
260  }
261 
262  return null;
263  }
264 
265  public function setItems(array $a_items): void
266  {
267  $this->items = $a_items;
268  }
269 
270  public function getItems(): array
271  {
272  return $this->items;
273  }
274 
279  public function getInputItemsRecursive(): array
280  {
281  $inputItems = array();
282 
283  foreach ($this->items as $item) {
284  if ($item->getType() == 'section_header') {
285  continue;
286  }
287 
288  $inputItems[] = $item;
289 
290  if ($item instanceof ilSubEnabledFormPropertyGUI) {
291  $inputItems = array_merge($inputItems, $item->getSubInputItemsRecursive());
292  }
293  }
294 
295  return $inputItems;
296  }
297 
298  public function setDisableStandardMessage(bool $a_val): void
299  {
300  $this->disable_standard_message = $a_val;
301  }
302 
303  public function getDisableStandardMessage(): bool
304  {
306  }
307 
308  // Get a value indicating whether the labels should be hidden or not.
309  public function getHideLabels(): bool
310  {
311  return $this->hide_labels;
312  }
313 
314  public function setHideLabels(bool $a_value = true): void
315  {
316  $this->hide_labels = $a_value;
317  }
318 
319  public function setValuesByArray(
320  array $a_values,
321  bool $a_restrict_to_value_keys = false
322  ): void {
323  foreach ($this->items as $item) {
324  if (!($a_restrict_to_value_keys) ||
325  in_array($item->getPostVar(), array_keys($a_values))) {
326  $item->setValueByArray($a_values);
327  }
328  }
329  }
330 
331  public function setValuesByPost()
332  {
333  global $DIC;
334 
335  if (!isset($DIC["http"])) {
336  return null;
337  }
338 
339  foreach ($this->items as $item) {
340  $item->setValueByArray($DIC->http()->request()->getParsedBody());
341  }
342  }
343 
344  public function checkInput(): bool
345  {
346  global $DIC;
347 
348  if ($this->check_input_called) {
349  die("Error: ilPropertyFormGUI->checkInput() called twice.");
350  }
351 
352  $ok = true;
353  foreach ($this->items as $item) {
354  $item_ok = $item->checkInput();
355  if (!$item_ok) {
356  $ok = false;
357  }
358  }
359 
360  // check if POST is missing completely (if post_max_size exceeded)
361  $post = $this->http->request()->getParsedBody();
362  if (count($this->items) > 0 && count($post) === 0) {
363  $ok = false;
364  }
365 
366  $this->check_input_called = true;
367 
368  // try to keep uploads for another try
369  $filehash = $this->getFileHash();
370  if (!$ok && !is_null($filehash) && $filehash && count($_FILES)) {
371  $hash = $filehash;
372 
373  foreach ($_FILES as $field => $data) {
374  // only try to keep files that are ok
375  // see 25484: Wrong error handling when uploading icon instead of tile
376  $item = $this->getItemByPostVar($field);
377  if (is_null($item) || !$item->checkInput()) {
378  continue;
379  }
380  // we support up to 2 nesting levels (see test/assessment)
381  if (is_array($data["tmp_name"])) {
382  foreach ($data["tmp_name"] as $idx => $upload) {
383  if (is_array($upload)) {
384  foreach ($upload as $idx2 => $file) {
385  if ($file && is_uploaded_file($file)) {
386  $file_name = $data["name"][$idx][$idx2];
387  $file_type = $data["type"][$idx][$idx2];
388  $this->keepFileUpload($hash, $field, $file, $file_name, $file_type, (string) $idx, (string) $idx2);
389  }
390  }
391  } elseif ($upload && is_uploaded_file($upload)) {
392  $file_name = $data["name"][$idx];
393  $file_type = $data["type"][$idx];
394  $this->keepFileUpload($hash, $field, $upload, $file_name, $file_type, (string) $idx);
395  }
396  }
397  } else {
398  $this->keepFileUpload($hash, $field, $data["tmp_name"], $data["name"], $data["type"]);
399  }
400  }
401  }
402  $http = $DIC->http();
403  $txt = $DIC->language()->txt("form_input_not_valid");
404  switch ($http->request()->getHeaderLine('Accept')) {
405  // When JS asks for a valid JSON-Response, we send the success and message as JSON
406  case 'application/json':
407  $stream = \ILIAS\Filesystem\Stream\Streams::ofString(json_encode([
408  'success' => $ok,
409  'message' => $txt,
410  ]));
411  $http->saveResponse($http->response()->withBody($stream));
412 
413  return $ok;
414 
415  // Otherwise, we send it using ilUtil, and it will be rendered in the Template
416  default:
417 
418  if (!$ok && !$this->getDisableStandardMessage()) {
419  $this->global_tpl->setOnScreenMessage('failure', $txt);
420  }
421 
422  return $ok;
423  }
424  }
425 
426  protected function getFileHash(): ?string
427  {
428  if (is_null($this->refinery)) {
429  return null;
430  }
431  // try to keep uploads for another try
432  $t = $this->refinery->kindlyTo()->string();
433  $w = $this->http->wrapper();
434  $filehash = null;
435  if ($w->post()->has("ilfilehash")) {
436  $filehash = $w->post()->retrieve("ilfilehash", $t);
437  }
438  return $filehash;
439  }
440 
448  public function getInput(
449  string $a_post_var,
450  bool $ensureValidation = true
451  ) {
452  // this check ensures, that checkInput has been called (incl. stripSlashes())
453  if (!$this->check_input_called && $ensureValidation) {
454  throw new LogicException('Error: ilPropertyFormGUI->getInput() called without calling checkInput() first.');
455  }
456 
457  $item = $this->getItemByPostVar($a_post_var);
458  if (is_object($item) && method_exists($item, "getInput")) {
459  return $item->getInput();
460  }
461 
462  $post = $this->http->request()->getParsedBody();
463  return $post[$a_post_var] ?? '';
464  }
465 
466  public function addCommandButton(
467  string $a_cmd,
468  string $a_text,
469  string $a_id = ""
470  ): void {
471  $this->buttons[] = array("cmd" => $a_cmd, "text" => $a_text, "id" => $a_id);
472  }
473 
474 
475  public function getCommandButtons(): array
476  {
477  return $this->buttons;
478  }
479 
480  public function clearCommandButtons(): void
481  {
482  $this->buttons = array();
483  }
484 
485  public function getContent(): string
486  {
487  global $DIC;
488  $lng = $this->lng;
489  $tpl = $DIC["tpl"];
491 
494 
495  $tpl->addJavaScript("assets/js/Basic.js");
496  $tpl->addJavaScript("assets/js/Form.js");
497 
498  $this->tpl = new ilTemplate("tpl.property_form.html", true, true, "components/ILIAS/Form");
499 
500  // check if form has not title and first item is a section header
501  // -> use section header for title and remove section header
502  // -> command buttons are presented on top
503  $fi = $this->items[0] ?? null;
504  if ($this->getMode() == "std" &&
505  $this->getTitle() == "" &&
506  is_object($fi) && $fi->getType() == "section_header"
507  ) {
508  $this->setTitle($fi->getTitle());
509  unset($this->items[0]);
510  }
511 
512 
513  // title icon
514  if ($this->getTitleIcon() != "" && is_file($this->getTitleIcon())) {
515  $this->tpl->setCurrentBlock("title_icon");
516  $this->tpl->setVariable("IMG_ICON", $this->getTitleIcon());
517  $this->tpl->parseCurrentBlock();
518  }
519 
520  // title
521  if ($this->getTitle() != "") {
522  // commands on top
523  if (count($this->buttons) > 0 && $this->getShowTopButtons() && (count($this->items) > 2 || $this->force_top_buttons)) {
524  // command buttons
525  foreach ($this->buttons as $button) {
526  $this->tpl->setCurrentBlock("cmd2");
527  $this->tpl->setVariable("CMD", $button["cmd"]);
528  $this->tpl->setVariable("CMD_TXT", $button["text"]);
529  if ($button["id"] != "") {
530  $this->tpl->setVariable("CMD2_ID", " id='" . $button["id"] . "_top'");
531  }
532  $this->tpl->parseCurrentBlock();
533  }
534  $this->tpl->setCurrentBlock("commands2");
535  $this->tpl->parseCurrentBlock();
536  }
537 
538  // required top
539  $this->tpl->setCurrentBlock("header");
540  if ($this->checkForRequiredField()) {
541  $this->tpl->setCurrentBlock("required_text_top");
542  $this->tpl->setVariable("TXT_REQUIRED_TOP", $lng->txt("required_field"));
543  $this->tpl->parseCurrentBlock();
544  }
545 
546  $this->tpl->setVariable("TXT_TITLE", $this->getTitle());
547  //$this->tpl->setVariable("LABEL", $this->getTopAnchor());
548  $this->tpl->setVariable("TXT_DESCRIPTION", $this->getDescription());
549  $this->tpl->parseCurrentBlock();
550  } elseif (!$this->required_text && $this->getMode() == "std") {
551  $this->tpl->setCurrentBlock("header");
552  // required top
553  $this->tpl->setCurrentBlock("required_text_top");
554  $this->tpl->setVariable("TXT_REQUIRED_TOP", $lng->txt("required_field"));
555  $this->tpl->parseCurrentBlock();
556  }
557  $this->tpl->touchBlock("item");
558 
559  // properties
560  $this->required_text = false;
561  foreach ($this->items as $item) {
562  if ($item->getType() != "hidden") {
563  $this->insertItem($item);
564  }
565  }
566 
567  // required
568  if ($this->required_text && $this->getMode() == "std") {
569  $this->tpl->setCurrentBlock("required_text");
570  $this->tpl->setVariable("TXT_REQUIRED", $lng->txt("required_field"));
571  $this->tpl->parseCurrentBlock();
572  }
573 
574  // command buttons
575  foreach ($this->buttons as $button) {
576  $this->tpl->setCurrentBlock("cmd");
577  $this->tpl->setVariable("CMD", $button["cmd"]);
578  $this->tpl->setVariable("CMD_TXT", $button["text"]);
579 
580  if ($button["id"] != "") {
581  $this->tpl->setVariable("CMD_ID", " id='" . $button["id"] . "'");
582  }
583 
584  $this->tpl->parseCurrentBlock();
585  }
586 
587  // #18808
588  if ($this->getMode() != "subform") {
589  // try to keep uploads even if checking input fails
590  if ($this->getMultipart()) {
591  $hash = $this->getFileHash() ?? null;
592  if (!$hash) {
593  $hash = md5(uniqid((string) mt_rand(), true));
594  }
595  $fhash = new ilHiddenInputGUI("ilfilehash");
596  $fhash->setValue($hash);
597  $this->addItem($fhash);
598  }
599  }
600 
601  // hidden properties
602  $hidden_fields = false;
603  foreach ($this->items as $item) {
604  if ($item->getType() == "hidden") {
605  $item->insert($this->tpl);
606  $hidden_fields = true;
607  }
608  }
609 
610  if ($this->required_text || count($this->buttons) > 0 || $hidden_fields) {
611  $this->tpl->setCurrentBlock("commands");
612  $this->tpl->parseCurrentBlock();
613  }
614 
615 
616  if ($this->getMode() == "subform") {
617  $this->tpl->touchBlock("sub_table");
618  } else {
619  $this->tpl->touchBlock("std_table");
620  $this->tpl->setVariable('STD_TABLE_WIDTH', $this->getTableWidth());
621  }
622 
623  return $this->tpl->get();
624  }
625 
626  protected function hideRequired(string $a_type): bool
627  {
628  // #15818
629  return $a_type == "non_editable_value";
630  }
631 
635  public function insertItem(
636  $item,
637  bool $a_sub_item = false
638  ): void {
639  global $DIC;
640  $tpl = $DIC["tpl"];
641  $lng = $this->lng;
642 
643 
644  //$cfg = array();
645 
646  //if(method_exists($item, "getMulti") && $item->getMulti())
647  if ($item instanceof ilMultiValuesItem && $item->getMulti()) {
648  $tpl->addJavascript("assets/js/ServiceFormMulti.js");
649 
650  $this->tpl->setCurrentBlock("multi_in");
651  $this->tpl->setVariable("ID", $item->getFieldId());
652  $this->tpl->parseCurrentBlock();
653 
654  $this->tpl->touchBlock("multi_out");
655 
656 
657  // add hidden item to enable preset multi items
658  // not used yet, should replace hidden field stuff
659  $multi_values = $item->getMultiValues();
660  if (is_array($multi_values) && sizeof($multi_values) > 1) {
661  $multi_value = new ilHiddenInputGUI("ilMultiValues~" . $item->getPostVar());
662  $multi_value->setValue(base64_encode(json_encode($multi_values)));
663  $this->addItem($multi_value);
664  }
665  //$cfg["multi_values"] = $multi_values;
666  }
667 
668  $item->insert($this->tpl);
669 
670  if ($item->getType() == "file" || $item->getType() == "image_file") {
671  $this->setMultipart(true);
672  }
673 
674  if ($item->getType() != "section_header") {
675  //$cfg["id"] = $item->getFieldId();
676 
677  // info text
678  if ($item->getInfo() != "") {
679  $this->tpl->setCurrentBlock("description");
680  $this->tpl->setVariable(
681  "PROPERTY_DESCRIPTION",
682  $item->getInfo()
683  );
684  $this->tpl->setVariable(
685  "DESCRIPTION_FOR_ID",
686  $item->getFieldId()
687  );
688  $this->tpl->parseCurrentBlock();
689  }
690 
691  if ($this->getMode() == "subform") {
692  // required
693  if (!$this->hideRequired($item->getType())) {
694  if ($item->getRequired()) {
695  $this->tpl->touchBlock("sub_required");
696  $this->required_text = true;
697  }
698  }
699 
700  // hidden title (for accessibility, e.g. file upload)
701  if ($item->getHiddenTitle() != "") {
702  $this->tpl->setCurrentBlock("sub_hid_title");
703  $this->tpl->setVariable(
704  "SPHID_TITLE",
705  $item->getHiddenTitle()
706  );
707  $this->tpl->parseCurrentBlock();
708  }
709 
710  $this->tpl->setCurrentBlock("sub_prop_start");
711  $this->tpl->setVariable("PROPERTY_TITLE", $item->getTitle());
712  $this->tpl->setVariable("PROPERTY_CLASS", "il_" . $item->getType());
713  if ($item->getType() != "non_editable_value" && $item->getFormLabelFor() != "") {
714  $this->tpl->setVariable("FOR_ID", ' for="' . $item->getFormLabelFor() . '" ');
715  }
716  $this->tpl->setVariable("LAB_ID", $item->getFieldId());
717  } else {
718  // required
719  if (!$this->hideRequired($item->getType())) {
720  if ($item->getRequired()) {
721  $this->tpl->touchBlock("required");
722  $this->required_text = true;
723  }
724  }
725 
726  // hidden title (for accessibility, e.g. file upload)
727  if ($item->getHiddenTitle() != "") {
728  $this->tpl->setCurrentBlock("std_hid_title");
729  $this->tpl->setVariable(
730  "PHID_TITLE",
731  $item->getHiddenTitle()
732  );
733  $this->tpl->parseCurrentBlock();
734  }
735 
736  $this->tpl->setCurrentBlock("std_prop_start");
737  $this->tpl->setVariable("PROPERTY_TITLE", $item->getTitle());
738  if ($item->getType() != "non_editable_value" && $item->getFormLabelFor() != "") {
739  $this->tpl->setVariable("FOR_ID", ' for="' . $item->getFormLabelFor() . '" ');
740  }
741  $this->tpl->setVariable("LAB_ID", $item->getFieldId());
742  if ($this->getHideLabels()) {
743  $this->tpl->setVariable("HIDE_LABELS_STYLE", " ilFormOptionHidden");
744  }
745  }
746  $this->tpl->parseCurrentBlock();
747 
748  // alert
749  if ($item->getType() != "non_editable_value" && $item->getAlert() != "") {
750  $this->tpl->setCurrentBlock("alert");
751  $this->tpl->setVariable(
752  "IMG_ALERT",
753  ilUtil::getImagePath("standard/icon_alert.svg")
754  );
755  $this->tpl->setVariable(
756  "ALT_ALERT",
757  $lng->txt("alert")
758  );
759  $this->tpl->setVariable(
760  "TXT_ALERT",
761  $item->getAlert()
762  );
763  $this->tpl->setVariable(
764  "ALERT_FOR_ID",
765  $item->getFieldId()
766  );
767  $this->tpl->parseCurrentBlock();
768  }
769 
770  // subitems
771  $sf = null;
772  if ($item->getType() != "non_editable_value" or 1) {
773  $sf = $item->getSubForm();
774  if ($item->hideSubForm() && is_object($sf)) {
775  if ($this->global_tpl) {
776  $dsfid = $item->getFieldId();
777  $this->global_tpl->addOnloadCode(
778  "il.Form.hideSubForm('subform_$dsfid');"
779  );
780  }
781  $this->addAsyncOnloadCode("il.Form.hideSubForm('subform_$dsfid');");
782  }
783  }
784 
785  $sf_content = "";
786  if (is_object($sf)) {
787  $sf_content = $sf->getContent();
788  if ($sf->getMultipart()) {
789  $this->setMultipart(true);
790  }
791  $this->tpl->setCurrentBlock("sub_form");
792  $this->tpl->setVariable("PROP_SUB_FORM", $sf_content);
793  $this->tpl->setVariable("SFID", $item->getFieldId());
794  $this->tpl->parseCurrentBlock();
795  }
796 
797  $this->tpl->setCurrentBlock("prop");
798  /* not used yet
799  $this->tpl->setVariable("ID", $item->getFieldId());
800  $this->tpl->setVariable("CFG", json_encode($cfg, JSON_THROW_ON_ERROR));*/
801  $this->tpl->parseCurrentBlock();
802  }
803 
804 
805  $this->tpl->touchBlock("item");
806  }
807 
808  public function addAsyncOnloadCode(string $code): void
809  {
810  $this->onload_code[] = $code;
811  }
812 
813  public function getHTML(): string
814  {
815  $html = parent::getHTML();
816 
817  // #13531 - get content that has to reside outside of the parent form tag, e.g. panels/layers
818  foreach ($this->items as $item) {
819  // #13536 - ilFormSectionHeaderGUI does NOT extend ilFormPropertyGUI ?!
820  if (method_exists($item, "getContentOutsideFormTag")) {
821  $outside = $item->getContentOutsideFormTag();
822  if ($outside) {
823  $html .= $outside;
824  }
825  }
826  }
827  if ($this->ctrl->isAsynch()) {
828  $html = $this->appendOnloadCode($html);
829  }
830  return $html;
831  }
832 
833  public function getHTMLAsync(): string
834  {
835  $html = $this->getHTML();
836  if (!$this->ctrl->isAsynch()) {
837  $html = $this->appendOnloadCode($html);
838  }
839  return $html;
840  }
841 
842  protected function appendOnloadCode(string $html): string
843  {
844  if (count($this->onload_code) > 0) {
845  $html .= "<script>";
846  foreach ($this->onload_code as $code) {
847  $html .= $code . "\n";
848  }
849  $html .= "</script>";
850  }
851  return $html;
852  }
853 
854  //
855  // UPLOAD HANDLING
856  //
857 
870  protected function keepFileUpload(
871  string $a_hash,
872  string $a_field,
873  string $a_tmp_name,
874  string $a_name,
875  string $a_type,
876  ?string $a_index = null,
877  ?string $a_sub_index = null
878  ): void {
879  if (in_array($a_tmp_name, $this->kept_uploads)) {
880  return; // already kept
881  }
882 
883  if (trim($a_tmp_name) == "") {
884  return;
885  }
886 
887  $a_name = ilFileUtils::getASCIIFilename($a_name);
888 
889  $tmp_file_name = implode("~~", array(session_id(),
890  $a_hash,
891  $a_field,
892  $a_index,
893  $a_sub_index,
894  str_replace("/", "~~", $a_type),
895  str_replace("~~", "_", $a_name)));
896 
897  // make sure temp directory exists
898  $temp_path = ilFileUtils::getDataDir() . "/temp";
899  if (!is_dir($temp_path)) {
900  ilFileUtils::createDirectory($temp_path);
901  }
902 
903  ilFileUtils::moveUploadedFile($a_tmp_name, $tmp_file_name, $temp_path . "/" . $tmp_file_name);
904 
906  $file_input = $this->getItemByPostVar($a_field);
907  $file_input->setPending($a_name);
908  $this->kept_uploads[] = $a_tmp_name;
909  }
910 
919  public function getFileUpload(
920  string $a_field,
921  ?string $a_index = null,
922  ?string $a_sub_index = null
923  ): array {
924  $res = array();
925  if ($a_index) {
926  if ($_FILES[$a_field]["tmp_name"][$a_index][$a_sub_index] ?? false) {
927  $res = array(
928  "tmp_name" => $_FILES[$a_field]["tmp_name"][$a_index][$a_sub_index],
929  "name" => $_FILES[$a_field]["name"][$a_index][$a_sub_index],
930  "type" => $_FILES[$a_field]["type"][$a_index][$a_sub_index],
931  "error" => $_FILES[$a_field]["error"][$a_index][$a_sub_index],
932  "size" => $_FILES[$a_field]["size"][$a_index][$a_sub_index],
933  "is_upload" => $_FILES[$a_field]["is_upload"][$a_index][$a_sub_index] ?? true
934  );
935  }
936  } elseif ($a_sub_index) {
937  if ($_FILES[$a_field]["tmp_name"][$a_index] ?? false) {
938  $res = array(
939  "tmp_name" => $_FILES[$a_field]["tmp_name"][$a_index],
940  "name" => $_FILES[$a_field]["name"][$a_index],
941  "type" => $_FILES[$a_field]["type"][$a_index],
942  "error" => $_FILES[$a_field]["error"][$a_index],
943  "size" => $_FILES[$a_field]["size"][$a_index],
944  "is_upload" => $_FILES[$a_field]["is_upload"][$a_index] ?? true
945  );
946  }
947  } else {
948  if ($_FILES[$a_field]["tmp_name"] ?? false) {
949  $res = array(
950  "tmp_name" => $_FILES[$a_field]["tmp_name"],
951  "name" => $_FILES[$a_field]["name"],
952  "type" => $_FILES[$a_field]["type"],
953  "error" => $_FILES[$a_field]["error"],
954  "size" => $_FILES[$a_field]["size"],
955  "is_upload" => $_FILES[$a_field]["is_upload"] ?? true
956  );
957  }
958  }
959  return $res;
960  }
961 
962  public function hasFileUpload(
963  string $a_field,
964  ?string $a_index = null,
965  ?string $a_sub_index = null
966  ): bool {
967  $data = $this->getFileUpload($a_field, $a_index, $a_sub_index);
968  return (bool) ($data["tmp_name"] ?? false);
969  }
970 
982  public function moveFileUpload(
983  string $a_target_directory,
984  string $a_field,
985  ?string $a_target_name = null,
986  ?string $a_index = null,
987  ?string $a_sub_index = null
988  ): string {
989  if (!is_dir($a_target_directory)) {
990  return "";
991  }
992 
993  $data = $this->getFileUpload($a_field, $a_index, $a_sub_index);
994  if ($data["tmp_name"] && file_exists($data["tmp_name"])) {
995  if ($a_target_name) {
996  $data["name"] = $a_target_name;
997  }
998 
999  $target_file = $a_target_directory . "/" . $data["name"];
1000  $target_file = str_replace("//", "/", $target_file);
1001 
1002  if ($data["is_upload"]) {
1003  if (!ilFileUtils::moveUploadedFile($data["tmp_name"], $data["name"], $target_file)) {
1004  return "";
1005  }
1006  } else {
1007  if (!ilFileUtils::rename($data["tmp_name"], $target_file)) {
1008  return "";
1009  }
1010  }
1011 
1012  return $target_file;
1013  }
1014  return "";
1015  }
1016 
1017  protected function rebuildUploadedFiles(): void
1018  {
1019  $file_hash = (string) $this->getFileHash();
1020  if ($file_hash != "") {
1021  $temp_path = ilFileUtils::getDataDir() . "/temp";
1022  if (is_dir($temp_path)) {
1023  $temp_files = glob($temp_path . "/" . session_id() . "~~" . $file_hash . "~~*");
1024  if (is_array($temp_files)) {
1025  foreach ($temp_files as $full_file) {
1026  $file = explode("~~", basename($full_file));
1027  $field = $file[2];
1028  $idx = $file[3];
1029  $idx2 = $file[4];
1030  $type = $file[5] . "/" . $file[6];
1031  $name = $file[7];
1032 
1033  if ($idx2 != "") {
1034  if (!isset($_FILES[$field]["tmp_name"][$idx][$idx2])) {
1035  $_FILES[$field]["tmp_name"][$idx][$idx2] = $full_file;
1036  $_FILES[$field]["name"][$idx][$idx2] = $name;
1037  $_FILES[$field]["type"][$idx][$idx2] = $type;
1038  $_FILES[$field]["error"][$idx][$idx2] = 0;
1039  $_FILES[$field]["size"][$idx][$idx2] = filesize($full_file);
1040  $_FILES[$field]["is_upload"][$idx][$idx2] = false;
1041  }
1042  } elseif ($idx != "") {
1043  if (!isset($_FILES[$field]["tmp_name"][$idx])) {
1044  $_FILES[$field]["tmp_name"][$idx] = $full_file;
1045  $_FILES[$field]["name"][$idx] = $name;
1046  $_FILES[$field]["type"][$idx] = $type;
1047  $_FILES[$field]["error"][$idx] = 0;
1048  $_FILES[$field]["size"][$idx] = filesize($full_file);
1049  $_FILES[$field]["is_upload"][$idx] = false;
1050  }
1051  } else {
1052  if (!isset($_FILES[$field]["tmp_name"])) {
1053  $_FILES[$field]["tmp_name"] = $full_file;
1054  $_FILES[$field]["name"] = $name;
1055  $_FILES[$field]["type"] = $type;
1056  $_FILES[$field]["error"] = 0;
1057  $_FILES[$field]["size"] = filesize($full_file);
1058  $_FILES[$field]["is_upload"] = false;
1059  }
1060  }
1061  }
1062  }
1063  }
1064  }
1065  }
1066 
1067  protected function checkForRequiredField(): bool
1068  {
1069  foreach ($this->items as $item) {
1070  if ($item instanceof ilFormSectionHeaderGUI) {
1071  return false;
1072  } elseif ($item->getType() != "hidden") {
1073  if ($this->getMode() == "subform") {
1074  if (!$this->hideRequired($item->getType())) {
1075  if ($item->getRequired()) {
1076  return true;
1077  }
1078  }
1079  } elseif (!$this->hideRequired($item->getType())) {
1080  if ($item->getRequired()) {
1081  return true;
1082  }
1083  }
1084  }
1085  }
1086 
1087  return false;
1088  }
1089 }
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setMultipart(bool $a_multipart)
$res
Definition: ltiservices.php:66
txt(string $a_topic, string $a_default_lang_fallback_mod="")
gets the text for a given topic if the topic is not in the list, the topic itself with "-" will be re...
getItemByPostVar(string $a_post_var)
removeItemByPostVar(string $a_post_var, bool $a_remove_unused_headers=false)
This class represents a form user interface.
setValue(string $a_value)
moveFileUpload(string $a_target_directory, string $a_field, ?string $a_target_name=null, ?string $a_index=null, ?string $a_sub_index=null)
Move upload to target directory.
loadLanguageModule(string $a_module)
Load language module.
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-...
static getASCIIFilename(string $a_filename)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static initDom(?ilGlobalTemplateInterface $a_main_tpl=null)
Init YUI Dom.
This class represents a hidden form property in a property form.
getNextClass($a_gui_class=null)
static http()
Fetches the global http state from ILIAS.
ilGlobalTemplateInterface $global_tpl
setHideLabels(bool $a_value=true)
setPreventDoubleSubmission(bool $a_val)
global $DIC
Definition: shib_login.php:22
static createDirectory(string $a_dir, int $a_mod=0755)
create directory
static getImagePath(string $image_name, string $module_path="", string $mode="output", bool $offline=false)
get image path (for images located in a template directory)
static moveUploadedFile(string $a_file, string $a_name, string $a_target, bool $a_raise_errors=true, string $a_mode="move_uploaded")
move uploaded file
static getDataDir()
get data directory (outside webspace)
setValuesByArray(array $a_values, bool $a_restrict_to_value_keys=false)
$txt
Definition: error.php:31
setTitleIcon(string $a_titleicon)
setTableWidth(string $a_width)
addCommandButton(string $a_cmd, string $a_text, string $a_id="")
Interface for multi values support.
static ofString(string $string)
Creates a new stream with an initial value.
Definition: Streams.php:41
setCurrentBlock(string $part=ilGlobalTemplateInterface::DEFAULT_BLOCK)
insertItem( $item, bool $a_sub_item=false)
getFileUpload(string $a_field, ?string $a_index=null, ?string $a_sub_index=null)
Get file upload data.
getInputItemsRecursive()
returns a flat array of all input items including the possibly existing subitems recursively ...
global $ilSetting
Definition: privfeed.php:31
This class represents a property that may include a sub form.
static rename(string $a_source, string $a_target)
static initEvent(?ilGlobalTemplateInterface $a_main_tpl=null)
Init YUI Event.
hasFileUpload(string $a_field, ?string $a_index=null, ?string $a_sub_index=null)
setDisableStandardMessage(bool $a_val)
$post
Definition: ltitoken.php:46
Refinery Factory $refinery