ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
Input.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
34 use LogicException;
35 use Generator;
38 
42 abstract class Input implements InputInternal
43 {
44  use ComponentHelper;
45 
52  protected $value = null;
53 
57  protected ?string $error = null;
58 
59  private ?string $name = null;
60 
61  protected ?string $dedicated_name = null;
62 
68  protected ?Result $content = null;
69 
73  protected array $operations = [];
74 
78  public function __construct(
79  protected DataFactory $data_factory,
80  protected Refinery $refinery,
81  ) {
82  }
83 
89  public function getValue()
90  {
91  return $this->value;
92  }
93 
101  public function withValue($value): self
102  {
103  $this->checkArg("value", $this->isClientSideValueOk($value), "Display value does not match input type.");
104  $clone = clone $this;
105  $clone->value = $value;
106  return $clone;
107  }
108 
114  abstract protected function isClientSideValueOk($value): bool;
115 
119  public function getError(): ?string
120  {
121  return $this->error;
122  }
123 
127  public function withError(string $error): self
128  {
129  $clone = clone $this;
130  $clone->setError($error);
131  return $clone;
132  }
133 
137  private function setError(string $error): void
138  {
139  $this->error = $error;
140  }
141 
145  public function withAdditionalTransformation(Transformation $trafo): self
146  {
147  $clone = clone $this;
148  $clone->setAdditionalTransformation($trafo);
149  return $clone;
150  }
151 
158  protected function setAdditionalTransformation(Transformation $trafo): void
159  {
160  $this->operations[] = $trafo;
161  if ($this->content !== null) {
162  if (!$this->content->isError()) {
163  $this->content = $trafo->applyTo($this->content);
164  }
165  if ($this->content->isError()) {
166  $this->setError($this->content->error());
167  }
168  }
169  }
170 
174  final public function getDedicatedName(): ?string
175  {
176  return $this->dedicated_name;
177  }
178 
182  final public function withDedicatedName(string $dedicated_name): self
183  {
184  $clone = clone $this;
185  $clone->dedicated_name = $dedicated_name;
186  return $clone;
187  }
188 
192  final public function getName(): ?string
193  {
194  return $this->name;
195  }
196 
200  public function withNameFrom(NameSource $source, ?string $parent_name = null): self
201  {
202  $clone = clone $this;
203  if ($source instanceof DynamicInputsNameSource) {
204  $clone->name = '';
205  } else {
206  $clone->name = ($parent_name !== null) ? $parent_name . '/' : '';
207  }
208  $clone->name .= ($clone->dedicated_name !== null)
209  ? $source->getNewDedicatedName($clone->dedicated_name)
210  : $source->getNewName();
211  return $clone;
212  }
213 
220  public function withInput(InputData $input): self
221  {
222  if ($this->getName() === null) {
223  throw new LogicException("Can only collect if input has a name.");
224  }
225 
226 
227  //TODO: Discuss, is this correct here. If there is no input contained in this post
228  //We assign null. Note that unset checkboxes are not contained in POST.
229  $value = $input->getOr($this->getName(), null);
230  // ATTENTION: There was a special case for the Filter Input Container here,
231  // which lead to #27909. The issue will most certainly appear again in. If
232  // you are the one debugging it and came here: Please don't put knowledge
233  // of the special case for the filter in this general class. Have a look
234  // into https://mantis.ilias.de/view.php?id=27909 for the according discussion.
235  $clone = $this->withValue($value);
236 
237  $clone->content = $this->applyOperationsTo($clone->getValue());
238  if ($clone->content->isError()) {
239  $error = $clone->content->error();
240  if ($error instanceof \Exception) {
241  $error = $error->getMessage();
242  }
243  return $clone->withError("" . $error);
244  }
245 
246  return $clone;
247  }
248 
254  protected function applyOperationsTo($res): Result
255  {
256  $res = $this->data_factory->ok($res);
257  foreach ($this->getOperations() as $op) {
258  if ($res->isError()) {
259  return $res;
260  }
261  $res = $op->applyTo($res);
262  }
263 
264  return $res;
265  }
266 
272  protected function getOperations(): Generator
273  {
274  foreach ($this->operations as $op) {
275  yield $op;
276  }
277  }
278 
282  public function getContent(): Result
283  {
284  if (is_null($this->content)) {
285  throw new LogicException("No content of this field has been evaluated yet. Seems withRequest was not called.");
286  }
287  return $this->content;
288  }
289 }
withValue($value)
Get an input like this with another value displayed on the client side.
Definition: Input.php:101
$res
Definition: ltiservices.php:66
getError()
The error of the input as used in HTML.
Definition: Input.php:119
setError(string $error)
Set an error on this input.
Definition: Input.php:137
Describes how Input-Elements want to interact with posted data.
Definition: InputData.php:29
getOr(string $name, $default)
Get a named value from the data and fallback to default if that name does not exist.
trait JavaScriptBindable
Trait for components implementing JavaScriptBindable providing standard implementation.
withAdditionalTransformation(Transformation $trafo)
Apply a transformation to the current or future content.
Definition: Input.php:145
applyOperationsTo($res)
Applies the operations in this instance to the value.
Definition: Input.php:254
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
Result $content
This is the current content of the input in the abstraction.
Definition: Input.php:68
isClientSideValueOk($value)
Check if the value is good to be displayed client side.
withNameFrom(NameSource $source, ?string $parent_name=null)
Definition: Input.php:200
Other than the FormInputNameSource this name source is for inputs that can be dynamically added multi...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
withDedicatedName(string $dedicated_name)
Definition: Input.php:182
Describes the interface of inputs that is used for internal processing of data from the client...
string $error
This is an error on the input as displayed client side.
Definition: Input.php:57
getNewName()
Generates a unique name on every call.
setAdditionalTransformation(Transformation $trafo)
Apply a transformation to the current or future content.
Definition: Input.php:158
getOperations()
Get the operations that should be performed on the input.
Definition: Input.php:272
getValue()
Get the value that is displayed in the input client side.
Definition: Input.php:89
withInput(InputData $input)
Collects the input, applies trafos on the input and returns a new input reflecting the data that was ...
Definition: Input.php:220
A transformation is a function from one datatype to another.
This describes commonalities between all inputs.
Definition: Input.php:46
__construct(protected DataFactory $data_factory, protected Refinery $refinery,)
Input constructor.
Definition: Input.php:78
Describes a source for input names.
Definition: NameSource.php:26
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
withError(string $error)
Get an input like this one, with a different error.
Definition: Input.php:127
applyTo(Result $result)
Perform the transformation and reify possible failures.