You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
551 lines
18 KiB
551 lines
18 KiB
<?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/>.
|
|
|
|
/**
|
|
* This page shows the answers of the current game
|
|
*
|
|
* @package mod_game
|
|
* @copyright 2007 Vasilis Daloukas
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
|
|
require_once("../../config.php");
|
|
require_once( "headergame.php");
|
|
|
|
require_login();
|
|
|
|
if (!has_capability('mod/game:viewreports', $context)) {
|
|
print_error( get_string( 'only_teachers', 'game'));
|
|
}
|
|
|
|
$PAGE->navbar->add(get_string('showanswers', 'game'));
|
|
|
|
$action = optional_param('action', "", PARAM_ALPHANUM); // Action.
|
|
if ($action == 'delstats') {
|
|
$DB->delete_records('game_repetitions', array('gameid' => $game->id, 'userid' => $USER->id));
|
|
}
|
|
if ($action == 'computestats') {
|
|
game_compute_repetitions($game);
|
|
}
|
|
|
|
echo '<b>'.get_string('repetitions', 'game').': </b>';
|
|
echo get_string('user').': ';
|
|
game_showusers($game);
|
|
echo " <a href=\"{$CFG->wwwroot}/mod/game/showanswers.php?q=$q&action=delstats\">".
|
|
get_string('clearrepetitions', 'game').'</a>';
|
|
echo " <a href=\"{$CFG->wwwroot}/mod/game/showanswers.php?q=$q&action=computestats\">".
|
|
get_string('computerepetitions', 'game').'</a>';
|
|
echo '<br><br>';
|
|
|
|
$existsbook = ($DB->get_record( 'modules', array( 'name' => 'book'), 'id,id'));
|
|
game_showanswers( $game, $existsbook, $context);
|
|
if ($game->gamekind == 'millionaire') {
|
|
game_showanswers_extra_millionaire( $game);
|
|
}
|
|
|
|
echo $OUTPUT->footer();
|
|
|
|
/**
|
|
* Compute repetitions
|
|
*
|
|
* @param stdClass $game
|
|
*/
|
|
function game_compute_repetitions($game) {
|
|
global $DB, $USER;
|
|
|
|
$DB->delete_records('game_repetitions', array('gameid' => $game->id, 'userid' => $USER->id));
|
|
|
|
$sql = "INSERT INTO {game_repetitions}( gameid,userid,questionid,glossaryentryid,repetitions) ".
|
|
"SELECT $game->id,$USER->id,questionid,glossaryentryid,COUNT(*) ".
|
|
"FROM {game_queries} WHERE gameid=$game->id AND userid=$USER->id GROUP BY questionid,glossaryentryid";
|
|
|
|
if (!$DB->execute( $sql)) {
|
|
print_error('Problem on computing statistics for repetitions');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Show users
|
|
*
|
|
* @param stdClass $game
|
|
*/
|
|
function game_showusers($game) {
|
|
global $CFG, $USER;
|
|
|
|
$users = array();
|
|
|
|
$context = game_get_context_course_instance( $game->course);
|
|
|
|
if ($courseusers = get_enrolled_users($context)) {
|
|
foreach ($courseusers as $courseuser) {
|
|
$users[$courseuser->id] = fullname($courseuser, has_capability('moodle/site:viewfullnames', $context));
|
|
}
|
|
}
|
|
if ($guest = guest_user()) {
|
|
$users[$guest->id] = fullname($guest);
|
|
}
|
|
$href = $CFG->wwwroot.'/mod/game/showanswers.php?q='.$game->id.'&userid=';
|
|
?>
|
|
<script type="text/javascript">
|
|
function onselectuser()
|
|
{
|
|
window.location.href =
|
|
"<?php echo $href;?>" + document.getElementById('menuuser').value;
|
|
}
|
|
</script>
|
|
<?php
|
|
|
|
$attributes = 'onchange="javascript:onselectuser();"';
|
|
$name = 'user';
|
|
$id = 'menu'.$name;
|
|
$class = 'menu'.$name;
|
|
$class = 'select ' . $class; // Add 'select' selector always.
|
|
$nothing = get_string("allparticipants");
|
|
$nothingvalue = '0';
|
|
$options = $users;
|
|
$selected = optional_param('userid', $USER->id, PARAM_INT);
|
|
|
|
$output = '<select id="'. $id .'" class="'. $class .'" name="'. $name .'" '. $attributes .'>' . "\n";
|
|
$output .= ' <option value="'. s($nothingvalue) .'"'. "\n";
|
|
if ($nothingvalue === $selected) {
|
|
$output .= ' selected="selected"';
|
|
}
|
|
$output .= '>'. $nothing .'</option>' . "\n";
|
|
|
|
if (!empty($options)) {
|
|
foreach ($options as $value => $label) {
|
|
$output .= ' <option value="'. s($value) .'"';
|
|
if ((string)$value == (string)$selected ||
|
|
(is_array($selected) && in_array($value, $selected))) {
|
|
$output .= ' selected="selected"';
|
|
}
|
|
if ($label === '') {
|
|
$output .= '>'. $value .'</option>' . "\n";
|
|
} else {
|
|
$output .= '>'. $label .'</option>' . "\n";
|
|
}
|
|
}
|
|
}
|
|
echo $output . '</select>' . "\n";
|
|
}
|
|
|
|
/**
|
|
* Show answers
|
|
*
|
|
* @param stdClass $game
|
|
* @param boolean $existsbook
|
|
* @param stdClass $context
|
|
*/
|
|
function game_showanswers( $game, $existsbook, $context) {
|
|
if ($game->gamekind == 'bookquiz' and $existsbook) {
|
|
game_showanswers_bookquiz( $game, $context);
|
|
return;
|
|
}
|
|
|
|
switch ($game->sourcemodule) {
|
|
case 'question':
|
|
game_showanswers_question( $game, $context);
|
|
break;
|
|
case 'glossary':
|
|
game_showanswers_glossary( $game);
|
|
break;
|
|
case 'quiz':
|
|
game_showanswers_quiz( $game, $context);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* append select to SQL
|
|
*
|
|
* @param stdClass $game
|
|
*/
|
|
function game_showanswers_appendselect( $game) {
|
|
switch ($game->gamekind) {
|
|
case 'hangman':
|
|
case 'cross':
|
|
case 'crypto':
|
|
return " AND qtype='shortanswer'";
|
|
case 'millionaire':
|
|
return " AND qtype = 'multichoice'";
|
|
case 'sudoku':
|
|
case 'bookquiz':
|
|
case 'snakes':
|
|
return " AND qtype in ('shortanswer', 'truefalse', 'multichoice')";
|
|
}
|
|
|
|
return '';
|
|
}
|
|
|
|
/**
|
|
* Show answers question
|
|
*
|
|
* @param stdClass $game
|
|
* @param stdClass $context
|
|
*/
|
|
function game_showanswers_question( $game, $context) {
|
|
global $DB;
|
|
|
|
if ($game->gamekind != 'bookquiz') {
|
|
$select = ' category='.$game->questioncategoryid;
|
|
|
|
if ($game->subcategories) {
|
|
$cats = question_categorylist( $game->questioncategoryid);
|
|
if (strpos( $cats, ',') > 0) {
|
|
$select = ' category in ('.$cats.')';
|
|
}
|
|
}
|
|
} else {
|
|
$context2 = get_context_instance(50, $COURSE->id);
|
|
$select = " contextid in ($context2->id)";
|
|
$select2 = '';
|
|
if ($recs = $DB->get_records_select( 'question_categories', $select, null, 'id,id')) {
|
|
foreach ($recs as $rec) {
|
|
$select2 .= ','.$rec->id;
|
|
}
|
|
}
|
|
$select = ' AND category IN ('.substr( $select2, 1).')';
|
|
}
|
|
|
|
$select .= ' AND hidden = 0 ';
|
|
$select .= game_showanswers_appendselect( $game);
|
|
|
|
$gamekind = $game->gamekind;
|
|
$showcategories = ($gamekind == 'bookquiz');
|
|
$order = ($showcategories ? 'category,questiontext' : 'questiontext');
|
|
$table = '{question} q';
|
|
if ($gamekind == 'millionaire') {
|
|
if (game_get_moodle_version() < '02.06') {
|
|
$table .= ',{question_multichoice} qmo';
|
|
$select .= " AND q.qtype='multichoice' AND qmo.single=1 AND qmo.question=q.id";
|
|
} else {
|
|
$table .= ',{qtype_multichoice_options} qmo';
|
|
$select .= " AND q.qtype='multichoice' AND qmo.single=1 AND qmo.questionid=q.id";
|
|
}
|
|
} else if ( ($gamekind == 'hangman') or ($gamekind == 'cryptex') or ($gamekind == 'cross')) {
|
|
$select .= " AND q.qtype = 'shortanswer'";
|
|
}
|
|
game_showanswers_question_select( $game, $table, $select, '*', $order, $showcategories, $game->course, $context);
|
|
}
|
|
|
|
/**
|
|
* Show answers quiz
|
|
*
|
|
* @param stdClass $game
|
|
* @param stdClass $context
|
|
*/
|
|
function game_showanswers_quiz( $game, $context) {
|
|
global $CFG;
|
|
|
|
if (game_get_moodle_version() < '02.07') {
|
|
$select = "quiz='$game->quizid' ".
|
|
' AND qqi.question=q.id'.
|
|
' AND q.hidden=0'.
|
|
game_showanswers_appendselect( $game);
|
|
$table = '{question} q,{quiz_question_instances} qqi';
|
|
} else {
|
|
$select = "qs.quizid='$game->quizid' ".
|
|
" AND qs.questionid=q.id ".game_showanswers_appendselect( $game);
|
|
$table = "{question} q,{quiz_slots} qs";
|
|
}
|
|
|
|
game_showanswers_question_select( $game, $table, $select, 'q.*', 'category,questiontext', false, $game->course, $context);
|
|
}
|
|
|
|
/**
|
|
* Create the select for SQL
|
|
*
|
|
* @param stdClass $game
|
|
* @param string $table
|
|
* @param string $select
|
|
* @param string $fields
|
|
* @param string $order
|
|
* @param boolean $showcategoryname
|
|
* @param int $courseid
|
|
* @param stdClass $context
|
|
*/
|
|
function game_showanswers_question_select( $game, $table, $select, $fields, $order, $showcategoryname, $courseid, $context) {
|
|
global $CFG, $DB, $OUTPUT;
|
|
|
|
$sql = "SELECT $fields FROM $table WHERE $select ORDER BY $order";
|
|
if (($questions = $DB->get_records_sql( $sql)) === false) {
|
|
return;
|
|
}
|
|
|
|
$table .= ",{game_repetitions} gr";
|
|
$select .= " AND gr.questionid=q.id AND gr.glossaryentryid=0 AND gr.gameid=".$game->id;
|
|
$userid = optional_param('userid', 0, PARAM_INT);
|
|
if ($userid) {
|
|
$select .= " AND gr.userid=$userid";
|
|
}
|
|
$sql = "SELECT q.id as id,SUM(repetitions) as c FROM {$table} WHERE $select GROUP BY q.id";
|
|
$reps = $DB->get_records_sql( $sql);
|
|
|
|
$categorynames = array();
|
|
if ($showcategoryname) {
|
|
$select = '';
|
|
$recs = $DB->get_records( 'question_categories', null, '', '*', 0, 1);
|
|
foreach ($recs as $rec) {
|
|
if (array_key_exists( 'course', $rec)) {
|
|
$select = "course=$courseid";
|
|
} else {
|
|
$context = get_context_instance(50, $courseid);
|
|
$select = " contextid in ($context->id)";
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (($categories = $DB->get_records_select( 'question_categories', $select, null, '', 'id,name'))) {
|
|
foreach ($categories as $rec) {
|
|
$categorynames[ $rec->id] = $rec->name;
|
|
}
|
|
}
|
|
}
|
|
|
|
echo '<table border="1">';
|
|
echo '<tr><td></td>';
|
|
if ($showcategoryname) {
|
|
echo '<td><b>'.get_string( 'categories', 'quiz').'</b></td>';
|
|
}
|
|
echo '<td><b>'.get_string( 'questions', 'quiz').'</b></td>';
|
|
echo '<td><b>'.get_string( 'answers', 'quiz').'</b></td>';
|
|
echo '<td><b>'.get_string( 'feedbacks', 'game').'</b></td>';
|
|
if ($reps) {
|
|
echo '<td><b>'.get_string( 'repetitions', 'game').'</b></td>';
|
|
}
|
|
echo "</tr>\r\n";
|
|
$line = 0;
|
|
foreach ($questions as $question) {
|
|
echo '<tr>';
|
|
echo '<td>'.(++$line);
|
|
echo '</td>';
|
|
|
|
if ($showcategoryname) {
|
|
echo '<td>';
|
|
if (array_key_exists( $question->category, $categorynames)) {
|
|
echo $categorynames[ $question->category];
|
|
} else {
|
|
echo ' ';
|
|
}
|
|
echo '</td>';
|
|
}
|
|
|
|
echo '<td>';
|
|
echo "<a title=\"Edit\" ".
|
|
"href=\"{$CFG->wwwroot}/question/question.php?inpopup=1&id=$question->id&courseid=$courseid\" target=\"_blank\">";
|
|
echo "<img src=\"".game_pix_url('t/edit')."\" alt=\"Edit\" /></a> ";
|
|
|
|
echo game_filterquestion(str_replace( array( "\'", '\"'), array( "'", '"'),
|
|
$question->questiontext), $question->id, $context->id, $game->course);
|
|
|
|
switch ($question->qtype) {
|
|
case 'shortanswer':
|
|
$recs = $DB->get_records( 'question_answers',
|
|
array( 'question' => $question->id), 'fraction DESC', 'id,answer,feedback');
|
|
if ($recs == false) {
|
|
$rec = false;
|
|
} else {
|
|
foreach ($recs as $rec) {
|
|
break;
|
|
}
|
|
}
|
|
echo "<td>$rec->answer</td>";
|
|
if ($rec->feedback == '') {
|
|
$rec->feedback = ' ';
|
|
}
|
|
echo "<td>$rec->feedback</td>";
|
|
break;
|
|
case 'multichoice':
|
|
case 'truefalse':
|
|
$recs = $DB->get_records( 'question_answers', array( 'question' => $question->id));
|
|
$feedback = '';
|
|
echo '<td>';
|
|
$i = 0;
|
|
foreach ($recs as $rec) {
|
|
if ($i++ > 0) {
|
|
echo '<br>';
|
|
}
|
|
if ($rec->fraction == 1) {
|
|
echo " <b>$rec->answer</b>";
|
|
} else {
|
|
echo " $rec->answer";
|
|
}
|
|
|
|
if ($rec->feedback == '') {
|
|
$feedback .= '<br>';
|
|
} else {
|
|
$feedback .= "<b>{$rec->feedback}</b><br>";
|
|
}
|
|
}
|
|
echo '</td>';
|
|
if ($feedback == '') {
|
|
$feedback = ' ';
|
|
}
|
|
echo "<td>$feedback</td>";
|
|
break;
|
|
default:
|
|
echo "<td>$question->qtype</td>";
|
|
break;
|
|
}
|
|
|
|
// Show repetitions.
|
|
if ($reps) {
|
|
if (array_key_exists( $question->id, $reps)) {
|
|
$rep = $reps[ $question->id];
|
|
echo '<td><center>'.$rep->c.'</td>';
|
|
} else {
|
|
echo '<td> </td>';
|
|
}
|
|
}
|
|
|
|
echo "</tr>\r\n";
|
|
}
|
|
echo "</table><br>\r\n\r\n";
|
|
}
|
|
|
|
/**
|
|
* Show answers glossary
|
|
*
|
|
* @param stdClass $game
|
|
*/
|
|
function game_showanswers_glossary( $game) {
|
|
global $CFG, $DB;
|
|
|
|
$table = '{glossary_entries} ge';
|
|
$select = "glossaryid={$game->glossaryid}";
|
|
if ($game->glossarycategoryid) {
|
|
$select .= " AND gec.entryid = ge.id ".
|
|
" AND gec.categoryid = {$game->glossarycategoryid}";
|
|
$table .= ",{glossary_entries_categories} gec";
|
|
}
|
|
if ($game->glossaryonlyapproved) {
|
|
// Only approved glossary entries will be used.
|
|
$select .= ' AND (ge.approved=1 OR teacherentry=1)';
|
|
}
|
|
$sql = "SELECT ge.id,definition,concept FROM $table WHERE $select ORDER BY definition";
|
|
if (($questions = $DB->get_records_sql( $sql)) === false) {
|
|
return;
|
|
}
|
|
|
|
// Show repetitions of questions.
|
|
$table = "{glossary_entries} ge, {game_repetitions} gr";
|
|
$select = "glossaryid={$game->glossaryid} AND gr.glossaryentryid=ge.id AND gr.gameid=".$game->id;
|
|
$userid = optional_param('userid', 0, PARAM_INT);
|
|
if ($userid) {
|
|
$select .= " AND gr.userid=$userid";
|
|
}
|
|
if ($game->glossarycategoryid) {
|
|
$select .= " AND gec.entryid = ge.id ".
|
|
" AND gec.categoryid = {$game->glossarycategoryid}";
|
|
$table .= ",{glossary_entries_categories} gec";
|
|
}
|
|
$sql = "SELECT ge.id,SUM(repetitions) as c FROM {$table} WHERE $select GROUP BY ge.id";
|
|
$reps = $DB->get_records_sql( $sql);
|
|
|
|
echo '<table border="1">';
|
|
echo '<tr><td></td>';
|
|
echo '<td><b>'.get_string( 'questions', 'quiz').'</b></td>';
|
|
echo '<td><b>'.get_string( 'answers', 'quiz').'</b></td>';
|
|
if ($reps != false) {
|
|
echo '<td><b>'.get_string( 'repetitions', 'game').'</b></td>';
|
|
}
|
|
echo "</tr>\r\n";
|
|
$line = 0;
|
|
foreach ($questions as $question) {
|
|
if ($game->param7 == 0) { // Not allowed spaces.
|
|
if (!( strpos( $question->concept, ' ') === false)) {
|
|
continue;
|
|
}
|
|
}
|
|
if ($game->param8 == 0) { // Not allowed -.
|
|
if (!( strpos( $question->concept, '-') === false)) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
echo '<tr>';
|
|
echo '<td>'.(++$line);
|
|
echo '</td>';
|
|
|
|
$query = new StdClass;
|
|
$query->glossaryid = $game->glossaryid;
|
|
$query->glossaryentryid = $question->id;
|
|
echo '<td>'.game_show_query( $game, $query, $question->definition).'</td>';
|
|
|
|
echo '<td>'.$question->concept.'</td>';
|
|
if ($reps != false) {
|
|
if (array_key_exists( $question->id, $reps)) {
|
|
$rep = $reps[ $question->id];
|
|
echo '<td><center>'.$rep->c.'</td>';
|
|
} else {
|
|
echo '<td> </td>';
|
|
}
|
|
}
|
|
echo "</tr>\r\n";
|
|
}
|
|
echo "</table><br>\r\n\r\n";
|
|
}
|
|
|
|
/**
|
|
* Show answers bookquiz
|
|
*
|
|
* @param stdClass $game
|
|
* @param stdClass $context
|
|
*/
|
|
function game_showanswers_bookquiz( $game, $context) {
|
|
global $CFG;
|
|
|
|
$select = "gbq.questioncategoryid=q.category ".
|
|
" AND gbq.gameid = $game->id ".
|
|
" AND bc.id = gbq.chapterid";
|
|
$table = "{question} q,{game_bookquiz_questions} gbq,{book_chapters} bc";
|
|
|
|
$showcategories = ($game->gamekind == 'bookquiz');
|
|
game_showanswers_question_select( $game, $table, $select, "DISTINCT q.*",
|
|
"bc.pagenum,questiontext", $showcategories, $game->course, $context);
|
|
}
|
|
|
|
function game_showanswers_extra_millionaire( $game)
|
|
{
|
|
global $CFG, $DB;
|
|
|
|
if ($game->questioncategoryid == 0) {
|
|
print_error( get_string( 'must_select_questioncategory', 'game'));
|
|
}
|
|
|
|
// Include subcategories.
|
|
$select = 'category='.$game->questioncategoryid;
|
|
if ($game->subcategories) {
|
|
$cats = question_categorylist( $game->questioncategoryid);
|
|
if (count( $cats)) {
|
|
$select = 'q.category in ('.implode(',', $cats).')';
|
|
}
|
|
}
|
|
|
|
if (game_get_moodle_version() < '02.06') {
|
|
$table = "{$CFG->prefix}question q, {$CFG->prefix}question_multichoice qmo";
|
|
$select .= " AND qtype='multichoice' AND qmo.single <> 1 AND qmo.question=q.id";
|
|
} else {
|
|
$table = "{$CFG->prefix}question q, {$CFG->prefix}qtype_multichoice_options qmo";
|
|
$select .= " AND qtype='multichoice' AND qmo.single <> 1 AND qmo.questionid=q.id";
|
|
}
|
|
|
|
$sql = "SELECT COUNT(*) as c FROM $table WHERE $select";
|
|
$rec = $DB->get_record_sql( $sql);
|
|
if ($rec->c != 0) {
|
|
echo get_string( 'millionaire_also_multichoice', 'game').': '.$rec->c;
|
|
}
|
|
}
|
|
|