ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilTextAreaInputGUI.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
27 {
28  protected array $rteSupport = [];
29  protected string $value = "";
30  protected int $cols = 30;
31  protected int $rows = 5;
32  protected bool $usert = false;
33  protected array $rtetags = [];
34  protected array $plugins = [];
35  protected array $removeplugins = [];
36  protected array $buttons = [];
37  protected array $rtesupport = [];
38  protected bool $use_tags_for_rte_only = true;
39  protected int $max_num_chars = 0;
40  protected int $min_num_chars = 0;
41  protected int $initial_rte_width = 795;
42  protected array $disabled_buttons = array();
43  protected bool $usePurifier = false;
45  protected ?string $root_block_element = null;
46 
47  protected array $rte_tag_set = array(
48  "mini" => array("strong", "em", "u", "ol", "li", "ul", "blockquote", "a", "p", "span", "br"), // #13286/#17981
49  "standard" => array("strong", "em", "u", "ol", "li", "ul", "p", "div",
50  "i", "b", "code", "sup", "sub", "pre", "strike", "gap"),
51  "extended" => array(
52  "a","blockquote","br","cite","code","div","em","h1","h2","h3",
53  "h4","h5","h6","hr","li","ol","p",
54  "pre","span","strike","strong","sub","sup","u","ul",
55  "i", "b", "gap"),
56  "extended_img" => array(
57  "a","blockquote","br","cite","code","div","em","h1","h2","h3",
58  "h4","h5","h6","hr","img","li","ol","p",
59  "pre","span","strike","strong","sub","sup","u","ul",
60  "i", "b", "gap"),
61  "extended_table" => array(
62  "a","blockquote","br","cite","code","div","em","h1","h2","h3",
63  "h4","h5","h6","hr","li","ol","p",
64  "pre","span","strike","strong","sub","sup","table","td",
65  "tr","u","ul", "i", "b", "gap"),
66  "extended_table_img" => array(
67  "a","blockquote","br","cite","code","div","em","h1","h2","h3",
68  "h4","h5","h6","hr","img","li","ol","p",
69  "pre","span","strike","strong","sub","sup","table","td",
70  "tr","u","ul", "i", "b", "gap"),
71  "full" => array(
72  "a","blockquote","br","cite","code","div","em","h1","h2","h3",
73  "h4","h5","h6","hr","img","li","ol","p",
74  "pre","span","strike","strong","sub","sup","table","td",
75  "tr","u","ul","ruby","rbc","rtc","rb","rt","rp", "i", "b", "gap"));
76 
77  public function __construct(
78  string $a_title = "",
79  string $a_postvar = ""
80  ) {
81  global $DIC;
82 
83  $this->lng = $DIC->language();
84  parent::__construct($a_title, $a_postvar);
85  $this->setType("textarea");
86  $this->setRteTagSet("standard");
87  $this->plugins = array();
88  $this->removeplugins = array();
89  $this->buttons = array();
90  $this->rteSupport = array();
91  }
92 
93  public function setValue(string $a_value): void
94  {
95  $this->value = $a_value;
96  }
97 
98  public function getValue(): string
99  {
100  return $this->value;
101  }
102 
106  public function setCols(int $a_cols): void
107  {
108  // obsolete because of bootstrap
109  $this->cols = $a_cols;
110  }
111 
112  public function getCols(): int
113  {
114  return $this->cols;
115  }
116 
117  public function setRows(int $a_rows): void
118  {
119  $this->rows = $a_rows;
120  }
121 
122  public function getRows(): int
123  {
124  return $this->rows;
125  }
126 
127  // Set Maximum number of characters allowed.
128  public function setMaxNumOfChars(int $a_number): void
129  {
130  $this->max_num_chars = $a_number;
131  }
132 
133  public function getMaxNumOfChars(): int
134  {
135  return $this->max_num_chars;
136  }
137 
138  public function setMinNumOfChars(int $a_number): void
139  {
140  $this->min_num_chars = $a_number;
141  }
142 
143  public function getMinNumOfChars(): int
144  {
145  return $this->min_num_chars;
146  }
147 
148  public function setUseRte(bool $a_usert, string $version = ''): void
149  {
150  $this->usert = $a_usert;
151  $this->rteSupport['version'] = $version;
152  }
153 
154  public function getUseRte(): bool
155  {
156  return $this->usert;
157  }
158 
159  public function addPlugin(string $a_plugin): void
160  {
161  $this->plugins[$a_plugin] = $a_plugin;
162  }
163 
164  public function removePlugin(string $a_plugin): void
165  {
166  $this->removeplugins[$a_plugin] = $a_plugin;
167  }
168 
169  // Add RTE button.
170  public function addButton(string $a_button): void
171  {
172  $this->buttons[$a_button] = $a_button;
173  }
174 
175  public function removeButton(string $a_button): void
176  {
177  unset($this->buttons[$a_button]);
178  }
179 
183  public function setRTESupport(
184  int $obj_id,
185  string $obj_type,
186  string $module,
187  ?string $cfg_template = null,
188  bool $hide_switch = false,
189  ?string $version = null
190  ): void {
191  $this->rteSupport = array(
192  "obj_id" => $obj_id,
193  "obj_type" => $obj_type,
194  "module" => $module,
195  'cfg_template' => $cfg_template,
196  'hide_switch' => $hide_switch,
197  'version' => $version
198  );
199  }
200 
201  public function removeRTESupport(): void
202  {
203  $this->rteSupport = array();
204  }
205 
206  public function setRteTags(array $a_rtetags): void
207  {
208  $this->rtetags = $a_rtetags;
209  }
210 
211  public function getRteTags(): array
212  {
213  return $this->rtetags;
214  }
215 
220  public function setRteTagSet(string $a_set_name): void
221  {
222  $this->setRteTags($this->rte_tag_set[$a_set_name]);
223  }
224 
225  public function getRteTagSet($a_set_name): array
226  {
227  return $this->rte_tag_set[$a_set_name];
228  }
229 
230  public function getRteTagString(): string
231  {
232  $result = "";
233  foreach ($this->getRteTags() as $tag) {
234  $result .= "<$tag>";
235  }
236  return $result;
237  }
238 
242  public function setUseTagsForRteOnly(bool $a_val): void
243  {
244  $this->use_tags_for_rte_only = $a_val;
245  }
246 
247  public function getUseTagsForRteOnly(): bool
248  {
250  }
251 
252  public function setValueByArray(array $a_values): void
253  {
254  $this->setValue($a_values[$this->getPostVar()] ?? "");
255 
256  foreach ($this->getSubItems() as $item) {
257  $item->setValueByArray($a_values);
258  }
259  }
260 
261  public function checkInput(): bool
262  {
263  $lng = $this->lng;
264 
265  $value = $this->getInput();
266  if ($this->getRequired() && trim($value) == "") {
267  $this->setAlert($lng->txt("msg_input_is_required"));
268  return false;
269  }
270 
271  if ($this->isCharLimited()) {
272  //avoid whitespace surprises. #20630, #20674
273  $ascii_whitespaces = chr(194) . chr(160);
274  $ascii_breaklines = chr(13) . chr(10);
275 
276  $to_replace = array($ascii_whitespaces, $ascii_breaklines, "&lt;", "&gt;", "&amp;");
277  $replace_to = array(' ', '', "_", "_", "_");
278 
279  #20630 mbstring extension is mandatory for 5.4
280  $chars_entered = mb_strlen(strip_tags(str_replace($to_replace, $replace_to, $value)));
281 
282  if ($this->getMaxNumOfChars() && ($chars_entered > $this->getMaxNumOfChars())) {
283  $this->setAlert($lng->txt("msg_input_char_limit_max"));
284  return false;
285  } elseif ($this->getMinNumOfChars() && ($chars_entered < $this->getMinNumOfChars())) {
286  $this->setAlert($lng->txt("msg_input_char_limit_min"));
287  return false;
288  }
289  }
290 
291  return $this->checkSubItemsInput();
292  }
293 
294  public function getInput(): string
295  {
296  if ($this->usePurifier() && $this->getPurifier()) {
297  $value = $this->getPurifier()->purify($this->raw($this->getPostVar()));
298  } else {
299  $allowed = $this->getRteTagString();
300  if (isset($this->plugins["latex"]) && $this->plugins["latex"] == "latex" && !is_int(strpos($allowed, "<span>"))) {
301  $allowed .= "<span>";
302  }
303  $value = ($this->getUseRte() || !$this->getUseTagsForRteOnly())
304  ? ilUtil::stripSlashes($this->raw($this->getPostVar()), true, $allowed)
305  : $this->str($this->getPostVar());
306  }
307 
308  $value = self::removeProhibitedCharacters($value);
309  return $value;
310  }
311 
312  public function insert(ilTemplate $a_tpl): void
313  {
314  $lng = $this->lng;
315 
316  $ttpl = new ilTemplate("tpl.prop_textarea.html", true, true, "components/ILIAS/Form");
317 
318  if ($this->getInfo() !== '') {
319  $ttpl->setCurrentBlock('described_by_description');
320  $ttpl->setVariable('DESCRIBED_BY_DESCRIPTION_FIELD_ID', $this->getFieldId());
321  $ttpl->parseCurrentBlock();
322  }
323 
324  // disabled rte
325  if ($this->getUseRte() && $this->getDisabled()) {
326  $ttpl->setCurrentBlock("disabled_rte");
327  $ttpl->setVariable("DR_VAL", $this->getValue());
328  } else {
329  if ($this->getUseRte()) {
330  $ttpl->touchBlock("rteditor");
331  $rtestring = ilRTE::_getRTEClassname();
332  $rte = new $rtestring((string) $this->rteSupport['version']);
333  $rte->setInitialWidth($this->getInitialRteWidth());
334 
335  // @todo: Check this.
336  $rte->addPlugin("emoticons");
337  foreach ($this->plugins as $plugin) {
338  if (strlen($plugin)) {
339  $rte->addPlugin($plugin);
340  }
341  }
342  foreach ($this->removeplugins as $plugin) {
343  if (strlen($plugin)) {
344  $rte->removePlugin($plugin);
345  }
346  }
347 
348  foreach ($this->buttons as $button) {
349  if (strlen($button)) {
350  $rte->addButton($button);
351  }
352  }
353 
354  $rte->disableButtons($this->getDisabledButtons());
355 
356  if ($this->getRTERootBlockElement() !== null) {
357  $rte->setRTERootBlockElement($this->getRTERootBlockElement());
358  }
359 
360  if (count($this->rteSupport) >= 3) {
361  $rte->addRTESupport($this->rteSupport["obj_id"], $this->rteSupport["obj_type"], $this->rteSupport["module"], false, $this->rteSupport['cfg_template']);
362  } else {
363  // disable all plugins for mini-tagset
364  if (!array_diff($this->getRteTags(), $this->getRteTagSet("mini"))) {
365  $rte->removeAllPlugins();
366 
367  // #13603 - "paste from word" is essential
368  $rte->addPlugin("paste");
369  //Add plugins 'lists', 'code' and 'link': in tinymce 3 it wasnt necessary to configure these plugins
370  $rte->addPlugin("lists");
371  $rte->addPlugin("link");
372  $rte->addPlugin("code");
373 
374  if (method_exists($rte, 'removeAllContextMenuItems')) {
375  $rte->removeAllContextMenuItems(); //https://github.com/ILIAS-eLearning/ILIAS/pull/3088#issuecomment-805830050
376  }
377 
378  // #11980 - p-tag is mandatory but we do not want the icons it comes with
379  $rte->disableButtons(array("anchor", "alignleft", "aligncenter",
380  "alignright", "alignjustify", "formatselect", "removeformat",
381  "cut", "copy", "paste", "pastetext")); // JF, 2013-12-09
382  }
383  $rte->addCustomRTESupport(0, "", $this->getRteTags());
384  }
385 
386  $ttpl->touchBlock("prop_ta_w");
387  } else {
388  if ($this->getCols() > 5) {
389  $ttpl->setCurrentBlock("prop_ta_c");
390  $ttpl->setVariable("COLS", $this->getCols());
391  $ttpl->parseCurrentBlock();
392  } else {
393  $ttpl->touchBlock("prop_ta_w");
394  }
395  }
396  $ttpl->setCurrentBlock("prop_textarea");
397  $ttpl->setVariable("ROWS", $this->getRows());
398  $ttpl->setVariable("POST_VAR", $this->getPostVar());
399  $ttpl->setVariable("ID", $this->getFieldId());
400  if ($this->getDisabled()) {
401  $ttpl->setVariable('DISABLED', 'disabled="disabled" ');
402  }
403  $ttpl->setVariable("PROPERTY_VALUE", ilLegacyFormElementsUtil::prepareFormOutput($this->getValue()));
404 
405  if ($this->getRequired()) {
406  $ttpl->setVariable("REQUIRED", "required=\"required\"");
407  }
408 
409  if ($this->isCharLimited()) {
410  $ttpl->setVariable("MAXCHARS", $this->getMaxNumOfChars());
411  $ttpl->setVariable("MINCHARS", $this->getMinNumOfChars());
412 
413  $lng->toJS("form_chars_remaining");
414  }
415  }
416  $ttpl->parseCurrentBlock();
417 
418  if ($this->isCharLimited()) {
419  $ttpl->setVariable("FEEDBACK_MAX_LIMIT", $this->getMaxNumOfChars());
420  $ttpl->setVariable("FEEDBACK_ID", $this->getFieldId());
421  $ttpl->setVariable("CHARS_REMAINING", $lng->txt("form_chars_remaining"));
422  }
423 
424  if ($this->getDisabled()) {
425  $ttpl->setVariable(
426  "HIDDEN_INPUT",
427  $this->getHiddenTag($this->getPostVar(), $this->getValue())
428  );
429  }
430  $a_tpl->setCurrentBlock("prop_generic");
431  $a_tpl->setVariable("PROP_GENERIC", $ttpl->get());
432  $a_tpl->parseCurrentBlock();
433  }
434 
440  public function usePurifier(?bool $a_flag = null)
441  {
442  if (null === $a_flag) {
443  return $this->usePurifier;
444  }
445 
446  $this->usePurifier = $a_flag;
447  return $this;
448  }
449 
453  public function setPurifier(ilHtmlPurifierInterface $Purifier): self
454  {
455  $this->Purifier = $Purifier;
456  return $this;
457  }
458 
460  {
461  return $this->Purifier;
462  }
463 
464  public function setRTERootBlockElement(?string $a_root_block_element): self
465  {
466  $this->root_block_element = $a_root_block_element;
467  return $this;
468  }
469 
470  public function getRTERootBlockElement(): ?string
471  {
473  }
474 
480  public function disableButtons($a_button): self
481  {
482  if (is_array($a_button)) {
483  $this->disabled_buttons = array_unique(array_merge($this->disabled_buttons, $a_button));
484  } else {
485  $this->disabled_buttons = array_unique(array_merge($this->disabled_buttons, array($a_button)));
486  }
487  return $this;
488  }
489 
493  public function getDisabledButtons(bool $as_array = true)
494  {
495  if (!$as_array) {
496  return implode(',', $this->disabled_buttons);
497  } else {
499  }
500  }
501 
502  public function getInitialRteWidth(): int
503  {
505  }
506 
507  public function setInitialRteWidth(int $initial_rte_width): void
508  {
509  $this->initial_rte_width = $initial_rte_width;
510  }
511 
512  public function isCharLimited(): bool
513  {
514  if ($this->getMaxNumOfChars() || $this->getMinNumOfChars()) {
515  return true;
516  }
517 
518  return false;
519  }
520 }
static _getRTEClassname()
parseCurrentBlock(string $part=ilGlobalTemplateInterface::DEFAULT_BLOCK)
ilHtmlPurifierInterface $Purifier
removePlugin(string $a_plugin)
setValueByArray(array $a_values)
setRTERootBlockElement(?string $a_root_block_element)
$version
Definition: plugin.php:24
setRTESupport(int $obj_id, string $obj_type, string $module, ?string $cfg_template=null, bool $hide_switch=false, ?string $version=null)
Set RTE support for a special module.
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...
setInitialRteWidth(int $initial_rte_width)
static stripSlashes(string $a_str, bool $a_strip_html=true, string $a_allow="")
__construct(string $a_title="", string $a_postvar="")
static prepareFormOutput($a_str, bool $a_strip=false)
setRteTagSet(string $a_set_name)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
Interface for html sanitizing functionality.
disableButtons($a_button)
Sets buttons which should be disabled in TinyMCE.
setVariable($variable, $value='')
Sets a variable value.
Definition: IT.php:544
global $DIC
Definition: shib_login.php:22
getHiddenTag(string $a_post_var, string $a_value)
setCurrentBlock(string $part=ilGlobalTemplateInterface::DEFAULT_BLOCK)
setUseTagsForRteOnly(bool $a_val)
Set use tags for RTE only (default is true)
removeButton(string $a_button)
usePurifier(?bool $a_flag=null)
Setter/Getter for the html purifier usage.
__construct(Container $dic, ilPlugin $plugin)
This class represents a property that may include a sub form.
This class represents a text area property in a property form.
setPurifier(ilHtmlPurifierInterface $Purifier)
Setter for the html purifier.
setUseRte(bool $a_usert, string $version='')
toJS($a_lang_key, ?ilGlobalTemplateInterface $a_tpl=null)
Transfer text to Javascript.
getDisabledButtons(bool $as_array=true)