2002-10-04 02:59:05 +00:00
|
|
|
<?PHP // $Id$
|
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Library of function for module quiz
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
/// CONSTANTS ///////////////////////////////////////////////////////////////////
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
define("GRADEHIGHEST", "1");
|
|
|
|
define("GRADEAVERAGE", "2");
|
|
|
|
define("ATTEMPTFIRST", "3");
|
|
|
|
define("ATTEMPTLAST", "4");
|
|
|
|
$QUIZ_GRADE_METHOD = array ( GRADEHIGHEST => get_string("gradehighest", "quiz"),
|
|
|
|
GRADEAVERAGE => get_string("gradeaverage", "quiz"),
|
|
|
|
ATTEMPTFIRST => get_string("attemptfirst", "quiz"),
|
|
|
|
ATTEMPTLAST => get_string("attemptlast", "quiz"));
|
|
|
|
|
|
|
|
define("SHORTANSWER", "1");
|
|
|
|
define("TRUEFALSE", "2");
|
|
|
|
define("MULTICHOICE", "3");
|
2002-10-20 05:35:25 +00:00
|
|
|
define("RANDOM", "4");
|
2002-10-15 12:54:11 +00:00
|
|
|
$QUIZ_QUESTION_TYPE = array ( MULTICHOICE => get_string("multichoice", "quiz"),
|
2002-10-13 07:17:48 +00:00
|
|
|
TRUEFALSE => get_string("truefalse", "quiz"),
|
2002-10-21 15:00:51 +00:00
|
|
|
SHORTANSWER => get_string("shortanswer", "quiz") );
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// FUNCTIONS ///////////////////////////////////////////////////////////////////
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
function quiz_add_instance($quiz) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Given an object containing all the necessary data,
|
|
|
|
/// (defined by the form in mod.html) this function
|
|
|
|
/// will create a new instance and return the id number
|
|
|
|
/// of the new instance.
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-15 16:22:18 +00:00
|
|
|
$quiz->created = time();
|
2002-10-04 02:59:05 +00:00
|
|
|
$quiz->timemodified = time();
|
2002-10-15 16:22:18 +00:00
|
|
|
$quiz->timeopen = make_timestamp($quiz->openyear, $quiz->openmonth, $quiz->openday,
|
|
|
|
$quiz->openhour, $quiz->openminute, $quiz->opensecond);
|
|
|
|
$quiz->timeclose = make_timestamp($quiz->closeyear, $quiz->closemonth, $quiz->closeday,
|
|
|
|
$quiz->closehour, $quiz->closeminute, $quiz->closesecond);
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-14 12:21:18 +00:00
|
|
|
if (!$quiz->id = insert_record("quiz", $quiz)) {
|
|
|
|
return false; // some error occurred
|
|
|
|
}
|
|
|
|
|
2002-10-14 15:57:33 +00:00
|
|
|
// The grades for every question in this quiz are stored in an array
|
2002-10-14 12:21:18 +00:00
|
|
|
if ($quiz->grades) {
|
|
|
|
foreach ($quiz->grades as $question => $grade) {
|
2002-10-15 12:54:11 +00:00
|
|
|
if ($question and $grade) {
|
|
|
|
unset($questiongrade);
|
|
|
|
$questiongrade->quiz = $quiz->id;
|
|
|
|
$questiongrade->question = $question;
|
|
|
|
$questiongrade->grade = $grade;
|
|
|
|
if (!insert_record("quiz_question_grades", $questiongrade)) {
|
|
|
|
return false;
|
|
|
|
}
|
2002-10-14 12:21:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-14 12:21:18 +00:00
|
|
|
return $quiz->id;
|
2002-10-04 02:59:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function quiz_update_instance($quiz) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Given an object containing all the necessary data,
|
|
|
|
/// (defined by the form in mod.html) this function
|
|
|
|
/// will update an existing instance with new data.
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
$quiz->timemodified = time();
|
2002-10-15 16:22:18 +00:00
|
|
|
$quiz->timeopen = make_timestamp($quiz->openyear, $quiz->openmonth, $quiz->openday,
|
|
|
|
$quiz->openhour, $quiz->openminute, $quiz->opensecond);
|
|
|
|
$quiz->timeclose = make_timestamp($quiz->closeyear, $quiz->closemonth, $quiz->closeday,
|
|
|
|
$quiz->closehour, $quiz->closeminute, $quiz->closesecond);
|
2002-10-04 02:59:05 +00:00
|
|
|
$quiz->id = $quiz->instance;
|
|
|
|
|
2002-10-14 12:21:18 +00:00
|
|
|
if (!update_record("quiz", $quiz)) {
|
|
|
|
return false; // some error occurred
|
|
|
|
}
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-14 12:21:18 +00:00
|
|
|
|
2002-10-14 15:57:33 +00:00
|
|
|
// The grades for every question in this quiz are stored in an array
|
2002-10-14 12:21:18 +00:00
|
|
|
// Insert or update records as appropriate
|
|
|
|
|
|
|
|
$existing = get_records("quiz_question_grades", "quiz", $quiz->id, "", "question,grade,id");
|
|
|
|
|
|
|
|
if ($quiz->grades) {
|
|
|
|
foreach ($quiz->grades as $question => $grade) {
|
2002-10-15 12:54:11 +00:00
|
|
|
if ($question and $grade) {
|
|
|
|
unset($questiongrade);
|
|
|
|
$questiongrade->quiz = $quiz->id;
|
|
|
|
$questiongrade->question = $question;
|
|
|
|
$questiongrade->grade = $grade;
|
|
|
|
if (isset($existing[$question])) {
|
|
|
|
if ($existing[$question]->grade != $grade) {
|
|
|
|
$questiongrade->id = $existing[$question]->id;
|
|
|
|
if (!update_record("quiz_question_grades", $questiongrade)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!insert_record("quiz_question_grades", $questiongrade)) {
|
2002-10-14 12:21:18 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2002-10-04 02:59:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function quiz_delete_instance($id) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Given an ID of an instance of this module,
|
|
|
|
/// this function will permanently delete the instance
|
|
|
|
/// and any data that depends on it.
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
if (! $quiz = get_record("quiz", "id", "$id")) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$result = true;
|
|
|
|
|
2002-10-14 15:57:33 +00:00
|
|
|
if ($attempts = get_record("quiz_attempts", "quiz", "$quiz->id")) {
|
|
|
|
foreach ($attempts as $attempt) {
|
|
|
|
if (! delete_records("quiz_responses", "attempt", "$attempt->id")) {
|
|
|
|
$result = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! delete_records("quiz_attempts", "quiz", "$quiz->id")) {
|
|
|
|
$result = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! delete_records("quiz_grades", "quiz", "$quiz->id")) {
|
|
|
|
$result = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! delete_records("quiz_question_grades", "quiz", "$quiz->id")) {
|
|
|
|
$result = false;
|
|
|
|
}
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
if (! delete_records("quiz", "id", "$quiz->id")) {
|
|
|
|
$result = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_user_outline($course, $user, $mod, $quiz) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Return a small object with summary information about what a
|
|
|
|
/// user has done with a given particular instance of this module
|
|
|
|
/// Used for user activity reports.
|
|
|
|
/// $return->time = the time they did it
|
|
|
|
/// $return->info = a short text description
|
2002-12-22 13:55:30 +00:00
|
|
|
if ($grade = get_record("quiz_grades", "user", $user->id, "quiz", $quiz->id)) {
|
2002-10-17 08:28:18 +00:00
|
|
|
|
|
|
|
if ($grade->grade) {
|
|
|
|
$result->info = get_string("grade").": $grade->grade";
|
|
|
|
}
|
|
|
|
$result->time = $grade->timemodified;
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
return NULL;
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
return $return;
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_user_complete($course, $user, $mod, $quiz) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Print a detailed representation of what a user has done with
|
|
|
|
/// a given particular instance of this module, for user activity reports.
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_print_recent_activity(&$logs, $isteacher=false) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Given a list of logs, assumed to be those since the last login
|
|
|
|
/// this function prints a short list of changes related to this module
|
|
|
|
/// If isteacher is true then perhaps additional information is printed.
|
|
|
|
/// This function is called from course/lib.php: print_recent_activity()
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
global $CFG, $COURSE_TEACHER_COLOR;
|
|
|
|
|
|
|
|
return $content; // True if anything was printed, otherwise false
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_cron () {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Function to be run periodically according to the moodle cron
|
|
|
|
/// This function searches for things that need to be done, such
|
|
|
|
/// as sending out mail, toggling flags etc ...
|
2002-10-04 02:59:05 +00:00
|
|
|
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2002-10-17 07:23:51 +00:00
|
|
|
function quiz_grades($quizid) {
|
2002-10-17 07:55:54 +00:00
|
|
|
/// Must return an array of grades, indexed by user, and a max grade.
|
|
|
|
|
2002-12-22 13:55:30 +00:00
|
|
|
$return->grades = get_records_menu("quiz_grades", "quiz", $quizid, "", "user,grade");
|
2002-10-17 07:55:54 +00:00
|
|
|
$return->maxgrade = get_field("quiz", "grade", "id", "$quizid");
|
|
|
|
return $return;
|
2002-10-17 07:23:51 +00:00
|
|
|
}
|
|
|
|
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-12-22 13:55:30 +00:00
|
|
|
/// SQL FUNCTIONS ////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
function quiz_move_questions($category1, $category2) {
|
|
|
|
global $CFG;
|
|
|
|
return execute_sql("UPDATE {$CFG->prefix}quiz_questions
|
|
|
|
SET category = '$category2'
|
|
|
|
WHERE category = '$category1'",
|
|
|
|
false);
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_get_question_grades($quizid, $questionlist) {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
return get_records_sql("SELECT question,grade
|
|
|
|
FROM {$CFG->prefix}quiz_question_grades
|
|
|
|
WHERE quiz = '$quizid'
|
|
|
|
AND question IN ($questionlist)");
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_get_grade_records($quiz) {
|
|
|
|
/// Gets all info required to display the table of quiz results
|
|
|
|
/// for report.php
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
return get_records_sql("SELECT qg.*, u.firstname, u.lastname, u.picture
|
|
|
|
FROM {$CFG->prefix}quiz_grades qg,
|
|
|
|
{$CFG->prefix}user u
|
|
|
|
WHERE qg.quiz = '$quiz->id'
|
|
|
|
AND qg.user = u.id");
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_get_answers($question) {
|
|
|
|
// Given a question, returns the correct answers and grades
|
|
|
|
global $CFG;
|
|
|
|
switch ($question->type) {
|
|
|
|
case SHORTANSWER; // Could be multiple answers
|
|
|
|
return get_records_sql("SELECT a.*, sa.usecase, g.grade
|
|
|
|
FROM {$CFG->prefix}quiz_shortanswer sa,
|
|
|
|
{$CFG->prefix}quiz_answers a,
|
|
|
|
{$CFG->prefix}quiz_question_grades g
|
|
|
|
WHERE sa.question = '$question->id'
|
|
|
|
AND sa.question = a.question
|
|
|
|
AND sa.question = g.question");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TRUEFALSE; // Should be always two answers
|
|
|
|
return get_records_sql("SELECT a.*, g.grade
|
|
|
|
FROM {$CFG->prefix}quiz_answers a,
|
|
|
|
{$CFG->prefix}quiz_question_grades g
|
|
|
|
WHERE a.question = '$question->id'
|
|
|
|
AND a.question = g.question");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MULTICHOICE; // Should be multiple answers
|
|
|
|
return get_records_sql("SELECT a.*, mc.single, g.grade
|
|
|
|
FROM {$CFG->prefix}quiz_multichoice mc,
|
|
|
|
{$CFG->prefix}quiz_answers a,
|
|
|
|
{$CFG->prefix}quiz_question_grades g
|
|
|
|
WHERE mc.question = '$question->id'
|
|
|
|
AND mc.question = a.question
|
|
|
|
AND mc.question = g.question");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RANDOM:
|
|
|
|
return false; // Not done yet
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function quiz_get_attempt_responses($attempt) {
|
|
|
|
// Given an attempt object, this function gets all the
|
|
|
|
// stored responses and returns them in a format suitable
|
|
|
|
// for regrading using quiz_grade_attempt_results()
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
if (!$responses = get_records_sql("SELECT q.id, q.type, r.answer
|
|
|
|
FROM {$CFG->prefix}quiz_responses r,
|
|
|
|
{$CFG->prefix}quiz_questions q
|
|
|
|
WHERE r.attempt = '$attempt->id'
|
|
|
|
AND q.id = r.question")) {
|
|
|
|
notify("Could not find any responses for that attempt!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($responses as $key => $response) {
|
|
|
|
$responses[$key]->answer = explode(",",$response->answer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $responses;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2002-10-04 02:59:05 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////////////////////
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Any other quiz functions go here. Each of them must have a name that
|
|
|
|
/// starts with quiz_
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-16 09:35:04 +00:00
|
|
|
function quiz_print_comment($text) {
|
|
|
|
global $THEME;
|
|
|
|
|
|
|
|
echo "<FONT COLOR=\"$THEME->cellheading2\">".text_to_html($text, true, false)."</FONT>";
|
|
|
|
}
|
|
|
|
|
2002-10-20 05:35:25 +00:00
|
|
|
function quiz_print_correctanswer($text) {
|
|
|
|
global $THEME;
|
|
|
|
|
|
|
|
echo "<P ALIGN=RIGHT><SPAN CLASS=highlight>$text</SPAN></P>";
|
|
|
|
}
|
|
|
|
|
2002-10-20 17:15:39 +00:00
|
|
|
function quiz_print_question_icon($question) {
|
|
|
|
// Prints a question icon
|
2002-10-23 12:10:38 +00:00
|
|
|
|
|
|
|
global $QUIZ_QUESTION_TYPE;
|
|
|
|
|
|
|
|
echo "<A HREF=\"question.php?id=$question->id\" TITLE=\"".$QUIZ_QUESTION_TYPE[$question->type]."\">";
|
2002-10-20 17:15:39 +00:00
|
|
|
switch ($question->type) {
|
|
|
|
case SHORTANSWER:
|
2002-10-23 12:10:38 +00:00
|
|
|
echo "<IMG BORDER=0 HEIGHT=16 WIDTH=16 SRC=\"pix/sa.gif\">";
|
2002-10-20 17:15:39 +00:00
|
|
|
break;
|
|
|
|
case TRUEFALSE:
|
2002-10-23 12:10:38 +00:00
|
|
|
echo "<IMG BORDER=0 HEIGHT=16 WIDTH=16 SRC=\"pix/tf.gif\">";
|
2002-10-20 17:15:39 +00:00
|
|
|
break;
|
|
|
|
case MULTICHOICE:
|
2002-10-23 12:10:38 +00:00
|
|
|
echo "<IMG BORDER=0 HEIGHT=16 WIDTH=16 SRC=\"pix/mc.gif\">";
|
2002-10-20 17:15:39 +00:00
|
|
|
break;
|
|
|
|
case RANDOM:
|
2002-10-23 12:10:38 +00:00
|
|
|
echo "<IMG BORDER=0 HEIGHT=16 WIDTH=16 SRC=\"pix/rs.gif\">";
|
2002-10-20 17:15:39 +00:00
|
|
|
break;
|
|
|
|
}
|
2002-10-23 12:10:38 +00:00
|
|
|
echo "</A>\n";
|
2002-10-20 17:15:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_print_question($number, $questionid, $grade, $courseid,
|
2002-10-20 05:35:25 +00:00
|
|
|
$feedback=NULL, $response=NULL, $actualgrade=NULL, $correct=NULL) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Prints a quiz question, any format
|
2002-10-16 09:35:04 +00:00
|
|
|
|
2002-10-06 17:06:54 +00:00
|
|
|
if (!$question = get_record("quiz_questions", "id", $questionid)) {
|
|
|
|
notify("Error: Question not found!");
|
|
|
|
}
|
|
|
|
|
|
|
|
$stranswer = get_string("answer", "quiz");
|
|
|
|
$strmarks = get_string("marks", "quiz");
|
|
|
|
|
|
|
|
echo "<TABLE WIDTH=100% CELLSPACING=10><TR><TD NOWRAP WIDTH=100 VALIGN=top>";
|
2002-10-16 09:35:04 +00:00
|
|
|
echo "<P ALIGN=CENTER><B>$number</B></P>";
|
2002-10-16 15:52:29 +00:00
|
|
|
if ($feedback or $response) {
|
2002-10-16 09:35:04 +00:00
|
|
|
echo "<P ALIGN=CENTER><FONT SIZE=1>$strmarks: $actualgrade/$grade</FONT></P>";
|
|
|
|
} else {
|
|
|
|
echo "<P ALIGN=CENTER><FONT SIZE=1>$grade $strmarks</FONT></P>";
|
|
|
|
}
|
2002-10-06 17:06:54 +00:00
|
|
|
print_spacer(1,100);
|
|
|
|
echo "</TD><TD VALIGN=TOP>";
|
|
|
|
|
|
|
|
switch ($question->type) {
|
2002-10-13 07:17:48 +00:00
|
|
|
case SHORTANSWER:
|
2002-10-06 17:06:54 +00:00
|
|
|
if (!$options = get_record("quiz_shortanswer", "question", $question->id)) {
|
|
|
|
notify("Error: Missing question options!");
|
|
|
|
}
|
2002-10-15 10:04:28 +00:00
|
|
|
echo text_to_html($question->questiontext);
|
2002-10-06 17:06:54 +00:00
|
|
|
if ($question->image) {
|
|
|
|
print_file_picture($question->image, $courseid, 200);
|
|
|
|
}
|
2002-10-16 09:35:04 +00:00
|
|
|
if ($response) {
|
|
|
|
$value = "VALUE=\"$response[0]\"";
|
|
|
|
}
|
|
|
|
echo "<P ALIGN=RIGHT>$stranswer: <INPUT TYPE=TEXT NAME=q$question->id SIZE=20 $value></P>";
|
|
|
|
if ($feedback) {
|
|
|
|
quiz_print_comment("<P ALIGN=right>$feedback[0]</P>");
|
|
|
|
}
|
2002-10-20 05:35:25 +00:00
|
|
|
if ($correct) {
|
2002-10-20 17:15:39 +00:00
|
|
|
$correctanswers = implode(", ", $correct);
|
2002-10-20 05:35:25 +00:00
|
|
|
quiz_print_correctanswer($correctanswers);
|
|
|
|
}
|
2002-10-06 17:06:54 +00:00
|
|
|
break;
|
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
case TRUEFALSE:
|
2002-10-06 17:06:54 +00:00
|
|
|
if (!$options = get_record("quiz_truefalse", "question", $question->id)) {
|
|
|
|
notify("Error: Missing question options!");
|
|
|
|
}
|
|
|
|
if (!$true = get_record("quiz_answers", "id", $options->true)) {
|
|
|
|
notify("Error: Missing question answers!");
|
|
|
|
}
|
|
|
|
if (!$false = get_record("quiz_answers", "id", $options->false)) {
|
|
|
|
notify("Error: Missing question answers!");
|
|
|
|
}
|
|
|
|
if (!$true->answer) {
|
|
|
|
$true->answer = get_string("true", "quiz");
|
|
|
|
}
|
|
|
|
if (!$false->answer) {
|
|
|
|
$false->answer = get_string("false", "quiz");
|
|
|
|
}
|
2002-10-15 10:04:28 +00:00
|
|
|
echo text_to_html($question->questiontext);
|
2002-10-06 17:06:54 +00:00
|
|
|
if ($question->image) {
|
|
|
|
print_file_picture($question->image, $courseid, 200);
|
|
|
|
}
|
2002-10-16 09:35:04 +00:00
|
|
|
|
|
|
|
if ($response[$true->id]) {
|
|
|
|
$truechecked = "CHECKED";
|
|
|
|
$feedbackid = $true->id;
|
|
|
|
} else if ($response[$false->id]) {
|
|
|
|
$falsechecked = "CHECKED";
|
|
|
|
$feedbackid = $false->id;
|
|
|
|
}
|
2002-10-20 05:35:25 +00:00
|
|
|
if ($correct) {
|
|
|
|
if ($correct[$true->id]) {
|
|
|
|
$truecorrect = "CLASS=highlight";
|
|
|
|
}
|
|
|
|
if ($correct[$false->id]) {
|
|
|
|
$falsecorrect = "CLASS=highlight";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
echo "<TABLE ALIGN=right cellpadding=5><TR><TD align=right>$stranswer: ";
|
|
|
|
echo "<TD $truecorrect>";
|
2002-10-16 09:35:04 +00:00
|
|
|
echo "<INPUT $truechecked TYPE=RADIO NAME=\"q$question->id\" VALUE=\"$true->id\">$true->answer";
|
2002-10-20 05:35:25 +00:00
|
|
|
echo "</TD><TD $falsecorrect>";
|
2002-10-16 09:35:04 +00:00
|
|
|
echo "<INPUT $falsechecked TYPE=RADIO NAME=\"q$question->id\" VALUE=\"$false->id\">$false->answer</P>";
|
2002-10-20 05:35:25 +00:00
|
|
|
echo "</TD></TR></TABLE><BR CLEAR=ALL>";
|
2002-10-16 09:35:04 +00:00
|
|
|
if ($feedback) {
|
|
|
|
quiz_print_comment("<P ALIGN=right>$feedback[$feedbackid]</P>");
|
|
|
|
}
|
|
|
|
|
2002-10-06 17:06:54 +00:00
|
|
|
break;
|
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
case MULTICHOICE:
|
2002-10-06 17:06:54 +00:00
|
|
|
if (!$options = get_record("quiz_multichoice", "question", $question->id)) {
|
|
|
|
notify("Error: Missing question options!");
|
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
if (!$answers = get_records_list("quiz_answers", "id", $options->answers)) {
|
2002-10-06 17:06:54 +00:00
|
|
|
notify("Error: Missing question answers!");
|
|
|
|
}
|
2002-10-15 10:04:28 +00:00
|
|
|
echo text_to_html($question->questiontext);
|
2002-10-06 17:06:54 +00:00
|
|
|
if ($question->image) {
|
|
|
|
print_file_picture($question->image, $courseid, 200);
|
|
|
|
}
|
|
|
|
echo "<TABLE ALIGN=right>";
|
|
|
|
echo "<TR><TD valign=top>$stranswer: </TD><TD>";
|
2002-11-12 17:20:04 +00:00
|
|
|
echo "<TABLE>";
|
2002-10-06 17:06:54 +00:00
|
|
|
$answerids = explode(",", $options->answers);
|
2002-10-16 09:35:04 +00:00
|
|
|
|
2002-10-06 17:06:54 +00:00
|
|
|
foreach ($answerids as $key => $answerid) {
|
|
|
|
$answer = $answers[$answerid];
|
|
|
|
$qnum = $key + 1;
|
2002-10-16 09:35:04 +00:00
|
|
|
|
|
|
|
if ($feedback and $response[$answerid]) {
|
|
|
|
$checked = "CHECKED";
|
|
|
|
} else {
|
|
|
|
$checked = "";
|
|
|
|
}
|
2002-10-06 17:06:54 +00:00
|
|
|
echo "<TR><TD valign=top>";
|
2002-10-13 07:17:48 +00:00
|
|
|
if ($options->single) {
|
2002-10-16 09:35:04 +00:00
|
|
|
echo "<INPUT $checked TYPE=RADIO NAME=q$question->id VALUE=\"$answer->id\">";
|
2002-10-06 17:06:54 +00:00
|
|
|
} else {
|
2002-10-16 09:35:04 +00:00
|
|
|
echo "<INPUT $checked TYPE=CHECKBOX NAME=q$question->id"."a$answer->id VALUE=\"$answer->id\">";
|
2002-10-06 17:06:54 +00:00
|
|
|
}
|
|
|
|
echo "</TD>";
|
2002-10-20 05:35:25 +00:00
|
|
|
if ($feedback and $correct[$answer->id]) {
|
|
|
|
echo "<TD valign=top CLASS=highlight>$qnum. $answer->answer</TD>";
|
|
|
|
} else {
|
|
|
|
echo "<TD valign=top>$qnum. $answer->answer</TD>";
|
|
|
|
}
|
2002-10-16 09:35:04 +00:00
|
|
|
if ($feedback) {
|
|
|
|
echo "<TD valign=top> ";
|
|
|
|
if ($response[$answerid]) {
|
|
|
|
quiz_print_comment($feedback[$answerid]);
|
|
|
|
}
|
|
|
|
echo "</TD>";
|
|
|
|
}
|
2002-10-06 17:06:54 +00:00
|
|
|
echo "</TR>";
|
|
|
|
}
|
|
|
|
echo "</TABLE>";
|
|
|
|
echo "</TABLE>";
|
|
|
|
break;
|
|
|
|
|
2002-10-20 05:35:25 +00:00
|
|
|
case RANDOM:
|
|
|
|
echo "<P>Random questions not supported yet</P>";
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
2002-10-06 17:06:54 +00:00
|
|
|
default:
|
|
|
|
notify("Error: Unknown question type!");
|
|
|
|
}
|
|
|
|
|
|
|
|
echo "</TD></TR></TABLE>";
|
2002-10-06 03:23:16 +00:00
|
|
|
}
|
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
function quiz_print_quiz_questions($quiz, $results=NULL) {
|
|
|
|
// Prints a whole quiz on one page.
|
|
|
|
|
|
|
|
if (!$quiz->questions) {
|
2002-10-14 15:57:33 +00:00
|
|
|
notify("No questions have been defined!", "view.php?id=$cm->id");
|
|
|
|
return false;
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$questions = explode(",", $quiz->questions);
|
|
|
|
|
|
|
|
if (!$grades = get_records_list("quiz_question_grades", "question", $quiz->questions, "", "question,grade")) {
|
2002-10-14 15:57:33 +00:00
|
|
|
notify("No grades were found for these questions!");
|
|
|
|
return false;
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
|
|
|
|
2002-12-09 08:20:19 +00:00
|
|
|
$strconfirmattempt = addslashes(get_string("readytosend", "quiz"));
|
|
|
|
|
|
|
|
echo "<FORM METHOD=POST ACTION=attempt.php onsubmit=\"return confirm('$strconfirmattempt');\">";
|
2002-10-13 07:17:48 +00:00
|
|
|
echo "<INPUT TYPE=hidden NAME=q VALUE=\"$quiz->id\">";
|
2002-10-16 09:35:04 +00:00
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
foreach ($questions as $key => $questionid) {
|
2002-10-20 05:35:25 +00:00
|
|
|
$feedback = NULL;
|
|
|
|
$response = NULL;
|
|
|
|
$actualgrades = NULL;
|
|
|
|
$correct = NULL;
|
2002-10-16 09:35:04 +00:00
|
|
|
if ($results) {
|
2002-10-20 05:35:25 +00:00
|
|
|
$feedback = $results->feedback[$questionid];
|
|
|
|
$response = $results->response[$questionid];
|
|
|
|
$actualgrades = $results->grades[$questionid];
|
|
|
|
if ($quiz->correctanswers) {
|
|
|
|
$correct = $results->correct[$questionid];
|
|
|
|
}
|
2002-10-16 09:35:04 +00:00
|
|
|
}
|
2002-10-20 05:35:25 +00:00
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
print_simple_box_start("CENTER", "90%");
|
2002-10-16 09:35:04 +00:00
|
|
|
quiz_print_question($key+1, $questionid, $grades[$questionid]->grade, $quiz->course,
|
2002-10-20 05:35:25 +00:00
|
|
|
$feedback, $response, $actualgrades, $correct);
|
2002-10-13 07:17:48 +00:00
|
|
|
print_simple_box_end();
|
|
|
|
echo "<BR>";
|
|
|
|
}
|
2002-10-16 09:35:04 +00:00
|
|
|
|
|
|
|
if (!$results) {
|
|
|
|
echo "<CENTER><INPUT TYPE=submit VALUE=\"".get_string("savemyanswers", "quiz")."\"></CENTER>";
|
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
echo "</FORM>";
|
2002-10-14 15:57:33 +00:00
|
|
|
|
|
|
|
return true;
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
2002-10-14 09:07:13 +00:00
|
|
|
|
|
|
|
function quiz_get_default_category($courseid) {
|
|
|
|
if ($categories = get_records("quiz_categories", "course", $courseid, "id")) {
|
|
|
|
foreach ($categories as $category) {
|
|
|
|
return $category; // Return the first one (lowest id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, we need to make one
|
2002-10-14 15:57:33 +00:00
|
|
|
$category->name = get_string("default", "quiz");
|
|
|
|
$category->info = get_string("defaultinfo", "quiz");
|
2002-10-14 09:07:13 +00:00
|
|
|
$category->course = $courseid;
|
|
|
|
$category->publish = 0;
|
|
|
|
|
|
|
|
if (!$category->id = insert_record("quiz_categories", $category)) {
|
|
|
|
notify("Error creating a default category!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return $category;
|
|
|
|
}
|
|
|
|
|
2002-10-20 17:15:39 +00:00
|
|
|
function quiz_get_category_menu($courseid, $published=false) {
|
|
|
|
if ($published) {
|
|
|
|
$publish = "OR publish = '1'";
|
|
|
|
}
|
2002-12-22 13:55:30 +00:00
|
|
|
return get_records_select_menu("quiz_categories", "course='$courseid' $publish", "name ASC", "id,name");
|
2002-10-20 17:15:39 +00:00
|
|
|
}
|
|
|
|
|
2002-10-14 09:07:13 +00:00
|
|
|
function quiz_print_category_form($course, $current) {
|
|
|
|
// Prints a form to choose categories
|
|
|
|
|
2002-12-22 13:55:30 +00:00
|
|
|
if (!$categories = get_records_select("quiz_categories", "course='$course->id' OR publish = '1'", "name ASC")) {
|
2002-10-14 09:07:13 +00:00
|
|
|
if (!$category = quiz_get_default_category($course->id)) {
|
|
|
|
notify("Error creating a default category!");
|
|
|
|
return false;
|
|
|
|
}
|
2002-10-26 05:17:53 +00:00
|
|
|
$categories[$category->id] = $category;
|
2002-10-14 09:07:13 +00:00
|
|
|
}
|
2002-10-15 12:54:11 +00:00
|
|
|
foreach ($categories as $key => $category) {
|
|
|
|
if ($category->publish) {
|
2002-10-20 17:41:24 +00:00
|
|
|
if ($catcourse = get_record("course", "id", $category->course)) {
|
|
|
|
$category->name .= " ($catcourse->shortname)";
|
2002-10-15 12:54:11 +00:00
|
|
|
}
|
|
|
|
}
|
2002-10-20 17:41:24 +00:00
|
|
|
$catmenu[$category->id] = $category->name;
|
2002-10-15 12:54:11 +00:00
|
|
|
}
|
2002-10-14 09:07:13 +00:00
|
|
|
$strcategory = get_string("category", "quiz");
|
|
|
|
$strshow = get_string("show", "quiz");
|
2002-10-15 16:45:39 +00:00
|
|
|
$streditcats = get_string("editcategories", "quiz");
|
2002-10-14 09:07:13 +00:00
|
|
|
|
2002-12-07 08:39:24 +00:00
|
|
|
echo "<TABLE width=\"100%\"><TR><TD NOWRAP>";
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "<FORM METHOD=POST ACTION=edit.php>";
|
|
|
|
echo "<B>$strcategory:</B> ";
|
2002-10-20 17:41:24 +00:00
|
|
|
choose_from_menu($catmenu, "cat", "$current");
|
2002-10-15 17:32:50 +00:00
|
|
|
echo "<INPUT TYPE=submit VALUE=\"$strshow\">";
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "</FORM>";
|
2002-10-15 16:45:39 +00:00
|
|
|
echo "</TD><TD align=right>";
|
|
|
|
echo "<FORM METHOD=GET ACTION=category.php>";
|
|
|
|
echo "<INPUT TYPE=hidden NAME=id VALUE=\"$course->id\">";
|
|
|
|
echo "<INPUT TYPE=submit VALUE=\"$streditcats\">";
|
|
|
|
echo "</FORM>";
|
|
|
|
echo "</TD></TR></TABLE>";
|
2002-10-14 09:07:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-10-14 12:21:18 +00:00
|
|
|
function quiz_get_all_question_grades($questionlist, $quizid) {
|
|
|
|
// Given a list of question IDs, finds grades or invents them to
|
|
|
|
// create an array of matching grades
|
|
|
|
|
2002-12-22 13:55:30 +00:00
|
|
|
$questions = quiz_get_question_grades($quizid, $questionlist);
|
2002-10-14 12:21:18 +00:00
|
|
|
|
|
|
|
$list = explode(",", $questionlist);
|
|
|
|
$grades = array();
|
|
|
|
|
|
|
|
foreach ($list as $qid) {
|
|
|
|
if (isset($questions[$qid])) {
|
|
|
|
$grades[$qid] = $questions[$qid]->grade;
|
|
|
|
} else {
|
|
|
|
$grades[$qid] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $grades;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function quiz_print_question_list($questionlist, $grades) {
|
2002-10-14 09:07:13 +00:00
|
|
|
// Prints a list of quiz questions in a small layout form with knobs
|
2002-10-14 12:21:18 +00:00
|
|
|
// $questionlist is comma-separated list
|
|
|
|
// $grades is an array of corresponding grades
|
2002-10-14 09:07:13 +00:00
|
|
|
|
|
|
|
global $THEME;
|
|
|
|
|
|
|
|
if (!$questionlist) {
|
|
|
|
echo "<P align=center>";
|
|
|
|
print_string("noquestions", "quiz");
|
|
|
|
echo "</P>";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$order = explode(",", $questionlist);
|
|
|
|
|
2002-10-14 12:21:18 +00:00
|
|
|
if (!$questions = get_records_list("quiz_questions", "id", $questionlist)) {
|
2002-10-14 09:07:13 +00:00
|
|
|
error("No questions were found!");
|
|
|
|
}
|
|
|
|
|
|
|
|
$strorder = get_string("order");
|
|
|
|
$strquestionname = get_string("questionname", "quiz");
|
|
|
|
$strgrade = get_string("grade");
|
|
|
|
$strdelete = get_string("delete");
|
|
|
|
$stredit = get_string("edit");
|
|
|
|
$strmoveup = get_string("moveup");
|
|
|
|
$strmovedown = get_string("movedown");
|
|
|
|
$strsavegrades = get_string("savegrades", "quiz");
|
2002-10-20 17:15:39 +00:00
|
|
|
$strtype = get_string("type", "quiz");
|
2002-10-14 09:07:13 +00:00
|
|
|
|
2002-10-14 15:57:33 +00:00
|
|
|
for ($i=10; $i>=0; $i--) {
|
2002-10-14 12:21:18 +00:00
|
|
|
$gradesmenu[$i] = $i;
|
2002-10-14 09:07:13 +00:00
|
|
|
}
|
|
|
|
$count = 0;
|
|
|
|
$sumgrade = 0;
|
|
|
|
$total = count($order);
|
|
|
|
echo "<FORM METHOD=post ACTION=edit.php>";
|
|
|
|
echo "<TABLE BORDER=0 CELLPADDING=5 CELLSPACING=2 WIDTH=\"100%\">";
|
2002-12-07 08:39:24 +00:00
|
|
|
echo "<TR><TH WIDTH=\"*\" COLSPAN=3 NOWRAP>$strorder</TH><TH align=left WIDTH=\"100%\" NOWRAP>$strquestionname</TH><TH width=\"*\" NOWRAP>$strtype</TH><TH WIDTH=\"*\" NOWRAP>$strgrade</TH><TH WIDTH=\"*\" NOWRAP>$stredit</TH></TR>";
|
2002-10-14 09:07:13 +00:00
|
|
|
foreach ($order as $qnum) {
|
|
|
|
$count++;
|
|
|
|
echo "<TR BGCOLOR=\"$THEME->cellcontent\">";
|
|
|
|
echo "<TD>$count</TD>";
|
|
|
|
echo "<TD>";
|
|
|
|
if ($count != 1) {
|
|
|
|
echo "<A TITLE=\"$strmoveup\" HREF=\"edit.php?up=$qnum\"><IMG
|
|
|
|
SRC=\"../../pix/t/up.gif\" BORDER=0></A>";
|
|
|
|
}
|
|
|
|
echo "</TD>";
|
|
|
|
echo "<TD>";
|
|
|
|
if ($count != $total) {
|
|
|
|
echo "<A TITLE=\"$strmovedown\" HREF=\"edit.php?down=$qnum\"><IMG
|
|
|
|
SRC=\"../../pix/t/down.gif\" BORDER=0></A>";
|
|
|
|
}
|
|
|
|
echo "</TD>";
|
|
|
|
echo "<TD>".$questions[$qnum]->name."</TD>";
|
2002-12-07 08:39:24 +00:00
|
|
|
echo "<TD ALIGN=CENTER>";
|
2002-10-20 17:15:39 +00:00
|
|
|
quiz_print_question_icon($questions[$qnum]);
|
|
|
|
echo "</TD>";
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "<TD>";
|
2002-10-15 12:54:11 +00:00
|
|
|
choose_from_menu($gradesmenu, "q$qnum", (string)$grades[$qnum], "");
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "<TD>";
|
|
|
|
echo "<A TITLE=\"$strdelete\" HREF=\"edit.php?delete=$qnum\"><IMG
|
|
|
|
SRC=\"../../pix/t/delete.gif\" BORDER=0></A> ";
|
|
|
|
echo "<A TITLE=\"$stredit\" HREF=\"question.php?id=$qnum\"><IMG
|
|
|
|
SRC=\"../../pix/t/edit.gif\" BORDER=0></A>";
|
|
|
|
echo "</TD>";
|
|
|
|
|
2002-10-14 12:21:18 +00:00
|
|
|
$sumgrade += $grades[$qnum];
|
2002-10-14 09:07:13 +00:00
|
|
|
}
|
2002-10-20 17:15:39 +00:00
|
|
|
echo "<TR><TD COLSPAN=5 ALIGN=right>";
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "<INPUT TYPE=submit VALUE=\"$strsavegrades:\">";
|
2002-10-15 12:54:11 +00:00
|
|
|
echo "<INPUT TYPE=hidden NAME=setgrades VALUE=\"save\">";
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "<TD ALIGN=LEFT BGCOLOR=\"$THEME->cellcontent\">";
|
|
|
|
echo "<B>$sumgrade</B>";
|
|
|
|
echo "</TD><TD></TD></TR>";
|
|
|
|
echo "</TABLE>";
|
|
|
|
echo "</FORM>";
|
2002-10-14 15:57:33 +00:00
|
|
|
|
|
|
|
return $sumgrade;
|
2002-10-14 09:07:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function quiz_print_cat_question_list($categoryid) {
|
|
|
|
// Prints a form to choose categories
|
|
|
|
|
|
|
|
global $THEME, $QUIZ_QUESTION_TYPE;
|
|
|
|
|
2002-10-14 15:57:33 +00:00
|
|
|
$strcategory = get_string("category", "quiz");
|
2002-10-14 09:07:13 +00:00
|
|
|
$strquestion = get_string("question", "quiz");
|
|
|
|
$strnoquestions = get_string("noquestions", "quiz");
|
|
|
|
$strselect = get_string("select", "quiz");
|
|
|
|
$strcreatenewquestion = get_string("createnewquestion", "quiz");
|
|
|
|
$strquestionname = get_string("questionname", "quiz");
|
|
|
|
$strdelete = get_string("delete");
|
|
|
|
$stredit = get_string("edit");
|
|
|
|
$straddselectedtoquiz = get_string("addselectedtoquiz", "quiz");
|
2002-10-20 17:15:39 +00:00
|
|
|
$strtype = get_string("type", "quiz");
|
2002-10-14 09:07:13 +00:00
|
|
|
|
|
|
|
if (!$categoryid) {
|
|
|
|
echo "<P align=center>";
|
|
|
|
print_string("selectcategoryabove", "quiz");
|
|
|
|
echo "</P>";
|
|
|
|
return;
|
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
|
2002-10-14 09:07:13 +00:00
|
|
|
if (!$category = get_record("quiz_categories", "id", "$categoryid")) {
|
|
|
|
notify("Category not found!");
|
|
|
|
return;
|
|
|
|
}
|
2002-10-15 12:54:11 +00:00
|
|
|
echo "<CENTER>";
|
2002-10-14 15:57:33 +00:00
|
|
|
echo text_to_html($category->info);
|
2002-10-14 09:07:13 +00:00
|
|
|
|
2002-10-14 15:57:33 +00:00
|
|
|
echo "<FORM METHOD=GET ACTION=question.php>";
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "<B>$strquestion:</B> ";
|
|
|
|
choose_from_menu($QUIZ_QUESTION_TYPE, "type", "", "");
|
|
|
|
echo "<INPUT TYPE=hidden NAME=category VALUE=\"$category->id\">";
|
|
|
|
echo "<INPUT TYPE=submit NAME=new VALUE=\"$strcreatenewquestion\">";
|
2002-10-22 10:38:00 +00:00
|
|
|
helpbutton("questiontypes", $strcreatenewquestion, "quiz");
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "</FORM>";
|
2002-10-15 12:54:11 +00:00
|
|
|
echo "</CENTER>";
|
2002-10-14 09:07:13 +00:00
|
|
|
|
|
|
|
if (!$questions = get_records("quiz_questions", "category", $category->id)) {
|
|
|
|
echo "<P align=center>";
|
|
|
|
print_string("noquestions", "quiz");
|
|
|
|
echo "</P>";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2002-10-14 15:57:33 +00:00
|
|
|
$canedit = isteacher($category->course);
|
|
|
|
|
2002-10-14 09:07:13 +00:00
|
|
|
echo "<FORM METHOD=post ACTION=edit.php>";
|
|
|
|
echo "<TABLE BORDER=0 CELLPADDING=5 CELLSPACING=2 WIDTH=\"100%\">";
|
2002-12-07 08:39:24 +00:00
|
|
|
echo "<TR><TH width=\"*\" NOWRAP>$strselect</TH><TH width=\"100%\" align=left NOWRAP>$strquestionname</TH><TH WIDTH=\"*\" NOWRAP>$strtype</TH>";
|
2002-10-14 15:57:33 +00:00
|
|
|
if ($canedit) {
|
2002-12-07 08:39:24 +00:00
|
|
|
echo "<TH width=\"*\" NOWRAP>$stredit</TH>";
|
2002-10-14 15:57:33 +00:00
|
|
|
}
|
|
|
|
echo "</TR>";
|
2002-10-14 09:07:13 +00:00
|
|
|
foreach ($questions as $question) {
|
|
|
|
echo "<TR BGCOLOR=\"$THEME->cellcontent\">";
|
|
|
|
echo "<TD ALIGN=CENTER>";
|
|
|
|
echo "<INPUT TYPE=CHECKBOX NAME=q$question->id VALUE=\"1\">";
|
|
|
|
echo "</TD>";
|
|
|
|
echo "<TD>".$question->name."</TD>";
|
2002-12-07 08:39:24 +00:00
|
|
|
echo "<TD ALIGN=CENTER>";
|
2002-10-20 17:15:39 +00:00
|
|
|
quiz_print_question_icon($question);
|
|
|
|
echo "</TD>";
|
2002-10-14 15:57:33 +00:00
|
|
|
if ($canedit) {
|
|
|
|
echo "<TD>";
|
2002-10-21 15:00:51 +00:00
|
|
|
echo "<A TITLE=\"$strdelete\" HREF=\"question.php?id=$question->id&delete=$question->id\"><IMG
|
2002-10-14 15:57:33 +00:00
|
|
|
SRC=\"../../pix/t/delete.gif\" BORDER=0></A> ";
|
|
|
|
echo "<A TITLE=\"$stredit\" HREF=\"question.php?id=$question->id\"><IMG
|
|
|
|
SRC=\"../../pix/t/edit.gif\" BORDER=0></A>";
|
|
|
|
echo "</TD></TR>";
|
|
|
|
}
|
|
|
|
echo "</TR>";
|
2002-10-14 09:07:13 +00:00
|
|
|
}
|
|
|
|
echo "<TR><TD COLSPAN=3>";
|
|
|
|
echo "<INPUT TYPE=hidden NAME=add VALUE=\"1\">";
|
|
|
|
echo "<INPUT TYPE=submit VALUE=\"<< $straddselectedtoquiz\">";
|
|
|
|
echo "</TD></TR>";
|
|
|
|
echo "</TABLE>";
|
|
|
|
echo "</FORM>";
|
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
|
2002-10-06 03:23:16 +00:00
|
|
|
|
2002-10-18 16:05:21 +00:00
|
|
|
function quiz_start_attempt($quizid, $userid, $numattempt) {
|
|
|
|
$attempt->quiz = $quizid;
|
|
|
|
$attempt->user = $userid;
|
|
|
|
$attempt->attempt = $numattempt;
|
|
|
|
$attempt->timestart = time();
|
|
|
|
$attempt->timefinish = 0;
|
|
|
|
$attempt->timemodified = time();
|
|
|
|
|
|
|
|
return insert_record("quiz_attempts", $attempt);
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_get_user_attempt_unfinished($quizid, $userid) {
|
|
|
|
// Returns an object containing an unfinished attempt (if there is one)
|
2002-12-22 13:55:30 +00:00
|
|
|
return get_record("quiz_attempts", "quiz", $quizid, "user", $userid, "timefinish", 0);
|
2002-10-18 16:05:21 +00:00
|
|
|
}
|
|
|
|
|
2002-10-06 03:23:16 +00:00
|
|
|
function quiz_get_user_attempts($quizid, $userid) {
|
2002-10-13 07:17:48 +00:00
|
|
|
// Returns a list of all attempts by a user
|
2002-12-22 13:55:30 +00:00
|
|
|
return get_records_select("quiz_attempts", "quiz = '$quizid' AND user = '$userid' AND timefinish > 0",
|
|
|
|
"attempt ASC");
|
2002-10-06 03:23:16 +00:00
|
|
|
}
|
|
|
|
|
2002-10-15 12:54:11 +00:00
|
|
|
|
|
|
|
function quiz_get_user_attempts_string($quiz, $attempts, $bestgrade) {
|
|
|
|
/// Returns a simple little comma-separated list of all attempts,
|
2002-10-22 04:25:58 +00:00
|
|
|
/// with each grade linked to the feedback report and with the best grade highlighted
|
2002-10-15 12:54:11 +00:00
|
|
|
|
|
|
|
$bestgrade = format_float($bestgrade);
|
|
|
|
foreach ($attempts as $attempt) {
|
|
|
|
$attemptgrade = format_float(($attempt->sumgrades / $quiz->sumgrades) * $quiz->grade);
|
|
|
|
if ($attemptgrade == $bestgrade) {
|
2002-10-22 04:25:58 +00:00
|
|
|
$userattempts[] = "<SPAN CLASS=highlight><A HREF=\"report.php?q=$quiz->id&attempt=$attempt->id\">$attemptgrade</A></SPAN>";
|
2002-10-15 12:54:11 +00:00
|
|
|
} else {
|
2002-10-22 04:25:58 +00:00
|
|
|
$userattempts[] = "<A HREF=\"report.php?q=$quiz->id&attempt=$attempt->id\">$attemptgrade</A>";
|
2002-10-15 12:54:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return implode(",", $userattempts);
|
|
|
|
}
|
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
function quiz_get_best_grade($quizid, $userid) {
|
|
|
|
/// Get the best current grade for a particular user in a quiz
|
2002-12-22 13:55:30 +00:00
|
|
|
if (!$grade = get_record("quiz_grades", "quiz", $quizid, "user", $userid)) {
|
2002-10-06 03:23:16 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-10-21 13:13:33 +00:00
|
|
|
return (round($grade->grade,0));
|
2002-10-06 03:23:16 +00:00
|
|
|
}
|
|
|
|
|
2002-10-22 06:52:23 +00:00
|
|
|
function quiz_save_best_grade($quiz, $userid) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Calculates the best grade out of all attempts at a quiz for a user,
|
|
|
|
/// and then saves that grade in the quiz_grades table.
|
|
|
|
|
2002-10-22 06:52:23 +00:00
|
|
|
if (!$attempts = quiz_get_user_attempts($quiz->id, $userid)) {
|
2002-10-21 12:39:46 +00:00
|
|
|
notify("Could not find any user attempts");
|
2002-10-13 07:17:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$bestgrade = quiz_calculate_best_grade($quiz, $attempts);
|
|
|
|
$bestgrade = (($bestgrade / $quiz->sumgrades) * $quiz->grade);
|
|
|
|
|
2002-12-22 13:55:30 +00:00
|
|
|
if ($grade = get_record("quiz_grades", "quiz", $quiz->id, "user", $userid)) {
|
2002-10-13 07:17:48 +00:00
|
|
|
$grade->grade = $bestgrade;
|
|
|
|
$grade->timemodified = time();
|
|
|
|
if (!update_record("quiz_grades", $grade)) {
|
2002-10-21 12:39:46 +00:00
|
|
|
notify("Could not update best grade");
|
2002-10-13 07:17:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$grade->quiz = $quiz->id;
|
2002-10-22 06:52:23 +00:00
|
|
|
$grade->user = $userid;
|
2002-10-21 13:02:14 +00:00
|
|
|
$grade->grade = round($bestgrade, 2);
|
2002-10-13 07:17:48 +00:00
|
|
|
$grade->timemodified = time();
|
|
|
|
if (!insert_record("quiz_grades", $grade)) {
|
2002-10-21 12:39:46 +00:00
|
|
|
notify("Could not insert new best grade");
|
2002-10-13 07:17:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-10-06 03:23:16 +00:00
|
|
|
function quiz_calculate_best_grade($quiz, $attempts) {
|
2002-10-13 07:17:48 +00:00
|
|
|
/// Calculate the best grade for a quiz given a number of attempts by a particular user.
|
2002-10-06 03:23:16 +00:00
|
|
|
|
|
|
|
switch ($quiz->grademethod) {
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
case ATTEMPTFIRST:
|
2002-10-06 03:23:16 +00:00
|
|
|
foreach ($attempts as $attempt) {
|
2002-10-13 07:17:48 +00:00
|
|
|
return $attempt->sumgrades;
|
2002-10-06 03:23:16 +00:00
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ATTEMPTLAST:
|
|
|
|
foreach ($attempts as $attempt) {
|
|
|
|
$final = $attempt->sumgrades;
|
|
|
|
}
|
|
|
|
return $final;
|
2002-10-06 03:23:16 +00:00
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
case GRADEAVERAGE:
|
2002-10-06 03:23:16 +00:00
|
|
|
$sum = 0;
|
|
|
|
$count = 0;
|
|
|
|
foreach ($attempts as $attempt) {
|
2002-10-13 07:17:48 +00:00
|
|
|
$sum += $attempt->sumgrades;
|
2002-10-06 03:23:16 +00:00
|
|
|
$count++;
|
|
|
|
}
|
|
|
|
return (float)$sum/$count;
|
|
|
|
|
|
|
|
default:
|
2002-10-13 07:17:48 +00:00
|
|
|
case GRADEHIGHEST:
|
|
|
|
$max = 0;
|
2002-10-06 03:23:16 +00:00
|
|
|
foreach ($attempts as $attempt) {
|
2002-10-13 07:17:48 +00:00
|
|
|
if ($attempt->sumgrades > $max) {
|
|
|
|
$max = $attempt->sumgrades;
|
|
|
|
}
|
2002-10-06 03:23:16 +00:00
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
return $max;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function quiz_save_attempt($quiz, $questions, $result, $attemptnum) {
|
|
|
|
/// Given a quiz, a list of attempted questions and a total grade
|
|
|
|
/// this function saves EVERYTHING so it can be reconstructed later
|
|
|
|
/// if necessary.
|
|
|
|
|
|
|
|
global $USER;
|
|
|
|
|
2002-10-18 16:05:21 +00:00
|
|
|
// First find the attempt in the database (start of attempt)
|
|
|
|
|
|
|
|
if (!$attempt = quiz_get_user_attempt_unfinished($quiz->id, $USER->id)) {
|
|
|
|
notify("Trying to save an attempt that was not started!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($attempt->attempt != $attemptnum) { // Double check.
|
|
|
|
notify("Number of this attempt is different to the unfinished one!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now let's complete this record and save it
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
$attempt->sumgrades = $result->sumgrades;
|
2002-10-18 16:05:21 +00:00
|
|
|
$attempt->timefinish = time();
|
2002-10-13 07:17:48 +00:00
|
|
|
$attempt->timemodified = time();
|
|
|
|
|
2002-10-18 16:05:21 +00:00
|
|
|
if (! update_record("quiz_attempts", $attempt)) {
|
2002-10-15 17:13:11 +00:00
|
|
|
notify("Error while saving attempt");
|
2002-10-13 07:17:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now let's save all the questions for this attempt
|
|
|
|
|
|
|
|
foreach ($questions as $question) {
|
|
|
|
$response->attempt = $attempt->id;
|
|
|
|
$response->question = $question->id;
|
|
|
|
$response->grade = $result->grades[$question->id];
|
|
|
|
if ($question->answer) {
|
|
|
|
$response->answer = implode(",",$question->answer);
|
|
|
|
} else {
|
|
|
|
$response->answer = "";
|
|
|
|
}
|
|
|
|
if (!insert_record("quiz_responses", $response)) {
|
2002-10-15 17:13:11 +00:00
|
|
|
notify("Error while saving response");
|
2002-10-13 07:17:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
2002-10-06 03:23:16 +00:00
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
return true;
|
2002-10-06 03:23:16 +00:00
|
|
|
}
|
2002-10-04 02:59:05 +00:00
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
function quiz_grade_attempt_results($quiz, $questions) {
|
|
|
|
/// Given a list of questions (including answers for each one)
|
|
|
|
/// this function does all the hard work of calculating the
|
|
|
|
/// grades for each question, as well as a total grade for
|
|
|
|
/// for the whole quiz. It returns everything in a structure
|
|
|
|
/// that looks like:
|
|
|
|
/// $result->sumgrades (sum of all grades for all questions)
|
|
|
|
/// $result->percentage (Percentage of grades that were correct)
|
|
|
|
/// $result->grade (final grade result for the whole quiz)
|
|
|
|
/// $result->grades[] (array of grades, indexed by question id)
|
2002-10-16 09:35:04 +00:00
|
|
|
/// $result->response[] (array of response arrays, indexed by question id)
|
2002-10-13 07:17:48 +00:00
|
|
|
/// $result->feedback[] (array of feedback arrays, indexed by question id)
|
2002-10-20 05:35:25 +00:00
|
|
|
/// $result->correct[] (array of feedback arrays, indexed by question id)
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
if (!$questions) {
|
|
|
|
error("No questions!");
|
|
|
|
}
|
|
|
|
|
|
|
|
$result->sumgrades = 0;
|
|
|
|
|
|
|
|
foreach ($questions as $question) {
|
2002-10-15 10:04:28 +00:00
|
|
|
if (!$answers = quiz_get_answers($question)) {
|
|
|
|
error("No answers defined for question id $question->id!");
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$grade = 0; // default
|
2002-10-20 05:35:25 +00:00
|
|
|
$correct = array();
|
2002-10-16 15:52:29 +00:00
|
|
|
$feedback = array();
|
|
|
|
$response = array();
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
switch ($question->type) {
|
|
|
|
case SHORTANSWER:
|
|
|
|
if ($question->answer) {
|
2002-10-22 10:38:00 +00:00
|
|
|
$question->answer = trim($question->answer[0]);
|
2002-10-13 07:17:48 +00:00
|
|
|
} else {
|
2002-10-16 15:52:29 +00:00
|
|
|
$question->answer = "";
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
2002-10-16 09:35:04 +00:00
|
|
|
$response[0] = $question->answer;
|
2002-10-13 07:17:48 +00:00
|
|
|
foreach($answers as $answer) { // There might be multiple right answers
|
2002-10-20 05:35:25 +00:00
|
|
|
if ($answer->fraction > $bestshortanswer) {
|
|
|
|
$correct[$answer->id] = $answer->answer;
|
|
|
|
}
|
2002-10-15 10:04:28 +00:00
|
|
|
if (!$answer->usecase) { // Don't compare case
|
2002-10-13 07:17:48 +00:00
|
|
|
$answer->answer = strtolower($answer->answer);
|
|
|
|
$question->answer = strtolower($question->answer);
|
|
|
|
}
|
|
|
|
if ($question->answer == $answer->answer) {
|
2002-10-16 09:35:04 +00:00
|
|
|
$feedback[0] = $answer->feedback;
|
2002-10-13 07:17:48 +00:00
|
|
|
$grade = (float)$answer->fraction * $answer->grade;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case TRUEFALSE:
|
|
|
|
if ($question->answer) {
|
|
|
|
$question->answer = $question->answer[0];
|
|
|
|
} else {
|
|
|
|
$question->answer = NULL;
|
|
|
|
}
|
|
|
|
foreach($answers as $answer) { // There should be two answers (true and false)
|
|
|
|
$feedback[$answer->id] = $answer->feedback;
|
2002-10-20 05:35:25 +00:00
|
|
|
if ($answer->fraction > 0) {
|
|
|
|
$correct[$answer->id] = true;
|
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
if ($question->answer == $answer->id) {
|
|
|
|
$grade = (float)$answer->fraction * $answer->grade;
|
2002-10-16 09:35:04 +00:00
|
|
|
$response[$answer->id] = true;
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case MULTICHOICE:
|
|
|
|
foreach($answers as $answer) { // There will be multiple answers, perhaps more than one is right
|
|
|
|
$feedback[$answer->id] = $answer->feedback;
|
2002-10-20 05:35:25 +00:00
|
|
|
if ($answer->fraction > 0) {
|
|
|
|
$correct[$answer->id] = true;
|
|
|
|
}
|
2002-10-13 07:17:48 +00:00
|
|
|
if ($question->answer) {
|
|
|
|
foreach ($question->answer as $questionanswer) {
|
|
|
|
if ($questionanswer == $answer->id) {
|
|
|
|
if ($answer->single) {
|
|
|
|
$grade = (float)$answer->fraction * $answer->grade;
|
2002-10-16 09:35:04 +00:00
|
|
|
$response[$answer->id] = true;
|
2002-10-13 07:17:48 +00:00
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
$grade += (float)$answer->fraction * $answer->grade;
|
2002-10-16 09:35:04 +00:00
|
|
|
$response[$answer->id] = true;
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2002-10-20 05:35:25 +00:00
|
|
|
case RANDOM:
|
|
|
|
// Not done yet
|
|
|
|
break;
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
if ($grade < 0.0) { // No negative grades
|
|
|
|
$grade = 0.0;
|
|
|
|
}
|
2002-10-14 15:57:33 +00:00
|
|
|
|
2002-10-26 06:11:43 +00:00
|
|
|
$result->grades[$question->id] = round($grade, 2);
|
2002-10-13 07:17:48 +00:00
|
|
|
$result->sumgrades += $grade;
|
|
|
|
$result->feedback[$question->id] = $feedback;
|
2002-10-16 09:35:04 +00:00
|
|
|
$result->response[$question->id] = $response;
|
2002-10-20 05:35:25 +00:00
|
|
|
$result->correct[$question->id] = $correct;
|
2002-10-13 07:17:48 +00:00
|
|
|
}
|
|
|
|
|
2002-10-15 12:54:11 +00:00
|
|
|
$fraction = (float)($result->sumgrades / $quiz->sumgrades);
|
|
|
|
$result->percentage = format_float($fraction * 100.0);
|
|
|
|
$result->grade = format_float($fraction * $quiz->grade);
|
2002-10-26 06:11:43 +00:00
|
|
|
$result->sumgrades = round($result->sumgrades, 2);
|
2002-10-13 07:17:48 +00:00
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
2002-10-22 04:25:58 +00:00
|
|
|
|
|
|
|
|
2002-10-13 07:17:48 +00:00
|
|
|
|
2002-10-04 02:59:05 +00:00
|
|
|
?>
|