ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.assErrorText.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
24 
38 {
39  protected const ERROR_TYPE_WORD = 1;
40  protected const ERROR_TYPE_PASSAGE = 2;
41  protected const DEFAULT_TEXT_SIZE = 100.0;
42  protected const ERROR_MAX_LENGTH = 150;
43 
44  protected const PARAGRAPH_SPLIT_REGEXP = '/[\n\r]+/';
45  protected const WORD_SPLIT_REGEXP = '/\s+/';
46  protected const FIND_PUNCTUATION_REGEXP = '/\p{P}/';
47  protected const ERROR_WORD_MARKER = '#';
48  protected const ERROR_PARAGRAPH_DELIMITERS = [
49  'start' => '((',
50  'end' => '))'
51  ];
52 
53  protected string $errortext = '';
54  protected array $parsed_errortext = [];
56  protected array $errordata = [];
57  protected float $textsize;
58  protected ?float $points_wrong = null;
59 
60  public function __construct(
61  string $title = '',
62  string $comment = '',
63  string $author = '',
64  int $owner = -1,
65  string $question = ''
66  ) {
68  $this->textsize = self::DEFAULT_TEXT_SIZE;
69  }
70 
71  public function isComplete(): bool
72  {
73  if (mb_strlen($this->title)
74  && ($this->author)
75  && ($this->question)
76  && ($this->getMaximumPoints() > 0)) {
77  return true;
78  } else {
79  return false;
80  }
81  }
82 
83  public function saveToDb(?int $original_id = null): void
84  {
88  parent::saveToDb();
89  }
90 
91  public function saveAnswerSpecificDataToDb()
92  {
93  $this->db->manipulateF(
94  "DELETE FROM qpl_a_errortext WHERE question_fi = %s",
95  ['integer'],
96  [$this->getId()]
97  );
98 
99  $sequence = 0;
100  foreach ($this->errordata as $error) {
101  $next_id = $this->db->nextId('qpl_a_errortext');
102  $this->db->manipulateF(
103  "INSERT INTO qpl_a_errortext (answer_id, question_fi, text_wrong, text_correct, points, sequence, position) VALUES (%s, %s, %s, %s, %s, %s, %s)",
104  ['integer', 'integer', 'text', 'text', 'float', 'integer', 'integer'],
105  [
106  $next_id,
107  $this->getId(),
108  $error->getTextWrong(),
109  $error->getTextCorrect(),
110  $error->getPoints(),
111  $sequence++,
112  $error->getPosition()
113  ]
114  );
115  }
116  }
117 
124  {
125  $this->db->manipulateF(
126  "DELETE FROM " . $this->getAdditionalTableName() . " WHERE question_fi = %s",
127  ["integer"],
128  [$this->getId()]
129  );
130 
131  $this->db->manipulateF(
132  "INSERT INTO " . $this->getAdditionalTableName() . " (question_fi, errortext, parsed_errortext, textsize, points_wrong) VALUES (%s, %s, %s, %s, %s)",
133  ["integer", "text", "text", "float", "float"],
134  [
135  $this->getId(),
136  $this->getErrorText(),
137  json_encode($this->getParsedErrorText()),
138  $this->getTextSize(),
139  $this->getPointsWrong()
140  ]
141  );
142  }
143 
150  public function loadFromDb($question_id): void
151  {
152  $db_question = $this->db->queryF(
153  "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",
154  ["integer"],
155  [$question_id]
156  );
157  if ($db_question->numRows() === 1) {
158  $data = $this->db->fetchAssoc($db_question);
159  $this->setId($question_id);
160  $this->setObjId($data["obj_fi"]);
161  $this->setTitle((string) $data["title"]);
162  $this->setComment((string) $data["description"]);
163  $this->setOriginalId($data["original_id"]);
164  $this->setNrOfTries($data['nr_of_tries']);
165  $this->setAuthor($data["author"]);
166  $this->setPoints($data["points"]);
167  $this->setOwner($data["owner"]);
168  $this->setQuestion(ilRTE::_replaceMediaObjectImageSrc((string) $data["question_text"], 1));
169  $this->setErrorText((string) $data["errortext"]);
170  $this->setParsedErrorText(json_decode($data['parsed_errortext'] ?? json_encode([]), true));
171  $this->setTextSize($data["textsize"]);
172  $this->setPointsWrong($data["points_wrong"]);
173 
174  try {
175  $this->setLifecycle(ilAssQuestionLifecycle::getInstance($data['lifecycle']));
178  }
179 
180  try {
181  $this->setAdditionalContentEditingMode($data['add_cont_edit_mode']);
182  } catch (ilTestQuestionPoolException $e) {
183  }
184  }
185 
186  $db_error_text = $this->db->queryF(
187  "SELECT * FROM qpl_a_errortext WHERE question_fi = %s ORDER BY sequence ASC",
188  ['integer'],
189  [$question_id]
190  );
191 
192  if ($db_error_text->numRows() > 0) {
193  while ($data = $this->db->fetchAssoc($db_error_text)) {
194  $this->errordata[] = new assAnswerErrorText(
195  (string) $data['text_wrong'],
196  (string) $data['text_correct'],
197  (float) $data['points'],
198  $data['position']
199  );
200  }
201  }
202 
204 
205  parent::loadFromDb($question_id);
206  }
207 
208  private function correctDataAfterParserUpdate(): void
209  {
210  if ($this->getErrorText() === '') {
211  return;
212  }
213  $needs_finalizing = false;
214  if ($this->getParsedErrorText() === []) {
215  $needs_finalizing = true;
216  $this->parseErrorText();
217  }
218 
219  if (isset($this->errordata[0])
220  && $this->errordata[0]->getPosition() === null) {
221  foreach ($this->errordata as $key => $error) {
222  $this->errordata[$key] = $this->addPositionToErrorAnswer($error);
223  }
225  }
226 
227  if ($needs_finalizing) {
230  }
231  }
232 
238  public function getMaximumPoints(): float
239  {
240  $maxpoints = 0.0;
241  foreach ($this->errordata as $error) {
242  if ($error->getPoints() > 0) {
243  $maxpoints += $error->getPoints();
244  }
245  }
246  return $maxpoints;
247  }
248 
249  public function calculateReachedPoints(
250  int $active_id,
251  ?int $pass = null,
252  bool $authorized_solution = true
253  ): float {
254  if ($pass === null) {
255  $pass = $this->getSolutionMaxPass($active_id);
256  }
257  $result = $this->getCurrentSolutionResultSet($active_id, $pass, $authorized_solution);
258 
259  $positions = [];
260  while ($row = $this->db->fetchAssoc($result)) {
261  $positions[] = $row['value1'];
262  }
263  $points = $this->getPointsForSelectedPositions($positions);
264  return $points;
265  }
266 
268  {
269  $reached_points = $this->getPointsForSelectedPositions($preview_session->getParticipantsSolution() ?? []);
270  $reached_points = $this->deductHintPointsFromReachedPoints($preview_session, $reached_points);
271  return $this->ensureNonNegativePoints($reached_points);
272  }
273 
274  public function saveWorkingData(
275  int $active_id,
276  ?int $pass = null,
277  bool $authorized = true
278  ): bool {
279  if (is_null($pass)) {
280  $pass = ilObjTest::_getPass($active_id);
281  }
282 
283  $this->getProcessLocker()->executeUserSolutionUpdateLockOperation(
284  function () use ($active_id, $pass, $authorized) {
285  $selected = $this->getAnswersFromRequest();
286  $this->removeCurrentSolution($active_id, $pass, $authorized);
287  foreach ($selected as $position) {
288  $this->saveCurrentSolution($active_id, $pass, $position, null, $authorized);
289  }
290  }
291  );
292 
293  return true;
294  }
295 
296  public function savePreviewData(ilAssQuestionPreviewSession $previewSession): void
297  {
298  $selection = $this->getAnswersFromRequest();
299  $previewSession->setParticipantsSolution($selection);
300  }
301 
302  private function getAnswersFromRequest(): array
303  {
304  return explode(
305  ',',
306  $this->questionpool_request->string('qst_' . $this->getId())
307  );
308  }
309 
310  public function getQuestionType(): string
311  {
312  return 'assErrorText';
313  }
314 
315  public function getAdditionalTableName(): string
316  {
317  return 'qpl_qst_errortext';
318  }
319 
320  public function getAnswerTableName(): string
321  {
322  return 'qpl_a_errortext';
323  }
324 
325  public function setErrorsFromParsedErrorText(): void
326  {
327  $current_error_data = $this->getErrorData();
328  $this->errordata = [];
329 
330  $has_too_long_errors = false;
331  foreach ($this->getParsedErrorText() as $paragraph) {
332  foreach ($paragraph as $position => $word) {
333  if ($word['error_type'] === 'in_passage'
334  || $word['error_type'] === 'passage_end'
335  || $word['error_type'] === 'none') {
336  continue;
337  }
338 
339  $text_wrong = $word['text_wrong'];
340  if (mb_strlen($text_wrong) > self::ERROR_MAX_LENGTH) {
341  $has_too_long_errors = true;
342  continue;
343  }
344 
345  list($text_correct, $points) =
346  $this->getAdditionalInformationFromExistingErrorDataByErrorText($current_error_data, $text_wrong);
347  $this->errordata[] = new assAnswerErrorText($text_wrong, $text_correct, $points, $position);
348  }
349  }
350 
351  if ($has_too_long_errors) {
352  $this->tpl->setOnScreenMessage(
353  'failure',
354  $this->lng->txt('qst_error_text_too_long')
355  );
356  }
357  }
358 
360  {
361  foreach ($this->getParsedErrorText() as $paragraph) {
362  foreach ($paragraph as $position => $word) {
363  if (isset($word['text_wrong'])
364  && ($word['text_wrong'] === $error->getTextWrong()
365  || mb_substr($word['text_wrong'], 0, -1) === $error->getTextWrong()
366  && preg_match(self::FIND_PUNCTUATION_REGEXP, mb_substr($word['text_wrong'], -1)) === 1)
367  && !array_key_exists($position, $this->generateArrayByPositionFromErrorData())
368  ) {
369  return $error->withPosition($position);
370  }
371  }
372  }
373 
374  return $error;
375  }
376 
377  private function completeParsedErrorTextFromErrorData(): void
378  {
379  foreach ($this->errordata as $error) {
380  $position = $error->getPosition();
381  foreach ($this->getParsedErrorText() as $key => $paragraph) {
382  if (array_key_exists($position, $paragraph)) {
383  $this->parsed_errortext[$key][$position]['text_correct'] =
384  $error->getTextCorrect();
385  $this->parsed_errortext[$key][$position]['points'] =
386  $error->getPoints();
387  break;
388  }
389  }
390  }
391  }
392 
397  public function setErrorData(array $errors): void
398  {
399  $this->errordata = [];
400 
401  foreach ($errors as $error) {
402  $answer = $this->addPositionToErrorAnswer($error);
403  $this->errordata[] = $answer;
404  }
406  }
407 
408  public function removeErrorDataWithoutPosition(): void
409  {
410  foreach ($this->getErrorData() as $index => $error) {
411  if ($error->getPosition() === null) {
412  unset($this->errordata[$index]);
413  }
414  }
415  $this->errordata = array_values($this->errordata);
416  }
417 
424  array $current_error_data,
425  string $text_wrong
426  ): array {
427  foreach ($current_error_data as $answer_object) {
428  if (strcmp($answer_object->getTextWrong(), $text_wrong) === 0) {
429  return[
430  $answer_object->getTextCorrect(),
431  $answer_object->getPoints()
432  ];
433  }
434  }
435  return ['', 0.0];
436  }
437 
438  public function assembleErrorTextOutput(
439  array $selections,
440  bool $graphical_output = false,
441  bool $show_correct_solution = false,
442  bool $use_link_tags = true,
443  array $correctness_icons = []
444  ): string {
445  $output_array = [];
446  foreach ($this->getParsedErrorText() as $paragraph) {
447  $array_reduce_function = fn(?string $carry, int $position)
448  => $carry . $this->generateOutputStringFromPosition(
449  $position,
450  $selections,
451  $paragraph,
452  $graphical_output,
453  $show_correct_solution,
454  $use_link_tags,
455  $correctness_icons
456  );
457  $output_array[] = '<p>' . trim(array_reduce(array_keys($paragraph), $array_reduce_function)) . '</p>';
458  }
459 
460  return implode("\n", $output_array);
461  }
462 
464  int $position,
465  array $selections,
466  array $paragraph,
467  bool $graphical_output,
468  bool $show_correct_solution,
469  bool $use_link_tags,
470  array $correctness_icons
471  ): string {
472  $text = $this->getTextForPosition($position, $paragraph, $show_correct_solution);
473  if ($text === '') {
474  return '';
475  }
476  $class = $this->getClassForPosition($position, $show_correct_solution, $selections);
477  $img = $this->getCorrectnessIconForPosition(
478  $position,
479  $graphical_output,
480  $selections,
481  $correctness_icons
482  );
483 
484  return ' ' . $this->getErrorTokenHtml($text, $class, $use_link_tags) . $img;
485  }
486 
487  private function getTextForPosition(
488  int $position,
489  array $paragraph,
490  bool $show_correct_solution
491  ): string {
492  $v = $paragraph[$position];
493  if ($show_correct_solution === true
494  && ($v['error_type'] === 'in_passage'
495  || $v['error_type'] === 'passage_end')) {
496  return '';
497  }
498  if ($show_correct_solution
499  && ($v['error_type'] === 'passage_start'
500  || $v['error_type'] === 'word')) {
501  return $v['text_correct'] ?? '';
502  }
503 
504  return $v['text'];
505  }
506 
507  private function getClassForPosition(
508  int $position,
509  bool $show_correct_solution,
510  array $selections
511  ): string {
512  if ($show_correct_solution !== true
513  && in_array($position, $selections['user'])) {
514  return 'ilc_qetitem_ErrorTextSelected';
515  }
516 
517  if ($show_correct_solution === true
518  && in_array($position, $selections['best'])) {
519  return 'ilc_qetitem_ErrorTextSelected';
520  }
521 
522  return 'ilc_qetitem_ErrorTextItem';
523  }
524 
526  int $position,
527  bool $graphical_output,
528  array $selections,
529  array $correctness_icons
530  ): string {
531  if ($graphical_output === true
532  && (in_array($position, $selections['user']) && !in_array($position, $selections['best'])
533  || !in_array($position, $selections['user']) && in_array($position, $selections['best']))) {
534  return $correctness_icons['not_correct'];
535  }
536 
537  if ($graphical_output === true
538  && in_array($position, $selections['user']) && in_array($position, $selections['best'])) {
539  return $correctness_icons['correct'];
540  }
541 
542  return '';
543  }
544 
545  public function createErrorTextExport(array $selections): string
546  {
547  if (!is_array($selections)) {
548  $selections = [];
549  }
550 
551  foreach ($this->getParsedErrorText() as $paragraph) {
552  $array_reduce_function = function ($carry, $k) use ($paragraph, $selections) {
553  $text = $paragraph[$k]['text'];
554  if (in_array($k, $selections)) {
555  $text = self::ERROR_WORD_MARKER . $paragraph[$k]['text'] . self::ERROR_WORD_MARKER;
556  }
557  return $carry . ' ' . $text;
558  };
559  $output_array[] = trim(array_reduce(array_keys($paragraph), $array_reduce_function));
560  }
561  return implode("\n", $output_array);
562  }
563 
564  public function getBestSelection(bool $with_positive_points_only = true): array
565  {
566  $positions_array = $this->generateArrayByPositionFromErrorData();
567  $selections = [];
568  foreach ($positions_array as $position => $position_data) {
569  if ($position === ''
570  || $with_positive_points_only && $position_data['points'] <= 0) {
571  continue;
572  }
573 
574  $selections[] = $position;
575  if ($position_data['length'] > 1) {
576  for ($i = 1;$i < $position_data['length'];$i++) {
577  $selections[] = $position + $i;
578  }
579  }
580  }
581 
582  return $selections;
583  }
584 
589  protected function getPointsForSelectedPositions(array $selected_word_positions): float
590  {
591  $points = 0;
592  $correct_positions = $this->generateArrayByPositionFromErrorData();
593 
594  foreach ($correct_positions as $correct_position => $correct_position_data) {
595  $selected_word_key = array_search($correct_position, $selected_word_positions);
596  if ($selected_word_key === false) {
597  continue;
598  }
599 
600  if ($correct_position_data['length'] === 1) {
601  $points += $correct_position_data['points'];
602  unset($selected_word_positions[$selected_word_key]);
603  continue;
604  }
605 
606  $passage_complete = true;
607  for ($i = 1;$i < $correct_position_data['length'];$i++) {
608  $selected_passage_element_key = array_search($correct_position + $i, $selected_word_positions);
609  if ($selected_passage_element_key === false) {
610  $passage_complete = false;
611  continue;
612  }
613  unset($selected_word_positions[$selected_passage_element_key]);
614  }
615 
616  if ($passage_complete) {
617  $points += $correct_position_data['points'];
618  unset($selected_word_positions[$selected_word_key]);
619  }
620  }
621 
622  foreach ($selected_word_positions as $word_position) {
623  if (!array_key_exists($word_position, $correct_positions)) {
624  $points += $this->getPointsWrong();
625  continue;
626  }
627  }
628 
629  return $points;
630  }
631 
632  public function flushErrorData(): void
633  {
634  $this->errordata = [];
635  }
636 
641  public function getErrorData(): array
642  {
643  return $this->errordata;
644  }
645 
650  private function getErrorDataAsArrayForJS(): array
651  {
652  $correct_answers = [];
653  foreach ($this->getErrorData() as $index => $answer_obj) {
654  $correct_answers[] = [
655  'answertext_wrong' => $answer_obj->getTextWrong(),
656  'answertext_correct' => $answer_obj->getTextCorrect(),
657  'points' => $answer_obj->getPoints(),
658  'length' => $answer_obj->getLength(),
659  'pos' => $this->getId() . '_' . $answer_obj->getPosition()
660  ];
661  }
662  return $correct_answers;
663  }
664 
665  public function getErrorText(): string
666  {
667  return $this->errortext ?? '';
668  }
669 
670  public function setErrorText(?string $text): void
671  {
672  $this->errortext = $text ?? '';
673  }
674 
675  public function getParsedErrorText(): array
676  {
678  }
679 
680  private function getParsedErrorTextForJS(): array
681  {
682  $answers = [];
683  foreach ($this->parsed_errortext as $paragraph) {
684  foreach ($paragraph as $position => $word) {
685  $answers[] = [
686  'answertext' => $word['text'],
687  'order' => $this->getId() . '_' . $position
688  ];
689  }
690  $answers[] = [
691  'answertext' => '###'
692  ];
693  }
694  array_pop($answers);
695 
696  return $answers;
697  }
698 
699  public function setParsedErrorText(array $parsed_errortext): void
700  {
701  $this->parsed_errortext = $parsed_errortext;
702  }
703 
704  public function getTextSize(): float
705  {
706  return $this->textsize;
707  }
708 
709  public function setTextSize($a_value): void
710  {
711  // in self-assesment-mode value should always be set (and must not be null)
712  if ($a_value === null) {
713  $a_value = 100;
714  }
715  $this->textsize = $a_value;
716  }
717 
718  public function getPointsWrong(): ?float
719  {
720  return $this->points_wrong;
721  }
722 
723  public function setPointsWrong($a_value): void
724  {
725  $this->points_wrong = $a_value;
726  }
727 
728  public function toJSON(): string
729  {
730  $result = [];
731  $result['id'] = $this->getId();
732  $result['type'] = (string) $this->getQuestionType();
733  $result['title'] = $this->getTitleForHTMLOutput();
734  $result['question'] = $this->formatSAQuestion($this->getQuestion());
735  $result['text'] = ilRTE::_replaceMediaObjectImageSrc($this->getErrorText(), 0);
736  $result['nr_of_tries'] = $this->getNrOfTries();
737  $result['shuffle'] = $this->getShuffle();
738  $result['feedback'] = [
739  'onenotcorrect' => $this->formatSAQuestion($this->feedbackOBJ->getGenericFeedbackTestPresentation($this->getId(), false)),
740  'allcorrect' => $this->formatSAQuestion($this->feedbackOBJ->getGenericFeedbackTestPresentation($this->getId(), true))
741  ];
742 
743  $result['correct_answers'] = $this->getErrorDataAsArrayForJS();
744  $result['answers'] = $this->getParsedErrorTextForJS();
745 
746  $mobs = ilObjMediaObject::_getMobsOfObject("qpl:html", $this->getId());
747  $result['mobs'] = $mobs;
748 
749  return json_encode($result);
750  }
751 
752  public function getOperators(string $expression): array
753  {
754  return ilOperatorsExpressionMapping::getOperatorsByExpression($expression);
755  }
756 
757  public function getExpressionTypes(): array
758  {
759  return [
764  ];
765  }
766 
767  public function getUserQuestionResult(
768  int $active_id,
769  int $pass
771  $result = new ilUserQuestionResult($this, $active_id, $pass);
772 
773  $data = $this->db->queryF(
774  "SELECT value1+1 as value1 FROM tst_solutions WHERE active_fi = %s AND pass = %s AND question_fi = %s AND step = (
775  SELECT MAX(step) FROM tst_solutions WHERE active_fi = %s AND pass = %s AND question_fi = %s
776  )",
777  ["integer", "integer", "integer","integer", "integer", "integer"],
778  [$active_id, $pass, $this->getId(), $active_id, $pass, $this->getId()]
779  );
780 
781  while ($row = $this->db->fetchAssoc($data)) {
782  $result->addKeyValue($row["value1"], $row["value1"]);
783  }
784 
785  $points = $this->calculateReachedPoints($active_id, $pass);
786  $max_points = $this->getMaximumPoints();
787 
788  $result->setReachedPercentage(($points / $max_points) * 100);
789 
790  return $result;
791  }
792 
793  public function parseErrorText(): void
794  {
795  $text_by_paragraphs = preg_split(self::PARAGRAPH_SPLIT_REGEXP, $this->getErrorText());
796  $text_array = [];
797  $offset = 0;
798  foreach ($text_by_paragraphs as $paragraph) {
799  $text_array[] = $this->addErrorInformationToTextParagraphArray(
800  preg_split(self::WORD_SPLIT_REGEXP, trim($paragraph)),
801  $offset
802  );
803  $offset += count(end($text_array));
804  }
805  $this->setParsedErrorText($text_array);
806  }
807 
813  private function addErrorInformationToTextParagraphArray(array $paragraph, int $offset): array
814  {
815  $paragraph_with_error_info = [];
816  $passage_start = null;
817  foreach ($paragraph as $position => $word) {
818  $actual_position = $position + $offset;
819  if ($passage_start !== null
820  && (mb_strrpos($word, self::ERROR_PARAGRAPH_DELIMITERS['end']) === mb_strlen($word) - 2
821  || mb_strrpos($word, self::ERROR_PARAGRAPH_DELIMITERS['end']) === mb_strlen($word) - 3
822  && preg_match(self::FIND_PUNCTUATION_REGEXP, mb_substr($word, -1)) === 1)) {
823  $actual_word = $this->parsePassageEndWord($word);
824 
825  $paragraph_with_error_info[$passage_start]['text_wrong'] .=
826  ' ' . $actual_word;
827  $paragraph_with_error_info[$actual_position] = [
828  'text' => $actual_word,
829  'error_type' => 'passage_end'
830  ];
831  $passage_start = null;
832  continue;
833  }
834  if ($passage_start !== null) {
835  $paragraph_with_error_info[$passage_start]['text_wrong'] .= ' ' . $word;
836  $paragraph_with_error_info[$actual_position] = [
837  'text' => $word,
838  'error_type' => 'in_passage'
839  ];
840  continue;
841  }
842  if (mb_strpos($word, self::ERROR_PARAGRAPH_DELIMITERS['start']) === 0) {
843  $paragraph_with_error_info[$actual_position] = [
844  'text' => substr($word, 2),
845  'text_wrong' => substr($word, 2),
846  'error_type' => 'passage_start',
847  'error_position' => $actual_position,
848  ];
849  $passage_start = $actual_position;
850  continue;
851  }
852  if (mb_strpos($word, self::ERROR_WORD_MARKER) === 0) {
853  $paragraph_with_error_info[$actual_position] = [
854  'text' => substr($word, 1),
855  'text_wrong' => substr($word, 1),
856  'error_type' => 'word',
857  'error_position' => $actual_position,
858  ];
859  continue;
860  }
861 
862  $paragraph_with_error_info[$actual_position] = [
863  'text' => $word,
864  'error_type' => 'none',
865  'points' => $this->getPointsWrong()
866  ];
867  }
868 
869  return $paragraph_with_error_info;
870  }
871 
872  private function parsePassageEndWord(string $word): string
873  {
874  if (mb_substr($word, -2) === self::ERROR_PARAGRAPH_DELIMITERS['end']) {
875  return mb_substr($word, 0, -2);
876  }
877  return mb_substr($word, 0, -3) . mb_substr($word, -1);
878  }
879 
887  public function getAvailableAnswerOptions($index = null): ?int
888  {
889  $error_text_array = array_reduce(
890  $this->parsed_errortext,
891  fn($c, $v) => $c + $v
892  );
893 
894  if ($index === null) {
895  return $error_text_array;
896  }
897 
898  if (array_key_exists($index, $error_text_array)) {
899  return $error_text_array[$index];
900  }
901 
902  return null;
903  }
904 
905  private function generateArrayByPositionFromErrorData(): array
906  {
907  $array_by_position = [];
908  foreach ($this->errordata as $error) {
909  $array_by_position[$error->getPosition()] = [
910  'length' => $error->getLength(),
911  'points' => $error->getPoints(),
912  'text' => $error->getTextWrong(),
913  'text_correct' => $error->getTextCorrect()
914  ];
915  }
916  ksort($array_by_position);
917  return $array_by_position;
918  }
919 
925  private function getErrorTokenHtml($item, $class, $useLinkTags): string
926  {
927  if ($useLinkTags) {
928  return '<a class="' . $class . '" href="#">' . ($item == '&nbsp;' ? $item : ilLegacyFormElementsUtil::prepareFormOutput(
929  $item
930  )) . '</a>';
931  }
932 
933  return '<span class="' . $class . '">' . ($item == '&nbsp;' ? $item : ilLegacyFormElementsUtil::prepareFormOutput(
934  $item
935  )) . '</span>';
936  }
937 
938  public function toLog(AdditionalInformationGenerator $additional_info): array
939  {
940  $result = [
941  AdditionalInformationGenerator::KEY_QUESTION_TYPE => (string) $this->getQuestionType(),
942  AdditionalInformationGenerator::KEY_QUESTION_TITLE => $this->getTitleForHTMLOutput(),
943  AdditionalInformationGenerator::KEY_QUESTION_TEXT => $this->formatSAQuestion($this->getQuestion()),
944  AdditionalInformationGenerator::KEY_QUESTION_ERRORTEXT_ERRORTEXT => ilRTE::_replaceMediaObjectImageSrc($this->getErrorText(), 0),
945  AdditionalInformationGenerator::KEY_QUESTION_SHUFFLE_ANSWER_OPTIONS => $additional_info
947  AdditionalInformationGenerator::KEY_FEEDBACK => [
948  AdditionalInformationGenerator::KEY_QUESTION_FEEDBACK_ON_INCOMPLETE => $this->formatSAQuestion($this->feedbackOBJ->getGenericFeedbackTestPresentation($this->getId(), false)),
949  AdditionalInformationGenerator::KEY_QUESTION_FEEDBACK_ON_COMPLETE => $this->formatSAQuestion($this->feedbackOBJ->getGenericFeedbackTestPresentation($this->getId(), true))
950  ]
951  ];
952 
953  $error_data = $this->getErrorData();
954  $result[AdditionalInformationGenerator::KEY_QUESTION_CORRECT_ANSWER_OPTIONS] = array_reduce(
955  array_keys($error_data),
956  static function (array $c, int $k) use ($error_data): array {
957  $c[$k + 1] = [
958  'text_wrong' => $error_data[$k]->getTextWrong(),
959  'text_correct' => $error_data[$k]->getTextCorrect(),
960  'points' => $error_data[$k]->getPoints()
961  ];
962  return $c;
963  },
964  []
965  );
966 
967  return $result;
968  }
969 
970  protected function solutionValuesToLog(
971  AdditionalInformationGenerator $additional_info,
972  array $solution_values
973  ): string {
974  return $this->createErrorTextExport(
975  array_map(
976  static fn(array $v): string => $v['value1'],
977  $solution_values
978  )
979  );
980  }
981 
982  public function solutionValuesToText(array $solution_values): string
983  {
984  return $this->createErrorTextExport(
985  array_map(
986  static fn(array $v): string => $v['value1'],
987  $solution_values
988  )
989  );
990  }
991 
992  public function getCorrectSolutionForTextOutput(int $active_id, int $pass): string
993  {
994  return $this->createErrorTextExport($this->getBestSelection());
995  }
996 }
getPointsForSelectedPositions(array $selected_word_positions)
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...
const FIND_PUNCTUATION_REGEXP
setNrOfTries(int $a_nr_of_tries)
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.
saveWorkingData(int $active_id, ?int $pass=null, bool $authorized=true)
loadFromDb($question_id)
Loads the object from the database.
setOwner(int $owner=-1)
addErrorInformationToTextParagraphArray(array $paragraph, int $offset)
calculateReachedPoints(int $active_id, ?int $pass=null, bool $authorized_solution=true)
ensureNonNegativePoints(float $points)
getCorrectSolutionForTextOutput(int $active_id, int $pass)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
getClassForPosition(int $position, bool $show_correct_solution, array $selections)
getTextForPosition(int $position, array $paragraph, bool $show_correct_solution)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
toLog(AdditionalInformationGenerator $additional_info)
static prepareFormOutput($a_str, bool $a_strip=false)
$c
Definition: deliver.php:25
setComment(string $comment="")
const ERROR_PARAGRAPH_DELIMITERS
calculateReachedPointsFromPreviewSession(ilAssQuestionPreviewSession $preview_session)
getCorrectnessIconForPosition(int $position, bool $graphical_output, array $selections, array $correctness_icons)
generateArrayByPositionFromErrorData()
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
setErrorData(array $errors)
__construct(string $title='', string $comment='', string $author='', int $owner=-1, string $question='')
createErrorTextExport(array $selections)
saveCurrentSolution(int $active_id, int $pass, $value1, $value2, bool $authorized=true, $tstamp=0)
parsePassageEndWord(string $word)
const string $errortext
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
assembleErrorTextOutput(array $selections, bool $graphical_output=false, bool $show_correct_solution=false, bool $use_link_tags=true, array $correctness_icons=[])
const PARAGRAPH_SPLIT_REGEXP
setPointsWrong($a_value)
Class for error text questions.
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.
getExpressionTypes()
Get all available expression types for a specific question.
setErrorText(?string $text)
addPositionToErrorAnswer(assAnswerErrorText $error)
savePreviewData(ilAssQuestionPreviewSession $previewSession)
setPoints(float $points)
setObjId(int $obj_id=0)
saveQuestionDataToDb(?int $original_id=null)
getBestSelection(bool $with_positive_points_only=true)
static _getMobsOfObject(string $a_type, int $a_id, int $a_usage_hist_nr=0, string $a_lang="-")
generateOutputStringFromPosition(int $position, array $selections, array $paragraph, bool $graphical_output, bool $show_correct_solution, bool $use_link_tags, array $correctness_icons)
getMaximumPoints()
Returns the maximum points, a learner can reach answering the question.
getSolutionMaxPass(int $active_id)
getOperators(string $expression)
Get all available operations for a specific question.
removeCurrentSolution(int $active_id, int $pass, bool $authorized=true)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setId(int $id=-1)
solutionValuesToLog(AdditionalInformationGenerator $additional_info, array $solution_values)
__construct(Container $dic, ilPlugin $plugin)
setOriginalId(?int $original_id)
solutionValuesToText(array $solution_values)
setTitle(string $title="")
saveToDb(?int $original_id=null)
setLifecycle(ilAssQuestionLifecycle $lifecycle)
getCurrentSolutionResultSet(int $active_id, int $pass, bool $authorized=true)
saveAdditionalQuestionDataToDb()
Saves the data for the additional data table.
completeParsedErrorTextFromErrorData()
getAvailableAnswerOptions($index=null)
If index is null, the function returns an array with all anwser options Else it returns the specific ...
setAuthor(string $author="")
setParsedErrorText(array $parsed_errortext)
setAdditionalContentEditingMode(?string $additionalContentEditingMode)
getAdditionalInformationFromExistingErrorDataByErrorText(array $current_error_data, string $text_wrong)
getErrorTokenHtml($item, $class, $useLinkTags)
getUserQuestionResult(int $active_id, int $pass)
Get the user solution for a question by active_id and the test pass.
setQuestion(string $question="")