ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.assErrorTextGUI.php
Go to the documentation of this file.
1 <?php
2 
34 {
35  private const DEFAULT_POINTS_WRONG = -1;
36 
37  private ilTabsGUI $tabs;
38 
39  public function __construct($id = -1)
40  {
41  global $DIC;
42  $this->tabs = $DIC->tabs();
43 
45  $this->object = new assErrorText();
46  $this->setErrorMessage($this->lng->txt("msg_form_save_error"));
47  if ($id >= 0) {
48  $this->object->loadFromDb($id);
49  }
50  }
51 
55  protected function writePostData(bool $always = false): int
56  {
57  $hasErrors = (!$always) ? $this->editQuestion(true) : false;
58  if (!$hasErrors) {
62  $this->saveTaxonomyAssignments();
63  return 0;
64  }
65  return 1;
66  }
67 
68  public function writeAnswerSpecificPostData(ilPropertyFormGUI $form): void
69  {
70  $data = $this->restructurePostDataForSaving($this->request_data_collector->raw('errordata') ?? []);
71  $this->object->setErrorData($data);
72  $this->object->removeErrorDataWithoutPosition();
73  }
74 
75  private function restructurePostDataForSaving(array $post): array
76  {
77  $keys = $post['key'] ?? [];
78  $restructured_array = [];
79  foreach ($keys as $key => $text_wrong) {
80  $restructured_array[] = new assAnswerErrorText(
81  $text_wrong,
82  $post['value'][$key],
83  (float) str_replace(',', '.', $post['points'][$key])
84  );
85  }
86  return $restructured_array;
87  }
88 
89  public function writeQuestionSpecificPostData(ilPropertyFormGUI $form): void
90  {
91  $this->object->setQuestion(
92  $this->request_data_collector->string('question')
93  );
94 
95  $this->object->setErrorText(
96  $this->request_data_collector->raw('errortext')
97  );
98 
99  $this->object->parseErrorText();
100 
101  $this->object->setPointsWrong(
102  $this->request_data_collector->float('points_wrong') ?? self::DEFAULT_POINTS_WRONG
103  );
104 
105  if (!$this->object->getSelfAssessmentEditingMode()) {
106  $this->object->setTextSize(
107  $this->request_data_collector->float('textsize')
108  );
109  }
110  }
111 
112  public function editQuestion(
113  bool $checkonly = false,
114  ?bool $is_save_cmd = null
115  ): bool {
116  $save = $is_save_cmd ?? $this->isSaveCommand();
117 
118  $form = new ilPropertyFormGUI();
119  $this->editForm = $form;
120 
121  $form->setFormAction($this->ctrl->getFormAction($this));
122  $form->setTitle($this->outQuestionType());
123  $form->setMultipart(false);
124  $form->setTableWidth("100%");
125  $form->setId("orderinghorizontal");
126 
127  $this->addBasicQuestionFormProperties($form);
128 
129  $this->populateQuestionSpecificFormPart($form);
130 
131  if (count($this->object->getErrorData()) || $checkonly) {
132  $this->populateAnswerSpecificFormPart($form);
133  }
134 
135  $this->populateTaxonomyFormSection($form);
136 
137  $form->addCommandButton("analyze", $this->lng->txt('analyze_errortext'));
138  $this->addQuestionFormCommandButtons($form);
139 
140  $errors = false;
141 
142  if ($save) {
143  $form->setValuesByPost();
144  $errors = !$form->checkInput();
145  $form->setValuesByPost(); // again, because checkInput now performs the whole stripSlashes handling and we need this if we don't want to have duplication of backslashes
146  if ($errors) {
147  $checkonly = false;
148  }
149  }
150 
151  if (!$checkonly) {
152  $this->renderEditForm($form);
153  }
154  return $errors;
155  }
156 
162  {
163  $header = new ilFormSectionHeaderGUI();
164  $header->setTitle($this->lng->txt("errors_section"));
165  $form->addItem($header);
166 
167  $errordata = new ilErrorTextWizardInputGUI($this->lng->txt("errors"), "errordata");
168  $errordata->setKeyName($this->lng->txt('text_wrong'));
169  $errordata->setValueName($this->lng->txt('text_correct'));
170  $errordata->setValues($this->object->getErrorData());
171  $form->addItem($errordata);
172 
173  // points for wrong selection
174  $points_wrong = new ilNumberInputGUI($this->lng->txt("points_wrong"), "points_wrong");
175  $points_wrong->allowDecimals(true);
176  $points_wrong->setMaxValue(0);
177  $points_wrong->setMaxvalueShouldBeLess(true);
178  $points_wrong->setValue($this->object->getPointsWrong());
179  $points_wrong->setInfo($this->lng->txt("points_wrong_info"));
180  $points_wrong->setSize(6);
181  $points_wrong->setRequired(true);
182  $form->addItem($points_wrong);
183  return $form;
184  }
185 
191  {
192  // errortext
193  $errortext = new ilTextAreaInputGUI($this->lng->txt("errortext"), "errortext");
194  $errortext->setValue($this->object->getErrorText());
195  $errortext->setRequired(true);
196  $errortext->setInfo($this->lng->txt("errortext_info"));
197  $errortext->setRows(10);
198  $errortext->setCols(80);
199  $form->addItem($errortext);
200 
201  if (!$this->object->getSelfAssessmentEditingMode()) {
202  // textsize
203  $textsize = new ilNumberInputGUI($this->lng->txt("textsize"), "textsize");
204  $textsize->setValue($this->object->getTextSize() ?? 100.0);
205  $textsize->setInfo($this->lng->txt("textsize_errortext_info"));
206  $textsize->setSize(6);
207  $textsize->setSuffix("%");
208  $textsize->setMinValue(10);
209  $textsize->setRequired(true);
210  $form->addItem($textsize);
211  }
212  return $form;
213  }
214 
218  public function analyze(): void
219  {
221  $this->writePostData(true);
222  $this->saveTaxonomyAssignments();
223  $this->object->setErrorsFromParsedErrorText();
224  $this->tabs->activateTab('edit_question');
225  $this->editQuestion();
226  }
227 
228  public function getSolutionOutput(
229  int $active_id,
230  ?int $pass = null,
231  bool $graphical_output = false,
232  bool $result_output = false,
233  bool $show_question_only = true,
234  bool $show_feedback = false,
235  bool $show_correct_solution = false,
236  bool $show_manual_scoring = false,
237  bool $show_question_text = true,
238  bool $show_inline_feedback = true
239  ): string {
240  $user_solutions = $this->getUsersSolutionFromPreviewOrDatabase($active_id, $pass);
241  return $this->renderSolutionOutput(
242  $user_solutions,
243  $active_id,
244  $pass,
245  $graphical_output,
246  $result_output,
247  $show_question_only,
248  $show_feedback,
249  $show_correct_solution,
250  $show_manual_scoring,
251  $show_question_text,
252  false,
253  $show_inline_feedback,
254  );
255  }
256 
257  public function renderSolutionOutput(
258  mixed $user_solutions,
259  int $active_id,
260  ?int $pass,
261  bool $graphical_output = false,
262  bool $result_output = false,
263  bool $show_question_only = true,
264  bool $show_feedback = false,
265  bool $show_correct_solution = false,
266  bool $show_manual_scoring = false,
267  bool $show_question_text = true,
268  bool $show_autosave_title = false,
269  bool $show_inline_feedback = false,
270  ): ?string {
271  $template = new ilTemplate("tpl.il_as_qpl_errortext_output_solution.html", true, true, "components/ILIAS/TestQuestionPool");
272 
273  $selections = [
274  'user' => $user_solutions ?
275  $user_solutions :
276  $this->getUsersSolutionFromPreviewOrDatabase($active_id, $pass)
277  ];
278  $selections['best'] = $this->object->getBestSelection();
279 
280  $reached_points = $this->object->getPoints();
281  if ($active_id > 0 && !$show_correct_solution) {
282  $reached_points = $this->object->getReachedPoints($active_id, $pass);
283  }
284 
285  if ($result_output === true) {
286  $resulttext = ($reached_points == 1) ? "(%s " . $this->lng->txt("point") . ")" : "(%s " . $this->lng->txt("points") . ")";
287  $template->setVariable("RESULT_OUTPUT", sprintf($resulttext, $reached_points));
288  }
289 
290  if ($this->object->getTextSize() >= 10) {
291  $template->setVariable("STYLE", " style=\"font-size: " . $this->object->getTextSize() . "%;\"");
292  }
293 
294  if ($show_question_text === true) {
295  $template->setVariable("QUESTIONTEXT", $this->object->getQuestionForHTMLOutput());
296  }
297 
298  $correctness_icons = [
299  'correct' => $this->generateCorrectnessIconsForCorrectness(self::CORRECTNESS_OK),
300  'not_correct' => $this->generateCorrectnessIconsForCorrectness(self::CORRECTNESS_NOT_OK)
301  ];
302  $errortext = $this->object->assembleErrorTextOutput($selections, $graphical_output, $show_correct_solution, false, $correctness_icons);
303 
304  $template->setVariable("ERRORTEXT", $errortext);
305  $questionoutput = $template->get();
306 
307  $solutiontemplate = new ilTemplate("tpl.il_as_tst_solution_output.html", true, true, "components/ILIAS/TestQuestionPool");
308 
309  $feedback = '';
310  if ($show_feedback) {
311  if (!$this->isTestPresentationContext()) {
312  $fb = $this->getGenericFeedbackOutput($active_id, $pass);
313  $feedback .= mb_strlen($fb) ? $fb : '';
314  }
315 
316  $fb = $this->getSpecificFeedbackOutput([]);
317  $feedback .= mb_strlen($fb) ? $fb : '';
318  }
319  if (mb_strlen($feedback)) {
320  $cssClass = (
321  $this->hasCorrectSolution($active_id, $pass) ?
323  );
324 
325  $solutiontemplate->setVariable("ILC_FB_CSS_CLASS", $cssClass);
326  $solutiontemplate->setVariable("FEEDBACK", ilLegacyFormElementsUtil::prepareTextareaOutput($feedback, true));
327  }
328 
329  $solutiontemplate->setVariable("SOLUTION_OUTPUT", $questionoutput);
330 
331  $solutionoutput = $solutiontemplate->get();
332  if (!$show_question_only) {
333  // get page object output
334  $solutionoutput = $this->getILIASPage($solutionoutput);
335  }
336  return $solutionoutput;
337  }
338 
339  public function getPreview(
340  bool $show_question_only = false,
341  bool $show_inline_feedback = false
342  ): string {
343  $selections = [
344  'user' => $this->getUsersSolutionFromPreviewOrDatabase()
345  ];
346 
347  return $this->generateQuestionOutput($selections, $show_question_only);
348  }
349 
350  public function getTestOutput(
351  int $active_id,
352  int $pass,
353  bool $is_question_postponed = false,
354  array|bool $user_post_solutions = false,
355  bool $show_specific_inline_feedback = false
356  ): string {
357  $selections = [
358  'user' => $this->getUsersSolutionFromPreviewOrDatabase($active_id, $pass)
359  ];
360 
361  return $this->outQuestionPage(
362  '',
363  $is_question_postponed,
364  $active_id,
365  $this->generateQuestionOutput($selections, false)
366  );
367  }
368 
369  private function generateQuestionOutput($selections, $show_question_only): string
370  {
371  $template = new ilTemplate("tpl.il_as_qpl_errortext_output.html", true, true, "components/ILIAS/TestQuestionPool");
372 
373  if ($this->object->getTextSize() >= 10) {
374  $template->setVariable("STYLE", " style=\"font-size: " . $this->object->getTextSize() . "%;\"");
375  }
376  $template->setVariable("QUESTIONTEXT", $this->object->getQuestionForHTMLOutput());
377  $errortext = $this->object->assembleErrorTextOutput($selections);
378  if ($this->getTargetGuiClass() !== null) {
379  $this->ctrl->setParameterByClass($this->getTargetGuiClass(), 'errorvalue', '');
380  }
381  $template->setVariable("ERRORTEXT", $errortext);
382  $template->setVariable("ERRORTEXT_ID", "qst_" . $this->object->getId());
383  $template->setVariable("ERRORTEXT_VALUE", join(',', $selections['user']));
384 
385  $this->tpl->addOnLoadCode('il.test.player.errortext.init()');
386  $this->tpl->addJavascript('assets/js/errortext.js');
387  $questionoutput = $template->get();
388 
389  if ($show_question_only) {
390  return $questionoutput;
391  }
392 
393  return $this->getILIASPage($questionoutput);
394  }
395 
396  private function getUsersSolutionFromPreviewOrDatabase(int $active_id = 0, ?int $pass = null): array
397  {
398  if (is_object($this->getPreviewSession())) {
399  return (array) $this->getPreviewSession()->getParticipantsSolution();
400  }
401 
402  if ($active_id > 0) {
403  $selections = [];
404  $solutions = $this->object->getSolutionValues($active_id, $pass ?? 0, true);
405  foreach ($solutions as $solution) {
406  $selections[] = $solution['value1'];
407  }
408  return $selections;
409  }
410 
411  return [];
412  }
413 
414  public function getSpecificFeedbackOutput(array $user_solution): string
415  {
416  if (!$this->object->feedbackOBJ->specificAnswerFeedbackExists()) {
417  return '';
418  }
419 
420  $feedback = '<table class="test_specific_feedback"><tbody>';
421  $elements = $this->object->getErrorData();
422  foreach ($elements as $index => $element) {
423  $feedback .= '<tr>';
424  $feedback .= '<td class="text-nowrap">' . $index . '. ' . $element->getTextWrong() . ':</td>';
425  $feedback .= '<td>' . $this->object->feedbackOBJ->getSpecificAnswerFeedbackTestPresentation(
426  $this->object->getId(),
427  0,
428  $index
429  ) . '</td>';
430 
431  $feedback .= '</tr>';
432  }
433  $feedback .= '</tbody></table>';
434 
435  return ilLegacyFormElementsUtil::prepareTextareaOutput($feedback, true);
436  }
437 
448  {
449  return [];
450  }
451 
462  {
463  return [];
464  }
465 
472  public function getAggregatedAnswersView(array $relevant_answers): string
473  {
474  $errortext = $this->object->getErrorText();
475 
476  $passdata = []; // Regroup answers into units of passes.
477  foreach ($relevant_answers as $answer_chosen) {
478  $passdata[$answer_chosen['active_fi'] . '-' . $answer_chosen['pass']][$answer_chosen['value2']][] = $answer_chosen['value1'];
479  }
480 
481  $html = '';
482  foreach ($passdata as $key => $pass) {
483  $passdata[$key] = $this->object->createErrorTextOutput($pass);
484  $html .= $passdata[$key] . '<hr /><br />';
485  }
486 
487  return $html;
488  }
489 
490  public function getAnswersFrequency($relevant_answers, $question_index): array
491  {
492  $answers_by_active_and_pass = [];
493 
494  foreach ($relevant_answers as $row) {
495  $key = $row['active_fi'] . ':' . $row['pass'];
496 
497  if (!isset($answers_by_active_and_pass[$key])) {
498  $answers_by_active_and_pass[$key] = ['user' => []];
499  }
500 
501  $answers_by_active_and_pass[$key]['user'][] = $row['value1'];
502  }
503 
504  $answers = [];
505 
506  foreach ($answers_by_active_and_pass as $answer) {
507  $error_text = '<div class="errortext">' . $this->object->assembleErrorTextOutput($answer) . '</div>';
508  $error_text_hashed = md5($error_text);
509 
510  if (!isset($answers[$error_text_hashed])) {
511  $answers[$error_text_hashed] = [
512  'answer' => $error_text, 'frequency' => 0
513  ];
514  }
515 
516  $answers[$error_text_hashed]['frequency']++;
517  }
518 
519  return array_values($answers);
520  }
521 
523  {
524  $errordata = new ilAssErrorTextCorrectionsInputGUI($this->lng->txt('errors'), 'errordata');
525  $errordata->setKeyName($this->lng->txt('text_wrong'));
526  $errordata->setValueName($this->lng->txt('text_correct'));
527  $errordata->setValues($this->object->getErrorData());
528  $form->addItem($errordata);
529 
530  // points for wrong selection
531  $points_wrong = new ilNumberInputGUI($this->lng->txt('points_wrong'), 'points_wrong');
532  $points_wrong->allowDecimals(true);
533  $points_wrong->setMaxValue(0);
534  $points_wrong->setMaxvalueShouldBeLess(true);
535  $points_wrong->setValue($this->object->getPointsWrong());
536  $points_wrong->setInfo($this->lng->txt('points_wrong_info'));
537  $points_wrong->setSize(6);
538  $points_wrong->setRequired(true);
539  $form->addItem($points_wrong);
540  }
541 
546  {
547  $existing_errordata = $this->object->getErrorData();
548  $this->object->flushErrorData();
549  $new_errordata = $this->request_data_collector->raw('errordata');
550  $errordata = [];
551  foreach ($new_errordata['points'] as $index => $points) {
552  $errordata[$index] = $existing_errordata[$index]->withPoints(
553  (float) str_replace(',', '.', $points)
554  );
555  }
556  $this->object->setErrorData($errordata);
557  $this->object->setPointsWrong((float) str_replace(',', '.', $form->getInput('points_wrong')));
558  }
559 }
generateQuestionOutput($selections, $show_question_only)
hasCorrectSolution($activeId, $passIndex)
getTestOutput(int $active_id, int $pass, bool $is_question_postponed=false, array|bool $user_post_solutions=false, bool $show_specific_inline_feedback=false)
generateCorrectnessIconsForCorrectness(int $correctness)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
populateQuestionSpecificFormPart(ilPropertyFormGUI $form)
populateCorrectionsFormProperties(ilPropertyFormGUI $form)
addBasicQuestionFormProperties(ilPropertyFormGUI $form)
populateAnswerSpecificFormPart(ilPropertyFormGUI $form)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
getAfterParticipationSuppressionAnswerPostVars()
Returns a list of postvars which will be suppressed in the form output when used in scoring adjustmen...
populateTaxonomyFormSection(ilPropertyFormGUI $form)
getInput(string $a_post_var, bool $ensureValidation=true)
Returns the input of an item, if item provides getInput method and as fallback the value of the HTTP-...
writeQuestionSpecificPostData(ilPropertyFormGUI $form)
Extracts the question specific values from the request and applies them to the data object...
addQuestionFormCommandButtons(ilPropertyFormGUI $form)
editQuestion(bool $checkonly=false, ?bool $is_save_cmd=null)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
allowDecimals(bool $a_value)
getSolutionOutput(int $active_id, ?int $pass=null, bool $graphical_output=false, bool $result_output=false, bool $show_question_only=true, bool $show_feedback=false, bool $show_correct_solution=false, bool $show_manual_scoring=false, bool $show_question_text=true, bool $show_inline_feedback=true)
getAfterParticipationSuppressionQuestionPostVars()
Returns a list of postvars which will be suppressed in the form output when used in scoring adjustmen...
setErrorMessage(string $errormessage)
This class represents a number property in a property form.
setValue(?string $a_value)
renderSolutionOutput(mixed $user_solutions, int $active_id, ?int $pass, bool $graphical_output=false, bool $result_output=false, bool $show_question_only=true, bool $show_feedback=false, bool $show_correct_solution=false, bool $show_manual_scoring=false, bool $show_question_text=true, bool $show_autosave_title=false, bool $show_inline_feedback=false,)
Class for error text questions.
global $DIC
Definition: shib_login.php:22
saveCorrectionsFormProperties(ilPropertyFormGUI $form)
writePostData(bool $always=false)
{}
getSpecificFeedbackOutput(array $user_solution)
getPreview(bool $show_question_only=false, bool $show_inline_feedback=false)
writeAnswerSpecificPostData(ilPropertyFormGUI $form)
Extracts the answer specific values from the request and applies them to the data object...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
analyze()
Parse the error text.
restructurePostDataForSaving(array $post)
getILIASPage(string $html="")
Returns the ILIAS Page around a question.
outQuestionPage($a_temp_var, $a_postponed=false, $active_id="", $html="", $inlineFeedbackEnabled=false)
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
__construct(Container $dic, ilPlugin $plugin)
This class represents a text area property in a property form.
This class represents a key value pair wizard property in a property form.
getUsersSolutionFromPreviewOrDatabase(int $active_id=0, ?int $pass=null)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
getAnswersFrequency($relevant_answers, $question_index)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
$post
Definition: ltitoken.php:46
static prepareTextareaOutput(string $txt_output, bool $prepare_for_latex_output=false, bool $omitNl2BrWhenTextArea=false)
Prepares a string for a text area output where latex code may be in it If the text is HTML-free...
renderEditForm(ilPropertyFormGUI $form)
getGenericFeedbackOutput(int $active_id, ?int $pass)
getAggregatedAnswersView(array $relevant_answers)
Returns an html string containing a question specific representation of the answers so far given in t...