ILIAS  release_8 Revision v8.19-1-g4e8f2f9140c
All Data Structures Namespaces Files Functions Variables Modules Pages
assOrderingQuestionImport Class Reference

This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Learning e.V. More...

+ Inheritance diagram for assOrderingQuestionImport:
+ Collaboration diagram for assOrderingQuestionImport:

Public Member Functions

 fromXML (&$item, $questionpool_id, &$tst_id, &$tst_object, &$question_counter, $import_mapping)
 Creates a question from a QTI file. More...
 
- Public Member Functions inherited from assQuestionImport
 __construct ($a_object)
 assQuestionImport constructor More...
 
 getQuestionId ()
 
 getFeedbackGeneric ($item)
 
 fromXML (&$item, $questionpool_id, &$tst_id, &$tst_object, &$question_counter, $import_mapping)
 Creates a question from a QTI file. More...
 
 importSuggestedSolution (int $question_id, string $value="", int $subquestion_index=0)
 

Data Fields

 $object
 
- Data Fields inherited from assQuestionImport
 $object
 

Protected Member Functions

 handleUploadedFile (array $answer)
 
 ensureImagePathExists ()
 
- Protected Member Functions inherited from assQuestionImport
 fetchIndexFromFeedbackIdent ($feedbackIdent, $prefix='response_')
 
 getFeedbackAnswerSpecific (ilQTIItem $item, $prefix='response_')
 
 addGeneralMetadata (ilQTIItem $item)
 
 fetchLifecycle (ilQTIItem $item)
 
 getQplImportArchivDirectory ()
 returns the full path to extracted qpl import archiv (qpl import dir + qpl archiv subdir) More...
 
 getTstImportArchivDirectory ()
 returns the full path to extracted tst import archiv (tst import dir + tst archiv subdir) More...
 
 processNonAbstractedImageReferences ($text, $sourceNic)
 
 fetchAdditionalContentEditingModeInformation ($qtiItem)
 fetches the "additional content editing mode" information from qti item and falls back to ADDITIONAL_CONTENT_EDITING_MODE_RTE when no or invalid information is given More...
 
 deduceThumbSizeFromImportValue (?int $size)
 

Detailed Description

This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Learning e.V.

ILIAS is licensed with the GPL-3.0, see https://www.gnu.org/licenses/gpl-3.0.en.html You should have received a copy of said license along with the source code, too.

If this is not the case or you just want to try ILIAS, you'll find us at: https://www.ilias.de https://github.com/ILIAS-eLearning Class for ordering question imports

assOrderingQuestionImport is a class for ordering question imports

Author
Helmut Schottmüller helmu.nosp@m.t.sc.nosp@m.hottm.nosp@m.uell.nosp@m.er@ma.nosp@m.c.co.nosp@m.m
Version
$Id$

Definition at line 28 of file class.assOrderingQuestionImport.php.

Member Function Documentation

◆ ensureImagePathExists()

assOrderingQuestionImport::ensureImagePathExists ( )
protected

Definition at line 369 of file class.assOrderingQuestionImport.php.

References ilFileUtils\makeDirParents(), and ILIAS\Repository\object().

Referenced by handleUploadedFile().

370  {
371  if (!file_exists($this->object->getImagePath())) {
372  ilFileUtils::makeDirParents($this->object->getImagePath());
373  }
374  }
static makeDirParents(string $a_dir)
Create a new directory and all parent directories.
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ fromXML()

assOrderingQuestionImport::fromXML ( $item,
  $questionpool_id,
$tst_id,
$tst_object,
$question_counter,
  $import_mapping 
)

Creates a question from a QTI file.

Receives parameters from a QTI parser and creates a valid ILIAS question object

Parameters
ilQtiItem$itemThe QTI item object
integer$questionpool_idThe id of the parent questionpool
integer$tst_idThe id of the parent test if the question is part of a test
object$tst_objectA reference to the parent test object
integer$question_counterA reference to a question counter to count the questions of an imported question pool
array$import_mappingAn array containing references to included ILIAS objects public

Definition at line 48 of file class.assOrderingQuestionImport.php.

References ILIAS\LTI\ToolProvider\$created, $DIC, $filename, $ilUser, $index, $response, $type, ilRTE\_replaceMediaObjectImageSrc(), ilObjMediaObject\_saveTempFileAsMediaObject(), ilObjMediaObject\_saveUsage(), assQuestionImport\addGeneralMetadata(), ilSession\clear(), assQuestionImport\deduceThumbSizeFromImportValue(), assQuestionImport\fetchAdditionalContentEditingModeInformation(), assQuestionImport\fetchIndexFromFeedbackIdent(), ilSession\get(), ILIAS\UI\Implementation\Component\Input\ViewControl\getContent(), assQuestionImport\getFeedbackAnswerSpecific(), assQuestionImport\getQplImportArchivDirectory(), assQuestionImport\getTstImportArchivDirectory(), IL_INST_ID, ILIAS\Repository\int(), ILIAS\Repository\object(), OQ_NESTED_PICTURES, OQ_NESTED_TERMS, OQ_PICTURES, and OQ_TERMS.

48  : array
49  {
50  global $DIC;
51  $ilUser = $DIC['ilUser'];
52 
53  // empty session variable for imported xhtml mobs
54  ilSession::clear('import_mob_xhtml');
55 
56  $presentation = $item->getPresentation();
57  $shuffle = 0;
58  $now = getdate();
59  $foundimage = false;
60  $created = sprintf("%04d%02d%02d%02d%02d%02d", $now['year'], $now['mon'], $now['mday'], $now['hours'], $now['minutes'], $now['seconds']);
61  $answers = [];
62  $type = OQ_TERMS;
63 
64  foreach ($presentation->order as $entry) {
65  switch ($entry["type"]) {
66  case "response":
67  $response = $presentation->response[$entry["index"]];
68  $type = $response->getIdent();
69  if ($response->getIdent() == 'OQP') {
71  } elseif ($response->getIdent() == 'OQNP') {
73  } elseif ($response->getIdent() == 'OQNT') {
75  } elseif ($response->getIdent() == 'OQT') {
76  $type = OQ_TERMS;
77  }
78 
79  $rendertype = $response->getRenderType();
80  switch (strtolower(get_class($rendertype))) {
81  case "ilqtirenderchoice":
82  $shuffle = $rendertype->getShuffle();
83  $answerorder = 0;
84  foreach ($rendertype->response_labels as $response_label) {
85  $ident = $response_label->getIdent();
86  $answertext = "";
87  $answerimage = [];
88  $answerdepth = 0;
89  foreach ($response_label->material as $mat) {
90  for ($m = 0; $m < $mat->getMaterialCount(); $m++) {
91  $foundmat = $mat->getMaterial($m);
92 
93  if (strcmp($foundmat["material"]->getLabel(), "answerdepth") == 0) {
94  $answerdepth = $foundmat["material"]->getContent();
95  }
96  if (strcmp($foundmat["type"], "mattext") == 0
97  && strcmp($foundmat["material"]->getLabel(), "answerdepth") != 0) {
98  $answertext .= $foundmat["material"]->getContent();
99  }
100  if (strcmp($foundmat["type"], "matimage") == 0
101  && strcmp($foundmat["material"]->getLabel(), "answerdepth") != 0) {
102  $foundimage = true;
103  $answerimage = array(
104  "imagetype" => $foundmat["material"]->getImageType(),
105  "label" => $foundmat["material"]->getLabel(),
106  "content" => $foundmat["material"]->getContent()
107  );
108  }
109  }
110  }
111  $answers[$answerorder] = array(
112  'ident' => $ident,
113  "answertext" => $answertext,
114  "answerimage" => $answerimage,
115  "points" => 0,
116  "answerorder" => $answerorder,
117  "answerdepth" => $answerdepth,
118  "correctness" => "",
119  "action" => ""
120  );
121  $answerorder++;
122  }
123  break;
124  }
125  break;
126  }
127  }
128 
129  $feedbacksgeneric = array();
130  foreach ($item->resprocessing as $resprocessing) {
131  foreach ($resprocessing->respcondition as $respcondition) {
132  $ident = "";
133  $correctness = 1;
134  $conditionvar = $respcondition->getConditionvar();
135  foreach ($conditionvar->order as $order) {
136  switch ($order["field"]) {
137  case "arr_not":
138  $correctness = 0;
139  break;
140  case "varequal":
141  $ident = $conditionvar->varequal[$order["index"]]->getContent();
142  $orderindex = $conditionvar->varequal[$order["index"]]->getIndex();
143  break;
144  }
145  }
146  foreach ($respcondition->setvar as $setvar) {
147  if (strcmp($ident, "") != 0) {
148  $answers[$ident]["solutionorder"] = $orderindex;
149  $answers[$ident]["action"] = $setvar->getAction();
150  $answers[$ident]["points"] = $setvar->getContent();
151  }
152  }
153  if (count($respcondition->displayfeedback)) {
154  foreach ($respcondition->displayfeedback as $feedbackpointer) {
155  if (strlen($feedbackpointer->getLinkrefid())) {
156  foreach ($item->itemfeedback as $ifb) {
157  if (strcmp($ifb->getIdent(), "response_allcorrect") == 0) {
158  // found a feedback for the identifier
159  if (count($ifb->material)) {
160  foreach ($ifb->material as $material) {
161  $feedbacksgeneric[1] = $material;
162  }
163  }
164  if ((count($ifb->flow_mat) > 0)) {
165  foreach ($ifb->flow_mat as $fmat) {
166  if (count($fmat->material)) {
167  foreach ($fmat->material as $material) {
168  $feedbacksgeneric[1] = $material;
169  }
170  }
171  }
172  }
173  } elseif (strcmp($ifb->getIdent(), "response_onenotcorrect") == 0) {
174  // found a feedback for the identifier
175  if (count($ifb->material)) {
176  foreach ($ifb->material as $material) {
177  $feedbacksgeneric[0] = $material;
178  }
179  }
180  if ((count($ifb->flow_mat) > 0)) {
181  foreach ($ifb->flow_mat as $fmat) {
182  if (count($fmat->material)) {
183  foreach ($fmat->material as $material) {
184  $feedbacksgeneric[0] = $material;
185  }
186  }
187  }
188  }
189  }
190  }
191  }
192  }
193  }
194  }
195  }
196 
197  $itemfeedbacks = $this->getFeedbackAnswerSpecific($item, 'link_');
198 
199  $this->addGeneralMetadata($item);
200  $this->object->setTitle($item->getTitle());
201  $this->object->setNrOfTries((int) $item->getMaxattempts());
202  $this->object->setComment($item->getComment());
203  $this->object->setAuthor($item->getAuthor());
204  $this->object->setOwner($ilUser->getId());
205  $this->object->setQuestion($this->object->QTIMaterialToString($item->getQuestiontext()));
206  $this->object->setOrderingType($type);
207  $this->object->setObjId($questionpool_id);
208  $this->object->setThumbSize(
209  $this->deduceThumbSizeFromImportValue((int) $item->getMetadataEntry('thumb_geometry'))
210  );
211  $this->object->setElementHeight($item->getMetadataEntry('element_height')
212  ? (int) $item->getMetadataEntry('element_height')
213  : null);
214  $this->object->setShuffle($shuffle);
215  $this->object->setPoints(0);
216  $this->object->saveQuestionDataToDb();
217  $points = 0;
218  $solanswers = array();
219 
220  foreach ($answers as $answer) {
221  $solanswers[$answer["solutionorder"] ?? null] = $answer;
222  }
223  ksort($solanswers);
224  $position = 0;
225  $element_list = $this->object->getOrderingElementList();
226  foreach ($solanswers as $answer) {
227  $points += $answer["points"];
228 
229  $element = new ilAssOrderingElement();
230 
231  if ($element->isExportIdent($answer['ident'])) {
232  $element->setExportIdent($answer['ident']);
233  } else {
234  $element = $element->withPosition($position++);
235  if (isset($answer['answerdepth'])) {
236  $element = $element->withIndentation((int) $answer['answerdepth']);
237  }
238  }
239 
240  if ($this->object->isImageOrderingType()) {
241  $filename = $this->handleUploadedfile($answer);
242  if ($filename !== null) {
243  $element = $element->withContent($filename);
244  }
245  } else {
246  $element = $element->withContent($answer["answertext"]);
247  }
248 
249  $element_list->addElement($element);
250  }
251  $this->object->setOrderingElementList($element_list);
252  $points = ($item->getMetadataEntry("points") > 0) ? $item->getMetadataEntry("points") : $points;
253  $this->object->setPoints($points);
254  // additional content editing mode information
255  $this->object->setAdditionalContentEditingMode(
257  );
258  $this->object->saveToDb();
259  if (count($item->suggested_solutions)) {
260  foreach ($item->suggested_solutions as $suggested_solution) {
261  $this->object->setSuggestedSolution($suggested_solution["solution"]->getContent(), $suggested_solution["gap_index"], true);
262  }
263  $this->object->saveToDb();
264  }
265 
266  foreach ($feedbacksgeneric as $correctness => $material) {
267  $m = $this->object->QTIMaterialToString($material);
268  $feedbacksgeneric[$correctness] = $m;
269  }
270  $questiontext = $this->object->getQuestion();
271 
272  // handle the import of media objects in XHTML code
273  if (is_array(ilSession::get("import_mob_xhtml"))) {
274  foreach (ilSession::get("import_mob_xhtml") as $mob) {
275  if ($tst_id > 0) {
276  $importfile = $this->getTstImportArchivDirectory() . '/' . $mob["uri"];
277  } else {
278  $importfile = $this->getQplImportArchivDirectory() . '/' . $mob["uri"];
279  }
280 
281  global $DIC; /* @var ILIAS\DI\Container $DIC */
282  $DIC['ilLog']->write(__METHOD__ . ': import mob from dir: ' . $importfile);
283 
284  $media_object = ilObjMediaObject::_saveTempFileAsMediaObject(basename($importfile), $importfile, false);
285  ilObjMediaObject::_saveUsage($media_object->getId(), "qpl:html", $this->object->getId());
286  $questiontext = str_replace("src=\"" . $mob["mob"] . "\"", "src=\"" . "il_" . IL_INST_ID . "_mob_" . $media_object->getId() . "\"", $questiontext);
287  foreach ($this->object->getOrderingElementList() as $element) {
288  $element->setContent(str_replace("src=\"" . $mob["mob"] . "\"", "src=\"" . "il_" . IL_INST_ID . "_mob_" . $media_object->getId() . "\"", $element->getContent()));
289  }
290  foreach ($feedbacksgeneric as $correctness => $material) {
291  $feedbacksgeneric[$correctness] = str_replace("src=\"" . $mob["mob"] . "\"", "src=\"" . "il_" . IL_INST_ID . "_mob_" . $media_object->getId() . "\"", $material);
292  }
293  foreach ($itemfeedbacks as $ident => $material) {
294  $itemfeedbacks[$ident] = str_replace("src=\"" . $mob["mob"] . "\"", "src=\"" . "il_" . IL_INST_ID . "_mob_" . $media_object->getId() . "\"", $material);
295  }
296  }
297  }
298  $this->object->setQuestion(ilRTE::_replaceMediaObjectImageSrc($questiontext, 1));
299  foreach ($this->object->getOrderingElementList() as $element) {
300  $element->setContent(ilRTE::_replaceMediaObjectImageSrc($element->getContent(), 1));
301  }
302  foreach ($feedbacksgeneric as $correctness => $material) {
303  $this->object->feedbackOBJ->importGenericFeedback(
304  $this->object->getId(),
305  $correctness,
307  );
308  }
309  foreach ($itemfeedbacks as $ident => $material) {
310  $index = $this->fetchIndexFromFeedbackIdent($ident, 'link_');
311 
312  $this->object->feedbackOBJ->importSpecificAnswerFeedback(
313  $this->object->getId(),
314  0,
315  $index,
317  );
318  }
319  $this->object->saveToDb();
320  if (isset($tst_id) && $tst_id !== $questionpool_id) {
321  $qplQid = $this->object->getId();
322  $tstQid = $this->object->duplicate(true, '', '', '', $tst_id);
323  $tst_object->questions[$question_counter++] = $tstQid;
324  $import_mapping[$item->getIdent()] = ["pool" => $qplQid, "test" => $tstQid];
325  return $import_mapping;
326  }
327 
328  if ($tst_id > 0) {
329  $tst_object->questions[$question_counter++] = $this->object->getId();
330  $import_mapping[$item->getIdent()] = ["pool" => 0, "test" => $this->object->getId()];
331  return $import_mapping;
332  }
333 
334  $import_mapping[$item->getIdent()] = ["pool" => $this->object->getId(), "test" => 0];
335  return $import_mapping;
336  }
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...
static get(string $a_var)
getFeedbackAnswerSpecific(ilQTIItem $item, $prefix='response_')
deduceThumbSizeFromImportValue(?int $size)
addGeneralMetadata(ilQTIItem $item)
const IL_INST_ID
Definition: constants.php:40
$type
const OQ_NESTED_TERMS
fetchAdditionalContentEditingModeInformation($qtiItem)
fetches the "additional content editing mode" information from qti item and falls back to ADDITIONAL_...
static _saveUsage(int $a_mob_id, string $a_type, int $a_id, int $a_usage_hist_nr=0, string $a_lang="-")
Save usage of mob within another container (e.g.
getQplImportArchivDirectory()
returns the full path to extracted qpl import archiv (qpl import dir + qpl archiv subdir) ...
$index
Definition: metadata.php:145
global $DIC
Definition: feed.php:28
const OQ_PICTURES
Ordering question constants.
$filename
Definition: buildRTE.php:78
static _saveTempFileAsMediaObject(string $name, string $tmp_name, bool $upload=true)
Create new media object and update page in db and return new media object.
const OQ_TERMS
const OQ_NESTED_PICTURES
$ilUser
Definition: imgupload.php:34
getTstImportArchivDirectory()
returns the full path to extracted tst import archiv (tst import dir + tst archiv subdir) ...
$response
static clear(string $a_var)
fetchIndexFromFeedbackIdent($feedbackIdent, $prefix='response_')
int $created
Timestamp for when the object was created.
Definition: System.php:151
+ Here is the call graph for this function:

◆ handleUploadedFile()

assOrderingQuestionImport::handleUploadedFile ( array  $answer)
protected

Definition at line 338 of file class.assOrderingQuestionImport.php.

References ilShellUtil\convertImage(), ensureImagePathExists(), ilFileUtils\ilTempnam(), ILIAS\Repository\object(), and assOrderingQuestion\VALID_UPLOAD_SUFFIXES.

338  : ?string
339  {
340  $image = base64_decode($answer["answerimage"]["content"] ?? '');
341  $image_file_name = $answer['answerimage']['label'] ?? '';
342  $tmp_path = ilFileUtils::ilTempnam();
343 
344  $file_handle = fopen($tmp_path, "wb");
345  if ($file_handle === false) {
346  return null;
347  }
348  fwrite($file_handle, $image);
349  fclose($file_handle);
350 
351  $filename_path_parts = explode(".", $image_file_name);
352  $suffix = strtolower(array_pop($filename_path_parts));
353  if (!in_array($suffix, assOrderingQuestion::VALID_UPLOAD_SUFFIXES)) {
354  return null;
355  }
356 
357  $this->ensureImagePathExists();
358  $target_filename = $this->object->buildHashedImageFilename($image_file_name, true);
359  $target_filepath = $this->object->getImagePath() . $target_filename;
360  if (rename($tmp_path, $target_filepath)) {
361  $thumb_path = $this->object->getImagePath() . $this->object->getThumbPrefix() . $target_filename;
362  ilShellUtil::convertImage($target_filepath, $thumb_path, "JPEG", $this->object->getThumbSize());
363  return $target_filename;
364  }
365 
366  return null;
367  }
static convertImage(string $a_from, string $a_to, string $a_target_format="", string $a_geometry="", string $a_background_color="")
convert image
static ilTempnam(?string $a_temp_path=null)
Returns a unique and non existing Path for e temporary file or directory.
+ Here is the call graph for this function:

Field Documentation

◆ $object

assOrderingQuestionImport::$object

Definition at line 33 of file class.assOrderingQuestionImport.php.


The documentation for this class was generated from the following file: