ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.assQuestion.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
38 use ILIAS\Notes\InternalDataService as NotesInternalDataService;
39 use ILIAS\Notes\NoteDBRepository as NotesRepo;
45 
50 abstract class assQuestion implements Question
51 {
52  protected const HAS_SPECIFIC_FEEDBACK = true;
53 
54  public const ADDITIONAL_CONTENT_EDITING_MODE_RTE = 'default';
55  public const ADDITIONAL_CONTENT_EDITING_MODE_IPE = 'pageobject';
56 
57  private const DEFAULT_THUMB_SIZE = 150;
58  private const MINIMUM_THUMB_SIZE = 20;
59  private const MAXIMUM_THUMB_SIZE = 8192;
60  public const TRIM_PATTERN = '/^[\p{C}\p{Z}]+|[\p{C}\p{Z}]+$/u';
61 
62  protected static $force_pass_results_update_enabled = false;
63 
67  protected \ilAssQuestionProcessLocker $processLocker;
70 
71  protected ILIAS $ilias;
73  protected ilLanguage $lng;
74  protected ilDBInterface $db;
77  protected HTTPServices $http;
78  protected Refinery $refinery;
80  protected LoggingServices $log;
81  protected Container $dic;
82 
84  protected \ilAssQuestionLifecycle $lifecycle;
85  public \ilAssQuestionFeedback $feedbackOBJ;
86  protected \ilAssQuestionPage $page;
87 
88  protected int $id;
89  protected string $title;
90  protected string $comment;
91  protected int $owner;
92  protected string $author;
93  protected int $thumb_size;
94  protected string $question;
95  protected float $points = 0.0;
96  protected bool $shuffle = true;
97  protected int $test_id;
98  protected int $obj_id = 0;
99  protected ?int $original_id = null;
100  private int $nr_of_tries;
101  protected ?int $lastChange = null;
102  private string $export_image_path;
103  protected ?string $external_id = null;
104  private string $additionalContentEditingMode = '';
105  public bool $prevent_rte_usage = false;
106  public bool $selfassessmenteditingmode = false;
107  public int $defaultnroftries = 0;
108  public string $questionActionCmd = 'handleQuestionAction';
109  protected ?int $step = null;
110 
114  protected array $suggested_solutions;
115 
116  public function __construct(
117  string $title = "",
118  string $comment = "",
119  string $author = "",
120  int $owner = -1,
121  string $question = ""
122  ) {
124  global $DIC;
125  $this->dic = $DIC;
126  $lng = $DIC['lng'];
127  $tpl = $DIC['tpl'];
128  $ilDB = $DIC['ilDB'];
129  $ilLog = $DIC->logger();
130  $local_dic = QuestionPoolDIC::dic();
131  $this->questionrepository = $local_dic['question.general_properties.repository'];
132  $this->questionpool_request = $local_dic['request_data_collector'];
133  $this->question_files = $local_dic['question_files'];
134  $this->suggestedsolution_repo = $local_dic['question.repo.suggestedsolutions'];
135  $this->current_user = $DIC['ilUser'];
136  $this->lng = $lng;
137  $this->tpl = $tpl;
138  $this->db = $ilDB;
139  $this->log = $ilLog;
140  $this->http = $DIC->http();
141  $this->refinery = $DIC->refinery();
142 
143  $this->thumb_size = self::DEFAULT_THUMB_SIZE;
144 
145  $this->title = $title;
146  $this->comment = $comment;
147  $this->setAuthor($author);
148  $this->setOwner($owner);
149 
150  $this->setQuestion($question);
151 
152  $this->id = -1;
153  $this->test_id = -1;
154  $this->suggested_solutions = [];
155  $this->shuffle = true;
156  $this->nr_of_tries = 0;
157  $this->setExternalId(null);
158 
159  $this->questionActionCmd = 'handleQuestionAction';
160  $this->export_image_path = '';
161  $this->shuffler = $DIC->refinery()->random()->dontShuffle();
162  $this->lifecycle = ilAssQuestionLifecycle::getDraftInstance();
163  $this->skillUsageService = $DIC->skills()->usage();
164  }
165 
166  abstract public function getQuestionType(): string;
167 
168  abstract public function isComplete(): bool;
169 
170  abstract public function saveWorkingData(int $active_id, ?int $pass = null, bool $authorized = true): bool;
171 
172  abstract public function calculateReachedPoints(
173  int $active_id,
174  ?int $pass = null,
175  bool $authorized_solution = true
176  ): float;
177 
178  abstract public function getAdditionalTableName(): string;
179  abstract public function getAnswerTableName(): string|array;
180 
189  abstract public function toLog(AdditionalInformationGenerator $additional_info): array;
190 
199  abstract protected function solutionValuesToLog(
200  AdditionalInformationGenerator $additional_info,
201  array $solution_values
202  ): array|string;
203 
208  abstract protected function solutionValuesToText(
209  array $solution_values
210  ): array|string;
211 
213  {
214  self::$force_pass_results_update_enabled = $force_pass_results_update_enabled;
215  }
216 
217  public static function isForcePassResultUpdateEnabled(): bool
218  {
219  return self::$force_pass_results_update_enabled;
220  }
221 
222  protected function getQuestionAction(): string
223  {
224  return $this->questionpool_request->getCmdIndex($this->questionActionCmd) ?? '';
225  }
226 
227  protected function isNonEmptyItemListPostSubmission(string $post_submission_field_name): bool
228  {
229  return !empty($this->questionpool_request->strArray($post_submission_field_name));
230  }
231 
232  public function getCurrentUser(): ilObjUser
233  {
234  return $this->current_user;
235  }
236 
237  public function getShuffler(): Transformation
238  {
239  return $this->shuffler;
240  }
241 
242  public function setShuffler(Transformation $shuffler): void
243  {
244  $this->shuffler = $shuffler;
245  }
246 
247  public function setProcessLocker(ilAssQuestionProcessLocker $processLocker): void
248  {
249  $this->processLocker = $processLocker;
250  }
251 
253  {
254  return $this->processLocker;
255  }
256 
257  final public function fromXML(
258  string $importdirectory,
259  int $user_id,
260  ilQTIItem $item,
261  int $questionpool_id,
262  ?int $tst_id,
263  ?ilObject &$tst_object,
264  int &$question_counter,
265  array $import_mapping,
266  array $solutionhints = []
267  ): array {
268  $classname = $this->getQuestionType() . "Import";
269  $import = new $classname($this);
270  $new_import_mapping = $import->fromXML(
271  $importdirectory,
272  $user_id,
273  $item,
274  $questionpool_id,
275  $tst_id,
276  $tst_object,
277  $question_counter,
278  $import_mapping
279  );
280 
281  foreach ($solutionhints as $hint) {
282  $this->importHint($import->getQuestionId(), $hint);
283  }
284 
285  return $new_import_mapping;
286  }
287 
288  private function importHint(int $question_id, array $hint_array): void
289  {
290  $hint = new ilAssQuestionHint();
291  $hint->setQuestionId($question_id);
292  $hint->setIndex($hint_array['index'] ?? '');
293  $hint->setPoints($hint_array['points'] ?? '');
294  if ($this->getAdditionalContentEditingMode() === self::ADDITIONAL_CONTENT_EDITING_MODE_IPE) {
295  $hint->save();
296  $hint_page = (new ilAssHintPage());
297  $hint_page->setParentId($question_id);
298  $hint_page->setId($hint->getId());
299  $hint_page->setXMLContent($hint_array['txt']);
300  $hint_page->createFromXML();
301  return;
302  }
303 
304  $hint->setText($hint_array['txt'] ?? '');
305  $hint->save();
306  }
307 
313  final public function toXML(
314  bool $a_include_header = true,
315  bool $a_include_binary = true,
316  bool $a_shuffle = false,
317  bool $test_output = false,
318  bool $force_image_references = false
319  ): string {
320  $classname = $this->getQuestionType() . "Export";
321  $export = new $classname($this);
322  return $export->toXML($a_include_header, $a_include_binary, $a_shuffle, $test_output, $force_image_references);
323  }
324 
325  public function setTitle(string $title = ""): void
326  {
327  $this->title = $title;
328  }
329 
330  public function setId(int $id = -1): void
331  {
332  $this->id = $id;
333  }
334 
335  public function setTestId(int $id = -1): void
336  {
337  $this->test_id = $id;
338  }
339 
340  public function setComment(string $comment = ""): void
341  {
342  $this->comment = $comment;
343  }
344 
345  public function setShuffle(?bool $shuffle = true): void
346  {
347  $this->shuffle = $shuffle ?? false;
348  }
349 
350  public function setAuthor(string $author = ""): void
351  {
352  if ($author === '') {
353  $author = $this->current_user->getFullname();
354  }
355  $this->author = $author;
356  }
357 
358  public function setOwner(int $owner = -1): void
359  {
360  $this->owner = $owner;
361  }
362 
363  public function getTitle(): string
364  {
365  return $this->title;
366  }
367 
368  public function getTitleForHTMLOutput(): string
369  {
370  return $this->refinery->encode()->htmlSpecialCharsAsEntities()->transform($this->title);
371  }
372 
373  public function getTitleFilenameCompliant(): string
374  {
375  return ilFileUtils::getASCIIFilename($this->getTitle());
376  }
377 
378  public function getId(): int
379  {
380  return $this->id;
381  }
382 
383  public function getShuffle(): bool
384  {
385  return $this->shuffle;
386  }
387 
388  public function getTestId(): int
389  {
390  return $this->test_id;
391  }
392 
393  public function getComment(): string
394  {
395  return $this->comment;
396  }
397 
398  public function getDescriptionForHTMLOutput(): string
399  {
400  return $this->refinery->encode()->htmlSpecialCharsAsEntities()->transform($this->comment);
401  }
402 
403  public function getThumbSize(): int
404  {
405  return $this->thumb_size;
406  }
407 
408  public function setThumbSize(int $a_size): void
409  {
410  if ($a_size >= self::MINIMUM_THUMB_SIZE) {
411  $this->thumb_size = $a_size;
412  } else {
413  throw new ilException("Thumb size must be at least " . self::MINIMUM_THUMB_SIZE . "px");
414  }
415  }
416 
417  public function getMinimumThumbSize(): int
418  {
419  return self::MINIMUM_THUMB_SIZE;
420  }
421 
422  public function getMaximumThumbSize(): int
423  {
424  return self::MAXIMUM_THUMB_SIZE;
425  }
426 
427  public function getAuthor(): string
428  {
429  return $this->author;
430  }
431 
432  public function getAuthorForHTMLOutput(): string
433  {
434  return $this->refinery->string()->stripTags()->transform($this->author);
435  }
436 
437  public function getOwner(): int
438  {
439  return $this->owner;
440  }
441 
442  public function getObjId(): int
443  {
444  return $this->obj_id;
445  }
446 
447  public function setObjId(int $obj_id = 0): void
448  {
449  $this->obj_id = $obj_id;
450  }
451 
453  {
454  return $this->lifecycle;
455  }
456 
457  public function setLifecycle(ilAssQuestionLifecycle $lifecycle): void
458  {
459  $this->lifecycle = $lifecycle;
460  }
461 
462  public function setExternalId(?string $external_id): void
463  {
464  $this->external_id = $external_id;
465  }
466 
467  public function getExternalId(): string
468  {
469  if ($this->external_id === null || $this->external_id === '') {
470  if ($this->getId() > 0) {
471  return 'il_' . IL_INST_ID . '_qst_' . $this->getId();
472  }
473  return uniqid('', true);
474  }
475  return $this->external_id;
476  }
477 
478  public static function _getSuggestedSolutionOutput(int $question_id): string
479  {
480  $question = self::instantiateQuestion($question_id);
481  if (!is_object($question)) {
482  return "";
483  }
484  return $question->getSuggestedSolutionOutput();
485  }
486 
487  public function getSuggestedSolutionOutput(): string
488  {
489  $output = [];
490  foreach ($this->suggested_solutions as $solution) {
491  switch ($solution->getType()) {
492  case SuggestedSolution::TYPE_LM:
493  case SuggestedSolution::TYPE_LM_CHAPTER:
494  case SuggestedSolution::TYPE_LM_PAGE:
495  case SuggestedSolution::TYPE_GLOSARY_TERM:
496  $output[] = '<a href="'
497  . $this->getInternalLinkHref($solution->getInternalLink())
498  . '">'
499  . $this->lng->txt("solution_hint")
500  . '</a>';
501  break;
502 
503  case SuggestedSolution::TYPE_FILE:
504  $possible_texts = array_values(
505  array_filter(
506  [
507  ilLegacyFormElementsUtil::prepareFormOutput($solution->getTitle()),
508  ilLegacyFormElementsUtil::prepareFormOutput($solution->getFilename()),
509  $this->lng->txt('tst_show_solution_suggested')
510  ]
511  )
512  );
513 
515  $output[] = '<a href="'
517  $this->getSuggestedSolutionPathWeb() . $solution->getFilename()
518  )
519  . '">'
520  . $possible_texts[0]
521  . '</a>';
522  break;
523  }
524  }
525  return implode("<br />", $output);
526  }
527 
528  public function getSuggestedSolutions(): array
529  {
531  }
532 
533  public static function _getReachedPoints(int $active_id, int $question_id, int $pass): float
534  {
535  global $DIC;
536  $ilDB = $DIC['ilDB'];
537 
538  $points = 0.0;
539 
540  $result = $ilDB->queryF(
541  "SELECT * FROM tst_test_result WHERE active_fi = %s AND question_fi = %s AND pass = %s",
542  ['integer','integer','integer'],
543  [$active_id, $question_id, $pass]
544  );
545  if ($result->numRows() == 1) {
546  $row = $ilDB->fetchAssoc($result);
547  $points = (float) $row["points"];
548  }
549  return $points;
550  }
551 
552  public function getReachedPoints(int $active_id, int $pass): float
553  {
554  return round(self::_getReachedPoints($active_id, $this->getId(), $pass), 2);
555  }
556 
557  public function getMaximumPoints(): float
558  {
559  return $this->points;
560  }
561 
562  final public function getAdjustedReachedPoints(int $active_id, int $pass, bool $authorized_solution = true): float
563  {
564  // determine reached points for submitted solution
565  $reached_points = $this->calculateReachedPoints($active_id, $pass, $authorized_solution);
566  $hint_tracking = new ilAssQuestionHintTracking($this->getId(), $active_id, $pass);
567  $requests_statistic_data = $hint_tracking->getRequestStatisticDataByQuestionAndTestpass();
568  $reached_points = $reached_points - $requests_statistic_data->getRequestsPoints();
569 
570  // adjust reached points regarding to tests scoring options
571  $reached_points = $this->adjustReachedPointsByScoringOptions($reached_points, $active_id);
572 
573  return $reached_points;
574  }
575 
579  final public function calculateResultsFromSolution(int $active_id, int $pass): void
580  {
581  // determine reached points for submitted solution
582  $reached_points = $this->calculateReachedPoints($active_id, $pass);
583  $questionHintTracking = new ilAssQuestionHintTracking($this->getId(), $active_id, $pass);
584  $requests_statistic_data = $questionHintTracking->getRequestStatisticDataByQuestionAndTestpass();
585  $reached_points = $reached_points - $requests_statistic_data->getRequestsPoints();
586 
587  // adjust reached points regarding to tests scoring options
588  $reached_points = $this->adjustReachedPointsByScoringOptions($reached_points, $active_id);
589 
590  if (is_null($reached_points)) {
591  $reached_points = 0.0;
592  }
593 
594  // fau: testNav - check for existing authorized solution to know if a result record should be written
595  $existing_solutions = $this->lookupForExistingSolutions($active_id, $pass);
596 
597  $this->getProcessLocker()->executeUserQuestionResultUpdateOperation(
598  function () use ($active_id, $pass, $reached_points, $requests_statistic_data, $existing_solutions) {
599  $query = "
600  DELETE FROM tst_test_result
601 
602  WHERE active_fi = %s
603  AND question_fi = %s
604  AND pass = %s
605  ";
606 
607  $types = ['integer', 'integer', 'integer'];
608  $values = [$active_id, $this->getId(), $pass];
609 
610  if ($this->getStep() !== null) {
611  $query .= "
612  AND step = %s
613  ";
614 
615  $types[] = 'integer';
616  $values[] = $this->getStep();
617  }
618  $this->db->manipulateF($query, $types, $values);
619 
620  if ($existing_solutions['authorized']) {
621  $next_id = $this->db->nextId("tst_test_result");
622  $fieldData = [
623  'test_result_id' => ['integer', $next_id],
624  'active_fi' => ['integer', $active_id],
625  'question_fi' => ['integer', $this->getId()],
626  'pass' => ['integer', $pass],
627  'points' => ['float', $reached_points],
628  'tstamp' => ['integer', time()],
629  'hint_count' => ['integer', $requests_statistic_data->getRequestsCount()],
630  'hint_points' => ['float', $requests_statistic_data->getRequestsPoints()],
631  'answered' => ['integer', true]
632  ];
633 
634  if ($this->getStep() !== null) {
635  $fieldData['step'] = ['integer', $this->getStep()];
636  }
637 
638  $this->db->insert('tst_test_result', $fieldData);
639  }
640  }
641  );
642 
643  // update test pass results
644  $test = new ilObjTest(
645  $this->getObjId(),
646  false
647  );
648  $test->updateTestPassResults($active_id, $pass, $this->getProcessLocker());
649  ilCourseObjectiveResult::_updateObjectiveResult($this->current_user->getId(), $active_id, $this->getId());
650  }
651 
656  final public function persistWorkingState(int $active_id, $pass, bool $authorized = true): bool
657  {
658  if (!$this instanceof QuestionPartiallySaveable && !$this->validateSolutionSubmit()) {
659  return false;
660  }
661 
662  $saveStatus = false;
663 
664  $this->getProcessLocker()->executePersistWorkingStateLockOperation(function () use ($active_id, $pass, $authorized, &$saveStatus) {
665  if ($pass === null) {
666  $pass = ilObjTest::_getPass($active_id);
667  }
668 
669  $saveStatus = $this->saveWorkingData($active_id, $pass, $authorized);
670 
671  if ($authorized) {
672  // fau: testNav - remove an intermediate solution if the authorized solution is saved
673  // the intermediate solution would set the displayed question status as "editing ..."
674  $this->removeIntermediateSolution($active_id, $pass);
675  // fau.
676  $this->calculateResultsFromSolution($active_id, $pass);
677  }
678  });
679 
680  return $saveStatus;
681  }
682 
686  final public function persistPreviewState(ilAssQuestionPreviewSession $preview_session): bool
687  {
688  $this->savePreviewData($preview_session);
689  return $this->validateSolutionSubmit();
690  }
691 
692  public function validateSolutionSubmit(): bool
693  {
694  return true;
695  }
696 
697  protected function savePreviewData(ilAssQuestionPreviewSession $preview_session): void
698  {
699  $preview_session->setParticipantsSolution($this->getSolutionSubmit());
700  }
701 
702  public function getSuggestedSolutionPath(): string
703  {
704  return CLIENT_WEB_DIR . "/assessment/$this->obj_id/$this->id/solution/";
705  }
706 
711  public function getImagePath($question_id = null, $object_id = null): string
712  {
713  if ($question_id === null) {
714  $question_id = $this->id;
715  }
716 
717  if ($object_id === null) {
718  $object_id = $this->obj_id;
719  }
720 
721  return $this->question_files->buildImagePath($question_id, $object_id);
722  }
723 
724  public function getSuggestedSolutionPathWeb(): string
725  {
727  . "/assessment/{$this->obj_id}/{$this->id}/solution/";
728  return str_replace(
729  ilFileUtils::removeTrailingPathSeparators(ILIAS_ABSOLUTE_PATH . '/public'),
731  $webdir
732  );
733  }
734 
740  public function getImagePathWeb(): string
741  {
742  if (!$this->export_image_path) {
744  . "/assessment/{$this->obj_id}/{$this->id}/images/";
745  return str_replace(
746  ilFileUtils::removeTrailingPathSeparators(ILIAS_ABSOLUTE_PATH . '/public'),
748  $webdir
749  );
750  }
752  }
753 
754  public function getTestOutputSolutions(int $activeId, int $pass): array
755  {
756  if ($this->getTestPresentationConfig()->isSolutionInitiallyPrefilled()) {
757  return $this->getSolutionValues($activeId, $pass, true);
758  }
759  return $this->getUserSolutionPreferingIntermediate($activeId, $pass);
760  }
761 
762 
764  int $active_id,
765  ?int $pass = null
766  ): array {
767  $solution = $this->getSolutionValues($active_id, $pass, false);
768 
769  if (!count($solution)) {
770  $solution = $this->getSolutionValues($active_id, $pass, true);
771  }
772 
773  return $solution;
774  }
775 
779  public function getSolutionValues(
780  int $active_id,
781  ?int $pass = null,
782  bool $authorized = true
783  ): array {
784  if ($pass === null && is_numeric($active_id)) {
785  $pass = $this->getSolutionMaxPass((int) $active_id);
786  }
787 
788  if ($this->getStep() !== null) {
789  $query = "
790  SELECT *
791  FROM tst_solutions
792  WHERE active_fi = %s
793  AND question_fi = %s
794  AND pass = %s
795  AND step = %s
796  AND authorized = %s
797  ORDER BY solution_id";
798 
799  $result = $this->db->queryF(
800  $query,
801  ['integer', 'integer', 'integer', 'integer', 'integer'],
802  [(int) $active_id, $this->getId(), $pass, $this->getStep(), (int) $authorized]
803  );
804  } else {
805  $query = "
806  SELECT *
807  FROM tst_solutions
808  WHERE active_fi = %s
809  AND question_fi = %s
810  AND pass = %s
811  AND authorized = %s
812  ORDER BY solution_id
813  ";
814 
815  $result = $this->db->queryF(
816  $query,
817  ['integer', 'integer', 'integer', 'integer'],
818  [(int) $active_id, $this->getId(), $pass, (int) $authorized]
819  );
820  }
821 
822  $values = [];
823 
824  while ($row = $this->db->fetchAssoc($result)) {
825  $values[] = $row;
826  }
827 
828  return $values;
829  }
830 
831  public function deleteAnswers(int $question_id): void
832  {
833  $answer_table_name = $this->getAnswerTableName();
834 
835  if (!is_array($answer_table_name)) {
836  $answer_table_name = [$answer_table_name];
837  }
838 
839  foreach ($answer_table_name as $table) {
840  if (strlen($table)) {
841  $this->db->manipulateF(
842  "DELETE FROM $table WHERE question_fi = %s",
843  ['integer'],
844  [$question_id]
845  );
846  }
847  }
848  }
849 
850  public function deleteAdditionalTableData(int $question_id): void
851  {
852  $additional_table_name = $this->getAdditionalTableName();
853 
854  if (!is_array($additional_table_name)) {
855  $additional_table_name = [$additional_table_name];
856  }
857 
858  foreach ($additional_table_name as $table) {
859  if (strlen($table)) {
860  $this->db->manipulateF(
861  "DELETE FROM $table WHERE question_fi = %s",
862  ['integer'],
863  [$question_id]
864  );
865  }
866  }
867  }
868 
869  protected function deletePageOfQuestion(int $question_id): void
870  {
871  if (ilAssQuestionPage::_exists('qpl', $question_id, "", true)) {
872  $page = new ilAssQuestionPage($question_id);
873  $page->delete();
874  }
875  }
876 
877  public function delete(int $question_id): void
878  {
879  if ($question_id < 1) {
880  return;
881  }
882 
883  $result = $this->db->queryF(
884  "SELECT obj_fi FROM qpl_questions WHERE question_id = %s",
885  ['integer'],
886  [$question_id]
887  );
888  if ($this->db->numRows($result) !== 1) {
889  return;
890  }
891 
892  $row = $this->db->fetchAssoc($result);
893  $obj_id = $row["obj_fi"];
894 
895  try {
896  $this->deletePageOfQuestion($question_id);
897  } catch (Exception $e) {
898  $this->log->root()->error("EXCEPTION: Could not delete page of question $question_id: $e");
899  return;
900  }
901 
902  $affectedRows = $this->db->manipulateF(
903  "DELETE FROM qpl_questions WHERE question_id = %s",
904  ['integer'],
905  [$question_id]
906  );
907  if ($affectedRows == 0) {
908  return;
909  }
910 
911  try {
912  $this->deleteAdditionalTableData($question_id);
913  $this->deleteAnswers($question_id);
914  $this->feedbackOBJ->deleteGenericFeedbacks($question_id, $this->isAdditionalContentEditingModePageObject());
915  $this->feedbackOBJ->deleteSpecificAnswerFeedbacks($question_id, $this->isAdditionalContentEditingModePageObject());
916  } catch (Exception $e) {
917  $this->log->root()->error("EXCEPTION: Could not delete additional table data of question {$question_id}: {$e}");
918  }
919 
920  try {
921  // delete the question in the tst_test_question table (list of test questions)
922  $affectedRows = $this->db->manipulateF(
923  "DELETE FROM tst_test_question WHERE question_fi = %s",
924  ['integer'],
925  [$question_id]
926  );
927  } catch (Exception $e) {
928  $this->log->root()->error("EXCEPTION: Could not delete delete question {$question_id} from a test: {$e}");
929  }
930 
931  try {
932  $this->getSuggestedSolutionsRepo()->deleteForQuestion($question_id);
933  } catch (Exception $e) {
934  $this->log->root()->error("EXCEPTION: Could not delete suggested solutions of question {$question_id}: {$e}");
935  }
936 
937  $directory = CLIENT_WEB_DIR . "/assessment/" . $obj_id . "/$question_id";
938  try {
939  if (is_dir($directory)) {
940  ilFileUtils::delDir($directory);
941  }
942  } catch (Exception $e) {
943  $this->log->root()->error("EXCEPTION: Could not delete question file directory {$directory} of question {$question_id}: {$e}");
944  }
945 
946  try {
947  $mobs = ilObjMediaObject::_getMobsOfObject("qpl:html", $question_id);
948  // remaining usages are not in text anymore -> delete them
949  // and media objects (note: delete method of ilObjMediaObject
950  // checks whether object is used in another context; if yes,
951  // the object is not deleted!)
952  foreach ($mobs as $mob) {
953  ilObjMediaObject::_removeUsage($mob, "qpl:html", $question_id);
954  if (ilObjMediaObject::_exists($mob)) {
955  $mob_obj = new ilObjMediaObject($mob);
956  $mob_obj->delete();
957  }
958  }
959  } catch (Exception $e) {
960  $this->log->root()->error("EXCEPTION: Error deleting the media objects of question {$question_id}: {$e}");
961  }
962  ilAssQuestionHintTracking::deleteRequestsByQuestionIds([$question_id]);
964  $assignmentList = new ilAssQuestionSkillAssignmentList($this->db);
965  $assignmentList->setParentObjId($obj_id);
966  $assignmentList->setQuestionIdFilter($question_id);
967  $assignmentList->loadFromDb();
968  foreach ($assignmentList->getAssignmentsByQuestionId($question_id) as $assignment) {
969  /* @var ilAssQuestionSkillAssignment $assignment */
970  $assignment->deleteFromDb();
971 
972  // remove skill usage
973  if (!$assignment->isSkillUsed()) {
974  $this->skillUsageService->removeUsage(
975  $assignment->getParentObjId(),
976  $assignment->getSkillBaseId(),
977  $assignment->getSkillTrefId()
978  );
979  }
980  }
981 
982  $this->deleteTaxonomyAssignments();
983  $this->deleteComments();
984 
985  try {
987  } catch (Exception $e) {
988  $this->log->root()->error(
989  "EXCEPTION: Error updating the question pool question count of"
990  . " question pool {$this->getObjId()} when deleting question {$question_id}: {$e}"
991  );
992  }
993  }
994 
995  private function deleteTaxonomyAssignments(): void
996  {
997  $taxIds = ilObjTaxonomy::getUsageOfObject($this->getObjId());
998 
999  foreach ($taxIds as $taxId) {
1000  $taxNodeAssignment = new ilTaxNodeAssignment('qpl', $this->getObjId(), 'quest', $taxId);
1001  $taxNodeAssignment->deleteAssignmentsOfItem($this->getId());
1002  }
1003  }
1004 
1005  public function getTotalAnswers(): int
1006  {
1007  // get all question references to the question id
1008  $result = $this->db->queryF(
1009  "SELECT question_id FROM qpl_questions WHERE original_id = %s OR question_id = %s",
1010  ['integer','integer'],
1011  [$this->id, $this->id]
1012  );
1013  if ($this->db->numRows($result) == 0) {
1014  return 0;
1015  }
1016  $found_id = [];
1017  while ($row = $this->db->fetchAssoc($result)) {
1018  $found_id[] = $row["question_id"];
1019  }
1020 
1021  $result = $this->db->query("SELECT * FROM tst_test_result WHERE " . $this->db->in('question_fi', $found_id, false, 'integer'));
1022 
1023  return $this->db->numRows($result);
1024  }
1025 
1026  public static function isFileAvailable(string $file): bool
1027  {
1028  if (!file_exists($file)) {
1029  return false;
1030  }
1031 
1032  if (!is_file($file)) {
1033  return false;
1034  }
1035 
1036  if (!is_readable($file)) {
1037  return false;
1038  }
1039 
1040  return true;
1041  }
1042 
1043  public function cloneXHTMLMediaObjectsOfQuestion(int $source_question_id): void
1044  {
1045  $mobs = ilObjMediaObject::_getMobsOfObject("qpl:html", $source_question_id);
1046  foreach ($mobs as $mob) {
1047  ilObjMediaObject::_saveUsage($mob, "qpl:html", $this->getId());
1048  }
1049  }
1050 
1051  public function createPageObject(): void
1052  {
1053  $qpl_id = $this->getObjId();
1054  $this->page = new ilAssQuestionPage(0);
1055  $this->page->setId($this->getId());
1056  $this->page->setParentId($qpl_id);
1057  $this->page->setXMLContent("<PageObject><PageContent>" .
1058  "<Question QRef=\"il__qst_" . $this->getId() . "\"/>" .
1059  "</PageContent></PageObject>");
1060  $this->page->create(false);
1061  }
1062 
1063  public function clonePageOfQuestion(int $a_q_id): void
1064  {
1065  if ($a_q_id > 0) {
1066  $page = new ilAssQuestionPage($a_q_id);
1067  $page->buildDom();
1068  ilPCPlugged::handleCopiedPluggedContent($page, $page->getDomDoc());
1069  $xml = str_replace("il__qst_" . $a_q_id, "il__qst_" . $this->id, $page->getXMLFromDom());
1070  $this->page->setXMLContent($xml);
1071  $this->page->updateFromXML();
1072  }
1073  }
1074 
1075  public function getPageOfQuestion(): string
1076  {
1077  $page = new ilAssQuestionPage($this->id);
1078  return $page->getXMLContent();
1079  }
1080 
1081  public function setOriginalId(?int $original_id): void
1082  {
1083  $this->original_id = $original_id;
1084  }
1085 
1086  public function getOriginalId(): ?int
1087  {
1088  return $this->original_id;
1089  }
1090 
1091  protected static $imageSourceFixReplaceMap = [
1092  'ok.svg' => 'ok.png',
1093  'not_ok.svg' => 'not_ok.png',
1094  'object/checkbox_checked.svg' => 'checkbox_checked.png',
1095  'object/checkbox_unchecked.svg' => 'checkbox_unchecked.png',
1096  'object/radiobutton_checked.svg' => 'radiobutton_checked.png',
1097  'object/radiobutton_unchecked.svg' => 'radiobutton_unchecked.png'
1098  ];
1099 
1100  public function fixSvgToPng(string $imageFilenameContainingString): string
1101  {
1102  $needles = array_keys(self::$imageSourceFixReplaceMap);
1103  $replacements = array_values(self::$imageSourceFixReplaceMap);
1104  return str_replace($needles, $replacements, $imageFilenameContainingString);
1105  }
1106 
1107  public function fixUnavailableSkinImageSources(string $html): string
1108  {
1109  $matches = null;
1110  if (preg_match_all('/src="(.*?)"/m', $html, $matches)) {
1111  $sources = $matches[1];
1112 
1113  $needleReplacementMap = [];
1114 
1115  foreach ($sources as $src) {
1116  $file = ilFileUtils::removeTrailingPathSeparators(ILIAS_ABSOLUTE_PATH) . DIRECTORY_SEPARATOR . $src;
1117 
1118  if (file_exists($file)) {
1119  continue;
1120  }
1121 
1122  $levels = explode(DIRECTORY_SEPARATOR, $src);
1123  if (count($levels) < 5 || $levels[0] !== 'Customizing' || $levels[2] !== 'skin') {
1124  continue;
1125  }
1126 
1127  $component = '';
1128 
1129  if ($levels[4] === 'components/ILIAS' || $levels[4] === 'components/ILIAS') {
1130  $component = $levels[4] . DIRECTORY_SEPARATOR . $levels[5];
1131  }
1132 
1133  $needleReplacementMap[$src] = ilUtil::getImagePath(basename($src), $component);
1134  }
1135 
1136  if (count($needleReplacementMap)) {
1137  $html = str_replace(array_keys($needleReplacementMap), array_values($needleReplacementMap), $html);
1138  }
1139  }
1140 
1141  return $html;
1142  }
1143 
1144  public function loadFromDb(int $question_id): void
1145  {
1146  $result = $this->db->queryF(
1147  'SELECT external_id FROM qpl_questions WHERE question_id = %s',
1148  ['integer'],
1149  [$question_id]
1150  );
1151  if ($this->db->numRows($result) === 1) {
1152  $data = $this->db->fetchAssoc($result);
1153  $this->external_id = $data['external_id'];
1154  }
1155 
1156  $suggested_solutions = $this->loadSuggestedSolutions();
1157  $this->suggested_solutions = [];
1158  if ($suggested_solutions) {
1159  foreach ($suggested_solutions as $solution) {
1160  $this->suggested_solutions[$solution->getSubquestionIndex()] = $solution;
1161  }
1162  }
1163  }
1164 
1171  public function createNewQuestion(bool $a_create_page = true): int
1172  {
1173  $complete = '0';
1174  $obj_id = $this->getObjId();
1175  if ($obj_id <= 0
1176  && $this->questionpool_request->hasRefId()) {
1177  $obj_id = $this->questionpool_request->getRefId();
1178  }
1179 
1180  if ($obj_id <= 0) {
1181  $obj_id = $this->questionpool_request->int('sel_qpl');
1182  }
1183 
1184  if ($obj_id <= 0) {
1185  return $this->getId();
1186  }
1187 
1188  $tstamp = time();
1189  if ($a_create_page) {
1190  $tstamp = 0;
1191  }
1192 
1193  $next_id = $this->db->nextId('qpl_questions');
1194  $this->db->insert("qpl_questions", [
1195  "question_id" => ["integer", $next_id],
1196  "question_type_fi" => ["integer", $this->getQuestionTypeID()],
1197  "obj_fi" => ["integer", $obj_id],
1198  "title" => ["text", ''],
1199  "description" => ["text", ''],
1200  "author" => ["text", $this->getAuthor()],
1201  "owner" => ["integer", $this->current_user->getId()],
1202  "question_text" => ["clob", ''],
1203  "points" => ["float", "0.0"],
1204  "nr_of_tries" => ["integer", $this->getDefaultNrOfTries()], // #10771
1205  "complete" => ["text", $complete],
1206  "created" => ["integer", time()],
1207  "original_id" => ["integer", null],
1208  "tstamp" => ["integer", $tstamp],
1209  "external_id" => ["text", $this->getExternalId()],
1210  'add_cont_edit_mode' => ['text', $this->getAdditionalContentEditingMode()]
1211  ]);
1212  $this->setId($next_id);
1213 
1214  if ($a_create_page) {
1215  // create page object of question
1216  $this->createPageObject();
1217  }
1218 
1219  return $this->getId();
1220  }
1221 
1222  public function saveQuestionDataToDb(?int $original_id = null): void
1223  {
1224  if ($this->getId() === -1) {
1225  $next_id = $this->db->nextId('qpl_questions');
1226  $this->db->insert("qpl_questions", [
1227  "question_id" => ["integer", $next_id],
1228  "question_type_fi" => ["integer", $this->getQuestionTypeID()],
1229  "obj_fi" => ["integer", $this->getObjId()],
1230  "title" => ["text", mb_substr($this->getTitle(), 0, 124)],
1231  "description" => ["text", mb_substr($this->getComment(), 0, 1000)],
1232  "author" => ["text", mb_substr($this->getAuthor(), 0, 512)],
1233  "owner" => ["integer", $this->getOwner()],
1234  "question_text" => ["clob", ilRTE::_replaceMediaObjectImageSrc($this->getQuestion(), 0)],
1235  "points" => ["float", $this->getMaximumPoints()],
1236  "nr_of_tries" => ["integer", $this->getNrOfTries()],
1237  "created" => ["integer", time()],
1238  "original_id" => ["integer", $original_id],
1239  "tstamp" => ["integer", time()],
1240  "external_id" => ["text", $this->getExternalId()],
1241  'add_cont_edit_mode' => ['text', $this->getAdditionalContentEditingMode()]
1242  ]);
1243  $this->setId($next_id);
1244  // create page object of question
1245  $this->createPageObject();
1246  return;
1247  }
1248 
1249  // Vorhandenen Datensatz aktualisieren
1250  $this->db->update("qpl_questions", [
1251  "obj_fi" => ["integer", $this->getObjId()],
1252  "title" => ["text", mb_substr($this->getTitle(), 0, 124)],
1253  "description" => ["text", mb_substr($this->getComment(), 0, 1000)],
1254  "author" => ["text", mb_substr($this->getAuthor(), 0, 512)],
1255  "question_text" => ["clob", ilRTE::_replaceMediaObjectImageSrc($this->getQuestion(), 0)],
1256  "points" => ["float", $this->getMaximumPoints()],
1257  "nr_of_tries" => ["integer", $this->getNrOfTries()],
1258  "tstamp" => ["integer", time()],
1259  'complete' => ['integer', $this->isComplete()],
1260  "external_id" => ["text", $this->getExternalId()]
1261  ], [
1262  "question_id" => ["integer", $this->getId()]
1263  ]);
1264  }
1265 
1266  public function duplicate(
1267  bool $for_test = true,
1268  string $title = '',
1269  string $author = '',
1270  int $owner = -1,
1271  $test_obj_id = null
1272  ): int {
1273  if ($this->id <= 0) {
1274  // The question has not been saved. It cannot be duplicated
1275  return -1;
1276  }
1277 
1278  $clone = clone $this;
1279  $clone->id = -1;
1280 
1281  if ((int) $test_obj_id > 0) {
1282  $clone->setObjId($test_obj_id);
1283  }
1284 
1285  if ($title) {
1286  $clone->setTitle($title);
1287  }
1288  if ($author) {
1289  $clone->setAuthor($author);
1290  }
1291  if ($owner) {
1292  $clone->setOwner($owner);
1293  }
1294  if ($for_test) {
1295  $clone->saveToDb($this->id);
1296  } else {
1297  $clone->saveToDb();
1298  }
1299 
1300  $clone->clonePageOfQuestion($this->getId());
1301  $clone->cloneXHTMLMediaObjectsOfQuestion($this->getId());
1302 
1303  $clone = $this->cloneQuestionTypeSpecificProperties($clone);
1304 
1305  $clone->onDuplicate($this->getObjId(), $this->getId(), $clone->getObjId(), $clone->getId());
1306 
1307  return $clone->id;
1308  }
1309 
1310  final public function copyObject(
1311  int $target_parent_id,
1312  string $title = ''
1313  ): int {
1314  if ($this->getId() <= 0) {
1315  throw new RuntimeException('The question has not been saved. It cannot be duplicated');
1316  }
1317  // duplicate the question in database
1318  $clone = clone $this;
1319  $clone->id = -1;
1320  $source_parent_id = $this->getObjId();
1321  $clone->setObjId($target_parent_id);
1322  if ($title) {
1323  $clone->setTitle($title);
1324  }
1325  $clone->saveToDb();
1326  $clone->clonePageOfQuestion($this->id);
1327  $clone->cloneXHTMLMediaObjectsOfQuestion($this->id);
1328  $clone = $this->cloneQuestionTypeSpecificProperties($clone);
1329 
1330  $clone->onCopy($source_parent_id, $this->id, $clone->getObjId(), $clone->getId());
1331 
1332  return $clone->id;
1333  }
1334 
1335  final public function createNewOriginalFromThisDuplicate(
1336  int $target_parent_id,
1337  string $target_question_title = ''
1338  ): int {
1339  if ($this->getId() <= 0) {
1340  throw new RuntimeException('The question has not been saved. It cannot be duplicated');
1341  }
1342 
1343  $source_question_id = $this->id;
1344  $source_parent_id = $this->getObjId();
1345 
1346  // duplicate the question in database
1347  $clone = clone $this;
1348  $clone->id = -1;
1349 
1350  $clone->setObjId($target_parent_id);
1351 
1352  if ($target_question_title) {
1353  $clone->setTitle($target_question_title);
1354  }
1355 
1356  $clone->saveToDb();
1357  $clone->clonePageOfQuestion($source_question_id);
1358  $clone->cloneXHTMLMediaObjectsOfQuestion($source_question_id);
1359 
1360  $clone = $this->cloneQuestionTypeSpecificProperties($clone);
1361 
1362  $clone->onCopy($source_parent_id, $source_question_id, $clone->getObjId(), $clone->getId());
1363 
1364  return $clone->id;
1365  }
1366 
1368  self $target
1369  ): self {
1370  return $target;
1371  }
1372 
1373  public function saveToDb(?int $original_id = null): void
1374  {
1375  // remove unused media objects from ILIAS
1376  $this->cleanupMediaObjectUsage();
1377 
1378  $complete = "0";
1379  if ($this->isComplete()) {
1380  $complete = "1";
1381  }
1382 
1383  $this->db->update(
1384  'qpl_questions',
1385  [
1386  'tstamp' => ['integer', time()],
1387  'owner' => ['integer', $this->getOwner()],
1388  'complete' => ['integer', $complete],
1389  'lifecycle' => ['text', $this->getLifecycle()->getIdentifier()],
1390  ],
1391  [
1392  'question_id' => ['integer', $this->getId()]
1393  ]
1394  );
1395 
1397  }
1398 
1399  protected function removeAllImageFiles(string $image_target_path): void
1400  {
1401  $target = opendir($image_target_path);
1402  while ($target_file = readdir($target)) {
1403  if ($target_file === '.' || $target_file === '..') {
1404  continue;
1405  }
1406  copy(
1407  $image_target_path . DIRECTORY_SEPARATOR . $target_file,
1408  $image_target_path . DIRECTORY_SEPARATOR . $target_file
1409  );
1410  }
1411  }
1412 
1413  public static function saveOriginalId(int $questionId, int $originalId): void
1414  {
1415  global $DIC;
1416  $ilDB = $DIC->database();
1417  $query = "UPDATE qpl_questions SET tstamp = %s, original_id = %s WHERE question_id = %s";
1418 
1419  $ilDB->manipulateF(
1420  $query,
1421  ['integer','integer', 'text'],
1422  [time(), $originalId, $questionId]
1423  );
1424  }
1425 
1426  public static function resetOriginalId(int $questionId): void
1427  {
1428  global $DIC;
1429  $ilDB = $DIC->database();
1430 
1431  $query = "UPDATE qpl_questions SET tstamp = %s, original_id = NULL WHERE question_id = %s";
1432 
1433  $ilDB->manipulateF(
1434  $query,
1435  ['integer', 'text'],
1436  [time(), $questionId]
1437  );
1438  }
1439 
1440  protected function onDuplicate(
1441  int $original_parent_id,
1442  int $original_question_id,
1443  int $duplicate_parent_id,
1444  int $duplicate_question_id
1445  ): void {
1446  $this->copySuggestedSolutions($duplicate_question_id);
1447  $this->cloneSuggestedSolutionFiles($original_parent_id, $original_question_id);
1448  $this->feedbackOBJ->duplicateFeedback($original_question_id, $duplicate_question_id);
1449  $this->duplicateQuestionHints($original_question_id, $duplicate_question_id);
1450  $this->duplicateSkillAssignments($original_parent_id, $original_question_id, $duplicate_parent_id, $duplicate_question_id);
1451  $this->duplicateComments($original_parent_id, $original_question_id, $duplicate_parent_id, $duplicate_question_id);
1452  }
1453 
1454  protected function afterSyncWithOriginal(
1455  int $original_question_id,
1456  int $clone_question_id,
1457  int $original_parent_id,
1458  int $clone_parent_id
1459  ): void {
1460  $this->feedbackOBJ->cloneFeedback($original_question_id, $clone_question_id);
1461  }
1462 
1463  protected function onCopy(int $sourceParentId, int $sourceQuestionId, int $targetParentId, int $targetQuestionId): void
1464  {
1465  $this->copySuggestedSolutions($targetQuestionId);
1466  $this->duplicateSuggestedSolutionFiles($sourceParentId, $sourceQuestionId);
1467  $this->feedbackOBJ->duplicateFeedback($sourceQuestionId, $targetQuestionId);
1468  $this->duplicateQuestionHints($sourceQuestionId, $targetQuestionId);
1469  $this->duplicateSkillAssignments($sourceParentId, $sourceQuestionId, $targetParentId, $targetQuestionId);
1470  $this->duplicateComments($sourceParentId, $sourceQuestionId, $targetParentId, $targetQuestionId);
1471  }
1472 
1473  protected function duplicateComments(
1474  int $parent_source_id,
1475  int $source_id,
1476  int $parent_target_id,
1477  int $target_id
1478  ): void {
1479  $manager = $this->getNotesManager();
1480  $data_service = $this->getNotesDataService();
1481  $notes = $manager->getNotesForRepositoryObjIds([$parent_source_id], Note::PUBLIC);
1482  $notes = array_filter(
1483  $notes,
1484  fn($n) => $n->getContext()->getSubObjId() === $source_id
1485  );
1486 
1487  foreach ($notes as $note) {
1488  $new_context = $data_service->context(
1489  $parent_target_id,
1490  $target_id,
1491  $note->getContext()->getType()
1492  );
1493  $new_note = $data_service->note(
1494  -1,
1495  $new_context,
1496  $note->getText(),
1497  $note->getAuthor(),
1498  $note->getType(),
1499  $note->getCreationDate(),
1500  $note->getUpdateDate(),
1501  $note->getRecipient()
1502  );
1503  $manager->createNote($new_note, [], true);
1504  }
1505  }
1506 
1507  protected function deleteComments(): void
1508  {
1509  $repo = $this->getNotesRepo();
1510  $manager = $this->getNotesManager();
1511  $source_id = $this->getId();
1512  $notes = $manager->getNotesForRepositoryObjIds([$this->getObjId()], Note::PUBLIC);
1513  $notes = array_filter(
1514  $notes,
1515  fn($n) => $n->getContext()->getSubObjId() === $source_id
1516  );
1517  foreach ($notes as $note) {
1518  $repo->deleteNote($note->getId());
1519  }
1520  }
1521 
1522  protected function getNotesManager(): NotesManager
1523  {
1524  $service = new NotesService($this->dic);
1525  return $service->internal()->domain()->notes();
1526  }
1527 
1528  protected function getNotesDataService(): NotesInternalDataService
1529  {
1530  $service = new NotesService($this->dic);
1531  return $service->internal()->data();
1532  }
1533 
1534  protected function getNotesRepo(): NotesRepo
1535  {
1536  $service = new NotesService($this->dic);
1537  return $service->internal()->repo()->note();
1538  }
1539 
1540  public function deleteSuggestedSolutions(): void
1541  {
1542  $this->getSuggestedSolutionsRepo()->deleteForQuestion($this->getId());
1545  $this->suggested_solutions = [];
1546  }
1547 
1548 
1549  public function getSuggestedSolution(int $subquestion_index = 0): ?SuggestedSolution
1550  {
1551  if (array_key_exists($subquestion_index, $this->suggested_solutions)) {
1552  return $this->suggested_solutions[$subquestion_index];
1553  }
1554  return null;
1555  }
1556 
1557  protected function cloneSuggestedSolutions(
1558  int $source_question_id,
1559  int $target_question_id
1560  ): void {
1561  $this->getSuggestedSolutionsRepo()->clone($source_question_id, $target_question_id);
1562  $this->cloneSuggestedSolutionFiles($source_question_id, $target_question_id);
1563  }
1564 
1568  protected function duplicateSuggestedSolutionFiles(int $parent_id, int $question_id): void
1569  {
1570  foreach ($this->suggested_solutions as $solution) {
1571  if (!$solution->isOfTypeFile()
1572  || $solution->getFilename() === '') {
1573  continue;
1574  }
1575 
1576  $filepath = $this->getSuggestedSolutionPath();
1577  $filepath_original = str_replace(
1578  "/{$this->obj_id}/{$this->id}/solution",
1579  "/{$parent_id}/{$question_id}/solution",
1580  $filepath
1581  );
1582  if (!file_exists($filepath)) {
1583  ilFileUtils::makeDirParents($filepath);
1584  }
1585  if (!is_file($filepath_original . $solution->getFilename())
1586  || !copy($filepath_original . $solution->getFilename(), $filepath . $solution->getFilename())) {
1587  $this->log->root()->error('File for suggested solutions could not be duplicated:');
1588  $this->log->root()->error("Question-Id: {$this->id}; Question-Title: {$this->title}; File: {$filepath_original}{$solution->getFilename()}");
1589  }
1590  }
1591  }
1592 
1593  protected function cloneSuggestedSolutionFiles(
1594  int $source_question_id,
1595  int $target_question_id
1596  ): void {
1597  $filepath_target = $this->getSuggestedSolutionPath();
1598  $filepath_original = str_replace("/$target_question_id/solution", "/$source_question_id/solution", $filepath_target);
1599  ilFileUtils::delDir($filepath_original);
1600  foreach ($this->suggested_solutions as $solution) {
1601  if (!$solution->isOfTypeFile()
1602  || $solution->getFilename() === '') {
1603  continue;
1604  }
1605 
1606  if (!file_exists($filepath_original)) {
1607  ilFileUtils::makeDirParents($filepath_original);
1608  }
1609 
1610  if (!is_file($filepath_original . $solution->getFilename())
1611  || copy($filepath_target . $solution->getFilename(), $filepath_target . $solution->getFilename())) {
1612  $this->log->root()->error('File for suggested solutions could not be cloned:');
1613  $this->log->root()->error("Question-Id: {$this->id}; Question-Title: {$this->title}; File: {$filepath_original}{$solution->getFilename()}");
1614  }
1615  }
1616  }
1617 
1618  protected function copySuggestedSolutions(int $target_question_id): void
1619  {
1620  $update = [];
1621  foreach ($this->getSuggestedSolutions() as $index => $solution) {
1622  $solution = $solution->withQuestionId($target_question_id);
1623  $update[] = $solution;
1624  }
1625  $this->getSuggestedSolutionsRepo()->update($update);
1626  }
1627 
1628  public function resolveInternalLink(string $internal_link): string
1629  {
1630  if (preg_match("/il_(\d+)_(\w+)_(\d+)/", $internal_link, $matches) === false) {
1631  return $internal_link;
1632  }
1633  switch ($matches[2]) {
1634  case "lm":
1635  $resolved_link = ilLMObject::_getIdForImportId($internal_link);
1636  break;
1637  case "pg":
1638  $resolved_link = ilInternalLink::_getIdForImportId("PageObject", $internal_link);
1639  break;
1640  case "st":
1641  $resolved_link = ilInternalLink::_getIdForImportId("StructureObject", $internal_link);
1642  break;
1643  case "git":
1644  $resolved_link = ilInternalLink::_getIdForImportId("GlossaryItem", $internal_link);
1645  break;
1646  case "mob":
1647  $resolved_link = ilInternalLink::_getIdForImportId("MediaObject", $internal_link);
1648  break;
1649  }
1650  if ($resolved_link === null) {
1651  return "il__{$matches[2]}_{$matches[3]}";
1652  }
1653  return $internal_link;
1654  }
1655 
1656 
1657  //TODO: move this to import or suggested solutions repo.
1658  //use in LearningModule and Survey as well ;(
1659  public function resolveSuggestedSolutionLinks(): void
1660  {
1661  $resolvedlinks = 0;
1662  $result_pre = $this->db->queryF(
1663  "SELECT internal_link, suggested_solution_id FROM qpl_sol_sug WHERE question_fi = %s",
1664  ['integer'],
1665  [$this->getId()]
1666  );
1667  if ($this->db->numRows($result_pre) < 1) {
1668  return;
1669  }
1670 
1671  while ($row = $this->db->fetchAssoc($result_pre)) {
1672  $internal_link = $row["internal_link"];
1673  $resolved_link = $this->resolveInternalLink($internal_link);
1674  if ($internal_link === $resolved_link) {
1675  continue;
1676  }
1677  // internal link was resolved successfully
1678  $this->db->manipulateF(
1679  "UPDATE qpl_sol_sug SET internal_link = %s WHERE suggested_solution_id = %s",
1680  ['text','integer'],
1681  [$resolved_link, $row["suggested_solution_id"]]
1682  );
1683  $resolvedlinks++;
1684  }
1685  if ($resolvedlinks === 0) {
1686  return;
1687  }
1688 
1690 
1691  $result_post = $this->db->queryF(
1692  "SELECT internal_link FROM qpl_sol_sug WHERE question_fi = %s",
1693  ['integer'],
1694  [$this->getId()]
1695  );
1696  if ($this->db->numRows($result_post) < 1) {
1697  return;
1698  }
1699 
1700  while ($row = $this->db->fetchAssoc($result_post)) {
1701  if (preg_match("/il_(\d*?)_(\w+)_(\d+)/", $row["internal_link"], $matches)) {
1702  ilInternalLink::_saveLink("qst", $this->getId(), $matches[2], $matches[3], $matches[1]);
1703  }
1704  }
1705  }
1706 
1707  public function getInternalLinkHref(string $target): string
1708  {
1709  $linktypes = [
1710  "lm" => "LearningModule",
1711  "pg" => "PageObject",
1712  "st" => "StructureObject",
1713  "git" => "GlossaryItem",
1714  "mob" => "MediaObject"
1715  ];
1716  $href = "";
1717  if (preg_match("/il__(\w+)_(\d+)/", $target, $matches)) {
1718  $type = $matches[1];
1719  $target_id = $matches[2];
1720  switch ($linktypes[$matches[1]]) {
1721  case "MediaObject":
1722  $href = "./ilias.php?baseClass=ilLMPresentationGUI&obj_type=" . $linktypes[$type]
1723  . "&cmd=media&ref_id=" . $this->questionpool_request->getRefId()
1724  . "&mob_id=" . $target_id;
1725  break;
1726  case "StructureObject":
1727  case "GlossaryItem":
1728  case "PageObject":
1729  case "LearningModule":
1730  default:
1731  $href = "./goto.php?target=" . $type . "_" . $target_id;
1732  break;
1733  }
1734  }
1735  return $href;
1736  }
1737 
1738  public function syncWithOriginal(): void
1739  {
1740  if ($this->getOriginalId() === null) {
1741  return;
1742  }
1743 
1744  $original_parent_id = self::lookupParentObjId($this->getOriginalId());
1745 
1746  if ($original_parent_id === null) {
1747  return;
1748  }
1749 
1750  $this->cloneSuggestedSolutions($this->getId(), $this->getOriginalId());
1751  $original = clone $this;
1752  // Now we become the original
1753  $original->setId($this->getOriginalId());
1754  $original->setOriginalId(null);
1755  $original->setObjId($original_parent_id);
1756 
1757  $original->saveToDb();
1758 
1759  $original->deletePageOfQuestion($this->getOriginalId());
1760  $original->createPageObject();
1761  $original->clonePageOfQuestion($this->getId());
1762  $original = $this->cloneQuestionTypeSpecificProperties($original);
1763  $this->cloneXHTMLMediaObjectsOfQuestion($original->getId());
1764  $this->afterSyncWithOriginal($this->getOriginalId(), $this->getId(), $this->getObjId(), $original_parent_id);
1765  $this->cloneHints($this->id, $this->original_id);
1766  }
1767 
1773  public static function instantiateQuestion(int $question_id): assQuestion
1774  {
1775  global $DIC;
1776  $ilCtrl = $DIC['ilCtrl'];
1777  $ilDB = $DIC['ilDB'];
1778  $lng = $DIC['lng'];
1779  $questionrepository = QuestionPoolDIC::dic()['question.general_properties.repository'];
1780  $question_type = $questionrepository->getForQuestionId($question_id)?->getClassName() ?? '';
1781  if ($question_type === '') {
1782  throw new InvalidArgumentException('No question with ID ' . $question_id . ' exists');
1783  }
1784 
1785  $question = new $question_type();
1786  $question->loadFromDb($question_id);
1787 
1788  $feedbackObjectClassname = self::getFeedbackClassNameByQuestionType($question_type);
1789  $question->feedbackOBJ = new $feedbackObjectClassname($question, $ilCtrl, $ilDB, $lng);
1790 
1791  return $question;
1792  }
1793 
1794  public function getPoints(): float
1795  {
1796  return $this->points;
1797  }
1798 
1799  public function setPoints(float $points): void
1800  {
1801  $this->points = $points;
1802  }
1803 
1804  public function getSolutionMaxPass(int $active_id): ?int
1805  {
1806  return self::_getSolutionMaxPass($this->getId(), $active_id);
1807  }
1808 
1812  public static function _getSolutionMaxPass(int $question_id, int $active_id): ?int
1813  {
1814  // the following code was the old solution which added the non answered
1815  // questions of a pass from the answered questions of the previous pass
1816  // with the above solution, only the answered questions of the last pass are counted
1817  global $DIC;
1818  $ilDB = $DIC['ilDB'];
1819 
1820  $result = $ilDB->queryF(
1821  "SELECT MAX(pass) maxpass FROM tst_test_result WHERE active_fi = %s AND question_fi = %s",
1822  ['integer','integer'],
1823  [$active_id, $question_id]
1824  );
1825  if ($result->numRows() === 1) {
1826  $row = $ilDB->fetchAssoc($result);
1827  return $row["maxpass"];
1828  }
1829 
1830  return null;
1831  }
1832 
1833  public function isWriteable(): bool
1834  {
1835  return ilObjQuestionPool::_isWriteable($this->getObjId(), $this->getCurrentUser()->getId());
1836  }
1837 
1838  public function deductHintPointsFromReachedPoints(ilAssQuestionPreviewSession $preview_session, $reached_points): ?float
1839  {
1840  $hint_tracking = new ilAssQuestionPreviewHintTracking($this->db, $preview_session);
1841  $requests_statistic_data = $hint_tracking->getRequestStatisticData();
1842  return $reached_points - $requests_statistic_data->getRequestsPoints();
1843  }
1844 
1846  {
1847  $reached_points = $this->deductHintPointsFromReachedPoints(
1848  $preview_session,
1849  $this->calculateReachedPointsForSolution($preview_session->getParticipantsSolution())
1850  );
1851 
1852  return $this->ensureNonNegativePoints($reached_points);
1853  }
1854 
1855  protected function ensureNonNegativePoints(float $points): float
1856  {
1857  return $points > 0.0 ? $points : 0.0;
1858  }
1859 
1860  public function isPreviewSolutionCorrect(ilAssQuestionPreviewSession $preview_session): bool
1861  {
1862  $reached_points = $this->calculateReachedPointsFromPreviewSession($preview_session);
1863 
1864  return !($reached_points < $this->getMaximumPoints());
1865  }
1866 
1867 
1878  final public function adjustReachedPointsByScoringOptions(float $points, int $active_id): float
1879  {
1880  $count_system = ilObjTest::_getCountSystem($active_id);
1881  if ($count_system == 1) {
1882  if (abs($this->getMaximumPoints() - $points) > 0.0000000001) {
1883  $points = 0;
1884  }
1885  }
1886  $score_cutting = ilObjTest::_getScoreCutting($active_id);
1887  if ($score_cutting == 0) {
1888  if ($points < 0) {
1889  $points = 0;
1890  }
1891  }
1892  return $points;
1893  }
1894  public function buildHashedImageFilename(string $plain_image_filename, bool $unique = false): string
1895  {
1896  $extension = "";
1897 
1898  if (preg_match("/.*\.(png|jpg|gif|jpeg)$/i", $plain_image_filename, $matches)) {
1899  $extension = "." . $matches[1];
1900  }
1901 
1902  if ($unique) {
1903  $plain_image_filename = uniqid($plain_image_filename . microtime(true), true);
1904  }
1905 
1906  return md5($plain_image_filename) . $extension;
1907  }
1908 
1919  public static function _setReachedPoints(
1920  int $active_id,
1921  int $question_id,
1922  float $points,
1923  float $maxpoints,
1924  int $pass,
1925  bool $manualscoring
1926  ): void {
1927  global $DIC;
1928  $ilDB = $DIC['ilDB'];
1929 
1930  if ($points > $maxpoints) {
1931  return;
1932  }
1933 
1934  if ($pass === null) {
1935  $pass = assQuestion::_getSolutionMaxPass($question_id, $active_id);
1936  }
1937 
1938  $old_points = 0;
1939  $result = $ilDB->queryF(
1940  "SELECT points FROM tst_test_result WHERE active_fi = %s AND question_fi = %s AND pass = %s",
1941  ['integer','integer','integer'],
1942  [$active_id, $question_id, $pass]
1943  );
1944  $manual = ($manualscoring) ? 1 : 0;
1945  $rowsnum = $result->numRows();
1946  if ($rowsnum > 0) {
1947  $row = $ilDB->fetchAssoc($result);
1948  $old_points = $row['points'];
1949  if ($old_points !== $points) {
1950  $affectedRows = $ilDB->manipulateF(
1951  "UPDATE tst_test_result SET points = %s, manual = %s, tstamp = %s WHERE active_fi = %s AND question_fi = %s AND pass = %s",
1952  ['float', 'integer', 'integer', 'integer', 'integer', 'integer'],
1953  [$points, $manual, time(), $active_id, $question_id, $pass]
1954  );
1955  }
1956  } else {
1957  $next_id = $ilDB->nextId('tst_test_result');
1958  $affectedRows = $ilDB->manipulateF(
1959  "INSERT INTO tst_test_result (test_result_id, active_fi, question_fi, points, pass, manual, tstamp) VALUES (%s, %s, %s, %s, %s, %s, %s)",
1960  ['integer', 'integer','integer', 'float', 'integer', 'integer','integer'],
1961  [$next_id, $active_id, $question_id, $points, $pass, $manual, time()]
1962  );
1963  }
1964 
1965  if (!self::isForcePassResultUpdateEnabled() && $old_points === $points && $rowsnum !== 0) {
1966  return;
1967  }
1968 
1969  $test_id = ilObjTest::_lookupTestObjIdForQuestionId($question_id);
1970  if ($test_id === null) {
1971  return;
1972  }
1973  $test = new ilObjTest(
1974  $test_id,
1975  false
1976  );
1977  $test->updateTestPassResults($active_id, $pass);
1979  }
1980 
1981  public function getQuestion(): string
1982  {
1983  return $this->question;
1984  }
1985 
1986  public function getQuestionForHTMLOutput(): string
1987  {
1988  return $this->purifyAndPrepareTextAreaOutput($this->question);
1989  }
1990 
1991  protected function purifyAndPrepareTextAreaOutput(string $content): string
1992  {
1993  $purified_content = $this->getHtmlQuestionContentPurifier()->purify($content);
1995  || !(new ilSetting('advanced_editing'))->get('advanced_editing_javascript_editor') === 'tinymce') {
1996  $purified_content = nl2br($purified_content);
1997  }
1999  $purified_content,
2000  true,
2001  true
2002  );
2003  }
2004 
2005  public function setQuestion(string $question = ""): void
2006  {
2007  $this->question = $question;
2008  }
2009 
2010  public function getQuestionTypeID(): int
2011  {
2012  $result = $this->db->queryF(
2013  "SELECT question_type_id FROM qpl_qst_type WHERE type_tag = %s",
2014  ['text'],
2015  [$this->getQuestionType()]
2016  );
2017  if ($this->db->numRows($result) == 1) {
2018  $row = $this->db->fetchAssoc($result);
2019  return (int) $row["question_type_id"];
2020  }
2021  return 0;
2022  }
2023 
2024  public function cloneHints(
2025  int $source_question_id,
2026  int $target_question_id
2027  ): void {
2028  // delete hints of the original
2029  $this->db->manipulateF(
2030  "DELETE FROM qpl_hints WHERE qht_question_fi = %s",
2031  ['integer'],
2032  [$target_question_id]
2033  );
2034 
2035  // get hints of the actual question
2036  $result = $this->db->queryF(
2037  "SELECT * FROM qpl_hints WHERE qht_question_fi = %s",
2038  ['integer'],
2039  [$source_question_id]
2040  );
2041 
2042  // save hints to the original
2043  if ($this->db->numRows($result) < 1) {
2044  return;
2045  }
2046 
2047  while ($row = $this->db->fetchAssoc($result)) {
2048  $next_id = $this->db->nextId('qpl_hints');
2049  $this->db->insert(
2050  'qpl_hints',
2051  [
2052  'qht_hint_id' => ['integer', $next_id],
2053  'qht_question_fi' => ['integer', $target_question_id],
2054  'qht_hint_index' => ['integer', $row["qht_hint_index"]],
2055  'qht_hint_points' => ['float', $row["qht_hint_points"]],
2056  'qht_hint_text' => ['text', $row["qht_hint_text"]],
2057  ]
2058  );
2059  }
2060  }
2061 
2062  protected function getRTETextWithMediaObjects(): string
2063  {
2064  // must be called in parent classes. add additional RTE text in the parent
2065  // classes and call this method to add the standard RTE text
2066  $collected = $this->getQuestion();
2067  $collected .= $this->feedbackOBJ->getGenericFeedbackContent($this->getId(), false);
2068  $collected .= $this->feedbackOBJ->getGenericFeedbackContent($this->getId(), true);
2069  $collected .= $this->feedbackOBJ->getAllSpecificAnswerFeedbackContents($this->getId());
2070 
2071  $questionHintList = ilAssQuestionHintList::getListByQuestionId($this->getId());
2072  foreach ($questionHintList as $questionHint) {
2073  /* @var $questionHint ilAssQuestionHint */
2074  $collected .= $questionHint->getText();
2075  }
2076 
2077  return $collected;
2078  }
2079 
2080  public function cleanupMediaObjectUsage(): void
2081  {
2082  $combinedtext = $this->getRTETextWithMediaObjects();
2083  ilRTE::_cleanupMediaObjectUsage($combinedtext, "qpl:html", $this->getId());
2084  }
2085 
2086  public function getInstances(): array
2087  {
2088  $result = $this->db->queryF(
2089  "SELECT question_id FROM qpl_questions WHERE original_id = %s",
2090  ["integer"],
2091  [$this->getId()]
2092  );
2093  $instances = [];
2094  $ids = [];
2095  while ($row = $this->db->fetchAssoc($result)) {
2096  $ids[] = $row["question_id"];
2097  }
2098  foreach ($ids as $question_id) {
2099  // check non random tests
2100  $result = $this->db->queryF(
2101  "SELECT tst_tests.obj_fi FROM tst_tests, tst_test_question WHERE tst_test_question.question_fi = %s AND tst_test_question.test_fi = tst_tests.test_id",
2102  ["integer"],
2103  [$question_id]
2104  );
2105  while ($row = $this->db->fetchAssoc($result)) {
2106  $instances[$row['obj_fi']] = ilObject::_lookupTitle($row['obj_fi']);
2107  }
2108  // check random tests
2109  $result = $this->db->queryF(
2110  "SELECT tst_tests.obj_fi FROM tst_tests, tst_test_rnd_qst, tst_active WHERE tst_test_rnd_qst.active_fi = tst_active.active_id AND tst_test_rnd_qst.question_fi = %s AND tst_tests.test_id = tst_active.test_fi",
2111  ["integer"],
2112  [$question_id]
2113  );
2114  while ($row = $this->db->fetchAssoc($result)) {
2115  $instances[$row['obj_fi']] = ilObject::_lookupTitle($row['obj_fi']);
2116  }
2117  }
2118  foreach ($instances as $key => $value) {
2119  $instances[$key] = ["obj_id" => $key, "title" => $value, "author" => ilObjTest::_lookupAuthor($key), "refs" => ilObject::_getAllReferences($key)];
2120  }
2121  return $instances;
2122  }
2123 
2130  public function getActiveUserData(int $active_id): array
2131  {
2132  $result = $this->db->queryF(
2133  "SELECT * FROM tst_active WHERE active_id = %s",
2134  ['integer'],
2135  [$active_id]
2136  );
2137  if ($this->db->numRows($result)) {
2138  $row = $this->db->fetchAssoc($result);
2139  return ["user_id" => $row["user_fi"], "test_id" => $row["test_fi"]];
2140  }
2141 
2142  return [];
2143  }
2144 
2145  public function hasSpecificFeedback(): bool
2146  {
2147  return static::HAS_SPECIFIC_FEEDBACK;
2148  }
2149 
2150  public static function getFeedbackClassNameByQuestionType(string $questionType): string
2151  {
2152  return str_replace('ass', 'ilAss', $questionType) . 'Feedback';
2153  }
2154 
2155 
2156 
2157  public static function instantiateQuestionGUI(int $question_id): ?assQuestionGUI
2158  {
2160  global $DIC;
2161  $ilCtrl = $DIC['ilCtrl'];
2162  $ilDB = $DIC['ilDB'];
2163  $lng = $DIC['lng'];
2164  $ilUser = $DIC['ilUser'];
2165  $ilLog = $DIC['ilLog'];
2166 
2167  if ($question_id <= 0) {
2168  $ilLog->warning('Instantiate question called without question id. (instantiateQuestionGUI@assQuestion)');
2169  throw new InvalidArgumentException('Instantiate question called without question id. (instantiateQuestionGUI@assQuestion)');
2170  }
2171 
2172  $questionrepository = QuestionPoolDIC::dic()['question.general_properties.repository'];
2173  $question_type = $questionrepository->getForQuestionId($question_id)?->getClassName();
2174 
2175  if ($question_type === null) {
2176  return null;
2177  }
2178 
2179  $question_type_gui = $question_type . 'GUI';
2180  $question_gui = new $question_type_gui($question_id);
2181 
2182  $feedback_object_classname = self::getFeedbackClassNameByQuestionType($question_type);
2183  $question = $question_gui->getObject();
2184  $question->feedbackOBJ = new $feedback_object_classname($question, $ilCtrl, $ilDB, $lng);
2185 
2186  $assSettings = new ilSetting('assessment');
2187  $processLockerFactory = new ilAssQuestionProcessLockerFactory($assSettings, $ilDB);
2188  $processLockerFactory->setQuestionId($question_gui->getObject()->getId());
2189  $processLockerFactory->setUserId($ilUser->getId());
2190  $question->setProcessLocker($processLockerFactory->getLocker());
2191  $question_gui->setObject($question);
2192 
2193  return $question_gui;
2194  }
2195 
2196  public function getNrOfTries(): int
2197  {
2198  return $this->nr_of_tries;
2199  }
2200 
2201  public function setNrOfTries(int $a_nr_of_tries): void
2202  {
2203  $this->nr_of_tries = $a_nr_of_tries;
2204  }
2205 
2206  public function setExportImagePath(string $path): void
2207  {
2208  $this->export_image_path = $path;
2209  }
2210 
2211  public static function _questionExistsInTest(int $question_id, int $test_id): bool
2212  {
2213  global $DIC;
2214  $ilDB = $DIC['ilDB'];
2215 
2216  if ($question_id < 1) {
2217  return false;
2218  }
2219 
2220  $result = $ilDB->queryF(
2221  "SELECT question_fi FROM tst_test_question WHERE question_fi = %s AND test_fi = %s",
2222  ['integer', 'integer'],
2223  [$question_id, $test_id]
2224  );
2225  return $ilDB->numRows($result) == 1;
2226  }
2227 
2228  public function formatSAQuestion($a_q): string
2229  {
2230  return $this->getSelfAssessmentFormatter()->format($a_q);
2231  }
2232 
2234  {
2235  return new \ilAssSelfAssessmentQuestionFormatter();
2236  }
2237 
2238  // scorm2004-start ???
2239 
2240  public function setPreventRteUsage(bool $prevent_rte_usage): void
2241  {
2242  $this->prevent_rte_usage = $prevent_rte_usage;
2243  }
2244 
2245  public function getPreventRteUsage(): bool
2246  {
2247  return $this->prevent_rte_usage;
2248  }
2249 
2251  {
2252  $this->lmMigrateQuestionTypeGenericContent($migrator);
2253  $this->lmMigrateQuestionTypeSpecificContent($migrator);
2254  $this->saveToDb();
2255 
2256  $this->feedbackOBJ->migrateContentForLearningModule($migrator, $this->getId());
2257  }
2258 
2260  {
2261  $this->setQuestion($migrator->migrateToLmContent($this->getQuestion()));
2262  }
2263 
2265  {
2266  // overwrite if any question type specific content except feedback needs to be migrated
2267  }
2268 
2269  public function setSelfAssessmentEditingMode(bool $selfassessmenteditingmode): void
2270  {
2271  $this->selfassessmenteditingmode = $selfassessmenteditingmode;
2272  }
2273 
2274  public function getSelfAssessmentEditingMode(): bool
2275  {
2277  }
2278 
2279  public function setDefaultNrOfTries(int $defaultnroftries): void
2280  {
2281  $this->defaultnroftries = $defaultnroftries;
2282  }
2283 
2284  public function getDefaultNrOfTries(): int
2285  {
2286  return $this->defaultnroftries;
2287  }
2288 
2289  // scorm2004-end ???
2290 
2291  public static function lookupParentObjId(int $question_id): ?int
2292  {
2293  global $DIC;
2294  $ilDB = $DIC['ilDB'];
2295 
2296  $query = "SELECT obj_fi FROM qpl_questions WHERE question_id = %s";
2297 
2298  $res = $ilDB->queryF($query, ['integer'], [$question_id]);
2299  $row = $ilDB->fetchAssoc($res);
2300 
2301  return $row['obj_fi'] ?? null;
2302  }
2303 
2304  protected function duplicateQuestionHints(int $original_question_id, int $duplicate_question_id): void
2305  {
2306  $hintIds = ilAssQuestionHintList::duplicateListForQuestion($original_question_id, $duplicate_question_id);
2307 
2309  foreach ($hintIds as $originalHintId => $duplicateHintId) {
2310  $this->ensureHintPageObjectExists($originalHintId);
2311  $originalPageObject = new ilAssHintPage($originalHintId);
2312  $originalXML = $originalPageObject->getXMLContent();
2313 
2314  $duplicatePageObject = new ilAssHintPage();
2315  $duplicatePageObject->setId($duplicateHintId);
2316  $duplicatePageObject->setParentId($this->getId());
2317  $duplicatePageObject->setXMLContent($originalXML);
2318  $duplicatePageObject->createFromXML();
2319  }
2320  }
2321  }
2322 
2323  protected function duplicateSkillAssignments(int $srcParentId, int $srcQuestionId, int $trgParentId, int $trgQuestionId): void
2324  {
2325  $assignmentList = new ilAssQuestionSkillAssignmentList($this->db);
2326  $assignmentList->setParentObjId($srcParentId);
2327  $assignmentList->setQuestionIdFilter($srcQuestionId);
2328  $assignmentList->loadFromDb();
2329 
2330  foreach ($assignmentList->getAssignmentsByQuestionId($srcQuestionId) as $assignment) {
2331  $assignment->setParentObjId($trgParentId);
2332  $assignment->setQuestionId($trgQuestionId);
2333  $assignment->saveToDb();
2334 
2335  // add skill usage
2336  $this->skillUsageService->addUsage(
2337  $trgParentId,
2338  $assignment->getSkillBaseId(),
2339  $assignment->getSkillTrefId()
2340  );
2341  }
2342  }
2343 
2344  public function syncSkillAssignments(int $srcParentId, int $srcQuestionId, int $trgParentId, int $trgQuestionId): void
2345  {
2346  $assignmentList = new ilAssQuestionSkillAssignmentList($this->db);
2347  $assignmentList->setParentObjId($trgParentId);
2348  $assignmentList->setQuestionIdFilter($trgQuestionId);
2349  $assignmentList->loadFromDb();
2350 
2351  foreach ($assignmentList->getAssignmentsByQuestionId($trgQuestionId) as $assignment) {
2352  $assignment->deleteFromDb();
2353 
2354  // remove skill usage
2355  if (!$assignment->isSkillUsed()) {
2356  $this->skillUsageService->removeUsage(
2357  $assignment->getParentObjId(),
2358  $assignment->getSkillBaseId(),
2359  $assignment->getSkillTrefId()
2360  );
2361  }
2362  }
2363 
2364  $this->duplicateSkillAssignments($srcParentId, $srcQuestionId, $trgParentId, $trgQuestionId);
2365  }
2366 
2367  public function ensureHintPageObjectExists($pageObjectId): void
2368  {
2369  if (!ilAssHintPage::_exists('qht', $pageObjectId)) {
2370  $pageObject = new ilAssHintPage();
2371  $pageObject->setParentId($this->getId());
2372  $pageObject->setId($pageObjectId);
2373  $pageObject->createFromXML();
2374  }
2375  }
2376 
2377  public function isAnswered(int $active_id, int $pass): bool
2378  {
2379  $numExistingSolutionRecords = assQuestion::getNumExistingSolutionRecords($active_id, $pass, $this->getId());
2380  return $numExistingSolutionRecords > 0;
2381  }
2382 
2383  protected static function getNumExistingSolutionRecords(int $activeId, int $pass, int $questionId): int
2384  {
2385  global $DIC;
2386  $ilDB = $DIC['ilDB'];
2387 
2388  $query = "
2389  SELECT count(active_fi) cnt
2390 
2391  FROM tst_solutions
2392 
2393  WHERE active_fi = %s
2394  AND question_fi = %s
2395  AND pass = %s
2396  ";
2397 
2398  $res = $ilDB->queryF(
2399  $query,
2400  ['integer','integer','integer'],
2401  [$activeId, $questionId, $pass]
2402  );
2403 
2404  $row = $ilDB->fetchAssoc($res);
2405 
2406  return (int) $row['cnt'];
2407  }
2408 
2409  public function getAdditionalContentEditingMode(): string
2410  {
2412  }
2413 
2414  public function setAdditionalContentEditingMode(?string $additionalContentEditingMode): void
2415  {
2416  if (!in_array($additionalContentEditingMode, $this->getValidAdditionalContentEditingModes())) {
2417  throw new ilTestQuestionPoolException('invalid additional content editing mode given: ' . $additionalContentEditingMode);
2418  }
2419 
2420  $this->additionalContentEditingMode = $additionalContentEditingMode;
2421  }
2422 
2424  {
2426  }
2427 
2428  public function isValidAdditionalContentEditingMode(string $additionalContentEditingMode): bool
2429  {
2430  if (in_array($additionalContentEditingMode, $this->getValidAdditionalContentEditingModes())) {
2431  return true;
2432  }
2433 
2434  return false;
2435  }
2436 
2437  public function getValidAdditionalContentEditingModes(): array
2438  {
2439  return [
2440  self::ADDITIONAL_CONTENT_EDITING_MODE_RTE,
2441  self::ADDITIONAL_CONTENT_EDITING_MODE_IPE
2442  ];
2443  }
2444 
2449  {
2450  return ilHtmlPurifierFactory::getInstanceByType('qpl_usersolution');
2451  }
2452 
2457  {
2458  return ilHtmlPurifierFactory::getInstanceByType('qpl_usersolution');
2459  }
2460 
2461  protected function buildQuestionDataQuery(): string
2462  {
2463  return "
2464  SELECT qpl_questions.*,
2465  {$this->getAdditionalTableName()}.*
2466  FROM qpl_questions
2467  LEFT JOIN {$this->getAdditionalTableName()}
2468  ON {$this->getAdditionalTableName()}.question_fi = qpl_questions.question_id
2469  WHERE qpl_questions.question_id = %s
2470  ";
2471  }
2472 
2473  public function setLastChange(int $lastChange): void
2474  {
2475  $this->lastChange = $lastChange;
2476  }
2477 
2478  public function getLastChange(): ?int
2479  {
2480  return $this->lastChange;
2481  }
2482 
2483  protected function getCurrentSolutionResultSet(int $active_id, int $pass, bool $authorized = true): \ilDBStatement
2484  {
2485  if ($this->getStep() !== null) {
2486  $query = "
2487  SELECT *
2488  FROM tst_solutions
2489  WHERE active_fi = %s
2490  AND question_fi = %s
2491  AND pass = %s
2492  AND step = %s
2493  AND authorized = %s
2494  ";
2495 
2496  return $this->db->queryF(
2497  $query,
2498  ['integer', 'integer', 'integer', 'integer', 'integer'],
2499  [$active_id, $this->getId(), $pass, $this->getStep(), (int) $authorized]
2500  );
2501  }
2502 
2503  $query = "
2504  SELECT *
2505  FROM tst_solutions
2506  WHERE active_fi = %s
2507  AND question_fi = %s
2508  AND pass = %s
2509  AND authorized = %s
2510  ";
2511 
2512  return $this->db->queryF(
2513  $query,
2514  ['integer', 'integer', 'integer', 'integer'],
2515  [$active_id, $this->getId(), $pass, (int) $authorized]
2516  );
2517  }
2518 
2519  protected function removeSolutionRecordById(int $solutionId): int
2520  {
2521  return $this->db->manipulateF(
2522  "DELETE FROM tst_solutions WHERE solution_id = %s",
2523  ['integer'],
2524  [$solutionId]
2525  );
2526  }
2527 
2528  // hey: prevPassSolutions - selected file reuse, copy solution records
2532  protected function getSolutionRecordById(int $solutionId): array
2533  {
2534  $result = $this->db->queryF(
2535  "SELECT * FROM tst_solutions WHERE solution_id = %s",
2536  ['integer'],
2537  [$solutionId]
2538  );
2539 
2540  if ($this->db->numRows($result) > 0) {
2541  return $this->db->fetchAssoc($result);
2542  }
2543  return [];
2544  }
2545  // hey.
2546 
2547  public function removeIntermediateSolution(int $active_id, int $pass): void
2548  {
2549  $this->getProcessLocker()->executeUserSolutionUpdateLockOperation(function () use ($active_id, $pass) {
2550  $this->removeCurrentSolution($active_id, $pass, false);
2551  });
2552  }
2553 
2557  public function removeCurrentSolution(int $active_id, int $pass, bool $authorized = true): int
2558  {
2559  if ($this->getStep() !== null) {
2560  $query = '
2561  DELETE FROM tst_solutions
2562  WHERE active_fi = %s
2563  AND question_fi = %s
2564  AND pass = %s
2565  AND step = %s
2566  AND authorized = %s
2567  ';
2568 
2569  return $this->db->manipulateF(
2570  $query,
2571  ['integer', 'integer', 'integer', 'integer', 'integer'],
2572  [$active_id, $this->getId(), $pass, $this->getStep(), (int) $authorized]
2573  );
2574  }
2575 
2576  $query = "
2577  DELETE FROM tst_solutions
2578  WHERE active_fi = %s
2579  AND question_fi = %s
2580  AND pass = %s
2581  AND authorized = %s
2582  ";
2583 
2584  return $this->db->manipulateF(
2585  $query,
2586  ['integer', 'integer', 'integer', 'integer'],
2587  [$active_id, $this->getId(), $pass, (int) $authorized]
2588  );
2589  }
2590 
2591  // fau: testNav - add timestamp as parameter to saveCurrentSolution
2592  public function saveCurrentSolution(int $active_id, int $pass, $value1, $value2, bool $authorized = true, $tstamp = 0): int
2593  {
2594  $next_id = $this->db->nextId("tst_solutions");
2595 
2596  $fieldData = [
2597  "solution_id" => ["integer", $next_id],
2598  "active_fi" => ["integer", $active_id],
2599  "question_fi" => ["integer", $this->getId()],
2600  "value1" => ["clob", $value1],
2601  "value2" => ["clob", $value2],
2602  "pass" => ["integer", $pass],
2603  "tstamp" => ["integer", ((int) $tstamp > 0) ? (int) $tstamp : time()],
2604  'authorized' => ['integer', (int) $authorized]
2605  ];
2606 
2607  if ($this->getStep() !== null) {
2608  $fieldData['step'] = ["integer", $this->getStep()];
2609  }
2610 
2611  return $this->db->insert("tst_solutions", $fieldData);
2612  }
2613  // fau.
2614 
2615  public function updateCurrentSolution(int $solutionId, $value1, $value2, bool $authorized = true): int
2616  {
2617  $fieldData = [
2618  "value1" => ["clob", $value1],
2619  "value2" => ["clob", $value2],
2620  "tstamp" => ["integer", time()],
2621  'authorized' => ['integer', (int) $authorized]
2622  ];
2623 
2624  if ($this->getStep() !== null) {
2625  $fieldData['step'] = ["integer", $this->getStep()];
2626  }
2627 
2628  return $this->db->update("tst_solutions", $fieldData, [
2629  'solution_id' => ['integer', $solutionId]
2630  ]);
2631  }
2632 
2633  // fau: testNav - added parameter to keep the timestamp (default: false)
2634  public function updateCurrentSolutionsAuthorization(int $activeId, int $pass, bool $authorized, bool $keepTime = false): int
2635  {
2636  $fieldData = [
2637  'authorized' => ['integer', (int) $authorized]
2638  ];
2639 
2640  if (!$keepTime) {
2641  $fieldData['tstamp'] = ['integer', time()];
2642  }
2643 
2644  $whereData = [
2645  'question_fi' => ['integer', $this->getId()],
2646  'active_fi' => ['integer', $activeId],
2647  'pass' => ['integer', $pass]
2648  ];
2649 
2650  if ($this->getStep() !== null) {
2651  $whereData['step'] = ["integer", $this->getStep()];
2652  }
2653 
2654  return $this->db->update('tst_solutions', $fieldData, $whereData);
2655  }
2656  // fau.
2657 
2658  // hey: prevPassSolutions - motivation slowly decreases on imagemap
2660 
2661  public static function implodeKeyValues(array $keyValues): string
2662  {
2663  return implode(assQuestion::KEY_VALUES_IMPLOSION_SEPARATOR, $keyValues);
2664  }
2665 
2666  public static function explodeKeyValues(string $keyValues): array
2667  {
2668  return explode(assQuestion::KEY_VALUES_IMPLOSION_SEPARATOR, $keyValues);
2669  }
2670 
2671  protected function deleteDummySolutionRecord(int $activeId, int $passIndex): void
2672  {
2673  foreach ($this->getSolutionValues($activeId, $passIndex, false) as $solutionRec) {
2674  if ($solutionRec['value1'] == '' && $solutionRec['value2'] == '') {
2675  $this->removeSolutionRecordById($solutionRec['solution_id']);
2676  }
2677  }
2678  }
2679 
2680  protected function isDummySolutionRecord(array $solutionRecord): bool
2681  {
2682  return !strlen($solutionRecord['value1']) && !strlen($solutionRecord['value2']);
2683  }
2684 
2685  protected function deleteSolutionRecordByValues(int $activeId, int $passIndex, bool $authorized, array $matchValues): void
2686  {
2687  $types = ["integer", "integer", "integer", "integer"];
2688  $values = [$activeId, $this->getId(), $passIndex, (int) $authorized];
2689  $valuesCondition = [];
2690 
2691  foreach ($matchValues as $valueField => $value) {
2692  switch ($valueField) {
2693  case 'value1':
2694  case 'value2':
2695  $valuesCondition[] = "{$valueField} = %s";
2696  $types[] = 'text';
2697  $values[] = $value;
2698  break;
2699 
2700  default:
2701  throw new ilTestQuestionPoolException('invalid value field given: ' . $valueField);
2702  }
2703  }
2704 
2705  $valuesCondition = implode(' AND ', $valuesCondition);
2706 
2707  $query = "
2708  DELETE FROM tst_solutions
2709  WHERE active_fi = %s
2710  AND question_fi = %s
2711  AND pass = %s
2712  AND authorized = %s
2713  AND $valuesCondition
2714  ";
2715 
2716  if ($this->getStep() !== null) {
2717  $query .= " AND step = %s ";
2718  $types[] = 'integer';
2719  $values[] = $this->getStep();
2720  }
2721 
2722  $this->db->manipulateF($query, $types, $values);
2723  }
2724 
2725  protected function duplicateIntermediateSolutionAuthorized(int $activeId, int $passIndex): void
2726  {
2727  foreach ($this->getSolutionValues($activeId, $passIndex, false) as $rec) {
2728  $this->saveCurrentSolution($activeId, $passIndex, $rec['value1'], $rec['value2'], true, $rec['tstamp']);
2729  }
2730  }
2731 
2732  protected function forceExistingIntermediateSolution(int $activeId, int $passIndex, bool $considerDummyRecordCreation): void
2733  {
2734  $intermediateSolution = $this->getSolutionValues($activeId, $passIndex, false);
2735 
2736  if (!count($intermediateSolution)) {
2737  $this->updateCurrentSolutionsAuthorization($activeId, $passIndex, false, true);
2738 
2739  // create a backup as authorized solution again (keeping timestamps)
2740  $this->duplicateIntermediateSolutionAuthorized($activeId, $passIndex);
2741 
2742  if ($considerDummyRecordCreation) {
2743  // create an additional dummy record to indicate the existence of an intermediate solution
2744  // even if all entries are deleted from the intermediate solution later
2745  $this->saveCurrentSolution($activeId, $passIndex, null, null, false);
2746  }
2747  }
2748  }
2749  // hey.
2750 
2754  public function setStep($step): void
2755  {
2756  $this->step = $step;
2757  }
2758 
2762  public function getStep(): ?int
2763  {
2764  return $this->step;
2765  }
2766 
2767  public static function convertISO8601FormatH_i_s_ExtendedToSeconds(string $time): int
2768  {
2769  $sec = 0;
2770  $time_array = explode(':', $time);
2771  if (count($time_array) == 3) {
2772  $sec += (int) $time_array[0] * 3600;
2773  $sec += (int) $time_array[1] * 60;
2774  $sec += (int) $time_array[2];
2775  }
2776  return $sec;
2777  }
2778 
2779  public function toJSON(): string
2780  {
2781  return json_encode([]);
2782  }
2783 
2784  // hey: prevPassSolutions - check for authorized solution
2785  public function intermediateSolutionExists(int $active_id, int $pass): bool
2786  {
2787  $solutionAvailability = $this->lookupForExistingSolutions($active_id, $pass);
2788  return (bool) $solutionAvailability['intermediate'];
2789  }
2790 
2791  public function authorizedSolutionExists(int $active_id, ?int $pass): bool
2792  {
2793  if ($pass === null) {
2794  return false;
2795  }
2796  $solutionAvailability = $this->lookupForExistingSolutions($active_id, $pass);
2797  return (bool) $solutionAvailability['authorized'];
2798  }
2799 
2800  public function authorizedOrIntermediateSolutionExists(int $active_id, int $pass): bool
2801  {
2802  $solutionAvailability = $this->lookupForExistingSolutions($active_id, $pass);
2803  return $solutionAvailability['authorized'] || $solutionAvailability['intermediate'];
2804  }
2805  // hey.
2806 
2807  protected function lookupMaxStep(int $active_id, int $pass): int
2808  {
2809  $result = $this->db->queryF(
2810  "SELECT MAX(step) max_step FROM tst_solutions WHERE active_fi = %s AND pass = %s AND question_fi = %s",
2811  ["integer", "integer", "integer"],
2812  [$active_id, $pass, $this->getId()]
2813  );
2814 
2815  $row = $this->db->fetchAssoc($result);
2816 
2817  return (int) $row['max_step'];
2818  }
2819 
2820  // fau: testNav - new function lookupForExistingSolutions
2825  public function lookupForExistingSolutions(int $activeId, int $pass): array
2826  {
2827  $return = [
2828  'authorized' => false,
2829  'intermediate' => false
2830  ];
2831 
2832  $query = "
2833  SELECT authorized, COUNT(*) cnt
2834  FROM tst_solutions
2835  WHERE active_fi = %s
2836  AND question_fi = %s
2837  AND pass = %s
2838  ";
2839 
2840  if ($this->getStep() !== null) {
2841  $query .= " AND step = " . $this->db->quote((int) $this->getStep(), 'integer') . " ";
2842  }
2843 
2844  $query .= "
2845  GROUP BY authorized
2846  ";
2847 
2848  $result = $this->db->queryF($query, ['integer', 'integer', 'integer'], [$activeId, $this->getId(), $pass]);
2849 
2850  while ($row = $this->db->fetchAssoc($result)) {
2851  if ($row['authorized']) {
2852  $return['authorized'] = $row['cnt'] > 0;
2853  } else {
2854  $return['intermediate'] = $row['cnt'] > 0;
2855  }
2856  }
2857  return $return;
2858  }
2859  // fau.
2860 
2861  public function isAddableAnswerOptionValue(int $qIndex, string $answerOptionValue): bool
2862  {
2863  return false;
2864  }
2865 
2866  public function addAnswerOptionValue(int $qIndex, string $answerOptionValue, float $points): void
2867  {
2868  }
2869 
2870  public function removeAllExistingSolutions(): void
2871  {
2872  $query = "DELETE FROM tst_solutions WHERE question_fi = %s";
2873  $this->db->manipulateF($query, ['integer'], [$this->getId()]);
2874  }
2875 
2876  public function removeExistingSolutions(int $activeId, int $pass): int
2877  {
2878  $query = "
2879  DELETE FROM tst_solutions
2880  WHERE active_fi = %s
2881  AND question_fi = %s
2882  AND pass = %s
2883  ";
2884 
2885  if ($this->getStep() !== null) {
2886  $query .= " AND step = " . $this->db->quote((int) $this->getStep(), 'integer') . " ";
2887  }
2888 
2889  return $this->db->manipulateF(
2890  $query,
2891  ['integer', 'integer', 'integer'],
2892  [$activeId, $this->getId(), $pass]
2893  );
2894  }
2895 
2896  public function resetUsersAnswer(int $activeId, int $pass): void
2897  {
2898  $this->removeExistingSolutions($activeId, $pass);
2899  $this->removeResultRecord($activeId, $pass);
2900 
2901  $test = new ilObjTest(
2902  $this->test_id,
2903  false
2904  );
2905  $test->updateTestPassResults(
2906  $activeId,
2907  $pass,
2908  $this->getProcessLocker(),
2909  $this->getTestId()
2910  );
2911  }
2912 
2913  public function removeResultRecord(int $activeId, int $pass): int
2914  {
2915  $query = "
2916  DELETE FROM tst_test_result
2917  WHERE active_fi = %s
2918  AND question_fi = %s
2919  AND pass = %s
2920  ";
2921 
2922  if ($this->getStep() !== null) {
2923  $query .= " AND step = " . $this->db->quote((int) $this->getStep(), 'integer') . " ";
2924  }
2925 
2926  return $this->db->manipulateF(
2927  $query,
2928  ['integer', 'integer', 'integer'],
2929  [$activeId, $this->getId(), $pass]
2930  );
2931  }
2932 
2933  public function fetchValuePairsFromIndexedValues(array $indexedValues): array
2934  {
2935  $valuePairs = [];
2936 
2937  foreach ($indexedValues as $value1 => $value2) {
2938  $valuePairs[] = ['value1' => $value1, 'value2' => $value2];
2939  }
2940 
2941  return $valuePairs;
2942  }
2943 
2944  public function fetchIndexedValuesFromValuePairs(array $value_pairs): array
2945  {
2946  $indexed_values = [];
2947 
2948  foreach ($value_pairs as $valuePair) {
2949  $indexed_values[$valuePair['value1']] = $valuePair['value2'];
2950  }
2951 
2952  return $indexed_values;
2953  }
2954 
2955  public function updateTimestamp(): void
2956  {
2957  $this->db->manipulateF(
2958  "UPDATE qpl_questions SET tstamp = %s WHERE question_id = %s",
2959  ['integer', 'integer'],
2960  [time(), $this->getId()]
2961  );
2962  }
2963 
2965  {
2966  if ($this->test_question_config === null) {
2967  $this->test_question_config = $this->buildTestPresentationConfig();
2968  }
2969 
2971  }
2972 
2974  {
2975  return new ilTestQuestionConfig();
2976  }
2977 
2979  {
2981  }
2982 
2983  protected function loadSuggestedSolutions(): array
2984  {
2985  $question_id = $this->getId();
2986  return $this->getSuggestedSolutionsRepo()->selectFor($question_id);
2987  }
2988 
3003  public static function extendedTrim(string $value): string
3004  {
3005  return preg_replace(self::TRIM_PATTERN, '', $value);
3006  }
3007 
3008  public function hasWritableOriginalInQuestionPool(): bool
3009  {
3010  return !is_null($this->original_id)
3011  && $this->questionrepository->questionExistsInPool($this->original_id)
3012  && assQuestion::instantiateQuestion($this->original_id)->isWriteable();
3013  }
3014 
3016  AdditionalInformationGenerator $additional_info,
3017  int $test_ref_id,
3018  int $active_id,
3019  int $pass,
3020  string $source_ip,
3021  TestParticipantInteractionTypes $interaction_type
3023  return new TestParticipantInteraction(
3024  $test_ref_id,
3025  $this->id,
3026  $this->current_user->getId(),
3027  $source_ip,
3028  $interaction_type,
3029  time(),
3030  $this->answerToLog($additional_info, $active_id, $pass)
3031  );
3032  }
3033 
3035  AdditionalInformationGenerator $additional_info,
3036  int $test_ref_id,
3040  $test_ref_id,
3041  $this->id,
3042  $this->current_user->getId(),
3043  $interaction_type,
3044  time(),
3045  $this->toLog($additional_info)
3046  );
3047  }
3048 
3049  protected function answerToLog(
3050  AdditionalInformationGenerator $additional_info,
3051  int $active_id,
3052  int $pass
3053  ): array {
3054  return [
3055  AdditionalInformationGenerator::KEY_PASS => $pass,
3056  AdditionalInformationGenerator::KEY_REACHED_POINTS => $this->getReachedPoints($active_id, $pass),
3057  AdditionalInformationGenerator::KEY_PAX_ANSWER => $this->solutionValuesToLog(
3058  $additional_info,
3059  $this->getSolutionValues($active_id, $pass)
3060  )
3061  ];
3062  }
3063 
3064  public function getSolutionForTextOutput(
3065  int $active_id,
3066  int $pass
3067  ): array|string {
3068  return $this->solutionValuesToText(
3069  $this->getSolutionValues($active_id, $pass)
3070  );
3071  }
3072 
3074  int $active_id,
3075  int $pass
3076  ): array|string {
3077  return $this->solutionValuesToText(
3078  $this->getSolutionValues($active_id, $pass)
3079  );
3080  }
3081 
3082  public function getVariablesAsTextArray(
3083  int $active_id,
3084  int $pass
3085  ): array {
3086  return [];
3087  }
3088 }
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...
loadFromDb(int $question_id)
const ADDITIONAL_CONTENT_EDITING_MODE_IPE
ilLanguage $lng
cloneXHTMLMediaObjectsOfQuestion(int $source_question_id)
setNrOfTries(int $a_nr_of_tries)
ilTestQuestionConfig $testQuestionConfig
ilAssQuestionFeedback $feedbackOBJ
syncSkillAssignments(int $srcParentId, int $srcQuestionId, int $trgParentId, int $trgQuestionId)
ilGlobalPageTemplate $tpl
static getListByQuestionId($questionId)
instantiates a question hint list for the passed question id
deletePageOfQuestion(int $question_id)
getSolutionValues(int $active_id, ?int $pass=null, bool $authorized=true)
Loads solutions of a given user from the database an returns it.
$res
Definition: ltiservices.php:66
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
SuggestedSolutionsDatabaseRepository $suggestedsolution_repo
authorizedOrIntermediateSolutionExists(int $active_id, int $pass)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
const MAXIMUM_THUMB_SIZE
getActiveUserData(int $active_id)
Returns the user id and the test id for a given active id.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
migrateContentForLearningModule(ilAssSelfAssessmentMigrator $migrator)
toXML(bool $a_include_header=true, bool $a_include_binary=true, bool $a_shuffle=false, bool $test_output=false, bool $force_image_references=false)
Returns a QTI xml representation of the question.
static _isWriteable($object_id, $user_id)
Returns true, if the question pool is writeable by a given user.
calculateResultsFromSolution(int $active_id, int $pass)
Calculates the question results from a previously saved question solution.
static _getPass($active_id)
Retrieves the actual pass of a given user for a given test.
const IL_INST_ID
Definition: constants.php:40
ilTestQuestionConfig $test_question_config
static _setReachedPoints(int $active_id, int $question_id, float $points, float $maxpoints, int $pass, bool $manualscoring)
Sets the points, a learner has reached answering the question Additionally objective results are upda...
isPreviewSolutionCorrect(ilAssQuestionPreviewSession $preview_session)
answerToLog(AdditionalInformationGenerator $additional_info, int $active_id, int $pass)
removeResultRecord(int $activeId, int $pass)
deleteAdditionalTableData(int $question_id)
deleteAnswers(int $question_id)
cloneHints(int $source_question_id, int $target_question_id)
duplicate(bool $for_test=true, string $title='', string $author='', int $owner=-1, $test_obj_id=null)
static _getSuggestedSolutionOutput(int $question_id)
updateCurrentSolutionsAuthorization(int $activeId, int $pass, bool $authorized, bool $keepTime=false)
onDuplicate(int $original_parent_id, int $original_question_id, int $duplicate_parent_id, int $duplicate_question_id)
resetUsersAnswer(int $activeId, int $pass)
static isFileAvailable(string $file)
removeAllImageFiles(string $image_target_path)
setProcessLocker(ilAssQuestionProcessLocker $processLocker)
Provides fluid interface to LoggingServices.
Interface Observer Contains several chained tasks and infos about them.
setOwner(int $owner=-1)
cloneSuggestedSolutionFiles(int $source_question_id, int $target_question_id)
ensureNonNegativePoints(float $points)
static _getAllReferences(int $id)
get all reference ids for object ID
ilAssQuestionProcessLocker $processLocker
ensureHintPageObjectExists($pageObjectId)
Refinery $refinery
setSelfAssessmentEditingMode(bool $selfassessmenteditingmode)
SkillUsageService $skillUsageService
isDummySolutionRecord(array $solutionRecord)
solutionValuesToText(array $solution_values)
MUST convert the given solution values into text.
toLog(AdditionalInformationGenerator $additional_info)
MUST return an array of the question settings that can be stored in the log.
adjustReachedPointsByScoringOptions(float $points, int $active_id)
Adjust the given reached points by checks for all special scoring options in the test container...
createNewOriginalFromThisDuplicate(int $target_parent_id, string $target_question_title='')
static _saveUsage(int $a_mob_id, string $a_type, int $a_id, int $a_usage_hist_nr=0, string $a_lang="-")
Save usage of mob within another container (e.g.
getAdditionalTableName()
static isForcePassResultUpdateEnabled()
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
getImagePathWeb()
Returns the web image path for web accessable images of a question.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setThumbSize(int $a_size)
static deleteHintsByQuestionIds(array $questionIds)
Deletes all question hints relating to questions included in given question ids.
static getUsageOfObject(int $a_obj_id, bool $a_include_titles=false)
cloneQuestionTypeSpecificProperties(self $target)
lmMigrateQuestionTypeSpecificContent(ilAssSelfAssessmentMigrator $migrator)
static getNumExistingSolutionRecords(int $activeId, int $pass, int $questionId)
duplicateSkillAssignments(int $srcParentId, int $srcQuestionId, int $trgParentId, int $trgQuestionId)
cloneSuggestedSolutions(int $source_question_id, int $target_question_id)
RequestDataCollector $questionpool_request
static _cleanupMediaObjectUsage(string $a_text, string $a_usage_type, int $a_usage_id)
Synchronises appearances of media objects in $a_text with media object usage table.
persistWorkingState(int $active_id, $pass, bool $authorized=true)
persists the working state for current testactive and testpass
importHint(int $question_id, array $hint_array)
static prepareFormOutput($a_str, bool $a_strip=false)
QuestionFiles $question_files
static setTokenMaxLifetimeInSeconds(int $token_max_lifetime_in_seconds)
persistPreviewState(ilAssQuestionPreviewSession $preview_session)
persists the preview state for current user and question
setPreventRteUsage(bool $prevent_rte_usage)
static makeDirParents(string $a_dir)
Create a new directory and all parent directories.
getSuggestedSolution(int $subquestion_index=0)
duplicateQuestionHints(int $original_question_id, int $duplicate_question_id)
setComment(string $comment="")
static _questionExistsInTest(int $question_id, int $test_id)
string $questionActionCmd
getValidAdditionalContentEditingModes()
HTTPServices $http
Customizing of pimple-DIC for ILIAS.
Definition: Container.php:35
static _lookupTestObjIdForQuestionId(int $q_id)
Get test Object ID for question ID.
getUserSolutionPreferingIntermediate(int $active_id, ?int $pass=null)
afterSyncWithOriginal(int $original_question_id, int $clone_question_id, int $original_parent_id, int $clone_parent_id)
fromXML(string $importdirectory, int $user_id, ilQTIItem $item, int $questionpool_id, ?int $tst_id, ?ilObject &$tst_object, int &$question_counter, array $import_mapping, array $solutionhints=[])
bool $selfassessmenteditingmode
$path
Definition: ltiservices.php:29
static implodeKeyValues(array $keyValues)
solutionValuesToLog(AdditionalInformationGenerator $additional_info, array $solution_values)
MUST convert the given solution values into an array or a string that can be stored in the log...
setExportImagePath(string $path)
static removeTrailingPathSeparators(string $path)
Repository internal data service.
calculateReachedPoints(int $active_id, ?int $pass=null, bool $authorized_solution=true)
fixUnavailableSkinImageSources(string $html)
isAddableAnswerOptionValue(int $qIndex, string $answerOptionValue)
updateCurrentSolution(int $solutionId, $value1, $value2, bool $authorized=true)
static getASCIIFilename(string $a_filename)
deleteDummySolutionRecord(int $activeId, int $passIndex)
getVariablesAsTextArray(int $active_id, int $pass)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
removeIntermediateSolution(int $active_id, int $pass)
static _getUserIdFromActiveId(int $active_id)
LoggingServices $log
static resetOriginalId(int $questionId)
resolveInternalLink(string $internal_link)
static setForcePassResultUpdateEnabled(bool $force_pass_results_update_enabled)
ilAssQuestionPage $page
static instantiateQuestion(int $question_id)
Interface for html sanitizing functionality.
saveCurrentSolution(int $active_id, int $pass, $value1, $value2, bool $authorized=true, $tstamp=0)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
purifyAndPrepareTextAreaOutput(string $content)
setTestId(int $id=-1)
static http()
Fetches the global http state from ILIAS.
clonePageOfQuestion(int $a_q_id)
getImagePath($question_id=null, $object_id=null)
Returns the image path for web accessable images of a question.
duplicateComments(int $parent_source_id, int $source_id, int $parent_target_id, int $target_id)
buildHashedImageFilename(string $plain_image_filename, bool $unique=false)
static _lookupTitle(int $obj_id)
lookupForExistingSolutions(int $activeId, int $pass)
Lookup if an authorized or intermediate solution exists.
Transformation $shuffler
copyObject(int $target_parent_id, string $title='')
getAdjustedReachedPoints(int $active_id, int $pass, bool $authorized_solution=true)
static _exists(string $a_parent_type, int $a_id, string $a_lang="", bool $a_no_cache=false)
Checks whether page exists.
answerToParticipantInteraction(AdditionalInformationGenerator $additional_info, int $test_ref_id, int $active_id, int $pass, string $source_ip, TestParticipantInteractionTypes $interaction_type)
removeSolutionRecordById(int $solutionId)
static _getReachedPoints(int $active_id, int $question_id, int $pass)
saveWorkingData(int $active_id, ?int $pass=null, bool $authorized=true)
getCorrectSolutionForTextOutput(int $active_id, int $pass)
static delDir(string $a_dir, bool $a_clean_only=false)
removes a dir and all its content (subdirs and files) recursively
setLastChange(int $lastChange)
const MINIMUM_THUMB_SIZE
getSolutionForTextOutput(int $active_id, int $pass)
static _getIdForImportId(string $a_import_id)
get current object id for import id (static)
static _getSolutionMaxPass(int $question_id, int $active_id)
Returns the maximum pass a users question solution.
deductHintPointsFromReachedPoints(ilAssQuestionPreviewSession $preview_session, $reached_points)
intermediateSolutionExists(int $active_id, int $pass)
static _getCountSystem($active_id)
global $DIC
Definition: shib_login.php:22
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 getInstanceByType(string $type)
const CLIENT_WEB_DIR
Definition: constants.php:47
static _lookupAuthor($obj_id)
Gets the authors name of the ilObjTest object.
setPoints(float $points)
setObjId(int $obj_id=0)
saveQuestionDataToDb(?int $original_id=null)
static _exists(int $id, bool $reference=false, ?string $type=null)
removeExistingSolutions(int $activeId, int $pass)
setDefaultNrOfTries(int $defaultnroftries)
saveToDb(?int $original_id=null)
static _getMobsOfObject(string $a_type, int $a_id, int $a_usage_hist_nr=0, string $a_lang="-")
fetchIndexedValuesFromValuePairs(array $value_pairs)
static saveOriginalId(int $questionId, int $originalId)
static _removeUsage(int $a_mob_id, string $a_type, int $a_id, int $a_usage_hist_nr=0, string $a_lang="-")
Remove usage of mob in another container.
const DEFAULT_THUMB_SIZE
static _updateQuestionCount(int $object_id)
savePreviewData(ilAssQuestionPreviewSession $preview_session)
ilAssQuestionLifecycle $lifecycle
fetchValuePairsFromIndexedValues(array $indexedValues)
createNewQuestion(bool $a_create_page=true)
Creates a new question without an owner when a new question is created This assures that an ID is giv...
static duplicateListForQuestion($originalQuestionId, $duplicateQuestionId)
duplicates a hint list from given original question id to given duplicate question id and returns an ...
isAdditionalContentEditingModePageObject()
duplicateIntermediateSolutionAuthorized(int $activeId, int $passIndex)
const KEY_VALUES_IMPLOSION_SEPARATOR
toQuestionAdministrationInteraction(AdditionalInformationGenerator $additional_info, int $test_ref_id, TestQuestionAdministrationInteractionTypes $interaction_type)
static $imageSourceFixReplaceMap
static _getScoreCutting(int $active_id)
Determines if the score of a question should be cut at 0 points or the score of the whole test...
getSolutionMaxPass(int $active_id)
isValidAdditionalContentEditingMode(string $additionalContentEditingMode)
static extendedTrim(string $value)
Trim non-printable characters from the beginning and end of a string.
removeCurrentSolution(int $active_id, int $pass, bool $authorized=true)
array $suggested_solutions
lmMigrateQuestionTypeGenericContent(ilAssSelfAssessmentMigrator $migrator)
static lookupParentObjId(int $question_id)
setId(int $id=-1)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
ilDBInterface $db
__construct(Container $dic, ilPlugin $plugin)
onCopy(int $sourceParentId, int $sourceQuestionId, int $targetParentId, int $targetQuestionId)
ilObjUser $current_user
const ADDITIONAL_CONTENT_EDITING_MODE_RTE
setOriginalId(?int $original_id)
setExternalId(?string $external_id)
getTestOutputSolutions(int $activeId, int $pass)
setShuffler(Transformation $shuffler)
static convertISO8601FormatH_i_s_ExtendedToSeconds(string $time)
setTitle(string $title="")
deleteSolutionRecordByValues(int $activeId, int $passIndex, bool $authorized, array $matchValues)
A transformation is a function from one datatype to another.
getAnswerTableName()
static signFile(string $path_to_file)
authorizedSolutionExists(int $active_id, ?int $pass)
setLifecycle(ilAssQuestionLifecycle $lifecycle)
comment()
description: > Example for rendring a comment glyph.
Definition: comment.php:41
getCurrentSolutionResultSet(int $active_id, int $pass, bool $authorized=true)
getReachedPoints(int $active_id, int $pass)
isAnswered(int $active_id, int $pass)
addAnswerOptionValue(int $qIndex, string $answerOptionValue, float $points)
forceExistingIntermediateSolution(int $activeId, int $passIndex, bool $considerDummyRecordCreation)
static $force_pass_results_update_enabled
static _updateObjectiveResult(int $a_user_id, int $a_active_id, int $a_question_id)
GeneralQuestionPropertiesRepository $questionrepository
static fixSvgToPng(string $imageFilenameContainingString)
const HAS_SPECIFIC_FEEDBACK
lookupMaxStep(int $active_id, int $pass)
calculateReachedPointsFromPreviewSession(ilAssQuestionPreviewSession $preview_session)
setAuthor(string $author="")
isNonEmptyItemListPostSubmission(string $post_submission_field_name)
static prepareTextareaOutput(string $txt_output, bool $prepare_for_latex_output=false, bool $omitNl2BrWhenTextArea=false)
Prepares a string for a text area output where latex code may be in it If the text is HTML-free...
duplicateSuggestedSolutionFiles(int $parent_id, int $question_id)
Duplicates the files of a suggested solution if the question is duplicated.
string $additionalContentEditingMode
$service
Definition: ltiservices.php:40
copySuggestedSolutions(int $target_question_id)
getSolutionRecordById(int $solutionId)
setShuffle(?bool $shuffle=true)
setAdditionalContentEditingMode(?string $additionalContentEditingMode)
static explodeKeyValues(string $keyValues)
getInternalLinkHref(string $target)
static getFeedbackClassNameByQuestionType(string $questionType)
setQuestion(string $question="")
string $export_image_path