19 declare(strict_types=1);
86 protected \ilAssQuestionPage
$page;
118 string $comment =
"",
121 string $question =
"" 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'];
140 $this->
http = $DIC->http();
143 $this->thumb_size = self::DEFAULT_THUMB_SIZE;
154 $this->suggested_solutions = [];
155 $this->shuffle =
true;
156 $this->nr_of_tries = 0;
159 $this->questionActionCmd =
'handleQuestionAction';
160 $this->export_image_path =
'';
161 $this->shuffler = $DIC->refinery()->random()->dontShuffle();
163 $this->skillUsageService = $DIC->skills()->usage();
170 abstract public function saveWorkingData(
int $active_id, ?
int $pass =
null,
bool $authorized =
true): bool;
175 bool $authorized_solution =
true 201 array $solution_values
209 array $solution_values
219 return self::$force_pass_results_update_enabled;
224 return $this->questionpool_request->getCmdIndex($this->questionActionCmd) ??
'';
229 return !empty($this->questionpool_request->strArray($post_submission_field_name));
258 string $importdirectory,
261 int $questionpool_id,
264 int &$question_counter,
265 array $import_mapping,
266 array $solutionhints = []
269 $import =
new $classname($this);
270 $new_import_mapping = $import->fromXML(
281 foreach ($solutionhints as $hint) {
282 $this->
importHint($import->getQuestionId(), $hint);
285 return $new_import_mapping;
288 private function importHint(
int $question_id, array $hint_array): void
291 $hint->setQuestionId($question_id);
292 $hint->setIndex($hint_array[
'index'] ??
'');
293 $hint->setPoints($hint_array[
'points'] ??
'');
297 $hint_page->setParentId($question_id);
298 $hint_page->setId($hint->getId());
299 $hint_page->setXMLContent($hint_array[
'txt']);
300 $hint_page->createFromXML();
304 $hint->setText($hint_array[
'txt'] ??
'');
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 321 $export =
new $classname($this);
322 return $export->toXML($a_include_header, $a_include_binary, $a_shuffle, $test_output, $force_image_references);
330 public function setId(
int $id = -1): void
337 $this->test_id =
$id;
347 $this->shuffle = $shuffle ??
false;
352 if ($author ===
'') {
353 $author = $this->current_user->getFullname();
370 return $this->
refinery->encode()->htmlSpecialCharsAsEntities()->transform($this->title);
400 return $this->
refinery->encode()->htmlSpecialCharsAsEntities()->transform($this->
comment);
410 if ($a_size >= self::MINIMUM_THUMB_SIZE) {
411 $this->thumb_size = $a_size;
413 throw new ilException(
"Thumb size must be at least " . self::MINIMUM_THUMB_SIZE .
"px");
419 return self::MINIMUM_THUMB_SIZE;
424 return self::MAXIMUM_THUMB_SIZE;
434 return $this->
refinery->string()->stripTags()->transform($this->author);
469 if ($this->external_id ===
null || $this->external_id ===
'') {
470 if ($this->
getId() > 0) {
473 return uniqid(
'',
true);
480 $question = self::instantiateQuestion($question_id);
481 if (!is_object($question)) {
484 return $question->getSuggestedSolutionOutput();
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="' 499 . $this->
lng->txt(
"solution_hint")
503 case SuggestedSolution::TYPE_FILE:
504 $possible_texts = array_values(
509 $this->
lng->txt(
'tst_show_solution_suggested')
515 $output[] =
'<a href="' 525 return implode(
"<br />", $output);
536 $ilDB = $DIC[
'ilDB'];
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]
545 if ($result->numRows() == 1) {
546 $row =
$ilDB->fetchAssoc($result);
547 $points = (float) $row[
"points"];
554 return round(self::_getReachedPoints($active_id, $this->
getId(), $pass), 2);
567 $requests_statistic_data = $hint_tracking->getRequestStatisticDataByQuestionAndTestpass();
568 $reached_points = $reached_points - $requests_statistic_data->getRequestsPoints();
573 return $reached_points;
584 $requests_statistic_data = $questionHintTracking->getRequestStatisticDataByQuestionAndTestpass();
585 $reached_points = $reached_points - $requests_statistic_data->getRequestsPoints();
590 if (is_null($reached_points)) {
591 $reached_points = 0.0;
598 function () use ($active_id, $pass, $reached_points, $requests_statistic_data, $existing_solutions) {
600 DELETE FROM tst_test_result 607 $types = [
'integer',
'integer',
'integer'];
608 $values = [$active_id, $this->
getId(), $pass];
615 $types[] =
'integer';
618 $this->db->manipulateF($query, $types, $values);
620 if ($existing_solutions[
'authorized']) {
621 $next_id = $this->db->nextId(
"tst_test_result");
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]
635 $fieldData[
'step'] = [
'integer', $this->
getStep()];
638 $this->db->insert(
'tst_test_result', $fieldData);
664 $this->
getProcessLocker()->executePersistWorkingStateLockOperation(
function () use ($active_id, $pass, $authorized, &$saveStatus) {
665 if ($pass ===
null) {
704 return CLIENT_WEB_DIR .
"/assessment/$this->obj_id/$this->id/solution/";
713 if ($question_id ===
null) {
717 if ($object_id ===
null) {
721 return $this->question_files->buildImagePath($question_id, $object_id);
727 .
"/assessment/{$this->obj_id}/{$this->id}/solution/";
742 if (!$this->export_image_path) {
744 .
"/assessment/{$this->obj_id}/{$this->id}/images/";
769 if (!count($solution)) {
782 bool $authorized =
true 784 if ($pass ===
null && is_numeric($active_id)) {
797 ORDER BY solution_id";
799 $result = $this->db->queryF(
801 [
'integer',
'integer',
'integer',
'integer',
'integer'],
802 [(
int) $active_id, $this->
getId(), $pass, $this->
getStep(), (
int) $authorized]
815 $result = $this->db->queryF(
817 [
'integer',
'integer',
'integer',
'integer'],
818 [(
int) $active_id, $this->
getId(), $pass, (
int) $authorized]
824 while ($row = $this->db->fetchAssoc($result)) {
835 if (!is_array($answer_table_name)) {
836 $answer_table_name = [$answer_table_name];
839 foreach ($answer_table_name as $table) {
840 if (strlen($table)) {
841 $this->db->manipulateF(
842 "DELETE FROM $table WHERE question_fi = %s",
854 if (!is_array($additional_table_name)) {
855 $additional_table_name = [$additional_table_name];
858 foreach ($additional_table_name as $table) {
859 if (strlen($table)) {
860 $this->db->manipulateF(
861 "DELETE FROM $table WHERE question_fi = %s",
877 public function delete(
int $question_id):
void 879 if ($question_id < 1) {
883 $result = $this->db->queryF(
884 "SELECT obj_fi FROM qpl_questions WHERE question_id = %s",
888 if ($this->db->numRows($result) !== 1) {
892 $row = $this->db->fetchAssoc($result);
893 $obj_id = $row[
"obj_fi"];
898 $this->log->root()->error(
"EXCEPTION: Could not delete page of question $question_id: $e");
902 $affectedRows = $this->db->manipulateF(
903 "DELETE FROM qpl_questions WHERE question_id = %s",
907 if ($affectedRows == 0) {
917 $this->log->root()->error(
"EXCEPTION: Could not delete additional table data of question {$question_id}: {$e}");
922 $affectedRows = $this->db->manipulateF(
923 "DELETE FROM tst_test_question WHERE question_fi = %s",
928 $this->log->root()->error(
"EXCEPTION: Could not delete delete question {$question_id} from a test: {$e}");
934 $this->log->root()->error(
"EXCEPTION: Could not delete suggested solutions of question {$question_id}: {$e}");
937 $directory =
CLIENT_WEB_DIR .
"/assessment/" . $obj_id .
"/$question_id";
939 if (is_dir($directory)) {
943 $this->log->root()->error(
"EXCEPTION: Could not delete question file directory {$directory} of question {$question_id}: {$e}");
952 foreach ($mobs as $mob) {
960 $this->log->root()->error(
"EXCEPTION: Error deleting the media objects of question {$question_id}: {$e}");
962 ilAssQuestionHintTracking::deleteRequestsByQuestionIds([$question_id]);
965 $assignmentList->setParentObjId($obj_id);
966 $assignmentList->setQuestionIdFilter($question_id);
967 $assignmentList->loadFromDb();
968 foreach ($assignmentList->getAssignmentsByQuestionId($question_id) as $assignment) {
970 $assignment->deleteFromDb();
973 if (!$assignment->isSkillUsed()) {
974 $this->skillUsageService->removeUsage(
975 $assignment->getParentObjId(),
976 $assignment->getSkillBaseId(),
977 $assignment->getSkillTrefId()
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}" 999 foreach ($taxIds as $taxId) {
1001 $taxNodeAssignment->deleteAssignmentsOfItem($this->
getId());
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]
1013 if ($this->db->numRows($result) == 0) {
1017 while ($row = $this->db->fetchAssoc($result)) {
1018 $found_id[] = $row[
"question_id"];
1021 $result = $this->db->query(
"SELECT * FROM tst_test_result WHERE " . $this->db->in(
'question_fi', $found_id,
false,
'integer'));
1023 return $this->db->numRows($result);
1028 if (!file_exists($file)) {
1032 if (!is_file($file)) {
1036 if (!is_readable($file)) {
1046 foreach ($mobs as $mob) {
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);
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();
1078 return $page->getXMLContent();
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' 1100 public function fixSvgToPng(
string $imageFilenameContainingString): string
1102 $needles = array_keys(self::$imageSourceFixReplaceMap);
1103 $replacements = array_values(self::$imageSourceFixReplaceMap);
1104 return str_replace($needles, $replacements, $imageFilenameContainingString);
1110 if (preg_match_all(
'/src="(.*?)"/m', $html, $matches)) {
1111 $sources = $matches[1];
1113 $needleReplacementMap = [];
1115 foreach ($sources as $src) {
1118 if (file_exists($file)) {
1122 $levels = explode(DIRECTORY_SEPARATOR, $src);
1123 if (count($levels) < 5 || $levels[0] !==
'Customizing' || $levels[2] !==
'skin') {
1129 if ($levels[4] ===
'components/ILIAS' || $levels[4] ===
'components/ILIAS') {
1130 $component = $levels[4] . DIRECTORY_SEPARATOR . $levels[5];
1136 if (count($needleReplacementMap)) {
1137 $html = str_replace(array_keys($needleReplacementMap), array_values($needleReplacementMap), $html);
1146 $result = $this->db->queryF(
1147 'SELECT external_id FROM qpl_questions WHERE question_id = %s',
1151 if ($this->db->numRows($result) === 1) {
1152 $data = $this->db->fetchAssoc($result);
1153 $this->external_id =
$data[
'external_id'];
1157 $this->suggested_solutions = [];
1158 if ($suggested_solutions) {
1159 foreach ($suggested_solutions as $solution) {
1160 $this->suggested_solutions[$solution->getSubquestionIndex()] = $solution;
1176 && $this->questionpool_request->hasRefId()) {
1177 $obj_id = $this->questionpool_request->getRefId();
1181 $obj_id = $this->questionpool_request->int(
'sel_qpl');
1185 return $this->
getId();
1189 if ($a_create_page) {
1193 $next_id = $this->db->nextId(
'qpl_questions');
1194 $this->db->insert(
"qpl_questions", [
1195 "question_id" => [
"integer", $next_id],
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"],
1205 "complete" => [
"text", $complete],
1206 "created" => [
"integer", time()],
1207 "original_id" => [
"integer",
null],
1208 "tstamp" => [
"integer", $tstamp],
1212 $this->
setId($next_id);
1214 if ($a_create_page) {
1219 return $this->
getId();
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],
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()],
1237 "created" => [
"integer", time()],
1238 "original_id" => [
"integer", $original_id],
1239 "tstamp" => [
"integer", time()],
1243 $this->
setId($next_id);
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)],
1258 "tstamp" => [
"integer", time()],
1259 'complete' => [
'integer', $this->
isComplete()],
1262 "question_id" => [
"integer", $this->
getId()]
1267 bool $for_test =
true,
1269 string $author =
'',
1273 if ($this->
id <= 0) {
1278 $clone = clone $this;
1281 if ((
int) $test_obj_id > 0) {
1282 $clone->setObjId($test_obj_id);
1286 $clone->setTitle($title);
1289 $clone->setAuthor($author);
1292 $clone->setOwner($owner);
1295 $clone->saveToDb($this->
id);
1300 $clone->clonePageOfQuestion($this->
getId());
1301 $clone->cloneXHTMLMediaObjectsOfQuestion($this->
getId());
1305 $clone->onDuplicate($this->
getObjId(), $this->
getId(), $clone->getObjId(), $clone->getId());
1311 int $target_parent_id,
1314 if ($this->
getId() <= 0) {
1315 throw new RuntimeException(
'The question has not been saved. It cannot be duplicated');
1318 $clone = clone $this;
1320 $source_parent_id = $this->
getObjId();
1321 $clone->setObjId($target_parent_id);
1323 $clone->setTitle($title);
1326 $clone->clonePageOfQuestion($this->
id);
1327 $clone->cloneXHTMLMediaObjectsOfQuestion($this->
id);
1330 $clone->onCopy($source_parent_id, $this->
id, $clone->getObjId(), $clone->getId());
1336 int $target_parent_id,
1337 string $target_question_title =
'' 1339 if ($this->
getId() <= 0) {
1340 throw new RuntimeException(
'The question has not been saved. It cannot be duplicated');
1344 $source_parent_id = $this->
getObjId();
1347 $clone = clone $this;
1350 $clone->setObjId($target_parent_id);
1352 if ($target_question_title) {
1353 $clone->setTitle($target_question_title);
1357 $clone->clonePageOfQuestion($source_question_id);
1358 $clone->cloneXHTMLMediaObjectsOfQuestion($source_question_id);
1362 $clone->onCopy($source_parent_id, $source_question_id, $clone->getObjId(), $clone->getId());
1386 'tstamp' => [
'integer', time()],
1387 'owner' => [
'integer', $this->
getOwner()],
1388 'complete' => [
'integer', $complete],
1389 'lifecycle' => [
'text', $this->
getLifecycle()->getIdentifier()],
1392 'question_id' => [
'integer', $this->
getId()]
1401 $target = opendir($image_target_path);
1402 while ($target_file = readdir($target)) {
1403 if ($target_file ===
'.' || $target_file ===
'..') {
1407 $image_target_path . DIRECTORY_SEPARATOR . $target_file,
1408 $image_target_path . DIRECTORY_SEPARATOR . $target_file
1416 $ilDB = $DIC->database();
1417 $query =
"UPDATE qpl_questions SET tstamp = %s, original_id = %s WHERE question_id = %s";
1421 [
'integer',
'integer',
'text'],
1422 [time(), $originalId, $questionId]
1429 $ilDB = $DIC->database();
1431 $query =
"UPDATE qpl_questions SET tstamp = %s, original_id = NULL WHERE question_id = %s";
1435 [
'integer',
'text'],
1436 [time(), $questionId]
1441 int $original_parent_id,
1442 int $original_question_id,
1443 int $duplicate_parent_id,
1444 int $duplicate_question_id
1448 $this->feedbackOBJ->duplicateFeedback($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);
1455 int $original_question_id,
1456 int $clone_question_id,
1457 int $original_parent_id,
1458 int $clone_parent_id
1460 $this->feedbackOBJ->cloneFeedback($original_question_id, $clone_question_id);
1463 protected function onCopy(
int $sourceParentId,
int $sourceQuestionId,
int $targetParentId,
int $targetQuestionId): void
1467 $this->feedbackOBJ->duplicateFeedback($sourceQuestionId, $targetQuestionId);
1470 $this->
duplicateComments($sourceParentId, $sourceQuestionId, $targetParentId, $targetQuestionId);
1474 int $parent_source_id,
1476 int $parent_target_id,
1481 $notes = $manager->getNotesForRepositoryObjIds([$parent_source_id], Note::PUBLIC);
1482 $notes = array_filter(
1484 fn($n) => $n->getContext()->getSubObjId() === $source_id
1487 foreach ($notes as $note) {
1488 $new_context = $data_service->context(
1491 $note->getContext()->getType()
1493 $new_note = $data_service->note(
1499 $note->getCreationDate(),
1500 $note->getUpdateDate(),
1501 $note->getRecipient()
1503 $manager->createNote($new_note, [],
true);
1511 $source_id = $this->
getId();
1512 $notes = $manager->getNotesForRepositoryObjIds([$this->
getObjId()], Note::PUBLIC);
1513 $notes = array_filter(
1515 fn($n) => $n->getContext()->getSubObjId() === $source_id
1517 foreach ($notes as $note) {
1518 $repo->deleteNote($note->getId());
1525 return $service->internal()->domain()->notes();
1531 return $service->internal()->data();
1537 return $service->internal()->repo()->note();
1545 $this->suggested_solutions = [];
1551 if (array_key_exists($subquestion_index, $this->suggested_solutions)) {
1552 return $this->suggested_solutions[$subquestion_index];
1558 int $source_question_id,
1559 int $target_question_id
1570 foreach ($this->suggested_solutions as $solution) {
1571 if (!$solution->isOfTypeFile()
1572 || $solution->getFilename() ===
'') {
1577 $filepath_original = str_replace(
1578 "/{$this->obj_id}/{$this->id}/solution",
1579 "/{$parent_id}/{$question_id}/solution",
1582 if (!file_exists($filepath)) {
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()}");
1594 int $source_question_id,
1595 int $target_question_id
1598 $filepath_original = str_replace(
"/$target_question_id/solution",
"/$source_question_id/solution", $filepath_target);
1600 foreach ($this->suggested_solutions as $solution) {
1601 if (!$solution->isOfTypeFile()
1602 || $solution->getFilename() ===
'') {
1606 if (!file_exists($filepath_original)) {
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()}");
1622 $solution = $solution->withQuestionId($target_question_id);
1623 $update[] = $solution;
1630 if (preg_match(
"/il_(\d+)_(\w+)_(\d+)/", $internal_link, $matches) ===
false) {
1631 return $internal_link;
1633 switch ($matches[2]) {
1650 if ($resolved_link ===
null) {
1651 return "il__{$matches[2]}_{$matches[3]}";
1653 return $internal_link;
1662 $result_pre = $this->db->queryF(
1663 "SELECT internal_link, suggested_solution_id FROM qpl_sol_sug WHERE question_fi = %s",
1667 if ($this->db->numRows($result_pre) < 1) {
1671 while ($row = $this->db->fetchAssoc($result_pre)) {
1672 $internal_link = $row[
"internal_link"];
1674 if ($internal_link === $resolved_link) {
1678 $this->db->manipulateF(
1679 "UPDATE qpl_sol_sug SET internal_link = %s WHERE suggested_solution_id = %s",
1681 [$resolved_link, $row[
"suggested_solution_id"]]
1685 if ($resolvedlinks === 0) {
1691 $result_post = $this->db->queryF(
1692 "SELECT internal_link FROM qpl_sol_sug WHERE question_fi = %s",
1696 if ($this->db->numRows($result_post) < 1) {
1700 while ($row = $this->db->fetchAssoc($result_post)) {
1701 if (preg_match(
"/il_(\d*?)_(\w+)_(\d+)/", $row[
"internal_link"], $matches)) {
1710 "lm" =>
"LearningModule",
1711 "pg" =>
"PageObject",
1712 "st" =>
"StructureObject",
1713 "git" =>
"GlossaryItem",
1714 "mob" =>
"MediaObject" 1717 if (preg_match(
"/il__(\w+)_(\d+)/", $target, $matches)) {
1718 $type = $matches[1];
1719 $target_id = $matches[2];
1720 switch ($linktypes[$matches[1]]) {
1722 $href =
"./ilias.php?baseClass=ilLMPresentationGUI&obj_type=" . $linktypes[$type]
1723 .
"&cmd=media&ref_id=" . $this->questionpool_request->getRefId()
1724 .
"&mob_id=" . $target_id;
1726 case "StructureObject":
1727 case "GlossaryItem":
1729 case "LearningModule":
1731 $href =
"./goto.php?target=" . $type .
"_" . $target_id;
1744 $original_parent_id = self::lookupParentObjId($this->
getOriginalId());
1746 if ($original_parent_id ===
null) {
1751 $original = clone $this;
1754 $original->setOriginalId(
null);
1755 $original->setObjId($original_parent_id);
1757 $original->saveToDb();
1760 $original->createPageObject();
1761 $original->clonePageOfQuestion($this->
getId());
1765 $this->
cloneHints($this->
id, $this->original_id);
1776 $ilCtrl = $DIC[
'ilCtrl'];
1777 $ilDB = $DIC[
'ilDB'];
1779 $questionrepository = QuestionPoolDIC::dic()[
'question.general_properties.repository'];
1780 $question_type = $questionrepository->
getForQuestionId($question_id)?->getClassName() ??
'';
1781 if ($question_type ===
'') {
1785 $question =
new $question_type();
1786 $question->loadFromDb($question_id);
1788 $feedbackObjectClassname = self::getFeedbackClassNameByQuestionType($question_type);
1789 $question->feedbackOBJ =
new $feedbackObjectClassname($question, $ilCtrl,
$ilDB, $lng);
1806 return self::_getSolutionMaxPass($this->
getId(), $active_id);
1818 $ilDB = $DIC[
'ilDB'];
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]
1825 if ($result->numRows() === 1) {
1826 $row =
$ilDB->fetchAssoc($result);
1827 return $row[
"maxpass"];
1841 $requests_statistic_data = $hint_tracking->getRequestStatisticData();
1842 return $reached_points - $requests_statistic_data->getRequestsPoints();
1857 return $points > 0.0 ?
$points : 0.0;
1881 if ($count_system == 1) {
1887 if ($score_cutting == 0) {
1898 if (preg_match(
"/.*\.(png|jpg|gif|jpeg)$/i", $plain_image_filename, $matches)) {
1899 $extension =
"." . $matches[1];
1903 $plain_image_filename = uniqid($plain_image_filename . microtime(
true),
true);
1906 return md5($plain_image_filename) . $extension;
1928 $ilDB = $DIC[
'ilDB'];
1930 if ($points > $maxpoints) {
1934 if ($pass ===
null) {
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]
1944 $manual = ($manualscoring) ? 1 : 0;
1945 $rowsnum = $result->numRows();
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]
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()]
1965 if (!self::isForcePassResultUpdateEnabled() && $old_points === $points && $rowsnum !== 0) {
1970 if ($test_id ===
null) {
1977 $test->updateTestPassResults($active_id, $pass);
1995 || !(
new ilSetting(
'advanced_editing'))->
get(
'advanced_editing_javascript_editor') ===
'tinymce') {
1996 $purified_content = nl2br($purified_content);
2012 $result = $this->db->queryF(
2013 "SELECT question_type_id FROM qpl_qst_type WHERE type_tag = %s",
2017 if ($this->db->numRows($result) == 1) {
2018 $row = $this->db->fetchAssoc($result);
2019 return (
int) $row[
"question_type_id"];
2025 int $source_question_id,
2026 int $target_question_id
2029 $this->db->manipulateF(
2030 "DELETE FROM qpl_hints WHERE qht_question_fi = %s",
2032 [$target_question_id]
2036 $result = $this->db->queryF(
2037 "SELECT * FROM qpl_hints WHERE qht_question_fi = %s",
2039 [$source_question_id]
2043 if ($this->db->numRows($result) < 1) {
2047 while ($row = $this->db->fetchAssoc($result)) {
2048 $next_id = $this->db->nextId(
'qpl_hints');
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"]],
2067 $collected .= $this->feedbackOBJ->getGenericFeedbackContent($this->
getId(),
false);
2068 $collected .= $this->feedbackOBJ->getGenericFeedbackContent($this->
getId(),
true);
2069 $collected .= $this->feedbackOBJ->getAllSpecificAnswerFeedbackContents($this->
getId());
2072 foreach ($questionHintList as $questionHint) {
2074 $collected .= $questionHint->getText();
2088 $result = $this->db->queryF(
2089 "SELECT question_id FROM qpl_questions WHERE original_id = %s",
2095 while ($row = $this->db->fetchAssoc($result)) {
2096 $ids[] = $row[
"question_id"];
2098 foreach ($ids as $question_id) {
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",
2105 while ($row = $this->db->fetchAssoc($result)) {
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",
2114 while ($row = $this->db->fetchAssoc($result)) {
2118 foreach ($instances as $key => $value) {
2132 $result = $this->db->queryF(
2133 "SELECT * FROM tst_active WHERE active_id = %s",
2137 if ($this->db->numRows($result)) {
2138 $row = $this->db->fetchAssoc($result);
2139 return [
"user_id" => $row[
"user_fi"],
"test_id" => $row[
"test_fi"]];
2147 return static::HAS_SPECIFIC_FEEDBACK;
2152 return str_replace(
'ass',
'ilAss', $questionType) .
'Feedback';
2157 public static function instantiateQuestionGUI(
int $question_id): ?
assQuestionGUI 2161 $ilCtrl = $DIC[
'ilCtrl'];
2162 $ilDB = $DIC[
'ilDB'];
2164 $ilUser = $DIC[
'ilUser'];
2165 $ilLog = $DIC[
'ilLog'];
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)');
2172 $questionrepository = QuestionPoolDIC::dic()[
'question.general_properties.repository'];
2173 $question_type = $questionrepository->
getForQuestionId($question_id)?->getClassName();
2175 if ($question_type ===
null) {
2179 $question_type_gui = $question_type .
'GUI';
2180 $question_gui =
new $question_type_gui($question_id);
2182 $feedback_object_classname = self::getFeedbackClassNameByQuestionType($question_type);
2183 $question = $question_gui->getObject();
2184 $question->feedbackOBJ =
new $feedback_object_classname($question, $ilCtrl,
$ilDB, $lng);
2186 $assSettings =
new ilSetting(
'assessment');
2188 $processLockerFactory->setQuestionId($question_gui->getObject()->getId());
2189 $processLockerFactory->setUserId($ilUser->getId());
2190 $question->setProcessLocker($processLockerFactory->getLocker());
2191 $question_gui->setObject($question);
2193 return $question_gui;
2203 $this->nr_of_tries = $a_nr_of_tries;
2208 $this->export_image_path =
$path;
2214 $ilDB = $DIC[
'ilDB'];
2216 if ($question_id < 1) {
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]
2225 return $ilDB->numRows($result) == 1;
2235 return new \ilAssSelfAssessmentQuestionFormatter();
2256 $this->feedbackOBJ->migrateContentForLearningModule($migrator, $this->
getId());
2294 $ilDB = $DIC[
'ilDB'];
2296 $query =
"SELECT obj_fi FROM qpl_questions WHERE question_id = %s";
2298 $res =
$ilDB->queryF($query, [
'integer'], [$question_id]);
2301 return $row[
'obj_fi'] ??
null;
2309 foreach ($hintIds as $originalHintId => $duplicateHintId) {
2312 $originalXML = $originalPageObject->getXMLContent();
2315 $duplicatePageObject->setId($duplicateHintId);
2316 $duplicatePageObject->setParentId($this->
getId());
2317 $duplicatePageObject->setXMLContent($originalXML);
2318 $duplicatePageObject->createFromXML();
2326 $assignmentList->setParentObjId($srcParentId);
2327 $assignmentList->setQuestionIdFilter($srcQuestionId);
2328 $assignmentList->loadFromDb();
2330 foreach ($assignmentList->getAssignmentsByQuestionId($srcQuestionId) as $assignment) {
2331 $assignment->setParentObjId($trgParentId);
2332 $assignment->setQuestionId($trgQuestionId);
2333 $assignment->saveToDb();
2336 $this->skillUsageService->addUsage(
2338 $assignment->getSkillBaseId(),
2339 $assignment->getSkillTrefId()
2344 public function syncSkillAssignments(
int $srcParentId,
int $srcQuestionId,
int $trgParentId,
int $trgQuestionId): void
2347 $assignmentList->setParentObjId($trgParentId);
2348 $assignmentList->setQuestionIdFilter($trgQuestionId);
2349 $assignmentList->loadFromDb();
2351 foreach ($assignmentList->getAssignmentsByQuestionId($trgQuestionId) as $assignment) {
2352 $assignment->deleteFromDb();
2355 if (!$assignment->isSkillUsed()) {
2356 $this->skillUsageService->removeUsage(
2357 $assignment->getParentObjId(),
2358 $assignment->getSkillBaseId(),
2359 $assignment->getSkillTrefId()
2371 $pageObject->setParentId($this->
getId());
2372 $pageObject->setId($pageObjectId);
2373 $pageObject->createFromXML();
2380 return $numExistingSolutionRecords > 0;
2386 $ilDB = $DIC[
'ilDB'];
2389 SELECT count(active_fi) cnt 2393 WHERE active_fi = %s 2394 AND question_fi = %s 2400 [
'integer',
'integer',
'integer'],
2401 [$activeId, $questionId, $pass]
2406 return (
int) $row[
'cnt'];
2440 self::ADDITIONAL_CONTENT_EDITING_MODE_RTE,
2441 self::ADDITIONAL_CONTENT_EDITING_MODE_IPE
2464 SELECT qpl_questions.*, 2465 {$this->getAdditionalTableName()}.* 2467 LEFT JOIN {$this->getAdditionalTableName()} 2468 ON {$this->getAdditionalTableName()}.question_fi = qpl_questions.question_id 2469 WHERE qpl_questions.question_id = %s 2489 WHERE active_fi = %s 2490 AND question_fi = %s 2496 return $this->db->queryF(
2498 [
'integer',
'integer',
'integer',
'integer',
'integer'],
2499 [$active_id, $this->
getId(), $pass, $this->
getStep(), (
int) $authorized]
2506 WHERE active_fi = %s 2507 AND question_fi = %s 2512 return $this->db->queryF(
2514 [
'integer',
'integer',
'integer',
'integer'],
2515 [$active_id, $this->
getId(), $pass, (
int) $authorized]
2521 return $this->db->manipulateF(
2522 "DELETE FROM tst_solutions WHERE solution_id = %s",
2534 $result = $this->db->queryF(
2535 "SELECT * FROM tst_solutions WHERE solution_id = %s",
2540 if ($this->db->numRows($result) > 0) {
2541 return $this->db->fetchAssoc($result);
2549 $this->
getProcessLocker()->executeUserSolutionUpdateLockOperation(
function () use ($active_id, $pass) {
2561 DELETE FROM tst_solutions 2562 WHERE active_fi = %s 2563 AND question_fi = %s 2569 return $this->db->manipulateF(
2571 [
'integer',
'integer',
'integer',
'integer',
'integer'],
2572 [$active_id, $this->
getId(), $pass, $this->
getStep(), (
int) $authorized]
2577 DELETE FROM tst_solutions 2578 WHERE active_fi = %s 2579 AND question_fi = %s 2584 return $this->db->manipulateF(
2586 [
'integer',
'integer',
'integer',
'integer'],
2587 [$active_id, $this->
getId(), $pass, (
int) $authorized]
2594 $next_id = $this->db->nextId(
"tst_solutions");
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]
2608 $fieldData[
'step'] = [
"integer", $this->
getStep()];
2611 return $this->db->insert(
"tst_solutions", $fieldData);
2618 "value1" => [
"clob", $value1],
2619 "value2" => [
"clob", $value2],
2620 "tstamp" => [
"integer", time()],
2621 'authorized' => [
'integer', (
int) $authorized]
2625 $fieldData[
'step'] = [
"integer", $this->
getStep()];
2628 return $this->db->update(
"tst_solutions", $fieldData, [
2629 'solution_id' => [
'integer', $solutionId]
2637 'authorized' => [
'integer', (
int) $authorized]
2641 $fieldData[
'tstamp'] = [
'integer', time()];
2645 'question_fi' => [
'integer', $this->
getId()],
2646 'active_fi' => [
'integer', $activeId],
2647 'pass' => [
'integer', $pass]
2651 $whereData[
'step'] = [
"integer", $this->
getStep()];
2654 return $this->db->update(
'tst_solutions', $fieldData, $whereData);
2673 foreach ($this->
getSolutionValues($activeId, $passIndex,
false) as $solutionRec) {
2674 if ($solutionRec[
'value1'] ==
'' && $solutionRec[
'value2'] ==
'') {
2682 return !strlen($solutionRecord[
'value1']) && !strlen($solutionRecord[
'value2']);
2687 $types = [
"integer",
"integer",
"integer",
"integer"];
2688 $values = [$activeId, $this->
getId(), $passIndex, (
int) $authorized];
2689 $valuesCondition = [];
2691 foreach ($matchValues as $valueField => $value) {
2692 switch ($valueField) {
2695 $valuesCondition[] =
"{$valueField} = %s";
2705 $valuesCondition = implode(
' AND ', $valuesCondition);
2708 DELETE FROM tst_solutions 2709 WHERE active_fi = %s 2710 AND question_fi = %s 2713 AND $valuesCondition 2717 $query .=
" AND step = %s ";
2718 $types[] =
'integer';
2722 $this->db->manipulateF($query, $types, $values);
2728 $this->
saveCurrentSolution($activeId, $passIndex, $rec[
'value1'], $rec[
'value2'],
true, $rec[
'tstamp']);
2736 if (!count($intermediateSolution)) {
2742 if ($considerDummyRecordCreation) {
2756 $this->step =
$step;
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];
2781 return json_encode([]);
2788 return (
bool) $solutionAvailability[
'intermediate'];
2793 if ($pass ===
null) {
2797 return (
bool) $solutionAvailability[
'authorized'];
2803 return $solutionAvailability[
'authorized'] || $solutionAvailability[
'intermediate'];
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()]
2815 $row = $this->db->fetchAssoc($result);
2817 return (
int) $row[
'max_step'];
2828 'authorized' =>
false,
2829 'intermediate' => false
2833 SELECT authorized, COUNT(*) cnt 2835 WHERE active_fi = %s 2836 AND question_fi = %s 2841 $query .=
" AND step = " . $this->db->quote((
int) $this->
getStep(),
'integer') .
" ";
2848 $result = $this->db->queryF($query, [
'integer',
'integer',
'integer'], [$activeId, $this->
getId(), $pass]);
2850 while ($row = $this->db->fetchAssoc($result)) {
2851 if ($row[
'authorized']) {
2852 $return[
'authorized'] = $row[
'cnt'] > 0;
2854 $return[
'intermediate'] = $row[
'cnt'] > 0;
2872 $query =
"DELETE FROM tst_solutions WHERE question_fi = %s";
2873 $this->db->manipulateF($query, [
'integer'], [$this->
getId()]);
2879 DELETE FROM tst_solutions 2880 WHERE active_fi = %s 2881 AND question_fi = %s 2886 $query .=
" AND step = " . $this->db->quote((
int) $this->
getStep(),
'integer') .
" ";
2889 return $this->db->manipulateF(
2891 [
'integer',
'integer',
'integer'],
2892 [$activeId, $this->
getId(), $pass]
2905 $test->updateTestPassResults(
2916 DELETE FROM tst_test_result 2917 WHERE active_fi = %s 2918 AND question_fi = %s 2923 $query .=
" AND step = " . $this->db->quote((
int) $this->
getStep(),
'integer') .
" ";
2926 return $this->db->manipulateF(
2928 [
'integer',
'integer',
'integer'],
2929 [$activeId, $this->
getId(), $pass]
2937 foreach ($indexedValues as $value1 => $value2) {
2938 $valuePairs[] = [
'value1' => $value1,
'value2' => $value2];
2946 $indexed_values = [];
2948 foreach ($value_pairs as $valuePair) {
2949 $indexed_values[$valuePair[
'value1']] = $valuePair[
'value2'];
2952 return $indexed_values;
2957 $this->db->manipulateF(
2958 "UPDATE qpl_questions SET tstamp = %s WHERE question_id = %s",
2959 [
'integer',
'integer'],
2960 [time(), $this->
getId()]
2966 if ($this->test_question_config ===
null) {
2985 $question_id = $this->
getId();
3005 return preg_replace(self::TRIM_PATTERN,
'', $value);
3010 return !is_null($this->original_id)
3011 && $this->questionrepository->questionExistsInPool($this->original_id)
3026 $this->current_user->getId(),
3030 $this->
answerToLog($additional_info, $active_id, $pass)
3042 $this->current_user->getId(),
3045 $this->
toLog($additional_info)
3055 AdditionalInformationGenerator::KEY_PASS => $pass,
3056 AdditionalInformationGenerator::KEY_REACHED_POINTS => $this->
getReachedPoints($active_id, $pass),
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
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.
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...
resolveSuggestedSolutionLinks()
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.
ilTestQuestionConfig $test_question_config
getSuggestedSolutionPath()
getTitleFilenameCompliant()
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)
buildTestPresentationConfig()
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.
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)
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 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)
migrateToLmContent($content)
lmMigrateQuestionTypeSpecificContent(ilAssSelfAssessmentMigrator $migrator)
static getNumExistingSolutionRecords(int $activeId, int $pass, int $questionId)
duplicateSkillAssignments(int $srcParentId, int $srcQuestionId, int $trgParentId, int $trgQuestionId)
getAdditionalContentEditingMode()
cloneSuggestedSolutions(int $source_question_id, int $target_question_id)
RequestDataCollector $questionpool_request
getParticipantsSolution()
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
getSelfAssessmentEditingMode()
importHint(int $question_id, array $hint_array)
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()
Customizing of pimple-DIC for ILIAS.
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
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.
static _saveLink(string $a_source_type, int $a_source_id, string $a_target_type, int $a_target_id, int $a_target_inst=0, string $a_source_lang="-")
save internal link information
calculateReachedPoints(int $active_id, ?int $pass=null, bool $authorized_solution=true)
fixUnavailableSkinImageSources(string $html)
getSuggestedSolutionsRepo()
isAddableAnswerOptionValue(int $qIndex, string $answerOptionValue)
updateCurrentSolution(int $solutionId, $value1, $value2, bool $authorized=true)
setParticipantsSolution($participantSolution)
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)
static resetOriginalId(int $questionId)
resolveInternalLink(string $internal_link)
getQuestionForHTMLOutput()
static setForcePassResultUpdateEnabled(bool $force_pass_results_update_enabled)
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...
TestQuestionAdministrationInteractionTypes
purifyAndPrepareTextAreaOutput(string $content)
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.
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.
static _deleteAllLinksOfSource(string $a_source_type, int $a_source_id, string $a_lang="-")
Delete all links of a given source.
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)
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.
static _getIdForImportId(string $a_type, string $a_target)
Get current id for an import id.
deductHintPointsFromReachedPoints(ilAssQuestionPreviewSession $preview_session, $reached_points)
intermediateSolutionExists(int $active_id, int $pass)
static _getCountSystem($active_id)
Repository for suggested solutions.
static getImagePath(string $image_name, string $module_path="", string $mode="output", bool $offline=false)
get image path (for images located in a template directory)
getDescriptionForHTMLOutput()
getHtmlUserSolutionPurifier()
hasWritableOriginalInQuestionPool()
static getInstanceByType(string $type)
static _lookupAuthor($obj_id)
Gets the authors name of the ilObjTest object.
getSuggestedSolutionPathWeb()
saveQuestionDataToDb(?int $original_id=null)
removeExistingSolutions(int $activeId, int $pass)
setDefaultNrOfTries(int $defaultnroftries)
saveToDb(?int $original_id=null)
cleanupMediaObjectUsage()
deleteSuggestedSolutions()
fetchIndexedValuesFromValuePairs(array $value_pairs)
static saveOriginalId(int $questionId, int $originalId)
TestParticipantInteractionTypes
static _updateQuestionCount(int $object_id)
savePreviewData(ilAssQuestionPreviewSession $preview_session)
ilAssQuestionLifecycle $lifecycle
fetchValuePairsFromIndexedValues(array $indexedValues)
getRTETextWithMediaObjects()
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)
__construct(Container $dic, ilPlugin $plugin)
onCopy(int $sourceParentId, int $sourceQuestionId, int $targetParentId, int $targetQuestionId)
const ADDITIONAL_CONTENT_EDITING_MODE_RTE
setOriginalId(?int $original_id)
getTestPresentationConfig()
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)
static signFile(string $path_to_file)
authorizedSolutionExists(int $active_id, ?int $pass)
setLifecycle(ilAssQuestionLifecycle $lifecycle)
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)
deleteTaxonomyAssignments()
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
getHtmlQuestionContentPurifier()
lookupMaxStep(int $active_id, int $pass)
calculateReachedPointsFromPreviewSession(ilAssQuestionPreviewSession $preview_session)
setAuthor(string $author="")
isNonEmptyItemListPostSubmission(string $post_submission_field_name)
duplicateSuggestedSolutionFiles(int $parent_id, int $question_id)
Duplicates the files of a suggested solution if the question is duplicated.
string $additionalContentEditingMode
copySuggestedSolutions(int $target_question_id)
getSolutionRecordById(int $solutionId)
setShuffle(?bool $shuffle=true)
setAdditionalContentEditingMode(?string $additionalContentEditingMode)
static explodeKeyValues(string $keyValues)
static getDraftInstance()
removeAllExistingSolutions()
getSuggestedSolutionOutput()
getSelfAssessmentFormatter()
getForQuestionId(int $question_id)
getInternalLinkHref(string $target)
static getFeedbackClassNameByQuestionType(string $questionType)
setQuestion(string $question="")
string $export_image_path