ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.assNumeric.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
23 
40 {
41  protected $lower_limit;
42  protected $upper_limit;
43  public $maxchars = 6;
44 
45  public function isComplete(): bool
46  {
47  if (
48  strlen($this->title)
49  && $this->author
50  && $this->question
51  && $this->getMaximumPoints() > 0
52  ) {
53  return true;
54  }
55  return false;
56  }
57 
58  public function saveToDb(?int $original_id = null): void
59  {
63  parent::saveToDb($original_id);
64  }
65 
66  public function loadFromDb(int $question_id): void
67  {
68  $result = $this->db->queryF(
69  "SELECT qpl_questions.*, " . $this->getAdditionalTableName() . ".* FROM qpl_questions LEFT JOIN " . $this->getAdditionalTableName() . " ON " . $this->getAdditionalTableName() . ".question_fi = qpl_questions.question_id WHERE qpl_questions.question_id = %s",
70  ["integer"],
71  [$question_id]
72  );
73  if ($result->numRows() == 1) {
74  $data = $this->db->fetchAssoc($result);
75  $this->setId($question_id);
76  $this->setObjId($data["obj_fi"]);
77  $this->setTitle((string) $data["title"]);
78  $this->setComment((string) $data["description"]);
79  $this->setNrOfTries($data['nr_of_tries']);
80  $this->setOriginalId($data["original_id"]);
81  $this->setAuthor($data["author"]);
82  $this->setPoints($data["points"]);
83  $this->setOwner($data["owner"]);
84  $this->setQuestion(ilRTE::_replaceMediaObjectImageSrc((string) $data["question_text"], 1));
85  $this->setMaxChars($data["maxnumofchars"]);
86 
87  try {
88  $this->setLifecycle(ilAssQuestionLifecycle::getInstance($data['lifecycle']));
91  }
92 
93  try {
94  $this->setAdditionalContentEditingMode($data['add_cont_edit_mode']);
95  } catch (ilTestQuestionPoolException $e) {
96  }
97  }
98 
99  $result = $this->db->queryF(
100  "SELECT * FROM qpl_num_range WHERE question_fi = %s ORDER BY aorder ASC",
101  ['integer'],
102  [$question_id]
103  );
104 
105  if ($result->numRows() > 0) {
107  while ($data = $this->db->fetchAssoc($result)) {
108  $this->setPoints($data['points']);
109  $this->setLowerLimit($data['lowerlimit']);
110  $this->setUpperLimit($data['upperlimit']);
111  }
112  }
113 
114  parent::loadFromDb($question_id);
115  }
116 
117  public function getLowerLimit()
118  {
119  return $this->lower_limit;
120  }
121 
122  public function getUpperLimit()
123  {
124  return $this->upper_limit;
125  }
126 
127  public function setLowerLimit(string $limit): void
128  {
129  $this->lower_limit = str_replace(',', '.', $limit);
130  }
131 
132  public function setUpperLimit(string $limit): void
133  {
134  $this->upper_limit = str_replace(',', '.', $limit);
135  }
136 
137  public function getMaximumPoints(): float
138  {
139  return $this->getPoints();
140  }
141 
143  {
144  $points = 0;
145  if ($this->contains($previewSession->getParticipantsSolution())) {
146  $points = $this->getPoints();
147  }
148 
149  $reachedPoints = $this->deductHintPointsFromReachedPoints($previewSession, $points);
150 
151  return $this->ensureNonNegativePoints($reachedPoints);
152  }
153 
154  public function calculateReachedPoints(
155  int $active_id,
156  ?int $pass = null,
157  bool $authorized_solution = true
158  ): float {
159  if ($pass === null) {
160  $pass = $this->getSolutionMaxPass($active_id);
161  }
162  $result = $this->getCurrentSolutionResultSet($active_id, $pass, $authorized_solution);
163  $data = $this->db->fetchAssoc($result);
164  $enteredvalue = '';
165  if (is_array($data) && array_key_exists('value1', $data)) {
166  $enteredvalue = $data["value1"];
167  }
168 
169  $points = 0.0;
170  if ($this->contains($enteredvalue)) {
171  $points = $this->getPoints();
172  }
173 
174  return $points;
175  }
176 
187  public function contains($value): bool
188  {
189  $eval = new EvalMath();
190  $eval->suppress_errors = true;
191  $result = $eval->e((string) $value);
192  if (($result === false) || ($result === true)) {
193  return false;
194  }
195 
196  if (($result >= $eval->e($this->getLowerLimit())) && ($result <= $eval->e($this->getUpperLimit()))) {
197  return true;
198  }
199  return false;
200  }
201 
202  public function validateSolutionSubmit(): bool
203  {
204  if ($this->getSolutionSubmit() === null) {
205  $this->tpl->setOnScreenMessage('failure', $this->lng->txt("err_no_numeric_value"), true);
206  return false;
207  }
208 
209  return true;
210  }
211 
212  protected function getSolutionSubmit(): ?float
213  {
214  return $this->questionpool_request->float('numeric_result') ?? null;
215  }
216 
217  public function isValidSolutionSubmit($numeric_solution): bool
218  {
219  $math = new EvalMath();
220  $math->suppress_errors = true;
221  $result = $math->evaluate($numeric_solution);
222 
223  return !(
224  ($result === false || $result === true) && strlen($numeric_solution) > 0
225  );
226  }
227 
228  public function saveWorkingData(
229  int $active_id,
230  ?int $pass = null,
231  bool $authorized = true
232  ): bool {
233  if (is_null($pass)) {
234  $pass = ilObjTest::_getPass($active_id);
235  }
236 
237  $answer = $this->getSolutionSubmit();
238  $this->getProcessLocker()->executeUserSolutionUpdateLockOperation(
239  function () use ($answer, $active_id, $pass, $authorized) {
240  $result = $this->getCurrentSolutionResultSet($active_id, $pass, $authorized);
241  $update = -1;
242  if ($this->db->numRows($result) !== 0) {
243  $row = $this->db->fetchAssoc($result);
244  $update = $row['solution_id'];
245  }
246 
247  if ($update !== -1
248  && $answer === '') {
249  $this->removeSolutionRecordById($update);
250  return;
251  }
252  if ($update !== -1) {
253  $this->updateCurrentSolution($update, $answer, null, $authorized);
254  return;
255  }
256 
257  if ($answer !== '') {
258  $this->saveCurrentSolution($active_id, $pass, $answer, null, $authorized);
259  }
260  }
261  );
262 
263  return true;
264  }
265 
266  protected function savePreviewData(ilAssQuestionPreviewSession $previewSession): void
267  {
268  $numericSolution = $this->getSolutionSubmit();
269  $previewSession->setParticipantsSolution($numericSolution);
270  }
271 
273  {
274  // save additional data
275  $this->db->manipulateF(
276  "DELETE FROM " . $this->getAdditionalTableName() . " WHERE question_fi = %s",
277  [ "integer" ],
278  [ $this->getId() ]
279  );
280 
281  $this->db->manipulateF(
282  "INSERT INTO " . $this->getAdditionalTableName(
283  ) . " (question_fi, maxnumofchars) VALUES (%s, %s)",
284  [ "integer", "integer" ],
285  [
286  $this->getId(),
287  ($this->getMaxChars()) ? $this->getMaxChars() : 0
288  ]
289  );
290  }
291 
292  public function saveAnswerSpecificDataToDb()
293  {
294  // Write range to the database
295  $this->db->manipulateF(
296  "DELETE FROM qpl_num_range WHERE question_fi = %s",
297  [ 'integer' ],
298  [ $this->getId() ]
299  );
300 
301  $next_id = $this->db->nextId('qpl_num_range');
302  $this->db->manipulateF(
303  "INSERT INTO qpl_num_range (range_id, question_fi, lowerlimit, upperlimit, points, aorder, tstamp)
304  VALUES (%s, %s, %s, %s, %s, %s, %s)",
305  [ 'integer', 'integer', 'text', 'text', 'float', 'integer', 'integer' ],
306  [ $next_id, $this->id, $this->getLowerLimit(), $this->getUpperLimit(
307  ), $this->getPoints(), 0, time() ]
308  );
309  }
310 
316  public function getQuestionType(): string
317  {
318  return "assNumeric";
319  }
320 
326  public function getMaxChars()
327  {
328  return $this->maxchars;
329  }
330 
336  public function setMaxChars($maxchars): void
337  {
338  $this->maxchars = $maxchars;
339  }
340 
346  public function getAdditionalTableName(): string
347  {
348  return "qpl_qst_numeric";
349  }
350 
355  public function getRTETextWithMediaObjects(): string
356  {
357  return parent::getRTETextWithMediaObjects();
358  }
359 
360  public function getOperators(string $expression): array
361  {
362  return ilOperatorsExpressionMapping::getOperatorsByExpression($expression);
363  }
364 
365  public function getExpressionTypes(): array
366  {
367  return [
371  ];
372  }
373 
374  public function getUserQuestionResult(
375  int $active_id,
376  int $pass
378  $result = new ilUserQuestionResult($this, $active_id, $pass);
379 
380  $maxStep = $this->lookupMaxStep($active_id, $pass);
381  if ($maxStep > 0) {
382  $data = $this->db->queryF(
383  "SELECT value1 FROM tst_solutions WHERE active_fi = %s AND pass = %s AND question_fi = %s AND step = %s",
384  ["integer", "integer", "integer","integer"],
385  [$active_id, $pass, $this->getId(), $maxStep]
386  );
387  } else {
388  $data = $this->db->queryF(
389  "SELECT value1 FROM tst_solutions WHERE active_fi = %s AND pass = %s AND question_fi = %s",
390  ["integer", "integer", "integer"],
391  [$active_id, $pass, $this->getId()]
392  );
393  }
394 
395  while ($row = $this->db->fetchAssoc($data)) {
396  $result->addKeyValue(1, $row["value1"]);
397  }
398 
399  $points = $this->calculateReachedPoints($active_id, $pass);
400  $max_points = $this->getMaximumPoints();
401 
402  $result->setReachedPercentage(($points / $max_points) * 100);
403 
404  return $result;
405  }
406 
407  public function getAvailableAnswerOptions(?int $index = null): array
408  {
409  return [
410  "lower" => $this->getLowerLimit(),
411  "upper" => $this->getUpperLimit()
412  ];
413  }
414 
415  public function getAnswerTableName(): string
416  {
417  return '';
418  }
419 
420  public function toLog(AdditionalInformationGenerator $additional_info): array
421  {
422  return [
423  AdditionalInformationGenerator::KEY_QUESTION_TYPE => (string) $this->getQuestionType(),
424  AdditionalInformationGenerator::KEY_QUESTION_TITLE => $this->getTitleForHTMLOutput(),
425  AdditionalInformationGenerator::KEY_QUESTION_TEXT => $this->formatSAQuestion($this->getQuestion()),
426  AdditionalInformationGenerator::KEY_QUESTION_SHUFFLE_ANSWER_OPTIONS => $additional_info
428  AdditionalInformationGenerator::KEY_QUESTION_MAXCHARS => $this->getMaxChars(),
429  AdditionalInformationGenerator::KEY_QUESTION_REACHABLE_POINTS => $this->getPoints(),
430  AdditionalInformationGenerator::KEY_QUESTION_LOWER_LIMIT => $this->getLowerLimit(),
431  AdditionalInformationGenerator::KEY_QUESTION_UPPER_LIMIT => $this->getUpperLimit(),
432  AdditionalInformationGenerator::KEY_FEEDBACK => [
433  AdditionalInformationGenerator::KEY_QUESTION_FEEDBACK_ON_INCOMPLETE => $this->formatSAQuestion($this->feedbackOBJ->getGenericFeedbackTestPresentation($this->getId(), false)),
434  AdditionalInformationGenerator::KEY_QUESTION_FEEDBACK_ON_COMPLETE => $this->formatSAQuestion($this->feedbackOBJ->getGenericFeedbackTestPresentation($this->getId(), true))
435  ]
436  ];
437  }
438 
439  protected function solutionValuesToLog(
440  AdditionalInformationGenerator $additional_info,
441  array $solution_values
442  ): string {
443  if (!array_key_exists(0, $solution_values) ||
444  !array_key_exists('value1', $solution_values[0])) {
445  return '';
446  }
447  return $solution_values[0]['value1'];
448  }
449 
450  public function solutionValuesToText(array $solution_values): string
451  {
452  if (!array_key_exists(0, $solution_values) ||
453  !array_key_exists('value1', $solution_values[0])) {
454  return '';
455  }
456  return $solution_values[0]['value1'];
457  }
458 
459  public function getCorrectSolutionForTextOutput(int $active_id, int $pass): string
460  {
461  return "{$this->getLowerLimit()}-{$this->getUpperLimit()}";
462  }
463 }
static _replaceMediaObjectImageSrc(string $a_text, int $a_direction=0, string $nic='')
Replaces image source from mob image urls with the mob id or replaces mob id with the correct image s...
setNrOfTries(int $a_nr_of_tries)
solutionValuesToText(array $solution_values)
getRTETextWithMediaObjects()
Collects all text in the question which could contain media objects which were created with the Rich ...
solutionValuesToLog(AdditionalInformationGenerator $additional_info, array $solution_values)
loadFromDb(int $question_id)
getUserQuestionResult(int $active_id, int $pass)
Get the user solution for a question by active_id and the test pass.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _getPass($active_id)
Retrieves the actual pass of a given user for a given test.
getExpressionTypes()
Get all available expression types for a specific question.
getMaxChars()
Returns the maximum number of characters for the numeric input field.
setOwner(int $owner=-1)
ensureNonNegativePoints(float $points)
setLowerLimit(string $limit)
saveWorkingData(int $active_id, ?int $pass=null, bool $authorized=true)
calculateReachedPointsFromPreviewSession(ilAssQuestionPreviewSession $previewSession)
getQuestionType()
Returns the question type of the question.
setComment(string $comment="")
setMaxChars($maxchars)
Sets the maximum number of characters for the numeric input field.
updateCurrentSolution(int $solutionId, $value1, $value2, bool $authorized=true)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
saveCurrentSolution(int $active_id, int $pass, $value1, $value2, bool $authorized=true, $tstamp=0)
getAvailableAnswerOptions(?int $index=null)
If index is null, the function returns an array with all anwser options else it returns the specific ...
getCorrectSolutionForTextOutput(int $active_id, int $pass)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
removeSolutionRecordById(int $solutionId)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
deductHintPointsFromReachedPoints(ilAssQuestionPreviewSession $preview_session, $reached_points)
saveAnswerSpecificDataToDb()
Saves the answer specific records into a question types answer table.
getAdditionalTableName()
Returns the name of the additional question data table in the database.
setPoints(float $points)
setObjId(int $obj_id=0)
saveQuestionDataToDb(?int $original_id=null)
saveToDb(?int $original_id=null)
isValidSolutionSubmit($numeric_solution)
calculateReachedPoints(int $active_id, ?int $pass=null, bool $authorized_solution=true)
getSolutionMaxPass(int $active_id)
toLog(AdditionalInformationGenerator $additional_info)
Class for numeric questions.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setId(int $id=-1)
setUpperLimit(string $limit)
setOriginalId(?int $original_id)
setTitle(string $title="")
setLifecycle(ilAssQuestionLifecycle $lifecycle)
getOperators(string $expression)
Get all available operations for a specific question.
getCurrentSolutionResultSet(int $active_id, int $pass, bool $authorized=true)
lookupMaxStep(int $active_id, int $pass)
setAuthor(string $author="")
savePreviewData(ilAssQuestionPreviewSession $previewSession)
setAdditionalContentEditingMode(?string $additionalContentEditingMode)
contains($value)
Checks for a given value within the range.
saveAdditionalQuestionDataToDb()
Saves a record to the question types additional data table.
setQuestion(string $question="")