MDL-38214 Questions: Add MCS, MCVS, MCHS Cloze subquestion types
[moodle.git] / question / type / multianswer / edit_multianswer_form.php
CommitLineData
aeb15530 1<?php
d3603157
TH
2// This file is part of Moodle - http://moodle.org/
3//
4// Moodle is free software: you can redistribute it and/or modify
5// it under the terms of the GNU General Public License as published by
6// the Free Software Foundation, either version 3 of the License, or
7// (at your option) any later version.
8//
9// Moodle is distributed in the hope that it will be useful,
10// but WITHOUT ANY WARRANTY; without even the implied warranty of
11// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12// GNU General Public License for more details.
13//
14// You should have received a copy of the GNU General Public License
15// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
16
271ffe3f 17/**
d3603157 18 * Defines the editing form for the multi-answer question type.
271ffe3f 19 *
b04a4319
TH
20 * @package qtype
21 * @subpackage multianswer
22 * @copyright 2007 Jamie Pratt me@jamiep.org
23 * @license http://www.gnu.org/copyleft/gpl.html GNU Public License
271ffe3f 24 */
25
d3603157 26
a17b297d
TH
27defined('MOODLE_INTERNAL') || die();
28
1649a4f7 29require_once($CFG->dirroot . '/question/type/numerical/questiontype.php');
30
a17b297d 31
271ffe3f 32/**
d3603157 33 * Form for editing multi-answer questions.
b04a4319
TH
34 *
35 * @copyright 2007 Jamie Pratt me@jamiep.org
36 * @license http://www.gnu.org/copyleft/gpl.html GNU Public License
271ffe3f 37 */
ab50232b 38class qtype_multianswer_edit_form extends question_edit_form {
271ffe3f 39
1649a4f7 40 // The variable $questiondisplay will contain the qtype_multianswer_extract_question from
41 // the questiontext.
59a3fcd3 42 public $questiondisplay;
1649a4f7 43 // The variable $savedquestiondisplay will contain the qtype_multianswer_extract_question
44 // from the questiontext in database.
59a3fcd3
TH
45 public $savedquestion;
46 public $savedquestiondisplay;
48b5b28f
PP
47 /** @var bool this question is used in quiz */
48 public $usedinquiz = false;
49 /** @var bool the qtype has been changed */
50 public $qtypechange = false;
51 /** @var integer number of questions that have been deleted */
52 public $negativediff = 0;
53 /** @var integer number of quiz that used this question */
54 public $nbofquiz = 0;
55 /** @var integer number of attempts that used this question */
56 public $nbofattempts = 0;
59a3fcd3
TH
57 public $confirm = 0;
58 public $reload = false;
1649a4f7 59 /** @var qtype_numerical_answer_processor used when validating numerical answers. */
60 protected $ap = null;
61
aeb15530 62
59a3fcd3
TH
63 public function __construct($submiturl, $question, $category, $contexts, $formeditable = true) {
64 global $SESSION, $CFG, $DB;
347fb175 65 $this->regenerate = true;
7d087744 66 $this->reload = optional_param('reload', false, PARAM_BOOL);
59a3fcd3 67
48b5b28f 68 $this->usedinquiz = false;
59a3fcd3
TH
69
70 if (isset($question->id) && $question->id != 0) {
e5c5f52e 71 // TODO MDL-43779 should not have quiz-specific code here.
59a3fcd3 72 $this->savedquestiondisplay = fullclone($question);
48b5b28f
PP
73 $this->nbofquiz = $DB->count_records('quiz_slots', array('questionid' => $question->id));
74 $this->usedinquiz = $this->nbofquiz > 0;
75 $this->nbofattempts = $DB->count_records_sql("
dc4a3ea1 76 SELECT count(1)
ccba5b88
TH
77 FROM {quiz_slots} slot
78 JOIN {quiz_attempts} quiza ON quiza.quiz = slot.quizid
79 WHERE slot.questionid = ?
dc4a3ea1 80 AND quiza.preview = 0", array($question->id));
d3e3bd6c 81 }
82
59a3fcd3 83 parent::__construct($submiturl, $question, $category, $contexts, $formeditable);
d3e3bd6c 84 }
85
c7df5006 86 protected function definition_inner($mform) {
347fb175
PP
87 $mform->addElement('hidden', 'reload', 1);
88 $mform->setType('reload', PARAM_INT);
705b5874 89
ab50232b
TH
90 // Remove meaningless defaultmark field.
91 $mform->removeElement('defaultmark');
7d087744 92 $this->confirm = optional_param('confirm', false, PARAM_BOOL);
f34488b2 93
1649a4f7 94 // Display the questions from questiontext.
b8cc71e7
PS
95 if ($questiontext = optional_param_array('questiontext', false, PARAM_RAW)) {
96 $this->questiondisplay = fullclone(qtype_multianswer_extract_question($questiontext));
f34488b2 97
59a3fcd3
TH
98 } else {
99 if (!$this->reload && !empty($this->savedquestiondisplay->id)) {
1649a4f7 100 // Use database data as this is first pass
101 // question->id == 0 so no stored datasets.
d3e3bd6c 102 $this->questiondisplay = fullclone($this->savedquestiondisplay);
59a3fcd3
TH
103 foreach ($this->questiondisplay->options->questions as $subquestion) {
104 if (!empty($subquestion)) {
fe6ce234 105 $subquestion->answer = array('');
59a3fcd3
TH
106 foreach ($subquestion->options->answers as $ans) {
107 $subquestion->answer[] = $ans->answer;
fe6ce234 108 }
d3e3bd6c 109 }
aeb15530 110 }
59a3fcd3 111 } else {
d3e3bd6c 112 $this->questiondisplay = "";
113 }
f34488b2 114 }
705b5874 115
59a3fcd3
TH
116 if (isset($this->savedquestiondisplay->options->questions) &&
117 is_array($this->savedquestiondisplay->options->questions)) {
118 $countsavedsubquestions = 0;
119 foreach ($this->savedquestiondisplay->options->questions as $subquestion) {
120 if (!empty($subquestion)) {
fe6ce234 121 $countsavedsubquestions++;
df79079f 122 }
f34488b2 123 }
705b5874 124 } else {
59a3fcd3 125 $countsavedsubquestions = 0;
705b5874 126 }
59a3fcd3
TH
127 if ($this->reload) {
128 if (isset($this->questiondisplay->options->questions) &&
129 is_array($this->questiondisplay->options->questions)) {
130 $countsubquestions = 0;
131 foreach ($this->questiondisplay->options->questions as $subquestion) {
132 if (!empty($subquestion)) {
fe6ce234 133 $countsubquestions++;
d3e3bd6c 134 }
135 }
fe6ce234 136 } else {
59a3fcd3 137 $countsubquestions = 0;
705b5874 138 }
59a3fcd3
TH
139 } else {
140 $countsubquestions = $countsavedsubquestions;
fe6ce234 141 }
aeb15530 142
59a3fcd3
TH
143 $mform->addElement('submit', 'analyzequestion',
144 get_string('decodeverifyquestiontext', 'qtype_multianswer'));
fe6ce234 145 $mform->registerNoSubmitButton('analyzequestion');
59a3fcd3 146 if ($this->reload) {
59a3fcd3 147 for ($sub = 1; $sub <= $countsubquestions; $sub++) {
aeb15530 148
59a3fcd3 149 if (isset($this->questiondisplay->options->questions[$sub]->qtype)) {
ab50232b 150 $this->editas[$sub] = $this->questiondisplay->options->questions[$sub]->qtype;
7d087744
TH
151 } else {
152 $this->editas[$sub] = optional_param('sub_'.$sub.'_qtype', 'unknown type', PARAM_PLUGIN);
fd97082c 153 }
928e7d2a 154
d3e3bd6c 155 $storemess = '';
59a3fcd3
TH
156 if (isset($this->savedquestiondisplay->options->questions[$sub]->qtype) &&
157 $this->savedquestiondisplay->options->questions[$sub]->qtype !=
158 $this->questiondisplay->options->questions[$sub]->qtype) {
48b5b28f 159 $this->qtypechange = true;
635971c7
TH
160 $storemess = ' ' . html_writer::tag('span', get_string(
161 'storedqtype', 'qtype_multianswer', question_bank::get_qtype_name(
162 $this->savedquestiondisplay->options->questions[$sub]->qtype)),
163 array('class' => 'error'));
59a3fcd3 164 }
48b5b28f 165 $mform->addElement('header', 'subhdr'.$sub, get_string('questionno', 'question',
59a3fcd3
TH
166 '{#'.$sub.'}').'&nbsp;'.question_bank::get_qtype_name(
167 $this->questiondisplay->options->questions[$sub]->qtype).$storemess);
aeb15530 168
7d087744 169 $mform->addElement('static', 'sub_'.$sub.'_questiontext',
928e7d2a 170 get_string('questiondefinition', 'qtype_multianswer'));
aeb15530 171
59a3fcd3 172 if (isset ($this->questiondisplay->options->questions[$sub]->questiontext)) {
7d087744 173 $mform->setDefault('sub_'.$sub.'_questiontext',
59a3fcd3 174 $this->questiondisplay->options->questions[$sub]->questiontext['text']);
f34488b2 175 }
aeb15530 176
7d087744 177 $mform->addElement('static', 'sub_'.$sub.'_defaultmark',
ab50232b 178 get_string('defaultmark', 'question'));
7d087744 179 $mform->setDefault('sub_'.$sub.'_defaultmark',
ab50232b 180 $this->questiondisplay->options->questions[$sub]->defaultmark);
aeb15530 181
59a3fcd3 182 if ($this->questiondisplay->options->questions[$sub]->qtype == 'shortanswer') {
7d087744 183 $mform->addElement('static', 'sub_'.$sub.'_usecase',
ab50232b 184 get_string('casesensitive', 'qtype_shortanswer'));
fe6ce234 185 }
aeb15530 186
59a3fcd3 187 if ($this->questiondisplay->options->questions[$sub]->qtype == 'multichoice') {
7d087744 188 $mform->addElement('static', 'sub_'.$sub.'_layout',
928e7d2a 189 get_string('layout', 'qtype_multianswer'));
fe6ce234 190 }
aeb15530 191
928e7d2a 192 foreach ($this->questiondisplay->options->questions[$sub]->answer as $key => $ans) {
7d087744 193 $mform->addElement('static', 'sub_'.$sub.'_answer['.$key.']',
928e7d2a 194 get_string('answer', 'question'));
aeb15530 195
59a3fcd3
TH
196 if ($this->questiondisplay->options->questions[$sub]->qtype == 'numerical' &&
197 $key == 0) {
7d087744 198 $mform->addElement('static', 'sub_'.$sub.'_tolerance['.$key.']',
ab50232b 199 get_string('acceptederror', 'qtype_numerical'));
d3e3bd6c 200 }
aeb15530 201
7d087744 202 $mform->addElement('static', 'sub_'.$sub.'_fraction['.$key.']',
59a3fcd3 203 get_string('grade'));
aeb15530 204
7d087744 205 $mform->addElement('static', 'sub_'.$sub.'_feedback['.$key.']',
59a3fcd3 206 get_string('feedback', 'question'));
d3e3bd6c 207 }
f34488b2 208 }
928e7d2a 209
48b5b28f
PP
210 $this->negativediff = $countsavedsubquestions - $countsubquestions;
211 if (($this->negativediff > 0) ||$this->qtypechange ||
212 ($this->usedinquiz && $this->negativediff != 0)) {
59a3fcd3
TH
213 $mform->addElement('header', 'additemhdr',
214 get_string('warningquestionmodified', 'qtype_multianswer'));
fe6ce234 215 }
48b5b28f 216 if ($this->negativediff > 0) {
59a3fcd3
TH
217 $mform->addElement('static', 'alert1', "<strong>".
218 get_string('questiondeleted', 'qtype_multianswer')."</strong>",
48b5b28f 219 get_string('questionsless', 'qtype_multianswer', $this->negativediff));
d3e3bd6c 220 }
48b5b28f 221 if ($this->qtypechange) {
59a3fcd3
TH
222 $mform->addElement('static', 'alert1', "<strong>".
223 get_string('questiontypechanged', 'qtype_multianswer')."</strong>",
224 get_string('questiontypechangedcomment', 'qtype_multianswer'));
d3e3bd6c 225 }
d3e3bd6c 226 }
48b5b28f
PP
227 if ($this->usedinquiz) {
228 if ($this->negativediff < 0) {
c4979f02 229 $diff = $countsubquestions - $countsavedsubquestions;
59a3fcd3
TH
230 $mform->addElement('static', 'alert1', "<strong>".
231 get_string('questionsadded', 'qtype_multianswer')."</strong>",
232 "<strong>".get_string('questionsmore', 'qtype_multianswer', $diff).
233 "</strong>");
c4979f02 234 }
59a3fcd3 235 $a = new stdClass();
48b5b28f
PP
236 $a->nbofquiz = $this->nbofquiz;
237 $a->nbofattempts = $this->nbofattempts;
59a3fcd3
TH
238 $mform->addElement('header', 'additemhdr2',
239 get_string('questionusedinquiz', 'qtype_multianswer', $a));
240 $mform->addElement('static', 'alertas',
241 get_string('youshouldnot', 'qtype_multianswer'));
d3e3bd6c 242 }
48b5b28f
PP
243 if (($this->negativediff > 0 || $this->usedinquiz &&
244 ($this->negativediff > 0 || $this->negativediff < 0 || $this->qtypechange)) &&
59a3fcd3
TH
245 $this->reload) {
246 $mform->addElement('header', 'additemhdr',
247 get_string('questionsaveasedited', 'qtype_multianswer'));
248 $mform->addElement('checkbox', 'confirm', '',
249 get_string('confirmquestionsaveasedited', 'qtype_multianswer'));
d3e3bd6c 250 $mform->setDefault('confirm', 0);
59a3fcd3
TH
251 } else {
252 $mform->addElement('hidden', 'confirm', 0);
e380d57f 253 $mform->setType('confirm', PARAM_BOOL);
705b5874 254 }
347fb175 255
e9af6091 256 $this->add_interactive_settings(true, true);
120e5cbf 257 }
705b5874 258
f34488b2 259
59a3fcd3 260 public function set_data($question) {
f34488b2 261 global $DB;
48b5b28f 262 $defaultvalues = array();
59a3fcd3
TH
263 if (isset($question->id) and $question->id and $question->qtype &&
264 $question->questiontext) {
c6fc9988 265
266 foreach ($question->options->questions as $key => $wrapped) {
59a3fcd3 267 if (!empty($wrapped)) {
fe6ce234 268 // The old way of restoring the definitions is kept to gradually
1649a4f7 269 // update all multianswer questions.
fe6ce234 270 if (empty($wrapped->questiontext)) {
ab50232b 271 $parsableanswerdef = '{' . $wrapped->defaultmark . ':';
fe6ce234 272 switch ($wrapped->qtype) {
59a3fcd3
TH
273 case 'multichoice':
274 $parsableanswerdef .= 'MULTICHOICE:';
275 break;
276 case 'shortanswer':
277 $parsableanswerdef .= 'SHORTANSWER:';
278 break;
279 case 'numerical':
280 $parsableanswerdef .= 'NUMERICAL:';
281 break;
282 default:
283 print_error('unknownquestiontype', 'question', '',
284 $wrapped->qtype);
c6fc9988 285 }
59a3fcd3 286 $separator = '';
fe6ce234
DC
287 foreach ($wrapped->options->answers as $subanswer) {
288 $parsableanswerdef .= $separator
48b5b28f 289 . '%' . round(100 * $subanswer->fraction) . '%';
c1f15d35
TH
290 if (is_array($subanswer->answer)) {
291 $parsableanswerdef .= $subanswer->answer['text'];
292 } else {
293 $parsableanswerdef .= $subanswer->answer;
294 }
fe6ce234 295 if (!empty($wrapped->options->tolerance)) {
1649a4f7 296 // Special for numerical answers.
fe6ce234
DC
297 $parsableanswerdef .= ":{$wrapped->options->tolerance}";
298 // We only want tolerance for the first alternative, it will
299 // be applied to all of the alternatives.
300 unset($wrapped->options->tolerance);
301 }
302 if ($subanswer->feedback) {
f4fe3968 303 $parsableanswerdef .= "#{$subanswer->feedback}";
fe6ce234
DC
304 }
305 $separator = '~';
c6fc9988 306 }
fe6ce234 307 $parsableanswerdef .= '}';
1649a4f7 308 // Fix the questiontext fields of old questions.
59a3fcd3
TH
309 $DB->set_field('question', 'questiontext', $parsableanswerdef,
310 array('id' => $wrapped->id));
fe6ce234
DC
311 } else {
312 $parsableanswerdef = str_replace('&#', '&\#', $wrapped->questiontext);
c6fc9988 313 }
59a3fcd3
TH
314 $question->questiontext = str_replace("{#$key}", $parsableanswerdef,
315 $question->questiontext);
c6fc9988 316 }
c6fc9988 317 }
318 }
347fb175 319
1649a4f7 320 // Set default to $questiondisplay questions elements.
59a3fcd3 321 if ($this->reload) {
fe6ce234 322 if (isset($this->questiondisplay->options->questions)) {
59a3fcd3 323 $subquestions = fullclone($this->questiondisplay->options->questions);
fe6ce234 324 if (count($subquestions)) {
59a3fcd3 325 $sub = 1;
fe6ce234 326 foreach ($subquestions as $subquestion) {
59a3fcd3 327 $prefix = 'sub_'.$sub.'_';
fe6ce234 328
1649a4f7 329 // Validate parameters.
fe6ce234
DC
330 $answercount = 0;
331 $maxgrade = false;
332 $maxfraction = -1;
59a3fcd3 333 if ($subquestion->qtype == 'shortanswer') {
fe6ce234 334 switch ($subquestion->usecase) {
59a3fcd3 335 case '1':
48b5b28f 336 $defaultvalues[$prefix.'usecase'] =
59a3fcd3
TH
337 get_string('caseyes', 'qtype_shortanswer');
338 break;
339 case '0':
340 default :
48b5b28f 341 $defaultvalues[$prefix.'usecase'] =
59a3fcd3 342 get_string('caseno', 'qtype_shortanswer');
fe6ce234 343 }
fd97082c 344 }
fd97082c 345
59a3fcd3 346 if ($subquestion->qtype == 'multichoice') {
48b5b28f 347 $defaultvalues[$prefix.'layout'] = $subquestion->layout;
fe6ce234 348 switch ($subquestion->layout) {
59a3fcd3 349 case '0':
48b5b28f 350 $defaultvalues[$prefix.'layout'] =
59a3fcd3
TH
351 get_string('layoutselectinline', 'qtype_multianswer');
352 break;
353 case '1':
48b5b28f 354 $defaultvalues[$prefix.'layout'] =
59a3fcd3
TH
355 get_string('layoutvertical', 'qtype_multianswer');
356 break;
357 case '2':
48b5b28f 358 $defaultvalues[$prefix.'layout'] =
59a3fcd3
TH
359 get_string('layouthorizontal', 'qtype_multianswer');
360 break;
361 default:
48b5b28f 362 $defaultvalues[$prefix.'layout'] =
59a3fcd3 363 get_string('layoutundefined', 'qtype_multianswer');
fe6ce234 364 }
aeb15530 365 }
59a3fcd3
TH
366 foreach ($subquestion->answer as $key => $answer) {
367 if ($subquestion->qtype == 'numerical' && $key == 0) {
48b5b28f 368 $defaultvalues[$prefix.'tolerance['.$key.']'] =
59a3fcd3 369 $subquestion->tolerance[0];
705b5874 370 }
c1f15d35
TH
371 if (is_array($answer)) {
372 $answer = $answer['text'];
373 }
fe6ce234
DC
374 $trimmedanswer = trim($answer);
375 if ($trimmedanswer !== '') {
376 $answercount++;
59a3fcd3 377 if ($subquestion->qtype == 'numerical' &&
1649a4f7 378 !($this->is_valid_number($trimmedanswer) || $trimmedanswer == '*')) {
59a3fcd3
TH
379 $this->_form->setElementError($prefix.'answer['.$key.']',
380 get_string('answermustbenumberorstar',
381 'qtype_numerical'));
fe6ce234
DC
382 }
383 if ($subquestion->fraction[$key] == 1) {
384 $maxgrade = true;
385 }
386 if ($subquestion->fraction[$key] > $maxfraction) {
59a3fcd3 387 $maxfraction = $subquestion->fraction[$key];
fe6ce234 388 }
f34488b2 389 }
fe6ce234 390
48b5b28f 391 $defaultvalues[$prefix.'answer['.$key.']'] =
59a3fcd3 392 htmlspecialchars($answer);
fe6ce234
DC
393 }
394 if ($answercount == 0) {
59a3fcd3
TH
395 if ($subquestion->qtype == 'multichoice') {
396 $this->_form->setElementError($prefix.'answer[0]',
397 get_string('notenoughanswers', 'qtype_multichoice', 2));
fe6ce234 398 } else {
59a3fcd3
TH
399 $this->_form->setElementError($prefix.'answer[0]',
400 get_string('notenoughanswers', 'question', 1));
705b5874 401 }
402 }
fe6ce234 403 if ($maxgrade == false) {
59a3fcd3
TH
404 $this->_form->setElementError($prefix.'fraction[0]',
405 get_string('fractionsnomax', 'question'));
705b5874 406 }
59a3fcd3 407 foreach ($subquestion->feedback as $key => $answer) {
f34488b2 408
48b5b28f 409 $defaultvalues[$prefix.'feedback['.$key.']'] =
59a3fcd3 410 htmlspecialchars ($answer['text']);
fe6ce234 411 }
59a3fcd3 412 foreach ($subquestion->fraction as $key => $answer) {
48b5b28f 413 $defaultvalues[$prefix.'fraction['.$key.']'] = $answer;
fe6ce234 414 }
f34488b2 415
fe6ce234
DC
416 $sub++;
417 }
705b5874 418 }
419 }
420 }
48b5b28f 421 $defaultvalues['alertas'] = "<strong>".get_string('questioninquiz', 'qtype_multianswer').
59a3fcd3 422 "</strong>";
347fb175 423
48b5b28f
PP
424 if ($defaultvalues != "") {
425 $question = (object)((array)$question + $defaultvalues);
705b5874 426 }
e9af6091 427 $question = $this->data_preprocessing_hints($question, true, true);
120e5cbf 428 parent::set_data($question);
429 }
430
1649a4f7 431 /**
432 * Validate that a string is a nubmer formatted correctly for the current locale.
433 * @param string $x a string
434 * @return bool whether $x is a number that the numerical question type can interpret.
435 */
436 protected function is_valid_number($x) {
437 if (is_null($this->ap)) {
438 $this->ap = new qtype_numerical_answer_processor(array());
439 }
440
441 list($value, $unit) = $this->ap->apply_units($x);
442
443 return !is_null($value) && !$unit;
444 }
445
446
59a3fcd3 447 public function validation($data, $files) {
a78890d5 448 $errors = parent::validation($data, $files);
8e8f9f3d 449
0bc0a265 450 $questiondisplay = qtype_multianswer_extract_question($data['questiontext']);
f34488b2 451
347fb175 452 if (isset($questiondisplay->options->questions)) {
59a3fcd3 453 $subquestions = fullclone($questiondisplay->options->questions);
705b5874 454 if (count($subquestions)) {
59a3fcd3 455 $sub = 1;
f34488b2 456 foreach ($subquestions as $subquestion) {
59a3fcd3 457 $prefix = 'sub_'.$sub.'_';
705b5874 458 $answercount = 0;
459 $maxgrade = false;
460 $maxfraction = -1;
59a3fcd3
TH
461 if (isset($this->savedquestiondisplay->options->questions[$sub]->qtype) &&
462 $this->savedquestiondisplay->options->questions[$sub]->qtype !=
463 $questiondisplay->options->questions[$sub]->qtype) {
464 $storemess = " STORED QTYPE ".question_bank::get_qtype_name(
465 $this->savedquestiondisplay->options->questions[$sub]->qtype);
466 }
467 foreach ($subquestion->answer as $key => $answer) {
c1f15d35
TH
468 if (is_array($answer)) {
469 $answer = $answer['text'];
470 }
705b5874 471 $trimmedanswer = trim($answer);
472 if ($trimmedanswer !== '') {
473 $answercount++;
e64e28d7
DP
474 if ($subquestion->qtype == 'numerical' &&
475 !($this->is_valid_number($trimmedanswer) || $trimmedanswer == '*')) {
476 $errors[$prefix.'answer['.$key.']'] =
59a3fcd3 477 get_string('answermustbenumberorstar', 'qtype_numerical');
fe6ce234 478 }
705b5874 479 if ($subquestion->fraction[$key] == 1) {
480 $maxgrade = true;
f34488b2 481 }
705b5874 482 if ($subquestion->fraction[$key] > $maxfraction) {
59a3fcd3 483 $maxfraction = $subquestion->fraction[$key];
705b5874 484 }
f34488b2 485 }
486 }
59a3fcd3
TH
487 if ($answercount == 0) {
488 if ($subquestion->qtype == 'multichoice') {
489 $errors[$prefix.'answer[0]'] =
490 get_string('notenoughanswers', 'qtype_multichoice', 2);
491 } else {
492 $errors[$prefix.'answer[0]'] =
493 get_string('notenoughanswers', 'question', 1);
705b5874 494 }
495 }
496 if ($maxgrade == false) {
59a3fcd3
TH
497 $errors[$prefix.'fraction[0]'] =
498 get_string('fractionsnomax', 'question');
f34488b2 499 }
500 $sub++;
705b5874 501 }
502 } else {
59a3fcd3 503 $errors['questiontext'] = get_string('questionsmissing', 'qtype_multianswer');
705b5874 504 }
120e5cbf 505 }
705b5874 506
48b5b28f
PP
507 if (($this->negativediff > 0 || $this->usedinquiz &&
508 ($this->negativediff > 0 || $this->negativediff < 0 ||
509 $this->qtypechange)) && !$this->confirm) {
59a3fcd3 510 $errors['confirm'] =
48b5b28f 511 get_string('confirmsave', 'qtype_multianswer', $this->negativediff);
59a3fcd3 512 }
fe6ce234
DC
513
514 return $errors;
c6fc9988 515 }
271ffe3f 516
59a3fcd3 517 public function qtype() {
271ffe3f 518 return 'multianswer';
519 }
520}