Multianswer restore now works.

Timelimit is rounded to integer before saving in the database
Some more diagnostic error messages
Towards more plugable question types: $QUIZ_MENU is now populated by the question types themselves.
This commit is contained in:
gustav_delius 2006-03-18 14:14:55 +00:00
parent a4b3fc9220
commit ccccf04f70
16 changed files with 132 additions and 105 deletions

View file

@ -33,9 +33,7 @@ define('QUESTION_EVENTSUBMIT', '7');
/**#@-*/ /**#@-*/
/**#@+ /**#@+
* The defined question types * The core question types
*
* @todo It would be nicer to have a fully automatic plug-in system
*/ */
define("SHORTANSWER", "1"); define("SHORTANSWER", "1");
define("TRUEFALSE", "2"); define("TRUEFALSE", "2");
@ -59,14 +57,24 @@ define("QUESTION_NUMANS", "10");
/** /**
* Array holding question type objects * Array holding question type objects
*/ */
$QTYPES= array(); global $QTYPES;
$QTYPES = array(); // This array will be populated when the questiontype.php files are loaded
/**
* Array of question types names translated to the user's language
*
* The $QTYPE_MENU array holds the names of all the question types that the user should
* be able to create directly. Some internal question types like random questions are excluded.
* The complete list of question types can be found in {@link $QTYPES}.
*/
$QTYPE_MENU = array(); // This array will be populated when the questiontype.php files are loaded
require_once("$CFG->dirroot/question/questiontypes/questiontype.php"); require_once("$CFG->dirroot/question/questiontypes/questiontype.php");
/* /*
* Load the questiontype.php file for each question type * Load the questiontype.php file for each question type
* These files in turn instantiate the corresponding question type class * These files in turn instantiate the corresponding question type class
* and adds it to the $QTYPES array * and add them to the $QTYPES array
*/ */
$qtypenames= get_list_of_plugins('question/questiontypes'); $qtypenames= get_list_of_plugins('question/questiontypes');
foreach($qtypenames as $qtypename) { foreach($qtypenames as $qtypename) {
@ -156,7 +164,9 @@ class cmoptions {
*/ */
function delete_question($question) { function delete_question($question) {
global $QTYPES; global $QTYPES;
if (isset($QTYPES[$question->qtype])) {
$QTYPES[$question->qtype]->delete_question($question); $QTYPES[$question->qtype]->delete_question($question);
} else {echo 'qtype: '.$question->qtype.'<br />';}
delete_records("question_answers", "question", $question->id); delete_records("question_answers", "question", $question->id);
delete_records("question_states", "question", $question->id); delete_records("question_states", "question", $question->id);
delete_records("question_sessions", "questionid", $question->id); delete_records("question_sessions", "questionid", $question->id);

View file

@ -43,6 +43,8 @@ function quiz_add_instance($quiz) {
$quiz->availableminute); $quiz->availableminute);
} }
$quiz->timelimit = round($quiz->timelimit);
if (empty($quiz->name)) { if (empty($quiz->name)) {
if (empty($quiz->intro)) { if (empty($quiz->intro)) {
$quiz->name = get_string('modulename', 'quiz'); $quiz->name = get_string('modulename', 'quiz');
@ -134,6 +136,8 @@ function quiz_update_instance($quiz) {
$quiz->availableminute); $quiz->availableminute);
} }
$quiz->timelimit = round($quiz->timelimit);
$quiz->id = $quiz->instance; $quiz->id = $quiz->instance;
if (!update_record("quiz", $quiz)) { if (!update_record("quiz", $quiz)) {
@ -268,7 +272,7 @@ function quiz_delete_instance($id) {
*/ */
function quiz_delete_course($course, $feedback=true) { function quiz_delete_course($course, $feedback=true) {
global $CFG; global $CFG, $QTYPES;
//To detect if we have created the "container category" //To detect if we have created the "container category"
$concatid = 0; $concatid = 0;

View file

@ -15,31 +15,6 @@
require_once($CFG->libdir.'/questionlib.php'); require_once($CFG->libdir.'/questionlib.php');
/**
* Array of question types names translated to the user's language
*
* The $QTYPE_MENU array holds the names of all the question types that the user should
* be able to create directly. Some internal question types like random questions are excluded.
* The complete list of question types can be found in {@link $QTYPES}.
*/
$QTYPE_MENU = array ( MULTICHOICE => get_string("multichoice", "quiz"),
TRUEFALSE => get_string("truefalse", "quiz"),
SHORTANSWER => get_string("shortanswer", "quiz"),
NUMERICAL => get_string("numerical", "quiz"),
CALCULATED => get_string("calculated", "quiz"),
MATCH => get_string("match", "quiz"),
DESCRIPTION => get_string("description", "quiz"),
RANDOMSAMATCH => get_string("randomsamatch", "quiz"),
MULTIANSWER => get_string("multianswer", "quiz"),
ESSAY => get_string("essay", "quiz")
);
// add remote question types
if ($rqp_types = get_records('question_rqp_types')) {
foreach($rqp_types as $type) {
$QTYPE_MENU[100+$type->id] = $type->name;
}
}
function question_category_form($course, $current, $recurse=1, $showhidden=false) { function question_category_form($course, $current, $recurse=1, $showhidden=false) {
global $CFG; global $CFG;

View file

@ -598,7 +598,10 @@ class question_calculated_qtype extends question_dataset_dependent_questiontype
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("CALCULATED", "10"); // already defined in questionlib.php
$QTYPES[CALCULATED]= new question_calculated_qtype(); $QTYPES[CALCULATED]= new question_calculated_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[CALCULATED] = get_string("calculated", "quiz");
function quiz_qtype_calculated_calculate_answer($formula, $individualdata, function quiz_qtype_calculated_calculate_answer($formula, $individualdata,
$tolerance, $tolerancetype, $answerlength, $answerformat='1', $unit='') { $tolerance, $tolerancetype, $answerlength, $answerformat='1', $unit='') {

View file

@ -66,6 +66,9 @@ class quiz_description_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("DESCRIPTION", "7"); // already defined in questionlib.php
$QTYPES[DESCRIPTION]= new quiz_description_qtype(); $QTYPES[DESCRIPTION]= new quiz_description_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[DESCRIPTION] = get_string("description", "quiz");
?> ?>

View file

@ -311,6 +311,9 @@ class question_essay_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("ESSAY", "12"); // already defined in questionlib.php
$QTYPES[ESSAY] = new question_essay_qtype(); $QTYPES[ESSAY] = new question_essay_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[ESSAY] = get_string("essay", "quiz");
?> ?>

View file

@ -337,6 +337,9 @@ class question_match_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("MATCH", "5"); // already defined in questionlib.php
$QTYPES[MATCH]= new question_match_qtype(); $QTYPES[MATCH]= new question_match_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[MATCH] = get_string("match", "quiz");
?> ?>

View file

@ -24,7 +24,7 @@ class quiz_embedded_cloze_qtype extends quiz_default_questiontype {
// Get relevant data indexed by positionkey from the multianswers table // Get relevant data indexed by positionkey from the multianswers table
if (!$sequence = get_field('question_multianswer', 'sequence', 'question', $question->id)) { if (!$sequence = get_field('question_multianswer', 'sequence', 'question', $question->id)) {
notify('Error: Missing question options!'); notify('Error: Cloze question '.$question->id.' is missing question options!');
return false; return false;
} }
@ -92,13 +92,13 @@ class quiz_embedded_cloze_qtype extends quiz_default_questiontype {
get_field('question_multianswer', 'id', 'question', $question->id)) { get_field('question_multianswer', 'id', 'question', $question->id)) {
$multianswer->id = $oldid; $multianswer->id = $oldid;
if (!update_record("question_multianswer", $multianswer)) { if (!update_record("question_multianswer", $multianswer)) {
$result->error = "Could not update quiz multianswer! " . $result->error = "Could not update cloze question options! " .
"(id=$multianswer->id)"; "(id=$multianswer->id)";
return $result; return $result;
} }
} else { } else {
if (!insert_record("question_multianswer", $multianswer)) { if (!insert_record("question_multianswer", $multianswer)) {
$result->error = "Could not insert quiz multianswer!"; $result->error = "Could not insert cloze question options!";
return $result; return $result;
} }
} }
@ -322,7 +322,7 @@ class quiz_embedded_cloze_qtype extends quiz_default_questiontype {
echo '</select>'; echo '</select>';
break; break;
default: default:
error("Unable to recognized questiontype ($wrapped->qtype) of error("Unable to recognize questiontype ($wrapped->qtype) of
question part $positionkey."); question part $positionkey.");
break; break;
} }
@ -380,8 +380,10 @@ class quiz_embedded_cloze_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("MULTIANSWER", "9"); // already defined in questionlib.php
$QTYPES[MULTIANSWER]= new quiz_embedded_cloze_qtype(); $QTYPES[MULTIANSWER]= new quiz_embedded_cloze_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[MULTIANSWER] = get_string("multianswer", "quiz");
///////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////
//// ADDITIONAL FUNCTIONS //// ADDITIONAL FUNCTIONS

View file

@ -372,6 +372,9 @@ class question_multichoice_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("MULTICHOICE", "3"); // already defined in questionlib.php
$QTYPES[MULTICHOICE]= new question_multichoice_qtype(); $QTYPES[MULTICHOICE]= new question_multichoice_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[MULTICHOICE] = get_string("multichoice", "quiz");
?> ?>

View file

@ -409,6 +409,9 @@ class question_numerical_qtype extends question_shortanswer_qtype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("NUMERICAL", "8"); // already defined in questionlib.php
$QTYPES[NUMERICAL]= new question_numerical_qtype(); $QTYPES[NUMERICAL]= new question_numerical_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[NUMERICAL] = get_string("numerical", "quiz");
?> ?>

View file

@ -237,6 +237,7 @@ class quiz_random_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("RANDOM", "4"); // already defined in questionlib.php
$QTYPES[RANDOM]= new quiz_random_qtype(); $QTYPES[RANDOM]= new quiz_random_qtype();
?> ?>

View file

@ -248,6 +248,9 @@ class question_randomsamatch_qtype extends question_match_qtype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("RANDOMSAMATCH", "6"); // already defined in questionlib.php
$QTYPES[RANDOMSAMATCH]= new question_randomsamatch_qtype(); $QTYPES[RANDOMSAMATCH]= new question_randomsamatch_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[RANDOMSAMATCH] = get_string("randomsamatch", "quiz");
?> ?>

View file

@ -437,6 +437,13 @@ class question_rqp_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("RQP", "11"); // already defined in questionlib.php
$QTYPES[RQP]= new question_rqp_qtype(); $QTYPES[RQP]= new question_rqp_qtype();
// The following adds the questiontype to the menu of types shown to teachers
if ($rqp_types = get_records('question_rqp_types')) {
foreach($rqp_types as $type) {
$QTYPE_MENU[100+$type->id] = $type->name;
}
}
?> ?>

View file

@ -246,6 +246,9 @@ class question_shortanswer_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("SHORTANSWER", "1"); // already defined in questionlib.php
$QTYPES[SHORTANSWER]= new question_shortanswer_qtype(); $QTYPES[SHORTANSWER]= new question_shortanswer_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[SHORTANSWER] = get_string("shortanswer", "quiz");
?> ?>

View file

@ -211,6 +211,9 @@ class question_truefalse_qtype extends quiz_default_questiontype {
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
//// INITIATION - Without this line the question type is not in use... /// //// INITIATION - Without this line the question type is not in use... ///
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// define("TRUEFALSE", "2"); // already defined in questionlib.php
$QTYPES[TRUEFALSE]= new question_truefalse_qtype(); $QTYPES[TRUEFALSE]= new question_truefalse_qtype();
// The following adds the questiontype to the menu of types shown to teachers
$QTYPE_MENU[TRUEFALSE] = get_string("truefalse", "quiz");
?> ?>

View file

@ -82,37 +82,6 @@
//----------------------------------------------------------- //-----------------------------------------------------------
include_once($CFG->libdir.'/questionlib.php'); include_once($CFG->libdir.'/questionlib.php');
// load questiontype-specific functions
unset($restorefns);
unset($restoremapfns);
unset($restorestatefns);
unset($recodeansfns);
//if ($qtypes = get_records('question_types')) {
if ($qtypes = get_list_of_plugins('question/questiontypes')) {
foreach ($qtypes as $name) {
$qtype->name = $name;
$restorelib = $CFG->dirroot.'/question/questiontypes/'.$qtype->name.'/restorelib.php';
if (file_exists($restorelib)) {
include_once($restorelib);
$restorefn = 'question_'.$qtype->name.'_restore';
if (function_exists($restorefn)) {
$restorefns[$qtype->name] = $restorefn;
}
$restoremapfn = 'question_'.$qtype->name.'_restore_map';
if (function_exists($restoremapfn)) {
$restoremapfns[$qtype->name] = $restoremapfn;
}
$restorestatefn = 'question_'.$qtype->name.'_states_restore';
if (function_exists($restorestatefn)) {
$restorestatefns[$qtype->name] = $restorestatefn;
}
$recodeansfn = 'question_'.$qtype->name.'_recode_answer';
if (function_exists($recodeansfn)) {
$recodeansfns[$qtype->name] = $recodeansfn;
}
}
}
}
function restore_question_categories($category,$restore) { function restore_question_categories($category,$restore) {
@ -188,8 +157,30 @@
function restore_questions ($old_category_id,$new_category_id,$info,$restore) { function restore_questions ($old_category_id,$new_category_id,$info,$restore) {
global $CFG; global $CFG, $QTYPES;
// load questiontype-specific functions
unset($restorefns);
unset($restoremapfns);
unset($recodeansfns);
foreach ($QTYPES as $key => $qtype) {
$restorelib = $CFG->dirroot.'/question/questiontypes/'.$qtype->name().'/restorelib.php';
if (file_exists($restorelib)) {
include_once($restorelib);
$restorefn = 'question_'.$qtype->name().'_restore';
if (function_exists($restorefn)) {
$restorefns[$key] = $restorefn;
}else {echo $restorefn;}
$restoremapfn = 'question_'.$qtype->name().'_restore_map';
if (function_exists($restoremapfn)) {
$restoremapfns[$key] = $restoremapfn;
}
$recodeansfn = 'question_'.$qtype->name().'_recode_answer';
if (function_exists($recodeansfn)) {
$recodeansfns[$key] = $recodeansfn;
}
}
}
$status = true; $status = true;
$restored_questions = array(); $restored_questions = array();
@ -273,8 +264,8 @@
//Now, restore every question_answers in this question //Now, restore every question_answers in this question
$status = question_restore_answers($oldid,$newid,$que_info,$restore); $status = question_restore_answers($oldid,$newid,$que_info,$restore);
//Now, depending of the type of questions, invoke different functions //Now, depending of the type of questions, invoke different functions
if (isset($restorefns[$question->type])) { if (isset($restorefns[$question->qtype])) {
$status = $restorefns[$question->type]->restore($oldid,$newid,$que_info,$restore); $status = $restorefns[$question->qtype]($oldid,$newid,$que_info,$restore);
} }
} else { } else {
//We are NOT creating the question, but we need to know every question_answers //We are NOT creating the question, but we need to know every question_answers
@ -284,8 +275,8 @@
//Now, depending of the type of questions, invoke different functions //Now, depending of the type of questions, invoke different functions
//to create the necessary mappings in backup_ids, because we are not //to create the necessary mappings in backup_ids, because we are not
//creating the question, but need some records in backup table //creating the question, but need some records in backup table
if (isset($restoremapfns[$question->type])) { if (isset($restoremapfns[$question->qtype])) {
$status = $restoremapfns[$question->type]->restore($oldid,$newid,$que_info,$restore); $status = $restoremapfns[$question->qtype]($oldid,$newid,$que_info,$restore);
} }
} }
@ -615,10 +606,12 @@
//We have to recode the answer field //We have to recode the answer field
//It depends of the question type !! //It depends of the question type !!
//We get the question first //We get the question first
$question = get_record("question","id",$state->question); if (!$question = get_record("question","id",$state->question)) {
error("Can't find the record for question $state->question for which I am trying to restore a state");
}
//It exists //It exists
if ($question) { if ($question) {
//Depending of the qtype, we make different recodes //Depending on the qtype, we make different recodes
switch ($question->qtype) { switch ($question->qtype) {
case 1: //SHORTANSWER QTYPE case 1: //SHORTANSWER QTYPE
//Nothing to do. The response is a text. //Nothing to do. The response is a text.
@ -728,41 +721,38 @@
//Nothing to do. The response is a text. //Nothing to do. The response is a text.
break; break;
case 9: //MULTIANSWER QTYPE case 9: //MULTIANSWER QTYPE
//The answer is a comma separated list of hypen separated multianswer_id and answers. We must recode them. //The answer is a comma separated list of hypen separated sequence number and answers. We may have to recode the answers
$answer_field = ""; $answer_field = "";
$in_first = true; $in_first = true;
$tok = strtok($state->answer,","); $tok = strtok($state->answer,",");
while ($tok) { while ($tok) {
//Extract the multianswer_id and the answer //Extract the multianswer_id and the answer
$exploded = explode("-",$tok); $exploded = explode("-",$tok);
$multianswer_id = $exploded[0]; $seqnum = $exploded[0];
$answer = $exploded[1]; $answer = $exploded[1];
//Get the multianswer from backup_ids // $sequence is an ordered array of the question ids.
$mul = backup_getid($restore->backup_unique_code,"question_multianswer",$multianswer_id); if (!$sequence = get_field('question_multianswer', 'sequence', 'question', $question->id)) {
if ($mul) { error("The cloze question $question->id is missing its options");
//Now, depending of the answertype field in question_multianswer }
//we do diferent things $sequence = explode(',', $sequence);
$mul_db = get_record ("question_multianswer","id",$mul->new_id); // The id of the current question.
if ($mul_db->answertype == "1") { $wrappedquestionid = $sequence[$seqnum-1];
//Shortanswer // now we can find the question
//The answer is text, do nothing if (!$wrappedquestion = get_record('question', 'id', $wrappedquestionid)) {
} else if ($mul_db->answertype == "3") { error("Can't find the subquestion $wrappedquestionid that is used as part $seqnum in cloze question $question->id");
//Multichoice }
// For multichoice question we need to recode the answer
if ($wrappedquestion->qtype == MULTICHOICE) {
//The answer is an answer_id, look for it in backup_ids //The answer is an answer_id, look for it in backup_ids
$ans = backup_getid($restore->backup_unique_code,"question_answers",$answer); $ans = backup_getid($restore->backup_unique_code,"question_answers",$answer);
$answer = $ans->new_id; $answer = $ans->new_id;
} else if ($mul_db->answertype == "8") {
//Numeric
//The answer is text, do nothing
} }
//build the new answer field for each pair
//Finaly, build the new answer field for each pair
if ($in_first) { if ($in_first) {
$answer_field .= $mul->new_id."-".$answer; $answer_field .= $seqnum."-".$answer;
$in_first = false; $in_first = false;
} else { } else {
$answer_field .= ",".$mul->new_id."-".$answer; $answer_field .= ",".$seqnum."-".$answer;
}
} }
//check for next //check for next
$tok = strtok(","); $tok = strtok(",");
@ -800,9 +790,20 @@
backup_putid($restore->backup_unique_code,"question_states",$oldid, backup_putid($restore->backup_unique_code,"question_states",$oldid,
$newid); $newid);
//Now process question type specific state information //Now process question type specific state information
foreach ($restorestatefns as $restorestatefn) {
if ($qtypes = get_list_of_plugins('question/questiontypes')) {
foreach ($qtypes as $name) {
$qtype->name = $name;
$restorelib = $CFG->dirroot.'/question/questiontypes/'.$qtype->name.'/restorelib.php';
if (file_exists($restorelib)) {
include_once($restorelib);
$restorestatefn = 'question_'.$qtype->name.'_states_restore';
if (function_exists($restorestatefn)) {
$restorestatefn($newid,$res_info,$restore); $restorestatefn($newid,$res_info,$restore);
} }
}
}
}
} else { } else {
$status = false; $status = false;
} }