moodle/mod/quiz/report/statistics/report.php

422 lines
21 KiB
PHP

<?php
/**
* This script calculates various statistics about student attempts
*
* @version $Id$
* @author Martin Dougiamas, Jamie Pratt, Tim Hunt and others.
* @license http://www.gnu.org/copyleft/gpl.html GNU Public License
* @package quiz
*//** */
require_once($CFG->dirroot.'/mod/quiz/report/statistics/statistics_form.php');
require_once($CFG->dirroot.'/mod/quiz/report/statistics/statistics_table.php');
class quiz_statistics_report extends quiz_default_report {
/**
* Display the report.
*/
function display($quiz, $cm, $course) {
global $CFG, $DB;
$context = get_context_instance(CONTEXT_MODULE, $cm->id);
$download = optional_param('download', '', PARAM_ALPHA);
$pageoptions = array();
$pageoptions['id'] = $cm->id;
$pageoptions['q'] = $quiz->id;
$pageoptions['mode'] = 'statistics';
$reporturl = new moodle_url($CFG->wwwroot.'/mod/quiz/report.php', $pageoptions);
$mform = new mod_quiz_report_statistics($reporturl);
if ($fromform = $mform->get_data()){
$useallattempts = $fromform->useallattempts;
if ($fromform->useallattempts){
set_user_preference('quiz_report_statistics_useallattempts', $fromform->useallattempts);
} else {
unset_user_preference('quiz_report_statistics_useallattempts');
}
} else {
$useallattempts = get_user_preferences('quiz_report_statistics_useallattempts', 0);
}
/// find out current groups mode
$currentgroup = groups_get_activity_group($cm, true);
$nostudentsingroup = false;//true if a group is selected and their is noeone in it.
if (!empty($currentgroup)) {
// all users who can attempt quizzes and who are in the currently selected group
$groupstudents = get_users_by_capability($context, 'mod/quiz:attempt','','','','',$currentgroup,'',false);
if (!$groupstudents){
$nostudentsingroup = true;
}
}
$questions = quiz_report_load_questions($quiz);
// Load the question type specific information
if (!get_question_options($questions)) {
print_error('cannotloadquestion', 'question');
}
$table = new quiz_report_statistics_table();
$table->is_downloading($download, get_string('reportstatistics','quiz_statistics'),
"$course->shortname ".format_string($quiz->name,true));
if (!$table->is_downloading()) {
// Only print headers if not asked to download data
$this->print_header_and_tabs($cm, $course, $quiz, "statistics");
}
if ($groupmode = groups_get_activity_groupmode($cm)) { // Groups are being used
if (!$table->is_downloading()) {
groups_print_activity_menu($cm, $reporturl->out());
if ($currentgroup && !$groupstudents){
notify(get_string('nostudentsingroup', 'quiz_statistics'));
}
}
}
// Print information on the number of existing attempts
if (!$table->is_downloading()) { //do not print notices when downloading
print_heading(get_string('quizinformation', 'quiz_statistics'));
$quizinformationtable = new object();
$quizinformationtable->align = array('center', 'center');
$quizinformationtable->width = '60%';
$quizinformationtable->class = 'generaltable titlesleft';
$quizinformationtable->data = array();
$quizinformationtable->data[] = array(get_string('quizname', 'quiz_statistics'), $quiz->name);
$quizinformationtable->data[] = array(get_string('coursename', 'quiz_statistics'), $course->fullname);
if ($cm->idnumber){
$quizinformationtable->data[] = array(get_string('coursename', 'quiz_statistics'), $cm->idnumber);
}
if ($quiz->timeopen){
$quizinformationtable->data[] = array(get_string('quizopen', 'quiz'), userdate($quiz->timeopen));
}
if ($quiz->timeclose){
$quizinformationtable->data[] = array(get_string('quizclose', 'quiz'), userdate($quiz->timeclose));
}
if ($quiz->timeopen && $quiz->timeclose){
$quizinformationtable->data[] = array(get_string('duration', 'quiz_statistics'), format_time($quiz->timeclose - $quiz->timeopen));
}
print_table($quizinformationtable);
}
if (!$table->is_downloading()) {
// Print display options
$mform->set_data(array('useallattempts' => $useallattempts));
$mform->display();
}
$fromqa = '{quiz_attempts} qa ';
$whereqa = 'quiz = :quizid AND preview=0 AND timefinish !=0 ';
$qaparams = array('quizid'=>$quiz->id);
if (!empty($currentgroup) && $groupstudents) {
list($grpsql, $grpparams) = $DB->get_in_or_equal(array_keys($groupstudents), SQL_PARAMS_NAMED, 'u0000');
$whereqa .= ' AND qa.userid '.$grpsql.' ';
$qaparams += $grpparams;
}
$sql = 'SELECT (attempt=1) AS isfirst, COUNT(1) AS countrecs, SUM(sumgrades) AS total ' .
'FROM '.$fromqa.
'WHERE ' .$whereqa.
'GROUP BY (attempt=1)';
if (!$nostudentsingroup){
//Calculating_MEAN_of_grades_for_all_attempts_by_students
//http://docs.moodle.org/en/Development:Quiz_item_analysis_calculations_in_practise#Calculating_MEAN_of_grades_for_all_attempts_by_students
if (!$attempttotals = $DB->get_records_sql($sql, $qaparams)){
print_heading(get_string('noattempts','quiz'));
$s = 0;
} else {
$firstattempt = $attempttotals[1];
$allattempts = new object();
$allattempts->countrecs = $firstattempt->countrecs +
(isset($attempttotals[0])?$attempttotals[0]->countrecs:0);
$allattempts->total = $firstattempt->total +
(isset($attempttotals[0])?$attempttotals[0]->total:0);
if ($useallattempts){
$usingattempts = $allattempts;
$usingattempts->attempts = get_string('allattempts', 'quiz_statistics');
$usingattempts->sql = '';
} else {
$usingattempts = $firstattempt;
$usingattempts->attempts = get_string('firstattempts', 'quiz_statistics');
$usingattempts->sql = 'AND qa.attempt=1 ';
}
$usingattempts->heading = get_string('statsfor', 'quiz_statistics', $usingattempts->attempts);
$s = $usingattempts->countrecs;
$sumgradesavg = $usingattempts->total / $usingattempts->countrecs;
}
} else {
$s = 0;
}
if ($s && !$table->is_downloading()) {
print_heading(get_string('quizoverallstatistics', 'quiz_statistics'));
$quizoverallstatistics = new object();
$quizoverallstatistics->align = array('center', 'center');
$quizoverallstatistics->width = '60%';
$quizoverallstatistics->class = 'generaltable titlesleft';
$quizoverallstatistics->data = array();
$quizoverallstatistics->data[] = array(get_string('nooffirstattempts', 'quiz_statistics'), $firstattempt->countrecs);
$quizoverallstatistics->data[] = array(get_string('noofallattempts', 'quiz_statistics'), $allattempts->countrecs);
$quizoverallstatistics->data[] = array(get_string('firstattemptsavg', 'quiz_statistics'), quiz_report_scale_sumgrades_as_percentage($firstattempt->total / $firstattempt->countrecs, $quiz));
$quizoverallstatistics->data[] = array(get_string('allattemptsavg', 'quiz_statistics'), quiz_report_scale_sumgrades_as_percentage($allattempts->total / $allattempts->countrecs, $quiz));
print_table($quizoverallstatistics);
}
$quizattsstatistics = new object();
$quizattsstatistics->align = array('center', 'center');
$quizattsstatistics->width = '60%';
$quizattsstatistics->class = 'generaltable titlesleft';
$quizattsstatistics->data = array();
//get the median
if ($s && !$table->is_downloading()) {
print_heading($usingattempts->heading);
if (($s%2)==0){
//even number of attempts
$limitoffset = ($s/2) - 1;
$limit = 2;
} else {
$limitoffset = (floor($s/2)) - 1;
$limit = 1;
}
$sql = 'SELECT id, sumgrades ' .
'FROM ' .$fromqa.
'WHERE ' .$whereqa.
$usingattempts->sql.
'ORDER BY sumgrades';
if (!$mediangrades = $DB->get_records_sql_menu($sql, $qaparams, $limitoffset, $limit)){
print_error('errormedian', 'quiz_statistics');
}
if (count($mediangrades)==1){
$median = array_shift($mediangrades);
} else {
$median = array_shift($mediangrades);
$median += array_shift($mediangrades);
$median = $median /2;
}
if ($s>1){
$quizattsstatistics->data[] = array(get_string('median', 'quiz_statistics'), quiz_report_scale_sumgrades_as_percentage($median, $quiz));
//fetch sum of squared, cubed and power 4d
//differences between grades and mean grade
$mean = $usingattempts->total / $s;
$sql = "SELECT " .
"SUM(POWER((qa.sumgrades - :mean1),2)) AS power2, " .
"SUM(POWER((qa.sumgrades - :mean2),3)) AS power3, ".
"SUM(POWER((qa.sumgrades - :mean3),4)) AS power4 ".
'FROM ' .$fromqa.
'WHERE ' .$whereqa.
$usingattempts->sql;
$params = array('mean1' => $mean, 'mean2' => $mean, 'mean3' => $mean)+$qaparams;
if (!$powers = $DB->get_record_sql($sql, $params)){
print_error('errorpowers', 'quiz_statistics');
}
//Standard_Deviation
//see http://docs.moodle.org/en/Development:Quiz_item_analysis_calculations_in_practise#Standard_Deviation
$sd = sqrt($powers->power2 / ($s -1));
$quizattsstatistics->data[] = array(get_string('standarddeviation', 'quiz_statistics'), quiz_report_scale_sumgrades_as_percentage($sd, $quiz));
//Skewness_and_Kurtosis
if ($s>2){
//see http://docs.moodle.org/en/Development:Quiz_item_analysis_calculations_in_practise#Skewness_and_Kurtosis
$m2= $powers->power2 / $s;
$m3= $powers->power3 / $s;
$m4= $powers->power4 / $s;
$k2= $s*$m2/($s-1);
$k3= $s*$s*$m3/(($s-1)*($s-2));
$skewness = $k3 / (pow($k2, 2/3));
$quizattsstatistics->data[] = array(get_string('skewness', 'quiz_statistics'), $skewness);
}
if ($s>3){
$k4= (($s*$s*$s)/(($s-1)*($s-2)*($s-3)))*((($s+1)*$m4)-(3*($s-1)*$m2*$m2));
$kurtosis = $k4 / ($k2*$k2);
$quizattsstatistics->data[] = array(get_string('kurtosis', 'quiz_statistics'), $kurtosis);
}
}
}
if ($s){
/* //CIC, ER and SE.
//http://docs.moodle.org/en/Development:Quiz_item_analysis_calculations_in_practise#CIC.2C_ER_and_SE
list($qsql, $sqlparams) = $DB->get_in_or_equal(array_keys($questions), SQL_PARAMS_NAMED);
$sqlparams += $qaparams;//put quiz id in at beginning of array
$qgradeavgsql = "SELECT qs.question, " .
"AVG(qs.grade) AS gradeaverage " .
"AVG(qa.sumgrades - qs.grade) AS sumgradeaverage " .
"FROM " .
"{question_sessions} qns, " .
"{question_states} qs, " .
$fromqa.' '.
'WHERE ' .$whereqa.
'AND qns.attemptid = qa.uniqueid '.
'AND qs.question '.$qsql.' ' .
$usingattempts->sql.
'AND qns.newgraded = qs.id GROUP BY qs.question';
$qgradeavgs = $DB->get_records_sql($qgradeavgsql, $sqlparams);
$sum = 0;
$sql = 'SELECT COUNT(1) as s,' .
'SUM(POWER((qs.grade - :mean),2)) AS power2 ' .
'FROM ' .
'{question_sessions} qns, ' .
'{question_states} qs, '.
$fromqa.' '.
'WHERE ' .$whereqa.
'AND qns.attemptid = qa.uniqueid '.
'AND qs.question = :qid ' .
$usingattempts->sql.
'AND qns.newgraded = qs.id';
foreach (array_keys($questions) as $qid){
$params = array('mean' => $qgradeavgs[$qid], 'qid' => $qid)+ $qaparams;
$fromdb = $DB->get_record_sql($sql, $params);
if ($fromdb === false){
print_error('errorpowerquestions', 'quiz_statistics');
}
$questions[$qid]->s = $fromdb->s;
if ($s>1){
$questions[$qid]->facility = $qgradeavgs[$qid] / $questions[$qid]->grade;
$questions[$qid]->sd = sqrt($fromdb->power2 / ($questions[$qid]->s -1));
}
$sum += $fromdb->power2;
}
//Discrimination index
$sql = 'SELECT qs.id, ' .
'qs.question, ' .
'qa.sumgrades - qs.grade AS sum, ' .
'qs.grade ' .
'FROM ' .
'{question_sessions} qns, ' .
'{question_states} qs, '.
$fromqa.' '.
'WHERE ' .$whereqa.
'AND qns.attemptid = qa.uniqueid '.
$usingattempts->sql.
'AND qns.newgraded = qs.id';
$fromdbrs = $DB->get_recordset_sql($sql, $qaparams);
if ($fromdbrs === false){
print_error('errorpowerquestions', 'quiz_statistics');
}
foreach ($fromdbrs as $record){
}*/
$sql = 'SELECT qs.id, ' .
'qs.question, ' .
'qa.sumgrades, ' .
'qs.grade ' .
'FROM ' .
'{question_sessions} qns, ' .
'{question_states} qs, '.
$fromqa.' '.
'WHERE ' .$whereqa.
'AND qns.attemptid = qa.uniqueid '.
$usingattempts->sql.
'AND qns.newgraded = qs.id';
$fromdbrs = $DB->get_recordset_sql($sql, $qaparams);
if ($fromdbrs === false){
print_error('errorstatisticsquestions', 'quiz_statistics');
}
foreach (array_keys($questions) as $qid){
$questions[$qid]->s = 0;
$questions[$qid]->totalgrades = 0;
$questions[$qid]->totalothergrades = 0;
$questions[$qid]->gradevariancesum = 0;
$questions[$qid]->othergradevariancesum = 0;
$questions[$qid]->covariancesum = 0;
$questions[$qid]->covariancemaxsum = 0;
$questions[$qid]->covariancewithoverallgradesum = 0;
$questions[$qid]->gradearray = array();
$questions[$qid]->othergradesarray = array();
}
foreach ($fromdbrs as $record){
$questions[$record->question]->s++;
$questions[$record->question]->totalgrades += $record->grade;
$questions[$record->question]->totalothergrades += $record->sumgrades - $record->grade;
//need to sort these to calculate max covariance :
$questions[$record->question]->gradearray[] = $record->grade;
$questions[$record->question]->othergradesarray[] = $record->sumgrades - $record->grade;
}
foreach (array_keys($questions) as $qid){
$questions[$qid]->gradeaverage = $questions[$qid]->totalgrades / $s;
$questions[$qid]->facility = $questions[$qid]->gradeaverage / $questions[$qid]->grade;
$questions[$qid]->othergradeaverage = $questions[$qid]->totalothergrades / $s;
sort($questions[$qid]->gradearray, SORT_NUMERIC);
sort($questions[$qid]->othergradesarray, SORT_NUMERIC);
}
//go through the records one more time
foreach ($fromdbrs as $record){
$gradedifference = ($record->grade - $questions[$record->question]->gradeaverage);
$othergradedifference = (($record->sumgrades - $record->grade) - $questions[$record->question]->othergradeaverage);
$overallgradedifference = $record->sumgrades - $sumgradesavg;
$sortedgradedifference = (array_shift($questions[$qid]->gradearray) - $questions[$record->question]->gradeaverage);
$sortedothergradedifference = (array_shift($questions[$qid]->othergradesarray) - $questions[$record->question]->othergradeaverage);
$questions[$record->question]->gradevariancesum += pow($gradedifference,2);
$questions[$record->question]->othergradevariancesum += pow($othergradedifference,2);
$questions[$record->question]->covariancesum += $gradedifference * $othergradedifference;
$questions[$record->question]->covariancemaxsum += $sortedgradedifference * $sortedothergradedifference;
$questions[$record->question]->covariancewithoverallgradesum += $gradedifference * $overallgradedifference;
}
$sumofcovariancewithoverallgrade = 0;
$sumofgradevariance =0;
foreach (array_keys($questions) as $qid){
$questions[$qid]->gradevariance = $questions[$qid]->gradevariancesum / ($s -1);
$questions[$qid]->othergradevariance = $questions[$qid]->othergradevariancesum / ($s -1);
$questions[$qid]->covariance = $questions[$qid]->covariancesum / ($s -1);
$questions[$qid]->covariancemax = $questions[$qid]->covariancemaxsum / ($s -1);
$sumofgradevariance += $questions[$qid]->gradevariance;
$questions[$qid]->covariancewithoverallgrade = $questions[$qid]->covariancewithoverallgradesum / ($s-1);
$sumofcovariancewithoverallgrade += sqrt($questions[$qid]->covariancewithoverallgrade);
$questions[$qid]->sd = sqrt($questions[$qid]->gradevariancesum / ($s -1));
//avoid divide by zero
if (sqrt($questions[$qid]->gradevariance * $questions[$qid]->othergradevariance)){
$questions[$qid]->discriminationindex = 100*$questions[$qid]->covariance
/ sqrt($questions[$qid]->gradevariance * $questions[$qid]->othergradevariance);
} else {
$questions[$qid]->discriminationindex = '';
}
$questions[$qid]->discriminativeefficiency = 100*$questions[$qid]->covariance / $questions[$qid]->covariancemax;
}
foreach (array_keys($questions) as $qid){
$questions[$qid]->effectiveweight = 100 * sqrt($questions[$qid]->covariancewithoverallgrade)/$sumofcovariancewithoverallgrade;
}
}
if (!$table->is_downloading()){
if ($s>1){
$p = count($questions);//no of positions
if ($p > 1){
$cic = (100 * $p / ($p -1)) * (1 - ($sumofgradevariance)/$k2);
$quizattsstatistics->data[] = array(get_string('cic', 'quiz_statistics'), number_format($cic, $quiz->decimalpoints).' %');
$errorratio = 100 * sqrt(1-($cic/100));
$quizattsstatistics->data[] = array(get_string('errorratio', 'quiz_statistics'), number_format($errorratio, $quiz->decimalpoints).' %');
$standarderror = ($errorratio * $sd / 100);
$quizattsstatistics->data[] = array(get_string('standarderror', 'quiz_statistics'),
quiz_report_scale_sumgrades_as_percentage($standarderror, $quiz));
}
}
if ($quizattsstatistics->data){
print_table($quizattsstatistics);
}
}
if (!$table->is_downloading()){
print_heading(get_string('quizstructureanalysis', 'quiz_statistics'));
}
$table->setup($quiz, $cm->id, $reporturl, $s);
foreach ($questions as $question){
$table->add_data_keyed($table->format_row($question));
}
$table->finish_output();
return true;
}
}
?>