25 require_once
'./Modules/Test/classes/inc.AssessmentConstants.php';
50 self::IMG_MIME_TYPE_JPG => [
'jpg',
'jpeg'],
51 self::IMG_MIME_TYPE_PNG => [
'png'],
52 self::IMG_MIME_TYPE_GIF => [
'gif']
56 self::IMG_MIME_TYPE_JPG => [
'binary'],
57 self::IMG_MIME_TYPE_PNG => [
'binary'],
58 self::IMG_MIME_TYPE_GIF => [
'binary']
182 'image/jpeg' => [
'jpg',
'jpeg'],
183 'image/png' => [
'png'],
184 'image/gif' => [
'gif']
194 string $comment =
"",
197 string $question =
"" 203 $ilDB = $DIC[
'ilDB'];
204 $ilLog = $DIC->logger();
206 $this->current_user = $DIC[
'ilUser'];
211 $this->
http = $DIC->http();
214 $this->thumb_size = self::DEFAULT_THUMB_SIZE;
225 $this->suggested_solutions = [];
227 $this->nr_of_tries = 0;
230 $this->questionActionCmd =
'handleQuestionAction';
231 $this->export_image_path =
'';
232 $this->shuffler = $DIC->refinery()->random()->dontShuffle();
245 return self::$forcePassResultsUpdateEnabled;
250 return (
bool) count(self::getAllowedFileExtensionsForMimeType($mimeType));
255 return current(explode(
';', $contentType));
260 foreach (self::$allowedFileExtensionsByMimeType as $allowedMimeType => $extensions) {
261 $rexCharsets = implode(
'|', self::$allowedCharsetsByMimeType[$allowedMimeType]);
262 $rexMimeType = preg_quote($allowedMimeType,
'/');
264 $rex =
'/^' . $rexMimeType .
'(;(\s)*charset=(' . $rexCharsets .
'))*$/';
266 if (!preg_match($rex, $mimeType)) {
278 return in_array(strtolower($fileExtension), self::getAllowedFileExtensionsForMimeType($mimeType),
true);
284 if ($question_id > 0) {
285 return 'il_' .
IL_INST_ID .
'_qst_' . $question_id;
288 return uniqid(
'',
true);
293 if (!isset($_POST[
'cmd']) || !isset($_POST[
'cmd'][$this->questionActionCmd])) {
297 if (!is_array($_POST[
'cmd'][$this->questionActionCmd]) || !count($_POST[
'cmd'][$this->questionActionCmd])) {
301 return key($_POST[
'cmd'][$this->questionActionCmd]);
306 if (!isset($_POST[$postSubmissionFieldname])) {
310 if (!is_array($_POST[$postSubmissionFieldname])) {
314 if (!count($_POST[$postSubmissionFieldname])) {
323 if (is_int($pass) && $pass >= 0) {
336 return \ilObjTest::_getPass($active_id);
344 $result = $this->db->queryF(
345 "SELECT test_fi FROM tst_active WHERE active_id = %s",
350 if ($this->db->numRows($result) > 0) {
351 $row = $this->db->fetchAssoc($result);
352 $test_id = (
int) $row[
"test_fi"];
358 protected function log(
int $active_id,
string $langVar): void
373 foreach (self::$allowedImageMaterialFileExtensionsByMimeType as $mimeType => $mimeExtensions) {
375 $extensions = array_merge($extensions, $mimeExtensions);
377 return array_unique($extensions);
410 public function fromXML($item,
int $questionpool_id, ?
int $tst_id, &$tst_object,
int &$question_counter, array $import_mapping, array &$solutionhints = []): array
413 $import =
new $classname($this);
414 $import_mapping = $import->fromXML($item, $questionpool_id, $tst_id, $tst_object, $question_counter, $import_mapping);
416 foreach ($solutionhints as $hint) {
418 $h->setQuestionId($import->getQuestionId());
419 $h->setIndex($hint[
'index'] ??
"");
420 $h->setPoints($hint[
'points'] ??
"");
421 $h->setText($hint[
'txt'] ??
"");
424 return $import_mapping;
433 bool $a_include_header =
true,
434 bool $a_include_binary =
true,
435 bool $a_shuffle =
false,
436 bool $test_output =
false,
437 bool $force_image_references =
false 440 $export =
new $classname($this);
441 return $export->toXML($a_include_header, $a_include_binary, $a_shuffle, $test_output, $force_image_references);
460 $ilDB = $DIC[
'ilDB'];
462 $result =
$ilDB->queryF(
463 "SELECT * FROM qpl_questions WHERE obj_fi = %s AND title = %s",
464 array(
'integer',
'text'),
465 array($questionpool_id, $title)
467 return ($result->numRows() > 0) ?
true :
false;
475 public function setId(
int $id = -1): void
482 $this->test_id =
$id;
497 $this->shuffle = $shuffle ??
false;
503 $author = $this->current_user->getFullname();
520 return $this->
refinery->string()->stripTags()->transform($this->title);
550 return $this->
refinery->string()->stripTags()->transform($this->
comment);
560 if ($a_size >= self::MINIMUM_THUMB_SIZE) {
561 $this->thumb_size = $a_size;
563 throw new ilException(
"Thumb size must be at least " . self::MINIMUM_THUMB_SIZE .
"px");
569 return self::MINIMUM_THUMB_SIZE;
574 return self::MAXIMUM_THUMB_SIZE;
605 return $this->
refinery->string()->stripTags()->transform($this->author);
640 if ($this->external_id === null || $this->external_id ===
'') {
653 $ilDB = $DIC[
'ilDB'];
656 $result =
$ilDB->queryF(
657 "SELECT points FROM qpl_questions WHERE question_id = %s",
661 if (
$ilDB->numRows($result) == 1) {
662 $row =
$ilDB->fetchAssoc($result);
663 $points = (float) $row[
"points"];
674 $ilDB = $DIC[
'ilDB'];
676 $result =
$ilDB->queryF(
677 "SELECT qpl_questions.*, qpl_qst_type.type_tag FROM qpl_qst_type, qpl_questions WHERE qpl_questions.question_id = %s AND qpl_questions.question_type_fi = qpl_qst_type.question_type_id",
682 if (
$ilDB->numRows($result)) {
683 return $ilDB->fetchAssoc($result);
691 $ilDB = $DIC[
'ilDB'];
693 $result =
$ilDB->queryF(
694 "SELECT suggested_solution_id FROM qpl_sol_sug WHERE question_fi = %s",
698 return $ilDB->numRows($result);
707 $question = self::_instantiateQuestion($question_id);
708 if (!is_object($question)) {
711 return $question->getSuggestedSolutionOutput();
721 foreach ($this->suggested_solutions as $solution) {
722 switch ($solution[
"type"]) {
730 $possible_texts = array_values(array_filter(array(
733 $this->
lng->txt(
'tst_show_solution_suggested')
739 $solutionValue = $solution[
"value"];
740 $solutionValue = $this->
fixSvgToPng($solutionValue);
746 return implode(
"<br />", $output);
765 $result = $this->db->queryF(
766 "SELECT * FROM qpl_sol_sug WHERE question_fi = %s AND subquestion_index = %s",
767 array(
'integer',
'integer'),
768 array($question_id, $subquestion_index)
770 if ($this->db->numRows($result) == 1) {
771 $row = $this->db->fetchAssoc($result);
773 "internal_link" => $row[
"internal_link"],
774 "import_id" => $row[
"import_id"]
791 $ilDB = $DIC[
'ilDB'];
795 $result =
$ilDB->queryF(
796 "SELECT * FROM tst_test_result WHERE active_fi = %s AND question_fi = %s AND pass = %s",
797 array(
'integer',
'integer',
'integer'),
798 array($active_id, $question_id, $pass)
800 if ($result->numRows() == 1) {
801 $row =
$ilDB->fetchAssoc($result);
802 $points = (float) $row[
"points"];
809 return round(self::_getReachedPoints($active_id, $this->
getId(), $pass), 2);
829 $requestsStatisticData = $hintTracking->getRequestStatisticDataByQuestionAndTestpass();
830 $reached_points = $reached_points - $requestsStatisticData->getRequestsPoints();
835 return $reached_points;
844 $ilDB = $DIC[
'ilDB'];
850 $requestsStatisticData = $questionHintTracking->getRequestStatisticDataByQuestionAndTestpass();
851 $reached_points = $reached_points - $requestsStatisticData->getRequestsPoints();
857 $isAnswered = $this->
isAnswered($active_id, $pass);
862 if (is_null($reached_points)) {
863 $reached_points = 0.0;
869 $this->
getProcessLocker()->executeUserQuestionResultUpdateOperation(
function () use (
$ilDB, $active_id, $pass, $reached_points, $requestsStatisticData, $isAnswered, $existingSolutions) {
871 DELETE FROM tst_test_result 878 $types = array(
'integer',
'integer',
'integer');
879 $values = array($active_id, $this->
getId(), $pass);
881 if ($this->
getStep() !== null) {
886 $types[] =
'integer';
891 if ($existingSolutions[
'authorized']) {
892 $next_id =
$ilDB->nextId(
"tst_test_result");
894 'test_result_id' => array(
'integer', $next_id),
895 'active_fi' => array(
'integer', $active_id),
896 'question_fi' => array(
'integer', $this->
getId()),
897 'pass' => array(
'integer', $pass),
898 'points' => array(
'float', $reached_points),
899 'tstamp' => array(
'integer', time()),
900 'hint_count' => array(
'integer', $requestsStatisticData->getRequestsCount()),
901 'hint_points' => array(
'float', $requestsStatisticData->getRequestsPoints()),
902 'answered' => array(
'integer', $isAnswered)
905 if ($this->
getStep() !== null) {
906 $fieldData[
'step'] = array(
'integer', $this->
getStep());
909 $ilDB->insert(
'tst_test_result', $fieldData);
918 "log_user_answered_question",
929 self::_updateTestPassResults($active_id, $pass, $obligationsEnabled, $this->
getProcessLocker());
937 final public function persistWorkingState(
int $active_id, $pass,
bool $obligationsEnabled =
false,
bool $authorized =
true): bool
945 $this->
getProcessLocker()->executePersistWorkingStateLockOperation(
function () use ($active_id, $pass, $authorized, $obligationsEnabled, &$saveStatus) {
946 if ($pass === null) {
986 abstract public function saveWorkingData(
int $active_id,
int $pass,
bool $authorized =
true): bool;
997 $ilDB = $DIC[
'ilDB'];
1001 if ($pass !== null) {
1003 SELECT tst_pass_result.* 1004 FROM tst_pass_result 1005 WHERE active_fi = %s 1009 $result =
$ilDB->queryF(
1011 array(
'integer',
'integer'),
1012 array($active_id, $pass)
1015 $test_pass_result_row =
$ilDB->fetchAssoc($result);
1017 if (!is_array($test_pass_result_row)) {
1018 $test_pass_result_row = [];
1020 $max = (float) ($test_pass_result_row[
'maxpoints'] ?? 0);
1021 $reached = (float) ($test_pass_result_row[
'points'] ?? 0);
1022 $percentage = ($max <= 0.0 || $reached <= 0.0) ? 0 : ($reached / $max) * 100.0;
1024 $obligationsAnswered = (
int) ($test_pass_result_row[
'obligations_answered'] ?? 1);
1026 $mark = ASS_MarkSchema::_getMatchingMarkFromActiveId($active_id, $percentage);
1027 $isPassed = isset($mark[
"passed"]) && $mark[
"passed"];
1029 $hint_count = $test_pass_result_row[
'hint_count'] ?? 0;
1030 $hint_points = $test_pass_result_row[
'hint_points'] ?? 0.0;
1032 $userTestResultUpdateCallback =
function () use (
$ilDB, $active_id, $pass, $max, $reached, $isPassed, $obligationsAnswered, $hint_count, $hint_points, $mark) {
1033 $passedOnceBefore = 0;
1034 $query =
"SELECT passed_once FROM tst_result_cache WHERE active_fi = %s";
1036 while ($passed_once_result_row =
$ilDB->fetchAssoc(
$res)) {
1037 $passedOnceBefore = (
int) $passed_once_result_row[
'passed_once'];
1040 $passedOnce = (
int) ($isPassed || $passedOnceBefore);
1043 "DELETE FROM tst_result_cache WHERE active_fi = %s",
1048 $ilDB->insert(
'tst_result_cache', array(
1049 'active_fi' => array(
'integer', $active_id),
1050 'pass' => array(
'integer', strlen($pass) ? $pass : 0),
1051 'max_points' => array(
'float', strlen($max) ? $max : 0),
1052 'reached_points' => array(
'float', strlen($reached) ? $reached : 0),
1053 'mark_short' => array(
'text', strlen($mark[
"short_name"] ??
'') ? $mark[
"short_name"] :
" "),
1054 'mark_official' => array(
'text', strlen($mark[
"official_name"] ??
'') ? $mark[
"official_name"] :
" "),
1055 'passed_once' => array(
'integer', $passedOnce),
1056 'passed' => array(
'integer', (
int) $isPassed),
1057 'failed' => array(
'integer', (
int) !$isPassed),
1058 'tstamp' => array(
'integer', time()),
1059 'hint_count' => array(
'integer', $hint_count),
1060 'hint_points' => array(
'float', $hint_points),
1061 'obligations_answered' => array(
'integer', $obligationsAnswered)
1065 if (is_object($processLocker)) {
1066 $processLocker->executeUserTestResultUpdateLockOperation($userTestResultUpdateCallback);
1068 $userTestResultUpdateCallback();
1074 public static function _updateTestPassResults(
1077 bool $obligationsEnabled =
false,
1079 int $test_obj_id = null
1082 $ilDB = $DIC[
'ilDB'];
1091 $result =
$ilDB->queryF(
1093 SELECT SUM(points) reachedpoints, 1094 SUM(hint_count) hint_count, 1095 SUM(hint_points) hint_points, 1096 COUNT(DISTINCT(question_fi)) answeredquestions 1097 FROM tst_test_result 1098 WHERE active_fi = %s 1101 array(
'integer',
'integer'),
1102 array($active_id, $pass)
1105 if ($result->numRows() > 0) {
1106 if ($obligationsEnabled) {
1108 SELECT answered answ 1109 FROM tst_test_question 1110 INNER JOIN tst_active 1112 AND tst_test_question.test_fi = tst_active.test_fi 1113 LEFT JOIN tst_test_result 1114 ON tst_test_result.active_fi = %s 1115 AND tst_test_result.pass = %s 1116 AND tst_test_question.question_fi = tst_test_result.question_fi 1117 WHERE obligatory = 1';
1119 $result_obligatory =
$ilDB->queryF(
1121 array(
'integer',
'integer',
'integer'),
1122 array($active_id, $active_id, $pass)
1125 $obligations_answered = 1;
1127 while ($row_obligatory =
$ilDB->fetchAssoc($result_obligatory)) {
1128 if (!(
int) $row_obligatory[
'answ']) {
1129 $obligations_answered = 0;
1134 $obligations_answered = 1;
1137 $row =
$ilDB->fetchAssoc($result);
1139 if ($row[
'reachedpoints'] === null) {
1140 $row[
'reachedpoints'] = 0.0;
1142 if ($row[
'hint_count'] === null) {
1143 $row[
'hint_count'] = 0;
1145 if ($row[
'hint_points'] === null) {
1146 $row[
'hint_points'] = 0.0;
1151 $updatePassResultCallback =
function () use (
$ilDB,
$data, $active_id, $pass, $row, $time, $obligations_answered, $exam_identifier) {
1157 'active_fi' => array(
'integer', $active_id),
1158 'pass' => array(
'integer', strlen($pass) ? $pass : 0)),
1160 'points' => array(
'float', $row[
'reachedpoints'] ?: 0),
1161 'maxpoints' => array(
'float',
$data[
'points']),
1162 'questioncount' => array(
'integer',
$data[
'count']),
1163 'answeredquestions' => array(
'integer', $row[
'answeredquestions']),
1164 'workingtime' => array(
'integer', $time),
1165 'tstamp' => array(
'integer', time()),
1166 'hint_count' => array(
'integer', $row[
'hint_count']),
1167 'hint_points' => array(
'float', $row[
'hint_points']),
1168 'obligations_answered' => array(
'integer', $obligations_answered),
1169 'exam_id' => array(
'text', $exam_identifier)
1175 $processLocker->executeUserPassResultUpdateLockOperation($updatePassResultCallback);
1177 $updatePassResultCallback();
1184 'active_fi' => $active_id,
1186 'points' => ($row[
"reachedpoints"]) ?: 0.0,
1187 'maxpoints' =>
$data[
"points"],
1188 'questioncount' =>
$data[
"count"],
1189 'answeredquestions' => $row[
"answeredquestions"],
1190 'workingtime' => $time,
1192 'hint_count' => $row[
'hint_count'],
1193 'hint_points' => $row[
'hint_points'],
1194 'obligations_answered' => $obligations_answered,
1195 'exam_id' => $exam_identifier
1199 public static function logAction(
string $logtext,
int $active_id,
int $question_id): void
1201 $original_id = self::_getOriginalId($question_id);
1222 if (!@is_dir($mediatempdir)) {
1225 $temp_name = tempnam($mediatempdir, $name .
"_____");
1226 $temp_name = str_replace(
"\\",
"/", $temp_name);
1227 @unlink($temp_name);
1236 return CLIENT_WEB_DIR .
"/assessment/$this->obj_id/$this->id/solution/";
1244 public function getImagePath($question_id = null, $object_id = null): string
1246 if ($question_id === null) {
1250 if ($object_id === null) {
1259 return CLIENT_WEB_DIR .
"/assessment/{$parentObjectId}/{$questionId}/images/";
1270 return CLIENT_WEB_DIR .
"/assessment/$this->obj_id/$this->id/flash/";
1290 if (!$this->export_image_path) {
1315 if (!count($solution)) {
1329 if ($pass === null && is_numeric($active_id)) {
1333 if ($this->
getStep() !== null) {
1337 WHERE active_fi = %s 1338 AND question_fi = %s 1342 ORDER BY solution_id";
1344 $result = $this->db->queryF(
1346 array(
'integer',
'integer',
'integer',
'integer',
'integer'),
1347 array((
int) $active_id, $this->
getId(), $pass, $this->
getStep(), (
int) $authorized)
1353 WHERE active_fi = %s 1354 AND question_fi = %s 1357 ORDER BY solution_id 1360 $result = $this->db->queryF(
1362 array(
'integer',
'integer',
'integer',
'integer'),
1363 array((
int) $active_id, $this->
getId(), $pass, (
int) $authorized)
1369 while ($row = $this->db->fetchAssoc($result)) {
1379 public function isInUse(
int $question_id = 0): bool
1389 if ($question_id < 1) {
1390 $question_id = $this->
getId();
1393 $result = $this->db->queryF(
1394 "SELECT COUNT(qpl_questions.question_id) question_count FROM qpl_questions, tst_test_question WHERE qpl_questions.original_id = %s AND qpl_questions.question_id = tst_test_question.question_fi",
1398 $row = $this->db->fetchAssoc($result);
1399 $count = (
int) $row[
"question_count"];
1401 $result = $this->db->queryF(
1403 SELECT tst_active.test_fi 1405 INNER JOIN tst_test_rnd_qst ON tst_test_rnd_qst.question_fi = qpl_questions.question_id 1406 INNER JOIN tst_active ON tst_active.active_id = tst_test_rnd_qst.active_fi 1407 WHERE qpl_questions.original_id = %s 1408 GROUP BY tst_active.test_fi",
1412 $count += (
int) $this->db->numRows($result);
1420 public function isClone(
int $question_id = 0): bool
1422 if ($question_id < 1) {
1425 $result = $this->db->queryF(
1426 "SELECT COUNT(original_id) cnt FROM qpl_questions WHERE question_id = %s",
1430 $row = $this->db->fetchAssoc($result);
1431 return ((
int) $row[
"cnt"]) > 0;
1437 $ilDB = $DIC[
'ilDB'];
1439 $result =
$ilDB->queryF(
1440 "SELECT qpl_qst_type.type_tag FROM qpl_qst_type, qpl_questions WHERE qpl_questions.question_id = %s AND qpl_questions.question_type_fi = qpl_qst_type.question_type_id",
1445 return $data[
"type_tag"] ??
'';
1468 if (!is_array($answer_table_name)) {
1469 $answer_table_name = array($answer_table_name);
1472 foreach ($answer_table_name as $table) {
1473 if (strlen($table)) {
1474 $this->db->manipulateF(
1475 "DELETE FROM $table WHERE question_fi = %s",
1487 if (!is_array($additional_table_name)) {
1488 $additional_table_name = array($additional_table_name);
1491 foreach ($additional_table_name as $table) {
1492 if (strlen($table)) {
1493 $this->db->manipulateF(
1494 "DELETE FROM $table WHERE question_fi = %s",
1510 public function delete(
int $question_id):
void 1512 if ($question_id < 1) {
1516 $result = $this->db->queryF(
1517 "SELECT obj_fi FROM qpl_questions WHERE question_id = %s",
1521 if ($this->db->numRows($result) == 1) {
1522 $row = $this->db->fetchAssoc($result);
1523 $obj_id = $row[
"obj_fi"];
1530 $this->
ilLog->root()->error(
"EXCEPTION: Could not delete page of question $question_id: $e");
1534 $affectedRows = $this->db->manipulateF(
1535 "DELETE FROM qpl_questions WHERE question_id = %s",
1539 if ($affectedRows == 0) {
1549 $this->
ilLog->root()->error(
"EXCEPTION: Could not delete additional table data of question $question_id: $e");
1555 $affectedRows = $this->db->manipulateF(
1556 "DELETE FROM tst_test_question WHERE question_fi = %s",
1561 $this->
ilLog->root()->error(
"EXCEPTION: Could not delete delete question $question_id from a test: $e");
1567 $affectedRows = $this->db->manipulateF(
1568 "DELETE FROM qpl_sol_sug WHERE question_fi = %s",
1573 $this->
ilLog->root()->error(
"EXCEPTION: Could not delete suggested solutions of question $question_id: $e");
1578 $directory =
CLIENT_WEB_DIR .
"/assessment/" . $obj_id .
"/$question_id";
1579 if (preg_match(
"/\d+/", $obj_id) and preg_match(
"/\d+/", $question_id) and is_dir($directory)) {
1583 $this->
ilLog->root()->error(
"EXCEPTION: Could not delete question file directory $directory of question $question_id: $e");
1593 foreach (
$mobs as $mob) {
1601 $this->
ilLog->root()->error(
"EXCEPTION: Error deleting the media objects of question $question_id: $e");
1604 ilAssQuestionHintTracking::deleteRequestsByQuestionIds(array($question_id));
1607 $assignmentList->setParentObjId($obj_id);
1608 $assignmentList->setQuestionIdFilter($question_id);
1609 $assignmentList->loadFromDb();
1610 foreach ($assignmentList->getAssignmentsByQuestionId($question_id) as $assignment) {
1612 $assignment->deleteFromDb();
1615 if (!$assignment->isSkillUsed()) {
1617 $assignment->getParentObjId(),
1618 $assignment->getSkillBaseId(),
1619 $assignment->getSkillTrefId(),
1630 $this->
ilLog->root()->error(
"EXCEPTION: Error updating the question pool question count of question pool " . $this->
getObjId() .
" when deleting question $question_id: $e");
1639 foreach ($taxIds as $taxId) {
1641 $taxNodeAssignment->deleteAssignmentsOfItem($this->
getId());
1648 $result = $this->db->queryF(
1649 "SELECT question_id FROM qpl_questions WHERE original_id = %s OR question_id = %s",
1650 array(
'integer',
'integer'),
1651 array($this->
id, $this->
id)
1653 if ($this->db->numRows($result) == 0) {
1657 while ($row = $this->db->fetchAssoc($result)) {
1658 $found_id[] = $row[
"question_id"];
1661 $result = $this->db->query(
"SELECT * FROM tst_test_result WHERE " . $this->db->in(
'question_fi', $found_id,
false,
'integer'));
1663 return $this->db->numRows($result);
1669 $ilDB = $DIC[
'ilDB'];
1671 $result =
$ilDB->queryF(
1672 "SELECT question_id FROM qpl_questions WHERE original_id = %s OR question_id = %s",
1673 array(
'integer',
'integer'),
1674 array($a_q_id, $a_q_id)
1676 if ($result->numRows() == 0) {
1681 while ($row =
$ilDB->fetchAssoc($result)) {
1682 $found_id[] = $row[
"question_id"];
1685 $result =
$ilDB->query(
"SELECT * FROM tst_test_result WHERE " .
$ilDB->in(
'question_fi', $found_id,
false,
'integer'));
1687 while ($row =
$ilDB->fetchAssoc($result)) {
1688 $reached = $row[
"points"];
1689 $max = self::_getMaximumPoints($row[
"question_fi"]);
1690 $answers[] = array(
"reached" => $reached,
"max" => $max);
1694 foreach ($answers as
$key => $value) {
1695 $max += $value[
"max"];
1696 $reached += $value[
"reached"];
1699 return $reached / $max;
1707 $ilDB = $DIC[
'ilDB'];
1709 $result =
$ilDB->queryF(
1710 "SELECT title FROM qpl_questions WHERE question_id = %s",
1715 if ($result->numRows() == 1) {
1716 $row =
$ilDB->fetchAssoc($result);
1717 return $row[
"title"];
1725 $ilDB = $DIC[
'ilDB'];
1727 $result =
$ilDB->queryF(
1728 "SELECT question_text FROM qpl_questions WHERE question_id = %s",
1733 if ($result->numRows() == 1) {
1734 $row =
$ilDB->fetchAssoc($result);
1735 return $row[
"question_text"] ??
'';
1743 if (!file_exists($file)) {
1747 if (!is_file($file)) {
1751 if (!is_readable($file)) {
1761 foreach (
$mobs as $mob) {
1769 foreach (
$mobs as $mob) {
1778 $this->page->setId($this->
getId());
1779 $this->page->setParentId($qpl_id);
1780 $this->page->setXMLContent(
"<PageObject><PageContent>" .
1781 "<Question QRef=\"il__qst_" . $this->
getId() .
"\"/>" .
1782 "</PageContent></PageObject>");
1783 $this->page->create(
false);
1791 $xml = str_replace(
"il__qst_" . $a_q_id,
"il__qst_" . $this->
id,
$page->getXMLContent());
1792 $this->page->setXMLContent(
$xml);
1793 $this->page->updateFromXML();
1800 return $page->getXMLContent();
1806 $ilDB = $DIC[
'ilDB'];
1808 if ($question_id < 1) {
1811 $result =
$ilDB->queryF(
1812 "SELECT type_tag FROM qpl_questions, qpl_qst_type WHERE qpl_questions.question_id = %s AND qpl_questions.question_type_fi = qpl_qst_type.question_type_id",
1816 if ($result->numRows() == 1) {
1818 return $data[
"type_tag"];
1827 $ilDB = $DIC[
'ilDB'];
1829 if ($question_id < 1) {
1833 $result =
$ilDB->queryF(
1834 "SELECT title FROM qpl_questions WHERE qpl_questions.question_id = %s",
1838 if ($result->numRows() == 1) {
1840 return $data[
"title"];
1857 'ok.svg' =>
'ok.png',
1858 'not_ok.svg' =>
'not_ok.png',
1859 'checkbox_checked.svg' =>
'checkbox_checked.png',
1860 'checkbox_unchecked.svg' =>
'checkbox_unchecked.png',
1861 'radiobutton_checked.svg' =>
'radiobutton_checked.png',
1862 'radiobutton_unchecked.svg' =>
'radiobutton_unchecked.png' 1865 public function fixSvgToPng(
string $imageFilenameContainingString): string
1867 $needles = array_keys(self::$imageSourceFixReplaceMap);
1868 $replacements = array_values(self::$imageSourceFixReplaceMap);
1869 return str_replace($needles, $replacements, $imageFilenameContainingString);
1875 if (preg_match_all(
'/src="(.*?)"/m', $html, $matches)) {
1876 $sources = $matches[1];
1878 $needleReplacementMap = [];
1880 foreach ($sources as $src) {
1883 if (file_exists($file)) {
1887 $levels = explode(DIRECTORY_SEPARATOR, $src);
1888 if (count($levels) < 5 || $levels[0] !==
'Customizing' || $levels[2] !==
'skin') {
1894 if ($levels[4] ===
'Modules' || $levels[4] ===
'Services') {
1895 $component = $levels[4] . DIRECTORY_SEPARATOR . $levels[5];
1901 if (count($needleReplacementMap)) {
1902 $html = str_replace(array_keys($needleReplacementMap), array_values($needleReplacementMap), $html);
1911 $result = $this->db->queryF(
1912 'SELECT external_id FROM qpl_questions WHERE question_id = %s',
1916 if ($this->db->numRows($result) === 1) {
1917 $data = $this->db->fetchAssoc($result);
1918 $this->external_id =
$data[
'external_id'];
1921 $result = $this->db->queryF(
1922 "SELECT internal_link, import_id, subquestion_index, type, value" .
1923 " FROM qpl_sol_sug WHERE question_fi = %s",
1928 $suggestedSolutions = [];
1930 while ($row = $this->db->fetchAssoc($result)) {
1931 $value = $row[
"value"];
1934 $unserializedValue = unserialize($value, [
'allowed_classes' =>
false]);
1935 if (is_array($unserializedValue)) {
1936 $value = $unserializedValue;
1941 if (is_string($value)) {
1945 $suggestedSolutions[$row[
"subquestion_index"]] = [
1946 "type" => $row[
"type"],
1948 "internal_link" => $row[
"internal_link"],
1949 "import_id" => $row[
"import_id"]
1952 $this->suggested_solutions = $suggestedSolutions;
1966 $obj_id = ($this->
getObjId() <= 0) ? (
ilObject::_lookupObjId((strlen($this->dic->testQuestionPool()->internal()->request()->getRefId())) ? $this->dic->testQuestionPool()->internal()->request()->getRefId() : $_POST[
"sel_qpl"])) : $this->
getObjId();
1968 if ($a_create_page) {
1975 $next_id = $this->db->nextId(
'qpl_questions');
1976 $this->db->insert(
"qpl_questions", array(
1977 "question_id" => array(
"integer", $next_id),
1979 "obj_fi" => array(
"integer", $obj_id),
1980 "title" => array(
"text", null),
1981 "description" => array(
"text", null),
1982 "author" => array(
"text", $this->
getAuthor()),
1983 "owner" => array(
"integer",
$ilUser->getId()),
1984 "question_text" => array(
"clob", null),
1985 "points" => array(
"float",
"0.0"),
1987 "complete" => array(
"text", $complete),
1988 "created" => array(
"integer", time()),
1989 "original_id" => array(
"integer", null),
1990 "tstamp" => array(
"integer", $tstamp),
1994 $this->
setId($next_id);
1996 if ($a_create_page) {
2002 return $this->
getId();
2007 if ($this->
getId() == -1) {
2008 $next_id = $this->db->nextId(
'qpl_questions');
2009 $this->db->insert(
"qpl_questions", array(
2010 "question_id" => array(
"integer", $next_id),
2012 "obj_fi" => array(
"integer", $this->
getObjId()),
2013 "title" => array(
"text", $this->
getTitle()),
2014 "description" => array(
"text", $this->
getComment()),
2015 "author" => array(
"text", $this->
getAuthor()),
2016 "owner" => array(
"integer", $this->
getOwner()),
2019 "nr_of_tries" => array(
"integer", $this->
getNrOfTries()),
2020 "created" => array(
"integer", time()),
2021 "original_id" => array(
"integer", ($original_id != -1) ? $original_id : null),
2022 "tstamp" => array(
"integer", time()),
2026 $this->
setId($next_id);
2031 $this->db->update(
"qpl_questions", array(
2032 "obj_fi" => array(
"integer", $this->
getObjId()),
2033 "title" => array(
"text", $this->
getTitle()),
2034 "description" => array(
"text", $this->
getComment()),
2035 "author" => array(
"text", $this->
getAuthor()),
2038 "nr_of_tries" => array(
"integer", $this->
getNrOfTries()),
2039 "tstamp" => array(
"integer", time()),
2040 'complete' => array(
'integer', $this->
isComplete()),
2043 "question_id" => array(
"integer", $this->
getId())
2060 $this->db->update(
'qpl_questions', array(
2061 'tstamp' => array(
'integer', time()),
2062 'owner' => array(
'integer', $this->
getOwner()),
2063 'complete' => array(
'integer', $complete),
2064 'lifecycle' => array(
'text', $this->
getLifecycle()->getIdentifier()),
2066 'question_id' => array(
'integer', $this->
getId())
2076 self::saveOriginalId($this->
getId(), $newId);
2082 $ilDB = $DIC->database();
2083 $query =
"UPDATE qpl_questions SET tstamp = %s, original_id = %s WHERE question_id = %s";
2087 array(
'integer',
'integer',
'text'),
2088 array(time(), $originalId, $questionId)
2095 $ilDB = $DIC->database();
2097 $query =
"UPDATE qpl_questions SET tstamp = %s, original_id = NULL WHERE question_id = %s";
2101 array(
'integer',
'text'),
2102 array(time(), $questionId)
2106 protected function onDuplicate(
int $originalParentId,
int $originalQuestionId,
int $duplicateParentId,
int $duplicateQuestionId): void
2109 $this->feedbackOBJ->duplicateFeedback($originalQuestionId, $duplicateQuestionId);
2114 protected function beforeSyncWithOriginal(
int $origQuestionId,
int $dupQuestionId,
int $origParentObjId,
int $dupParentObjId): void
2118 protected function afterSyncWithOriginal(
int $origQuestionId,
int $dupQuestionId,
int $origParentObjId,
int $dupParentObjId): void
2120 $this->feedbackOBJ->syncFeedback($origQuestionId, $dupQuestionId);
2123 protected function onCopy(
int $sourceParentId,
int $sourceQuestionId,
int $targetParentId,
int $targetQuestionId): void
2128 $this->feedbackOBJ->duplicateFeedback($sourceQuestionId, $targetQuestionId);
2140 $this->db->manipulateF(
2141 "DELETE FROM qpl_sol_sug WHERE question_fi = %s",
2143 array($this->
getId())
2146 $this->suggested_solutions = [];
2155 if (array_key_exists($subquestion_index, $this->suggested_solutions)) {
2156 return $this->suggested_solutions[$subquestion_index];
2170 if (array_key_exists($subquestion_index, $this->suggested_solutions)) {
2171 $title = $this->suggested_solutions[$subquestion_index][
"internal_link"];
2188 public function setSuggestedSolution(
string $solution_id =
"",
int $subquestion_index = 0,
bool $is_import =
false): void
2190 if (strcmp($solution_id,
"") != 0) {
2193 $import_id = $solution_id;
2196 $this->suggested_solutions[$subquestion_index] = array(
2197 "internal_link" => $solution_id,
2198 "import_id" => $import_id
2208 foreach ($this->suggested_solutions as
$index => $solution) {
2209 if (!is_array($solution) ||
2210 !array_key_exists(
"type", $solution) ||
2211 strcmp($solution[
"type"],
"file") !== 0) {
2216 $filepath_original = str_replace(
2217 "/{$this->obj_id}/{$this->id}/solution",
2218 "/$parent_id/$question_id/solution",
2221 if (!file_exists($filepath)) {
2226 if (!copy($filepath_original .
$filename, $filepath . $filename)) {
2227 $this->
ilLog->root()->error(
"File could not be duplicated!!!!");
2228 $this->
ilLog->root()->error(
"object: " . print_r($this,
true));
2235 int $target_question_id,
2239 $filepath_original = str_replace(
2240 "{$this->getObjId()}/{$this->id}/solution",
2241 "{$target_obj_id}/{$target_question_id}/solution",
2245 foreach ($this->suggested_solutions as
$index => $solution) {
2246 if (strcmp($solution[
"type"],
"file") == 0) {
2247 if (!file_exists($filepath_original)) {
2252 if (!@copy($filepath .
$filename, $filepath_original . $filename)) {
2253 $this->
ilLog->root()->error(
"File could not be duplicated!!!!");
2254 $this->
ilLog->root()->error(
"object: " . print_r($this,
true));
2263 foreach ($this->suggested_solutions as
$index => $solution) {
2264 if (strcmp($solution[
"type"],
"file") == 0) {
2266 $filepath_original = str_replace(
"/$this->obj_id/$this->id/solution",
"/$source_questionpool_id/$source_question_id/solution", $filepath);
2267 if (!file_exists($filepath)) {
2272 if (!copy($filepath_original .
$filename, $filepath . $filename)) {
2273 $this->
ilLog->root()->error(
"File could not be copied!!!!");
2274 $this->
ilLog->root()->error(
"object: " . print_r($this,
true));
2284 $this->db->manipulateF(
2285 "DELETE FROM qpl_sol_sug WHERE question_fi = %s",
2290 foreach ($this->suggested_solutions as
$index => $solution) {
2291 $next_id = $this->db->nextId(
'qpl_sol_sug');
2293 $value = $solution[
'value'] ??
'';
2294 if (is_array($value)) {
2295 $value = serialize($value);
2301 'suggested_solution_id' => array(
'integer', $next_id),
2302 'question_fi' => array(
'integer', $id ),
2303 'type' => array(
'text', $solution[
'type'] ??
''),
2305 'internal_link' => array(
'text', $solution[
'internal_link'] ??
''),
2306 'import_id' => array(
'text', null),
2307 'subquestion_index' => array(
'integer',
$index),
2308 'tstamp' => array(
'integer', time()),
2311 if (preg_match(
"/il_(\d*?)_(\w+)_(\d+)/", $solution[
"internal_link"], $matches)) {
2315 if ($original_id !== -1
2316 && $original_obj_id !== -1) {
2331 public function saveSuggestedSolution(
string $type, $solution_id =
"",
int $subquestion_index = 0, $value =
""):
void 2333 $this->db->manipulateF(
2334 "DELETE FROM qpl_sol_sug WHERE question_fi = %s AND subquestion_index = %s",
2335 array(
"integer",
"integer"),
2342 $next_id = $this->db->nextId(
'qpl_sol_sug');
2344 $affectedRows = $this->db->insert(
2347 'suggested_solution_id' => array(
'integer', $next_id ),
2348 'question_fi' => array(
'integer', $this->
getId() ),
2349 'type' => array(
'text', $type ),
2351 'internal_link' => array(
'text', $solution_id ),
2352 'import_id' => array(
'text', null ),
2353 'subquestion_index' => array(
'integer', $subquestion_index ),
2354 'tstamp' => array(
'integer', time() ),
2357 if ($affectedRows == 1) {
2358 $this->suggested_solutions[$subquestion_index] = array(
2361 "internal_link" => $solution_id,
2370 if (preg_match(
"/il_(\d+)_(\w+)_(\d+)/", $internal_link, $matches)) {
2371 switch ($matches[2]) {
2388 if (strcmp($resolved_link,
"") == 0) {
2389 $resolved_link = $internal_link;
2392 $resolved_link = $internal_link;
2394 return $resolved_link;
2400 $result = $this->db->queryF(
2401 "SELECT * FROM qpl_sol_sug WHERE question_fi = %s",
2405 if ($this->db->numRows($result) > 0) {
2406 while ($row = $this->db->fetchAssoc($result)) {
2407 $internal_link = $row[
"internal_link"];
2409 if (strcmp($internal_link, $resolved_link) != 0) {
2411 $affectedRows = $this->db->manipulateF(
2412 "UPDATE qpl_sol_sug SET internal_link = %s WHERE suggested_solution_id = %s",
2413 array(
'text',
'integer'),
2414 array($resolved_link, $row[
"suggested_solution_id"])
2420 if ($resolvedlinks) {
2423 $result = $this->db->queryF(
2424 "SELECT * FROM qpl_sol_sug WHERE question_fi = %s",
2428 if ($this->db->numRows($result) > 0) {
2429 while ($row = $this->db->fetchAssoc($result)) {
2430 if (preg_match(
"/il_(\d*?)_(\w+)_(\d+)/", $row[
"internal_link"], $matches)) {
2442 "lm" =>
"LearningModule",
2443 "pg" =>
"PageObject",
2444 "st" =>
"StructureObject",
2445 "git" =>
"GlossaryItem",
2446 "mob" =>
"MediaObject" 2449 if (preg_match(
"/il__(\w+)_(\d+)/", $target, $matches)) {
2450 $type = $matches[1];
2452 switch ($linktypes[$matches[1]]) {
2454 $href =
"./ilias.php?baseClass=ilLMPresentationGUI&obj_type=" . $linktypes[
$type]
2455 .
"&cmd=media&ref_id=" . $DIC->testQuestionPool()->internal()->request()->getRefId()
2458 case "StructureObject":
2459 case "GlossaryItem":
2461 case "LearningModule":
2463 $href =
"./goto.php?target=" . $type .
"_" .
$target_id;
2473 $ilDB = $DIC[
'ilDB'];
2474 $result =
$ilDB->queryF(
2475 "SELECT * FROM qpl_questions WHERE question_id = %s",
2479 if (
$ilDB->numRows($result) > 0) {
2480 $row =
$ilDB->fetchAssoc($result);
2481 if ($row[
"original_id"] > 0) {
2482 return $row[
"original_id"];
2485 return (
int) $row[
"question_id"];
2494 $ilDB = $DIC[
'ilDB'];
2497 SELECT COUNT(dupl.question_id) cnt 2498 FROM qpl_questions dupl 2499 INNER JOIN qpl_questions orig 2500 ON orig.question_id = dupl.original_id 2501 WHERE dupl.question_id = %s 2507 return $row[
'cnt'] > 0;
2515 $currentID = $this->
getId();
2518 $originalObjId = self::lookupParentObjId($this->
getOriginalId());
2520 if (!$originalObjId) {
2538 $this->
setId($currentID);
2558 if ($question_id < 1) {
2562 $result = $this->db->queryF(
2563 "SELECT question_id FROM qpl_questions WHERE question_id = %s",
2567 return $result->numRows() == 1;
2572 if ($question_id < 1) {
2576 $result = $this->db->queryF(
2577 "SELECT question_id FROM qpl_questions INNER JOIN object_data ON obj_fi = obj_id WHERE question_id = %s AND type = 'qpl'",
2581 return $this->db->numRows($result) == 1;
2590 return self::_instantiateQuestion($question_id);
2598 return self::instantiateQuestion($question_id);
2609 $ilCtrl = $DIC[
'ilCtrl'];
2610 $ilDB = $DIC[
'ilDB'];
2614 if ($question_type ===
'') {
2619 $question =
new $question_type();
2620 $question->loadFromDb($question_id);
2622 $feedbackObjectClassname = self::getFeedbackClassNameByQuestionType($question_type);
2623 $question->feedbackOBJ =
new $feedbackObjectClassname($question, $ilCtrl,
$ilDB, $lng);
2630 if (strcmp($this->points,
"") == 0) {
2644 return self::_getSolutionMaxPass($this->
getId(), $active_id);
2660 $ilDB = $DIC[
'ilDB'];
2662 $result =
$ilDB->queryF(
2663 "SELECT MAX(pass) maxpass FROM tst_test_result WHERE active_fi = %s AND question_fi = %s",
2664 array(
'integer',
'integer'),
2665 array($active_id, $question_id)
2667 if ($result->numRows() === 1) {
2668 $row =
$ilDB->fetchAssoc($result);
2669 return $row[
"maxpass"];
2678 $ilDB = $DIC[
'ilDB'];
2680 if (($question_id < 1) || ($user_id < 1)) {
2684 $result =
$ilDB->queryF(
2685 "SELECT obj_fi FROM qpl_questions WHERE question_id = %s",
2689 if (
$ilDB->numRows($result) == 1) {
2690 $row =
$ilDB->fetchAssoc($result);
2691 $qpl_object_id = (
int) $row[
"obj_fi"];
2701 $ilDB = $DIC[
'ilDB'];
2703 $result =
$ilDB->queryF(
2704 "SELECT test_random_question_id FROM tst_test_rnd_qst WHERE question_fi = %s",
2708 return $ilDB->numRows($result) > 0;
2722 abstract public function calculateReachedPoints($active_id, $pass = null, $authorizedSolution =
true, $returndetails =
false);
2730 $reachedPoints = $reachedPoints - $requestsStatisticData->getRequestsPoints();
2732 return $reachedPoints;
2745 return $points > 0 ?
$points : 0;
2769 if ($count_system == 1) {
2775 if ($score_cutting == 0) {
2789 return self::lookupResultRecordExist($active_id, $question_id, $pass);
2798 public static function _areAnswered(
int $a_user_id, array $a_question_ids): bool
2801 $ilDB = $DIC[
'ilDB'];
2804 "SELECT DISTINCT(question_fi) FROM tst_test_result JOIN tst_active " .
2805 "ON (active_id = active_fi) " .
2806 "WHERE " .
$ilDB->in(
'question_fi', $a_question_ids,
false,
'integer') .
2807 " AND user_fi = %s",
2811 return (
$res->numRows() == count($a_question_ids)) ?
true :
false;
2827 public function prepareTextareaOutput(
string $txt_output,
bool $prepare_for_latex_output =
false,
bool $omitNl2BrWhenTextArea =
false)
2831 $prepare_for_latex_output,
2832 $omitNl2BrWhenTextArea
2846 if (strcmp($material[
"type"],
"mattext") == 0) {
2847 $result .= $material[
"material"]->getContent();
2849 if (strcmp($material[
"type"],
"matimage") == 0) {
2850 $matimage = $material[
"material"];
2851 if (preg_match(
"/(il_([0-9]+)_mob_([0-9]+))/", $matimage->getLabel(), $matches)) {
2853 "mob" => $matimage->getLabel(),
2854 "uri" => $matimage->getUri()
2867 "texttype" =>
"text/plain" 2869 if ($this->
isHTML($a_material)) {
2870 $attrs[
"texttype"] =
"text/xhtml";
2875 foreach (
$mobs as $mob) {
2876 $moblabel =
"il_" .
IL_INST_ID .
"_mob_" . $mob;
2877 if (strpos($a_material,
"mm_$mob") !==
false) {
2881 "label" => $moblabel,
2882 "uri" =>
"objects/" .
"il_" .
IL_INST_ID .
"_mob_" . $mob .
"/" . $mob_obj->getTitle()
2885 $a_xml_writer->
xmlElement(
"matimage", $imgattrs, null);
2889 if ($close_material_tag) {
2898 if (preg_match(
"/.*\.(png|jpg|gif|jpeg)$/i", $plain_image_filename, $matches)) {
2899 $extension =
"." . $matches[1];
2903 $plain_image_filename = uniqid($plain_image_filename . microtime(
true),
true);
2906 return md5($plain_image_filename) . $extension;
2919 public static function _setReachedPoints(
int $active_id,
int $question_id,
float $points,
float $maxpoints,
int $pass,
bool $manualscoring,
bool $obligationsEnabled): bool
2922 $ilDB = $DIC[
'ilDB'];
2923 $refinery = $DIC[
'refinery'];
2925 $float_trafo = $refinery->kindlyTo()->float();
2927 $points = $float_trafo->transform($points);
2932 if ($points <= $maxpoints) {
2933 if ($pass === null) {
2940 if ($pass !== null) {
2941 $result =
$ilDB->queryF(
2942 "SELECT points FROM tst_test_result WHERE active_fi = %s AND question_fi = %s AND pass = %s",
2943 array(
'integer',
'integer',
'integer'),
2944 array($active_id, $question_id, $pass)
2946 $manual = ($manualscoring) ? 1 : 0;
2947 $rowsnum = $result->numRows();
2950 $row =
$ilDB->fetchAssoc($result);
2951 $old_points = $row[
"points"];
2952 if ($old_points != $points) {
2953 $affectedRows =
$ilDB->manipulateF(
2954 "UPDATE tst_test_result SET points = %s, manual = %s, tstamp = %s WHERE active_fi = %s AND question_fi = %s AND pass = %s",
2955 array(
'float',
'integer',
'integer',
'integer',
'integer',
'integer'),
2956 array($points, $manual, time(), $active_id, $question_id, $pass)
2960 $next_id =
$ilDB->nextId(
'tst_test_result');
2961 $affectedRows =
$ilDB->manipulateF(
2962 "INSERT INTO tst_test_result (test_result_id, active_fi, question_fi, points, pass, manual, tstamp) VALUES (%s, %s, %s, %s, %s, %s, %s)",
2963 array(
'integer',
'integer',
'integer',
'float',
'integer',
'integer',
'integer'),
2964 array($next_id, $active_id, $question_id, $points, $pass, $manual, time())
2968 if (self::isForcePassResultUpdateEnabled() || $old_points != $points || $rowsnum == 0) {
2969 assQuestion::_updateTestPassResults($active_id, $pass, $obligationsEnabled);
2979 "log_answer_changed_points",
2986 ), $active_id, $question_id);
3010 || !(
new ilSetting(
'advanced_editing'))->
get(
'advanced_editing_javascript_editor') ===
'tinymce') {
3011 $purified_content = nl2br($purified_content);
3034 $result = $this->db->queryF(
3035 "SELECT question_type_id FROM qpl_qst_type WHERE type_tag = %s",
3039 if ($this->db->numRows($result) == 1) {
3040 $row = $this->db->fetchAssoc($result);
3041 return (
int) $row[
"question_type_id"];
3049 $this->db->manipulateF(
3050 "DELETE FROM qpl_hints WHERE qht_question_fi = %s",
3052 array($this->original_id)
3056 $result = $this->db->queryF(
3057 "SELECT * FROM qpl_hints WHERE qht_question_fi = %s",
3059 array($this->
getId())
3063 if ($this->db->numRows($result) > 0) {
3064 while ($row = $this->db->fetchAssoc($result)) {
3065 $next_id = $this->db->nextId(
'qpl_hints');
3069 'qht_hint_id' => array(
'integer', $next_id),
3070 'qht_question_fi' => array(
'integer', $this->original_id),
3071 'qht_hint_index' => array(
'integer', $row[
"qht_hint_index"]),
3072 'qht_hint_points' => array(
'float', $row[
"qht_hint_points"]),
3073 'qht_hint_text' => array(
'text', $row[
"qht_hint_text"]),
3085 $collected .= $this->feedbackOBJ->getGenericFeedbackContent($this->
getId(),
false);
3086 $collected .= $this->feedbackOBJ->getGenericFeedbackContent($this->
getId(),
true);
3087 $collected .= $this->feedbackOBJ->getAllSpecificAnswerFeedbackContents($this->
getId());
3089 foreach ($this->suggested_solutions as $solution_array) {
3090 if (is_string([
'value'])) {
3091 $collected .= $solution_array[
"value"];
3095 foreach ($questionHintList as $questionHint) {
3097 $collected .= $questionHint->getText();
3111 $result = $this->db->queryF(
3112 "SELECT question_id FROM qpl_questions WHERE original_id = %s",
3114 array($this->
getId())
3118 while ($row = $this->db->fetchAssoc($result)) {
3119 $ids[] = $row[
"question_id"];
3121 foreach ($ids as $question_id) {
3123 $result = $this->db->queryF(
3124 "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",
3128 while ($row = $this->db->fetchAssoc($result)) {
3132 $result = $this->db->queryF(
3133 "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",
3137 while ($row = $this->db->fetchAssoc($result)) {
3141 foreach ($instances as
$key => $value) {
3151 if (in_array($questiontype, $scoring)) {
3166 $result = $this->db->queryF(
3167 "SELECT * FROM tst_active WHERE active_id = %s",
3171 if ($this->db->numRows($result)) {
3172 $row = $this->db->fetchAssoc($result);
3173 return array(
"user_id" => $row[
"user_fi"],
"test_id" => $row[
"test_fi"]);
3181 return static::HAS_SPECIFIC_FEEDBACK;
3186 if (self::isCoreQuestionType($question_type)) {
3187 self::includeCoreClass($question_type, $gui);
3193 return str_replace(
'ass',
'ilAss', $questionType) .
'Feedback';
3198 return file_exists(
"Modules/TestQuestionPool/classes/class.{$questionType}GUI.php");
3203 if ($withGuiClass) {
3208 $feedbackClassName = self::getFeedbackClassNameByQuestionType($questionType);
3214 if (file_exists(
"./Modules/TestQuestionPool/classes/class." . $type_tag .
".php")) {
3216 return $lng->
txt($type_tag);
3218 $component_factory = $DIC[
'component.factory'];
3220 foreach ($component_factory->getActivePluginsInSlot(
"qst") as $pl) {
3221 if ($pl->getQuestionType() === $type_tag) {
3222 return $pl->getQuestionTypeTranslation();
3234 return self::instantiateQuestionGUI($question_id);
3242 $ilCtrl = $DIC[
'ilCtrl'];
3243 $ilDB = $DIC[
'ilDB'];
3247 if (strcmp($a_question_id,
"") != 0) {
3252 $question_type_gui = $question_type .
'GUI';
3253 $question_gui =
new $question_type_gui();
3254 $question_gui->object->loadFromDb($a_question_id);
3256 $feedbackObjectClassname = self::getFeedbackClassNameByQuestionType($question_type);
3257 $question_gui->object->feedbackOBJ =
new $feedbackObjectClassname($question_gui->object, $ilCtrl,
$ilDB, $lng);
3259 $assSettings =
new ilSetting(
'assessment');
3261 $processLockerFactory->setQuestionId($question_gui->object->getId());
3262 $processLockerFactory->setUserId(
$ilUser->getId());
3264 $question_gui->object->setProcessLocker($processLockerFactory->getLocker());
3267 $ilLog = $DIC[
'ilLog'];
3268 $ilLog->write(
'Instantiate question called without question id. (instantiateQuestionGUI@assQuestion)', $ilLog->WARNING);
3269 throw new InvalidArgumentException(
'Instantiate question called without question id. (instantiateQuestionGUI@assQuestion)');
3271 return $question_gui;
3289 throw new BadMethodCallException(
'assQuestion::__get is discouraged, used with: ' . $value);
3299 throw new BadMethodCallException(
'assQuestion::__set is discouraged, used with: ' .
$key);
3309 throw new BadMethodCallException(
'assQuestion::__isset is discouraged, used with: ' .
$key);
3319 $this->nr_of_tries = $a_nr_of_tries;
3324 $this->export_image_path =
$path;
3330 $ilDB = $DIC[
'ilDB'];
3332 if ($question_id < 1) {
3336 $result =
$ilDB->queryF(
3337 "SELECT question_fi FROM tst_test_question WHERE question_fi = %s AND test_fi = %s",
3338 array(
'integer',
'integer'),
3339 array($question_id, $test_id)
3341 return $ilDB->numRows($result) == 1;
3351 return new \ilAssSelfAssessmentQuestionFormatter();
3372 $this->feedbackOBJ->migrateContentForLearningModule($migrator, $this->
getId());
3410 $ilDB = $DIC[
'ilDB'];
3412 $query =
"SELECT obj_fi FROM qpl_questions WHERE question_id = %s";
3417 return $row[
'obj_fi'];
3428 return self::lookupParentObjId($originalQuestionId);
3436 foreach ($hintIds as $originalHintId => $duplicateHintId) {
3439 $originalXML = $originalPageObject->getXMLContent();
3442 $duplicatePageObject->setId($duplicateHintId);
3443 $duplicatePageObject->setParentId($this->
getId());
3444 $duplicatePageObject->setXMLContent($originalXML);
3445 $duplicatePageObject->createFromXML();
3453 $assignmentList->setParentObjId($srcParentId);
3454 $assignmentList->setQuestionIdFilter($srcQuestionId);
3455 $assignmentList->loadFromDb();
3457 foreach ($assignmentList->getAssignmentsByQuestionId($srcQuestionId) as $assignment) {
3458 $assignment->setParentObjId($trgParentId);
3459 $assignment->setQuestionId($trgQuestionId);
3460 $assignment->saveToDb();
3465 $assignment->getSkillBaseId(),
3466 $assignment->getSkillTrefId()
3471 public function syncSkillAssignments(
int $srcParentId,
int $srcQuestionId,
int $trgParentId,
int $trgQuestionId): void
3474 $assignmentList->setParentObjId($trgParentId);
3475 $assignmentList->setQuestionIdFilter($trgQuestionId);
3476 $assignmentList->loadFromDb();
3478 foreach ($assignmentList->getAssignmentsByQuestionId($trgQuestionId) as $assignment) {
3479 $assignment->deleteFromDb();
3482 if (!$assignment->isSkillUsed()) {
3484 $assignment->getParentObjId(),
3485 $assignment->getSkillBaseId(),
3486 $assignment->getSkillTrefId(),
3499 $pageObject->setParentId($this->
getId());
3500 $pageObject->setId($pageObjectId);
3501 $pageObject->createFromXML();
3523 $ilDB = $DIC[
'ilDB'];
3526 SELECT count(active_fi) cnt 3530 WHERE active_fi = %s 3531 AND question_fi = %s 3537 array(
'integer',
'integer',
'integer'),
3538 array($activeId, $questionId, $pass)
3543 return (
int) $row[
'cnt'];
3577 self::ADDITIONAL_CONTENT_EDITING_MODE_RTE,
3578 self::ADDITIONAL_CONTENT_EDITING_MODE_IPE
3601 SELECT qpl_questions.*, 3602 {$this->getAdditionalTableName()}.* 3604 LEFT JOIN {$this->getAdditionalTableName()} 3605 ON {$this->getAdditionalTableName()}.question_fi = qpl_questions.question_id 3606 WHERE qpl_questions.question_id = %s 3622 if ($this->
getStep() !== null) {
3626 WHERE active_fi = %s 3627 AND question_fi = %s 3633 return $this->db->queryF(
3635 array(
'integer',
'integer',
'integer',
'integer',
'integer'),
3636 array($active_id, $this->
getId(), $pass, $this->
getStep(), (
int) $authorized)
3643 WHERE active_fi = %s 3644 AND question_fi = %s 3649 return $this->db->queryF(
3651 array(
'integer',
'integer',
'integer',
'integer'),
3652 array($active_id, $this->
getId(), $pass, (
int) $authorized)
3658 return $this->db->manipulateF(
3659 "DELETE FROM tst_solutions WHERE solution_id = %s",
3671 $result = $this->db->queryF(
3672 "SELECT * FROM tst_solutions WHERE solution_id = %s",
3677 if ($this->db->numRows($result) > 0) {
3678 return $this->db->fetchAssoc($result);
3686 $this->
getProcessLocker()->executeUserSolutionUpdateLockOperation(
function () use ($active_id, $pass) {
3696 if ($this->
getStep() !== null) {
3698 DELETE FROM tst_solutions 3699 WHERE active_fi = %s 3700 AND question_fi = %s 3706 return $this->db->manipulateF(
3708 array(
'integer',
'integer',
'integer',
'integer',
'integer'),
3709 array($active_id, $this->
getId(), $pass, $this->
getStep(), (
int) $authorized)
3714 DELETE FROM tst_solutions 3715 WHERE active_fi = %s 3716 AND question_fi = %s 3721 return $this->db->manipulateF(
3723 array(
'integer',
'integer',
'integer',
'integer'),
3724 array($active_id, $this->
getId(), $pass, (
int) $authorized)
3731 $next_id = $this->db->nextId(
"tst_solutions");
3734 "solution_id" => array(
"integer", $next_id),
3735 "active_fi" => array(
"integer", $active_id),
3736 "question_fi" => array(
"integer", $this->
getId()),
3737 "value1" => array(
"clob", $value1),
3738 "value2" => array(
"clob", $value2),
3739 "pass" => array(
"integer", $pass),
3740 "tstamp" => array(
"integer", ((
int) $tstamp > 0) ? (
int) $tstamp : time()),
3741 'authorized' => array(
'integer', (
int) $authorized)
3744 if ($this->
getStep() !== null) {
3745 $fieldData[
'step'] = array(
"integer", $this->
getStep());
3748 return $this->db->insert(
"tst_solutions", $fieldData);
3755 "value1" => array(
"clob", $value1),
3756 "value2" => array(
"clob", $value2),
3757 "tstamp" => array(
"integer", time()),
3758 'authorized' => array(
'integer', (
int) $authorized)
3761 if ($this->
getStep() !== null) {
3762 $fieldData[
'step'] = array(
"integer", $this->
getStep());
3765 return $this->db->update(
"tst_solutions", $fieldData, array(
3766 'solution_id' => array(
'integer', $solutionId)
3774 'authorized' => array(
'integer', (
int) $authorized)
3778 $fieldData[
'tstamp'] = array(
'integer', time());
3782 'question_fi' => array(
'integer', $this->
getId()),
3783 'active_fi' => array(
'integer', $activeId),
3784 'pass' => array(
'integer', $pass)
3787 if ($this->
getStep() !== null) {
3788 $whereData[
'step'] = array(
"integer", $this->
getStep());
3791 return $this->db->update(
'tst_solutions', $fieldData, $whereData);
3810 foreach ($this->
getSolutionValues($activeId, $passIndex,
false) as $solutionRec) {
3811 if ($solutionRec[
'value1'] ==
'' && $solutionRec[
'value2'] ==
'') {
3819 return !strlen($solutionRecord[
'value1']) && !strlen($solutionRecord[
'value2']);
3824 $types = array(
"integer",
"integer",
"integer",
"integer");
3825 $values = array($activeId, $this->
getId(), $passIndex, (
int) $authorized);
3826 $valuesCondition = [];
3828 foreach ($matchValues as $valueField => $value) {
3829 switch ($valueField) {
3832 $valuesCondition[] =
"{$valueField} = %s";
3842 $valuesCondition = implode(
' AND ', $valuesCondition);
3845 DELETE FROM tst_solutions 3846 WHERE active_fi = %s 3847 AND question_fi = %s 3850 AND $valuesCondition 3853 if ($this->
getStep() !== null) {
3854 $query .=
" AND step = %s ";
3855 $types[] =
'integer';
3859 $this->db->manipulateF(
$query, $types, $values);
3865 $this->
saveCurrentSolution($activeId, $passIndex, $rec[
'value1'], $rec[
'value2'],
true, $rec[
'tstamp']);
3873 if (!count($intermediateSolution)) {
3881 if ($considerDummyRecordCreation) {
3897 $this->step =
$step;
3911 $time_array = explode(
':', $time);
3912 if (count($time_array) == 3) {
3913 $sec += (
int) $time_array[0] * 3600;
3914 $sec += (
int) $time_array[1] * 60;
3915 $sec += (
int) $time_array[2];
3922 return json_encode([]);
3925 abstract public function duplicate(
bool $for_test =
true,
string $title =
"",
string $author =
"",
string $owner =
"", $testObjId = null):
int;
3931 return (
bool) $solutionAvailability[
'intermediate'];
3936 if ($pass === null) {
3940 return (
bool) $solutionAvailability[
'authorized'];
3946 return $solutionAvailability[
'authorized'] || $solutionAvailability[
'intermediate'];
3952 $result = $this->db->queryF(
3953 "SELECT MAX(step) max_step FROM tst_solutions WHERE active_fi = %s AND pass = %s AND question_fi = %s",
3954 array(
"integer",
"integer",
"integer"),
3955 array($active_id, $pass, $this->
getId())
3958 $row = $this->db->fetchAssoc($result);
3960 return (
int) $row[
'max_step'];
3971 'authorized' =>
false,
3972 'intermediate' =>
false 3976 SELECT authorized, COUNT(*) cnt 3978 WHERE active_fi = %s 3979 AND question_fi = %s 3983 if ($this->
getStep() !== null) {
3984 $query .=
" AND step = " . $this->db->quote((
int) $this->
getStep(),
'integer') .
" ";
3991 $result = $this->db->queryF(
$query, array(
'integer',
'integer',
'integer'), array($activeId, $this->
getId(), $pass));
3993 while ($row = $this->db->fetchAssoc($result)) {
3994 if ($row[
'authorized']) {
3995 $return[
'authorized'] = $row[
'cnt'] > 0;
3997 $return[
'intermediate'] = $row[
'cnt'] > 0;
4015 $query =
"DELETE FROM tst_solutions WHERE question_fi = %s";
4016 $this->db->manipulateF(
$query, array(
'integer'), array($this->
getId()));
4022 DELETE FROM tst_solutions 4023 WHERE active_fi = %s 4024 AND question_fi = %s 4028 if ($this->
getStep() !== null) {
4029 $query .=
" AND step = " . $this->db->quote((
int) $this->
getStep(),
'integer') .
" ";
4032 return $this->db->manipulateF(
4034 array(
'integer',
'integer',
'integer'),
4035 array($activeId, $this->
getId(), $pass)
4044 $this->
log($activeId,
"log_user_solution_willingly_deleted");
4046 self::_updateTestPassResults(
4058 DELETE FROM tst_test_result 4059 WHERE active_fi = %s 4060 AND question_fi = %s 4064 if ($this->
getStep() !== null) {
4065 $query .=
" AND step = " . $this->db->quote((
int) $this->
getStep(),
'integer') .
" ";
4068 return $this->db->manipulateF(
4070 array(
'integer',
'integer',
'integer'),
4071 array($activeId, $this->
getId(), $pass)
4078 $ilDB = $DIC[
'ilDB'];
4080 $IN_questionIds =
$ilDB->in(
'question_fi', $questionIds,
false,
'integer');
4084 FROM tst_test_result 4085 WHERE active_fi = %s 4092 array(
'integer',
'integer'),
4093 array($activeId, $pass)
4096 return $row[
'cnt'] < count($questionIds);
4102 $ilDB = $DIC[
'ilDB'];
4104 $IN_questionIds =
$ilDB->in(
'question_fi', $questionIds,
false,
'integer');
4108 FROM tst_test_result 4109 WHERE active_fi = %s 4116 array(
'integer',
'integer'),
4117 array($activeId, $pass)
4120 $questionsHavingResultRecord = [];
4122 while ($row =
$ilDB->fetchAssoc(
$res)) {
4123 $questionsHavingResultRecord[] = $row[
'question_fi'];
4126 $questionsMissingResultRecordt = array_diff(
4128 $questionsHavingResultRecord
4131 return $questionsMissingResultRecordt;
4137 $ilDB = $DIC[
'ilDB'];
4141 FROM tst_test_result 4142 WHERE active_fi = %s 4143 AND question_fi = %s 4147 $row =
$ilDB->fetchAssoc(
$ilDB->queryF(
$query, array(
'integer',
'integer',
'integer'), array($activeId, $questionId, $pass)));
4149 return $row[
'cnt'] > 0;
4156 foreach ($indexedValues as $value1 => $value2) {
4157 $valuePairs[] = array(
'value1' => $value1,
'value2' => $value2);
4165 $indexedValues = [];
4167 foreach ($valuePairs as $valuePair) {
4168 $indexedValues[ $valuePair[
'value1'] ] = $valuePair[
'value2'];
4171 return $indexedValues;
4186 $this->db->manipulateF(
4187 "UPDATE qpl_questions SET tstamp = %s WHERE question_id = %s",
4188 array(
'integer',
'integer'),
4189 array(time(), $this->
getId())
4203 if ($this->testQuestionConfigInstance === null) {
4231 $query =
'SELECT user_fi FROM tst_active ' . PHP_EOL
4232 .
'JOIN tst_test_question ON tst_test_question.test_fi = tst_active.test_fi ' . PHP_EOL
4233 .
'JOIN qpl_questions ON qpl_questions.question_id = tst_test_question.question_fi ' . PHP_EOL
4234 .
'WHERE qpl_questions.obj_fi = ' . $this->db->quote($this->
getObjId(),
'integer');
4237 return $res->numRows() > 0;
4256 return preg_replace(self::TRIM_PATTERN,
'', $value);
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)
static _getUserIdFromActiveId($active_id)
getSolutionValues($active_id, $pass=null, bool $authorized=true)
Loads solutions of a given user from the database an returns it.
const ADDITIONAL_CONTENT_EDITING_MODE_IPE
setNrOfTries(int $a_nr_of_tries)
ilTestQuestionConfig $testQuestionConfig
ilAssQuestionFeedback $feedbackOBJ
syncSkillAssignments(int $srcParentId, int $srcQuestionId, int $trgParentId, int $trgQuestionId)
static get(string $a_var)
ilGlobalPageTemplate $tpl
static _getManualScoringTypes()
Retrieve the manual scoring settings as type strings.
static getListByQuestionId($questionId)
instantiates a question hint list for the passed question id
deletePageOfQuestion(int $question_id)
static _getWorkingTimeOfParticipantForPass($active_id, $pass)
Returns the complete working time in seconds for a test participant.
static _getSuggestedSolutionCount(int $question_id)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
authorizedOrIntermediateSolutionExists(int $active_id, int $pass)
static getAllowedImageMaterialFileExtensions()
bool $obligationsToBeConsidered
getActiveUserData(int $active_id)
Returns the user id and the test id for a given active id.
static _addLog( $user_id, $object_id, $logtext, $question_id=0, $original_id=0, $test_only=false, $test_ref_id=0)
Add an assessment log entry.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
migrateContentForLearningModule(ilAssSelfAssessmentMigrator $migrator)
static _instanciateQuestionGUI(int $question_id)
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.
static _getPass($active_id)
Retrieves the actual pass of a given user for a given test.
static _getParticipantData($active_id)
Retrieves a participant name from active id.
setOutputType(int $outputType=OUTPUT_HTML)
static _getQuestionType(int $question_id)
getSuggestedSolutionPath()
static lookupResultRecordExist(int $activeId, int $questionId, int $pass)
getTitleFilenameCompliant()
txtlng(string $a_module, string $a_topic, string $a_language)
gets the text for a given topic in a given language if the topic is not in the list, the topic itself with "-" will be returned
txt(string $a_topic, string $a_default_lang_fallback_mod="")
gets the text for a given topic if the topic is not in the list, the topic itself with "-" will be re...
ILIAS HTTP Services $http
static _updateTestResultCache(int $active_id, ilAssQuestionProcessLocker $processLocker=null)
Move this to a proper place.
getQuestionType()
Returns the question type of the question.
removeResultRecord(int $activeId, int $pass)
deleteAdditionalTableData(int $question_id)
usageNumber(int $question_id=0)
Returns the number of place the question is in use in pools or tests.
static includeCoreClass($questionType, $withGuiClass)
deleteAnswers(int $question_id)
static _getSuggestedSolutionOutput(int $question_id)
updateCurrentSolutionsAuthorization(int $activeId, int $pass, bool $authorized, bool $keepTime=false)
static isHTML(string $a_text)
Checks if a given string contains HTML or not.
_questionExistsInPool(int $question_id)
buildTestPresentationConfig()
build basic test question configuration instance
static fetchMimeTypeIdentifier(string $contentType)
__set($key, $value)
Object setter.
resetUsersAnswer(int $activeId, int $pass)
__get($value)
Object getter.
Abstract basic class which is to be extended by the concrete assessment question type classes...
static isFileAvailable(string $file)
setProcessLocker(ilAssQuestionProcessLocker $processLocker)
copyPageOfQuestion(int $a_q_id)
Class ChatMainBarProvider .
static isObligationPossible(int $questionId)
persistWorkingState(int $active_id, $pass, bool $obligationsEnabled=false, bool $authorized=true)
persists the working state for current testactive and testpass
static lookupOriginalParentObjId(int $originalQuestionId)
returns the parent object id for given original question id (should be a qpl id, but theoretically it...
saveWorkingData(int $active_id, int $pass, bool $authorized=true)
Saves the learners input of the question to the database.
addQTIMaterial(ilXmlWriter $a_xml_writer, string $a_material, bool $close_material_tag=true, bool $add_mobs=true)
static _getQuestionTitle(int $question_id)
static _getAllReferences(int $id)
get all reference ids for object ID
ilAssQuestionProcessLocker $processLocker
static getImagePath(string $img, string $module_path="", string $mode="output", bool $offline=false)
get image path (for images located in a template directory)
ensureHintPageObjectExists($pageObjectId)
setSelfAssessmentEditingMode(bool $selfassessmenteditingmode)
getColumnCoord(int $a_col)
Get column "name" from number.
adjustReachedPointsByScoringOptions($points, $active_id, $pass=null)
Adjust the given reached points by checks for all special scoring options in the test container...
ensureNonNegativePoints($points)
isDummySolutionRecord(array $solutionRecord)
bool $shuffle
Indicates whether the answers will be shuffled or not.
static _getInternalLinkHref(string $target="")
__construct(string $title="", string $comment="", string $author="", int $owner=-1, string $question="")
assQuestion constructor
static _getQuestionTypeName($type_tag)
static isForcePassResultUpdateEnabled()
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static lookupParentObjId(int $questionId)
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)
savePreviewData(ilAssQuestionPreviewSession $previewSession)
static _getOriginalId(int $question_id)
static getUsageOfObject(int $a_obj_id, bool $a_include_titles=false)
static _isUsedInRandomTest(int $question_id)
migrateToLmContent($content)
lmMigrateQuestionTypeSpecificContent(ilAssSelfAssessmentMigrator $migrator)
static isAllowedImageFileExtension(string $mimeType, string $fileExtension)
static getNumExistingSolutionRecords(int $activeId, int $pass, int $questionId)
syncXHTMLMediaObjectsOfQuestion()
duplicateSkillAssignments(int $srcParentId, int $srcQuestionId, int $trgParentId, int $trgQuestionId)
static getQuestionsMissingResultRecord(int $activeId, int $pass, array $questionIds)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
getAdditionalContentEditingMode()
static setUsage(int $a_obj_id, int $a_skill_id, int $a_tref_id, bool $a_use=true)
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.
getSelfAssessmentEditingMode()
getMaterial(int $a_index)
static _getTotalRightAnswers(int $a_q_id)
static setTokenMaxLifetimeInSeconds(int $token_max_lifetime_in_seconds)
setPreventRteUsage(bool $prevent_rte_usage)
static makeDirParents(string $a_dir)
Create a new directory and all parent directories.
getSuggestedSolution(int $subquestion_index=0)
Returns a suggested solution for a given subquestion index.
isHTML($a_text)
Checks if a given string contains HTML or not.
static _getObjectIDFromActiveID($active_id)
Returns the ILIAS test object id for a given active id.
setComment(string $comment="")
persistPreviewState(ilAssQuestionPreviewSession $previewSession)
persists the preview state for current user and question
static _questionExistsInTest(int $question_id, int $test_id)
string $questionActionCmd
getValidAdditionalContentEditingModes()
float $points
The maximum available points for the question.
Customizing of pimple-DIC for ILIAS.
int $obj_id
Object id of the container object.
fetchIndexedValuesFromValuePairs(array $valuePairs)
bool $selfassessmenteditingmode
static implodeKeyValues(array $keyValues)
setExportImagePath(string $path)
static _areAnswered(int $a_user_id, array $a_question_ids)
Checks if an array of question ids is answered by a user or not.
static removeTrailingPathSeparators(string $path)
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
fixUnavailableSkinImageSources(string $html)
loadSuggestedSolution(int $question_id, int $subquestion_index=0)
Returns a suggested solution for a given subquestion index.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _lookupObjId(int $ref_id)
static instantiateQuestionGUI(int $a_question_id)
setExportDetailsXLS(ilAssExcelFormatHelper $worksheet, int $startrow, int $active_id, int $pass)
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)
xmlEndTag(string $tag)
Writes an endtag.
isPreviewSolutionCorrect(ilAssQuestionPreviewSession $previewSession)
lookupTestId(int $active_id)
Move to ilObjTest or similar
removeIntermediateSolution(int $active_id, int $pass)
calculateReachedPoints($active_id, $pass=null, $authorizedSolution=true, $returndetails=false)
Returns the points, a learner has reached answering the question.
static isQuestionObligatory($question_id)
checks wether the question with given id is marked as obligatory or not
static deleteHintsByQuestionIds($questionIds)
Deletes all question hints relating to questions included in given question ids.
static resetOriginalId(int $questionId)
calculateReachedPointsFromPreviewSession(ilAssQuestionPreviewSession $previewSession)
getQuestionForHTMLOutput()
supportsJavascriptOutput()
static instantiateQuestion(int $question_id)
getFlashPath()
Returns the image path for web accessable flash files of a question.
static isAllowedImageMimeType($mimeType)
Interface for html sanitizing functionality.
saveCurrentSolution(int $active_id, int $pass, $value1, $value2, bool $authorized=true, $tstamp=0)
static _needsManualScoring(int $question_id)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _enabledAssessmentLogging()
purifyAndPrepareTextAreaOutput(string $content)
static http()
Fetches the global http state from ILIAS.
getImagePath($question_id=null, $object_id=null)
Returns the image path for web accessable images of a question.
static _updateQuestionCount($object_id)
Updates the number of available questions for a question pool in the database.
ensureCurrentTestPass(int $active_id, int $pass)
buildHashedImageFilename(string $plain_image_filename, bool $unique=false)
static _lookupTitle(int $obj_id)
static _getQuestionCountAndPointsForPassOfParticipant($active_id, $pass)
areObligationsToBeConsidered()
lookupForExistingSolutions(int $activeId, int $pass)
Lookup if an authorized or intermediate solution exists.
static _isWorkedThrough(int $active_id, int $question_id, int $pass)
Returns true if the question was worked through in the given pass Worked through means that the user ...
_resolveIntLinks(int $question_id)
getAdjustedReachedPoints(int $active_id, int $pass, bool $authorizedSolution=true)
returns the reached points ...
static _getScoreCutting($active_id)
Determines if the score of a question should be cut at 0 points or the score of the whole test...
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.
static logAction(string $logtext, int $active_id, int $question_id)
static _getResultPass($active_id)
Retrieves the pass number that should be counted for a given user.
removeSolutionRecordById(int $solutionId)
static _getReachedPoints(int $active_id, int $question_id, int $pass)
static delDir(string $a_dir, bool $a_clean_only=false)
removes a dir and all its content (subdirs and files) recursively
setSuggestedSolution(string $solution_id="", int $subquestion_index=0, bool $is_import=false)
Sets a suggested solution for the question.
static _getIdForImportId(string $a_import_id)
get current object id for import id (static)
questionTitleExists(int $questionpool_id, string $title)
Returns TRUE if the question title exists in a question pool in the database.
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.
isComplete()
Returns true, if a question is complete for use.
intermediateSolutionExists(int $active_id, int $pass)
fromXML($item, int $questionpool_id, ?int $tst_id, &$tst_object, int &$question_counter, array $import_mapping, array &$solutionhints=[])
Receives parameters from a QTI parser and creates a valid ILIAS question object.
static _getCountSystem($active_id)
Gets the count system for the calculation of points.
_resolveInternalLink(string $internal_link)
static createDirectory(string $a_dir, int $a_mod=0755)
create directory
static $forcePassResultsUpdateEnabled
if(!defined('PATH_SEPARATOR')) $GLOBALS['_PEAR_default_error_mode']
static originalQuestionExists(int $questionId)
QTIMaterialToString(ilQTIMaterial $a_material)
Reads an QTI material tag and creates a text or XHTML string.
static _setReachedPoints(int $active_id, int $question_id, float $points, float $maxpoints, int $pass, bool $manualscoring, bool $obligationsEnabled)
Sets the points, a learner has reached answering the question Additionally objective results are upda...
static _getQuestionInfo(int $question_id)
getDescriptionForHTMLOutput()
getHtmlUserSolutionPurifier()
static getInstanceByType(string $type)
static _getTitle(int $a_q_id)
lookupCurrentTestPass(int $active_id, int $pass)
static moveUploadedFile(string $a_file, string $a_name, string $a_target, bool $a_raise_errors=true, string $a_mode="move_uploaded")
move uploaded file
static _lookupAuthor($obj_id)
Gets the authors name of the ilObjTest object.
getSuggestedSolutionPathWeb()
string $question
The question text.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
getSuggestedSolutionTitle(int $subquestion_index=0)
Returns the title of a suggested solution at a given subquestion_index.
isClone(int $question_id=0)
Checks whether the question is a clone of another question or not.
log(int $active_id, string $langVar)
removeExistingSolutions(int $activeId, int $pass)
setDefaultNrOfTries(int $defaultnroftries)
cleanupMediaObjectUsage()
deleteSuggestedSolutions()
isNonEmptyItemListPostSubmission(string $postSubmissionFieldname)
moveUploadedMediaFile(string $file, string $name)
Move an uploaded media file to an public accessible temp dir to present it.
static missingResultRecordExists(int $activeId, int $pass, array $questionIds)
static saveOriginalId(int $questionId, int $originalId)
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 ...
int $test_id
The database id of a test in which the question is contained.
isAdditionalContentEditingModePageObject()
int $outputType
Contains the output type of a question.
deductHintPointsFromReachedPoints(ilAssQuestionPreviewSession $previewSession, $reachedPoints)
static $allowedCharsetsByMimeType
setObligationsToBeConsidered(bool $obligationsToBeConsidered)
generateExternalId(int $question_id)
onDuplicate(int $originalParentId, int $originalQuestionId, int $duplicateParentId, int $duplicateQuestionId)
duplicateIntermediateSolutionAuthorized(int $activeId, int $passIndex)
updateSuggestedSolutions(int $original_id=-1, int $original_obj_id=-1)
const KEY_VALUES_IMPLOSION_SEPARATOR
static _getMaximumPoints(int $question_id)
Returns the maximum points, a learner can reach answering the question.
_questionExists($question_id)
Returns true if the question already exists in the database.
copySuggestedSolutionFiles(int $source_questionpool_id, int $source_question_id)
saveQuestionDataToDb(int $original_id=-1)
syncSuggestedSolutionFiles(int $target_question_id, int $target_obj_id)
static $imageSourceFixReplaceMap
static getAllowedFileExtensionsForMimeType(string $mimeType)
ilTestQuestionConfig $testQuestionConfigInstance
getSolutionMaxPass(int $active_id)
isValidAdditionalContentEditingMode(string $additionalContentEditingMode)
static extendedTrim(string $value)
Trim non-printable characters from the beginning and end of a string.
static _instantiateQuestion(int $question_id)
removeCurrentSolution(int $active_id, int $pass, bool $authorized=true)
array $suggested_solutions
lmMigrateQuestionTypeGenericContent(ilAssSelfAssessmentMigrator $migrator)
static getQuestionTypeFromDb(int $question_id)
onCopy(int $sourceParentId, int $sourceQuestionId, int $targetParentId, int $targetQuestionId)
const ADDITIONAL_CONTENT_EDITING_MODE_RTE
setOriginalId(?int $original_id)
getTestPresentationConfig()
setExternalId(?string $external_id)
buildImagePath($questionId, $parentObjectId)
getRequestStatisticData()
getTestOutputSolutions(int $activeId, int $pass)
_getSuggestedSolution(int $question_id, int $subquestion_index=0)
setShuffler(Transformation $shuffler)
static convertISO8601FormatH_i_s_ExtendedToSeconds(string $time)
setTitle(string $title="")
deleteSolutionRecordByValues(int $activeId, int $passIndex, bool $authorized, array $matchValues)
setLastChange($lastChange)
static _instanciateQuestion(int $question_id)
static signFile(string $path_to_file)
authorizedSolutionExists(int $active_id, ?int $pass)
static buildExamId($active_id, $pass, $test_obj_id=null)
xmlStartTag(string $tag, ?array $attrs=null, bool $empty=false, bool $encode=true, bool $escape=true)
Writes a starttag.
setLifecycle(ilAssQuestionLifecycle $lifecycle)
static _includeClass(string $question_type, int $gui=0)
prepareTextareaOutput(string $txt_output, bool $prepare_for_latex_output=false, bool $omitNl2BrWhenTextArea=false)
static _getQuestionText(int $a_q_id)
getCurrentSolutionResultSet(int $active_id, int $pass, bool $authorized=true)
static _isWriteable(int $question_id, int $user_id)
duplicate(bool $for_test=true, string $title="", string $author="", string $owner="", $testObjId=null)
calculateResultsFromSolution(int $active_id, int $pass, bool $obligationsEnabled=false)
Calculates the question results from a previously saved question solution.
getReachedPoints(int $active_id, int $pass)
isAnswered(int $active_id, int $pass)
addAnswerOptionValue(int $qIndex, string $answerOptionValue, float $points)
xmlElement(string $tag, $attrs=null, $data=null, $encode=true, $escape=true)
Writes a basic element (no children, just textual content)
deleteTaxonomyAssignments()
forceExistingIntermediateSolution(int $activeId, int $passIndex, bool $considerDummyRecordCreation)
static _updateObjectiveResult(int $a_user_id, int $a_active_id, int $a_question_id)
const HAS_SPECIFIC_FEEDBACK
ILIAS DI LoggingServices $ilLog
getHtmlQuestionContentPurifier()
lookupMaxStep(int $active_id, int $pass)
setAuthor(string $author="")
static $allowedImageMaterialFileExtensionsByMimeType
duplicateQuestionHints(int $originalQuestionId, int $duplicateQuestionId)
duplicateSuggestedSolutionFiles(int $parent_id, int $question_id)
Duplicates the files of a suggested solution if the question is duplicated.
setNewOriginalId(int $newId)
string $additionalContentEditingMode
afterSyncWithOriginal(int $origQuestionId, int $dupQuestionId, int $origParentObjId, int $dupParentObjId)
getSolutionRecordById(int $solutionId)
beforeSyncWithOriginal(int $origQuestionId, int $dupQuestionId, int $origParentObjId, int $dupParentObjId)
setShuffle(?bool $shuffle=true)
setAdditionalContentEditingMode(?string $additionalContentEditingMode)
static $allowedFileExtensionsByMimeType
static explodeKeyValues(string $keyValues)
static set(string $a_var, $a_val)
Set a value.
static getDraftInstance()
getUserSolutionPreferingIntermediate(int $active_id, $pass=null)
isInUse(int $question_id=0)
Checks whether the question is in use or not in pools or tests.
static isCoreQuestionType(string $questionType)
removeAllExistingSolutions()
__isset($key)
Object issetter.
copyXHTMLMediaObjectsOfQuestion(int $a_q_id)
getSuggestedSolutionOutput()
ILIAS Refinery Factory $refinery
getSelfAssessmentFormatter()
static getFeedbackClassNameByQuestionType(string $questionType)
fixSvgToPng(string $imageFilenameContainingString)
setQuestion(string $question="")
string $export_image_path
(Web) Path to images
static setForcePassResultUpdateEnabled(bool $forcePassResultsUpdateEnabled)