MDL-26511 Images in Cloze multichoice

vertical and horizontal answer and in feedback of all qtype answers
This commit is contained in:
ppichet 2013-06-04 23:59:11 -04:00 committed by Dan Poltawski
parent 56ad6d46d1
commit feec4f8941
9 changed files with 136 additions and 14 deletions

View File

@ -590,7 +590,8 @@ function question_move_questions_to_category($questionids, $newcategoryid) {
SELECT q.id, q.qtype, qc.contextid SELECT q.id, q.qtype, qc.contextid
FROM {question} q FROM {question} q
JOIN {question_categories} qc ON q.category = qc.id JOIN {question_categories} qc ON q.category = qc.id
WHERE q.id $questionidcondition", $params); WHERE q.id $questionidcondition OR q.parent $questionidcondition",
array_merge($params, $params));
foreach ($questions as $question) { foreach ($questions as $question) {
if ($newcontextid != $question->contextid) { if ($newcontextid != $question->contextid) {
question_bank::get_qtype($question->qtype)->move_files( question_bank::get_qtype($question->qtype)->move_files(

View File

@ -61,7 +61,8 @@ class qformat_multianswer extends qformat_default {
$question->length = 1; $question->length = 1;
$question->penalty = 0.3333333; $question->penalty = 0.3333333;
if (!empty($question)) { if (!empty($question) && isset($question->options) && isset($question->options->questions) &&
count($question->options->questions) != 0 ) {
$question->name = $this->create_default_question_name($question->questiontext, get_string('questionname', 'question')); $question->name = $this->create_default_question_name($question->questiontext, get_string('questionname', 'question'));
$questions[] = $question; $questions[] = $question;
} }

View File

@ -449,12 +449,12 @@ class qformat_xml extends qformat_default {
*/ */
public function import_multianswer($question) { public function import_multianswer($question) {
global $USER; global $USER;
question_bank::get_qtype('multianswer'); $qtypemultianswer = question_bank::get_qtype('multianswer');
$questiontext = $this->import_text_with_files($question, $questiontext = $this->import_text_with_files($question,
array('#', 'questiontext', 0)); array('#', 'questiontext', 0));
$qo = qtype_multianswer_extract_question($questiontext); $qo = qtype_multianswer_extract_question($questiontext);
$qo = $qtypemultianswer->set_subquestions_elements_itemid($qo, '@@PLUGINFILE@@');
// 'header' parts particular to multianswer // 'header' parts particular to multianswer
$qo->qtype = 'multianswer'; $qo->qtype = 'multianswer';
$qo->course = $this->course; $qo->course = $this->course;

View File

@ -189,8 +189,10 @@ class qtype_multianswer_edit_form extends question_edit_form {
$mform->addElement('static', 'sub_'.$sub.'_layout', $mform->addElement('static', 'sub_'.$sub.'_layout',
get_string('layout', 'qtype_multianswer')); get_string('layout', 'qtype_multianswer'));
} }
$nbanswer = 0;
$choices = array();
foreach ($this->questiondisplay->options->questions[$sub]->answer as $key => $ans) { foreach ($this->questiondisplay->options->questions[$sub]->answer as $key => $ans) {
$nbanswer++;
$mform->addElement('static', 'sub_'.$sub.'_answer['.$key.']', $mform->addElement('static', 'sub_'.$sub.'_answer['.$key.']',
get_string('answer', 'question')); get_string('answer', 'question'));
@ -205,6 +207,15 @@ class qtype_multianswer_edit_form extends question_edit_form {
$mform->addElement('static', 'sub_'.$sub.'_feedback['.$key.']', $mform->addElement('static', 'sub_'.$sub.'_feedback['.$key.']',
get_string('feedback', 'question')); get_string('feedback', 'question'));
if ($this->questiondisplay->options->questions[$sub]->qtype == 'multichoice' &&
$this->questiondisplay->options->questions[$sub]->layout == 0) {
$choices[] = $ans['text'];
}
}
if ($nbanswer > 0 && $this->questiondisplay->options->questions[$sub]->qtype == 'multichoice' &&
$this->questiondisplay->options->questions[$sub]->layout == 0) {
$mform->addElement('select', 'sub_'.$sub.'_layoutselectinline',
get_string('layoutselectinline', 'qtype_multianswer'), $choices);
} }
} }
@ -364,6 +375,7 @@ class qtype_multianswer_edit_form extends question_edit_form {
get_string('layoutundefined', 'qtype_multianswer'); get_string('layoutundefined', 'qtype_multianswer');
} }
} }
$choices = array();
foreach ($subquestion->answer as $key => $answer) { foreach ($subquestion->answer as $key => $answer) {
if ($subquestion->qtype == 'numerical' && $key == 0) { if ($subquestion->qtype == 'numerical' && $key == 0) {
$default_values[$prefix.'tolerance['.$key.']'] = $default_values[$prefix.'tolerance['.$key.']'] =
@ -388,9 +400,19 @@ class qtype_multianswer_edit_form extends question_edit_form {
$maxfraction = $subquestion->fraction[$key]; $maxfraction = $subquestion->fraction[$key];
} }
} }
if ($subquestion->qtype == 'multichoice' && $subquestion->layout == '0') {
$default_values[$prefix.'answer['.$key.']'] = $a = new stdClass();
htmlspecialchars($answer); $a->strip = strip_tags($trimmedanswer);
$a->html = htmlspecialchars($trimmedanswer);
$default_values[$prefix.'answer['.$key.']'] = get_string('answerselectelementshownas',
'qtype_multianswer', $a);
$choices[$answercount] = $trimmedanswer;
} else {
$default_values[$prefix.'answer['.$key.']'] = htmlspecialchars($answer);
}
}
if ($subquestion->qtype == 'multichoice' && $subquestion->layout == '0') {
$default_values[$prefix.'_layoutselectinline'] = $choices;
} }
if ($answercount == 0) { if ($answercount == 0) {
if ($subquestion->qtype == 'multichoice') { if ($subquestion->qtype == 'multichoice') {

View File

@ -23,6 +23,8 @@
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/ */
$string['answerselectelementempty'] = '{$a->html}<br />This answer will appear empty as the dropdown menu does not show HTML components.';
$string['answerselectelementshownas'] = '{$a->html}<br />This answer will be rendered in the dropdown menu as : {$a->strip}';
$string['confirmquestionsaveasedited'] = 'I confirm that I want the question to be saved as edited'; $string['confirmquestionsaveasedited'] = 'I confirm that I want the question to be saved as edited';
$string['confirmsave'] = 'Confirm then save {$a}'; $string['confirmsave'] = 'Confirm then save {$a}';
$string['correctanswer'] = 'Correct answer'; $string['correctanswer'] = 'Correct answer';

View File

@ -0,0 +1,47 @@
<?php
// This file is part of Moodle - http://moodle.org/
//
// Moodle is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Moodle is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
/**
* Serve question type files
*
* @since 2.0
* @package qtype_multianswer
* @copyright Pierre Pichet
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
defined('MOODLE_INTERNAL') || die();
/**
* Checks file access for multianswer questions.
* @package qtype_multianswer
* @category files
* @param stdClass $course course object
* @param stdClass $cm course module object
* @param stdClass $context context object
* @param string $filearea file area
* @param array $args extra arguments
* @param bool $forcedownload whether or not force download
* @param array $options additional options affecting the file serving
* @return bool
*/
function qtype_multianswer_pluginfile($course, $cm, $context, $filearea, $args, $forcedownload, array $options=array()) {
global $DB, $CFG;
require_once($CFG->libdir . '/questionlib.php');
question_pluginfile($course, $context, 'qtype_multianswer', $filearea, $args, $forcedownload, $options);
}

View File

@ -168,8 +168,45 @@ class qtype_multianswer extends question_type {
$this->save_hints($question, true); $this->save_hints($question, true);
} }
/**
* Set the itemid if the element contains a valid file identify by (draftfile.php).
* param array() $question from qtype_multianswer_extract_question().
* param string $searchcriteria i.e $searchcriteria ,@@PLUGINFILE@@.
* return $question.
*/
public function set_subquestions_elements_itemid($question, $searchcriteria = '') {
if (isset($question->options) && isset($question->options->questions) &&
$question->options->questions != '' && isset($question->questiontext['itemid'])
&& $question->questiontext['itemid'] != '' && $searchcriteria != '') {
$questiontextitemid = $question->questiontext['itemid'];
foreach ($question->options->questions as $wrapped) {
if (preg_match('/'.$searchcriteria.'/', $wrapped->questiontext['text'])) {
$wrapped->questiontext['itemid'] = $questiontextitemid;
} else {
$wrapped->questiontext['itemid'] = '';
}
foreach ($wrapped->answer as $key => $answer) {
if (is_array($answer)) {
if (preg_match('/'.$searchcriteria.'/', $answer['text'])) {
$wrapped->answer[$key]['itemid'] = $questiontextitemid;
} else {
$wrapped->answer[$key]['itemid'] = '';
}
}
if (preg_match('/'.$searchcriteria.'/', $wrapped->feedback[$key]['text'])) {
$wrapped->feedback[$key]['itemid'] = $questiontextitemid;
} else {
$wrapped->feedback[$key]['itemid'] = '';
}
}
}
}
return $question;
}
public function save_question($authorizedquestion, $form) { public function save_question($authorizedquestion, $form) {
$question = qtype_multianswer_extract_question($form->questiontext); $question = qtype_multianswer_extract_question($form->questiontext);
$question = $this->set_subquestions_elements_itemid($question, 'draftfile.php');
if (isset($authorizedquestion->id)) { if (isset($authorizedquestion->id)) {
$question->id = $authorizedquestion->id; $question->id = $authorizedquestion->id;
} }
@ -296,9 +333,15 @@ define('ANSWER_REGEX_ALTERNATIVES', 9);
function qtype_multianswer_extract_question($text) { function qtype_multianswer_extract_question($text) {
// Variable $text is an array [text][format][itemid]. // Variable $text is an array [text][format][itemid].
// If the main question contains a file then its [itemid]
// will not be empty.
// The other subquestions question , answers feedback and
// multiple choice vertical and horizontal answers should be declared
// array and should have [itemid] set to [text][itemid].
$question = new stdClass(); $question = new stdClass();
$question->qtype = 'multianswer'; $question->qtype = 'multianswer';
$question->questiontext = $text; $question->questiontext = $text;
$question->questiontext['text'] = str_replace(array('<pre>', '</pre>'), array('', ''), $question->questiontext['text']);
$question->generalfeedback['text'] = ''; $question->generalfeedback['text'] = '';
$question->generalfeedback['format'] = FORMAT_HTML; $question->generalfeedback['format'] = FORMAT_HTML;
$question->generalfeedback['itemid'] = ''; $question->generalfeedback['itemid'] = '';
@ -314,6 +357,9 @@ function qtype_multianswer_extract_question($text) {
$wrapped->generalfeedback['text'] = ''; $wrapped->generalfeedback['text'] = '';
$wrapped->generalfeedback['format'] = FORMAT_HTML; $wrapped->generalfeedback['format'] = FORMAT_HTML;
$wrapped->generalfeedback['itemid'] = ''; $wrapped->generalfeedback['itemid'] = '';
$wrapped->questiontext['text'] = $answerregs[0];
$wrapped->questiontext['format'] = $question->questiontext['format'];
$wrapped->questiontext['itemid'] = $question->questiontext['itemid'];
if (isset($answerregs[ANSWER_REGEX_NORM])&& $answerregs[ANSWER_REGEX_NORM]!== '') { if (isset($answerregs[ANSWER_REGEX_NORM])&& $answerregs[ANSWER_REGEX_NORM]!== '') {
$wrapped->defaultmark = $answerregs[ANSWER_REGEX_NORM]; $wrapped->defaultmark = $answerregs[ANSWER_REGEX_NORM];
} else { } else {
@ -388,9 +434,6 @@ function qtype_multianswer_extract_question($text) {
$wrapped->answer = array(); $wrapped->answer = array();
$wrapped->fraction = array(); $wrapped->fraction = array();
$wrapped->feedback = array(); $wrapped->feedback = array();
$wrapped->questiontext['text'] = $answerregs[0];
$wrapped->questiontext['format'] = FORMAT_HTML;
$wrapped->questiontext['itemid'] = '';
$answerindex = 0; $answerindex = 0;
$remainingalts = $answerregs[ANSWER_REGEX_ALTERNATIVES]; $remainingalts = $answerregs[ANSWER_REGEX_ALTERNATIVES];
@ -408,7 +451,7 @@ function qtype_multianswer_extract_question($text) {
$feedback = str_replace('\}', '}', $feedback); $feedback = str_replace('\}', '}', $feedback);
$wrapped->feedback["$answerindex"]['text'] = str_replace('\#', '#', $feedback); $wrapped->feedback["$answerindex"]['text'] = str_replace('\#', '#', $feedback);
$wrapped->feedback["$answerindex"]['format'] = FORMAT_HTML; $wrapped->feedback["$answerindex"]['format'] = FORMAT_HTML;
$wrapped->feedback["$answerindex"]['itemid'] = ''; $wrapped->feedback["$answerindex"]['itemid'] = $question->questiontext['itemid'];
} else { } else {
$wrapped->feedback["$answerindex"]['text'] = ''; $wrapped->feedback["$answerindex"]['text'] = '';
$wrapped->feedback["$answerindex"]['format'] = FORMAT_HTML; $wrapped->feedback["$answerindex"]['format'] = FORMAT_HTML;
@ -436,6 +479,11 @@ function qtype_multianswer_extract_question($text) {
'text' => $wrapped->answer["$answerindex"], 'text' => $wrapped->answer["$answerindex"],
'format' => FORMAT_HTML, 'format' => FORMAT_HTML,
'itemid' => ''); 'itemid' => '');
if ($wrapped->answer["$answerindex"]['text'] != '' &&
($wrapped->layout == qtype_multichoice_base::LAYOUT_HORIZONTAL ||
$wrapped->layout == qtype_multichoice_base::LAYOUT_VERTICAL )) {
$wrapped->answer["$answerindex"]['itemid'] = $question->questiontext['itemid'];
}
} }
} }
$tmp = explode($altregs[0], $remainingalts, 2); $tmp = explode($altregs[0], $remainingalts, 2);

View File

@ -388,7 +388,7 @@ class qtype_multianswer_multichoice_vertical_renderer extends qtype_multianswer_
question_state::$gradedright) { question_state::$gradedright) {
$feedback[] = get_string('correctansweris', 'qtype_multichoice', $feedback[] = get_string('correctansweris', 'qtype_multichoice',
$subq->format_text($ans->answer, $ans->answerformat, $subq->format_text($ans->answer, $ans->answerformat,
$qa, 'question', 'answer', $ansid)); $qa, 'question', 'answer', $ans->id));
break; break;
} }
} }

View File

@ -206,7 +206,8 @@ class qtype_numerical extends question_type {
} }
$options->question = $question->id; $options->question = $question->id;
$options->answer = $answer->id; $options->answer = $answer->id;
if (trim($question->tolerance[$key]) == '') { // When the answer is * the tolerance can be absent i.e. in multianswer.
if (!isset($question->tolerance[$key]) || trim($question->tolerance[$key]) == '') {
$options->tolerance = ''; $options->tolerance = '';
} else { } else {
$options->tolerance = $this->apply_unit($question->tolerance[$key], $options->tolerance = $this->apply_unit($question->tolerance[$key],