Source for file lp_stats.php
Documentation is available at lp_stats.php
==============================================================================
Dokeos - elearning and course management software
Copyright (c) 2008 Dokeos SPRL
For a full list of contributors, see "credits.txt".
The full license can be read in "license.txt".
This program 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 2
of the License, or (at your option) any later version.
See the GNU General Public License for more details.
Contact address: Dokeos, rue du Corbeau, 108, B-1030 Brussels, Belgium
==============================================================================
* This script displays statistics on the current learning path (scorm)
* This script must be included by lp_controller.php to get basic initialisation
* @package dokeos.learnpath
* @author Yannick Warnier <ywarnier@beeznest.org>
require_once('learnpath.class.php');
//require_once('scorm.class.php');
require_once ('resourcelinker.inc.php');
if(empty($_SESSION['_course']['id']) && isset ($_GET['course']))
$course_code = $_SESSION['_course']['id'];
//The two following variables have to be declared by the includer script
//$lp_id = $_SESSION['oLP']->get_id();
//$list = $_SESSION['oLP']->get_flat_ordered_items_list($lp_id);
//$user_id = $_user['user_id'];
//$stats_charset = $_SESSION['oLP']->encoding
if($origin != 'tracking')
if (!empty ($stats_charset)) {
$charset_lang = $stats_charset;
$charset = $charset_lang;
$htmlHeadXtra[] = ''. '<style type="text/css" media="screen, projection">
@import "../css/public_admin/scorm.css";
include_once ('../inc/reduced_header.inc.php');
//if display in fullscreen required
if (strcmp($_GET["fs"], "true") == 0)
$output .= '<table width="100%" align="center">';
$output .= '<table width="100%">';
//check if the user asked for the "extend all" option
if($origin == 'tracking')
$url_suffix = '&course='. $_GET['course']. '&student_id='. $_GET['student_id']. '&lp_id='. $_GET['lp_id']. '&origin='. $_GET['origin'];
if (!empty ($_GET['extend_all']))
$extend_all_link = '<a href="'. api_get_self(). '?action=stats'. $url_suffix. '"><img src="../img/view_less_stats.gif" alt="fold_view" border="0"></a>';
$extend_all_link = '<a href="'. api_get_self(). '?action=stats&extend_all=1'. $url_suffix. '"><img src="../img/view_more_stats.gif" alt="extend_view" border="0"></a>';
if($origin != 'tracking')
$output .= "<tr><td><div class='title'>". htmlentities(get_lang('ScormMystatus'), ENT_QUOTES, $dokeos_charset). "</div></td></tr>";
$output .= "<tr><td> </td></tr>". "<tr><td>". "<table border='0' class='data_table'><tr>\n". '<td width="16">'. $extend_all_link. '</td>'. '<td colspan="4" class="title"><div class="mystatusfirstrow">'. htmlentities(get_lang('ScormLessonTitle'), ENT_QUOTES, $dokeos_charset). "</div></td>\n". '<td colspan="2" class="title"><div class="mystatusfirstrow">'. htmlentities(get_lang('ScormStatus'), ENT_QUOTES, $dokeos_charset). "</div></td>\n". '<td colspan="2" class="title"><div class="mystatusfirstrow">'. htmlentities(get_lang('ScormScore'), ENT_QUOTES, $dokeos_charset). "</div></td>\n". '<td colspan="2" class="title"><div class="mystatusfirstrow">'. htmlentities(get_lang('ScormTime'), ENT_QUOTES, $dokeos_charset). "</div></td><td class='title'><div class='mystatusfirstrow'>". htmlentities(get_lang('Actions'), ENT_QUOTES, $dokeos_charset). "</div></td></tr>\n";
//going through the items using the $items[] array instead of the database order ensures
// we get them in the same order as in the imsmanifest file, which is rather random when using
$sql = "SELECT max(view_count) FROM $TBL_LP_VIEW " .
"WHERE lp_id = $lp_id AND user_id = '". $user_id. "'";
//error_log('New LP - Querying views for latest attempt: '.$sql,0);
foreach ($list as $my_item_id)
if ((!empty ($_GET['extend_id']) and $_GET['extend_id'] == $my_item_id) OR $extend_all) {
//prepare statement to go through each attempt
$sql = "SELECT iv.status as mystatus, v.view_count as mycount, " .
"iv.score as myscore, iv.total_time as mytime, i.id as myid, " .
"i.title as mytitle, i.max_score as mymaxscore, " .
"iv.max_score as myviewmaxscore, " .
"i.item_type as item_type, iv.view_count as iv_view_count, " .
"iv.id as iv_id, path as path ".
" FROM $TBL_LP_ITEM as i, $TBL_LP_ITEM_VIEW as iv, $TBL_LP_VIEW as v ".
" WHERE i.id = iv.lp_item_id ".
" AND i.id = $my_item_id ".
" AND iv.lp_view_id = v.id ".
" AND i.lp_id = $lp_id ".
" AND v.user_id = ". $user_id. " ".
" AND v.view_count = $view ".
" ORDER BY iv.view_count $qry_order ";
$sql = "SELECT iv.status as mystatus, v.view_count as mycount, " .
"iv.score as myscore, iv.total_time as mytime, i.id as myid, " .
"i.title as mytitle, i.max_score as mymaxscore, " .
"iv.max_score as myviewmaxscore, " .
"i.item_type as item_type, iv.view_count as iv_view_count, " .
"iv.id as iv_id, path as path ".
" FROM $TBL_LP_ITEM as i, $TBL_LP_ITEM_VIEW as iv, $TBL_LP_VIEW as v ".
" WHERE i.id = iv.lp_item_id ".
" AND i.id = $my_item_id ".
" AND iv.lp_view_id = v.id ".
" AND i.lp_id = $lp_id ".
" AND v.user_id = ". $user_id. " ".
" ORDER BY iv.view_count $qry_order ";
if (($extend_this OR $extend_all) && $num > 0)
//if there are several attempts, and the link to extend has been clicked, show each attempt...
$extend_link = '<a href="'. api_get_self(). '?action=stats&fold_id='. $my_item_id. $url_suffix. '"><img src="../img/visible.gif" alt="fold_view" border="0"></a>'. "\n";
$title = $row['mytitle'];
if ($row['item_type'] != 'dokeos_chapter')
$correct_test_link = array();
if($row['item_type'] == 'quiz')
if($origin != 'tracking')
$sql_last_attempt= 'SELECT exe_id FROM '. $tbl_stats_exercices. ' WHERE exe_exo_id="'. $row['path']. '" AND exe_user_id="'. api_get_user_id(). '" AND exe_cours_id="'. $course_code. '" ORDER BY exe_date DESC limit 1';
$sql_last_attempt= 'SELECT exe_id FROM '. $tbl_stats_exercices. ' WHERE exe_exo_id="'. $row['path']. '" AND exe_user_id="'. $_GET['student_id']. '" AND exe_cours_id="'. $course_code. '" ORDER BY exe_date DESC limit 1';
$resultLastAttempt = api_sql_query($sql_last_attempt,__FILE__ ,__LINE__ );
$id_last_attempt = $rowLA[0];
if($origin != 'tracking')
$correct_test_link = '<a href="../exercice/exercise_show.php?origin=student_progress&id='. $id_last_attempt. '&cidReq='. $course_code. '" target="_parent"><img src="'. api_get_path(WEB_IMG_PATH). 'quiz.gif"></a>';
$correct_test_link = '<a href="../exercice/exercise_show.php?origin=tracking_course&id='. $id_last_attempt. '&cidReq='. $course_code. '&student='. $_GET['student_id']. '" target="_parent"><img src="'. api_get_path(WEB_IMG_PATH). 'quiz.gif"></a>';
if($origin != 'tracking')
$correct_test_link = '<a href="../exercice/exercise_show.php?origin=student_progress&id='. $id_last_attempt. '&cidReq='. $course_code. '" target="_parent"><img src="'. api_get_path(WEB_IMG_PATH). 'quiz.gif"></a>';
$correct_test_link = '<a href="../exercice/exercise_show.php?origin=tracking_course&id='. $id_last_attempt. '&cidReq='. $course_code. '&student='. $_GET['student_id']. '" target="_parent"><img src="'. api_get_path(WEB_IMG_PATH). 'quiz.gif"></a>';
$output .= "<tr class='$oddclass'>\n". "<td>$extend_link</td>\n". '<td colspan="4" class="content"><div class="mystatus">'. htmlentities($title,ENT_QUOTES,$charset_lang). "</div></td>\n". '<td colspan="2" class="content"></td>'. "\n". '<td colspan="2" class="content"></td>'. "\n". '<td colspan="2" class="content"></td><td class="content"></td>'. "\n". "</tr>\n";
//check if there are interactions below
$extend_attempt_link = '';
$extend_this_attempt = 0;
if (!empty ($_GET['extend_attempt_id']) && $_GET['extend_attempt_id'] == $row['iv_id']) {
//the extend button for this attempt has been clicked
$extend_this_attempt = 1;
$extend_attempt_link = '<a href="'. api_get_self(). '?action=stats&extend_id='. $my_item_id. '&fold_attempt_id='. $row['iv_id']. $url_suffix. '"><img src="../img/visible.gif" alt="fold_attempt_view" border="0"></a>'. "\n";
} else { //same case if fold_attempt_id is set, so not implemented explicitly
//the extend button for this attempt has not been clicked
$extend_attempt_link = '<a href="'. api_get_self(). '?action=stats&extend_id='. $my_item_id. '&extend_attempt_id='. $row['iv_id']. $url_suffix. '"><img src="../img/invisible.gif" alt="extend_attempt_view" border="0"></a>'. "\n";
$lesson_status = $row['mystatus'];
$score = $row['myscore'];
$time_for_total = $row['mytime'];
$scoIdentifier = $row['myid'];
if($row['item_type'] == 'sco')
if(!empty($row['myviewmaxscore']) and $row['myviewmaxscore']> 0)
$maxscore= $row['myviewmaxscore'];
elseif($row['myviewmaxscore'] === '')
$maxscore = $row['mymaxscore'];
$maxscore = $row['mymaxscore'];
//Remove "NaN" if any (@todo: locate the source of these NaN)
if (($lesson_status == 'completed') or ($lesson_status == 'passed')) {
$mylanglist = array ('completed' => 'ScormCompstatus', 'incomplete' => 'ScormIncomplete', 'failed' => 'ScormFailed', 'passed' => 'ScormPassed', 'browsed' => 'ScormBrowsed', 'not attempted' => 'ScormNotAttempted',);
$my_lesson_status = htmlentities(get_lang($mylanglist[$lesson_status]), ENT_QUOTES, $dokeos_charset);
//$my_lesson_status = get_lang($mylanglist[$lesson_status]);
if ($row['item_type'] != 'dokeos_chapter') {
$output .= "<tr class='$oddclass'>\n". "<td></td>\n". "<td>$extend_attempt_link</td>\n". '<td colspan="3">'. htmlentities(get_lang('Attempt'), ENT_QUOTES, $dokeos_charset). ' '. $row['iv_view_count']. "</td>\n"
//."<td><font color='$color'><div class='mystatus'>".htmlentities($array_status[$lesson_status],ENT_QUOTES,$charset_lang)."</div></font></td>\n"
. '<td colspan="2"><font color="'. $color. '"><div class="mystatus">'. $my_lesson_status. "</div></font></td>\n". '<td colspan="2"><div class="mystatus" align="center">'. ($score == 0 ? '-' : ($maxscore== 0? $score: $score. '/'. $maxscore)). "</div></td>\n". '<td colspan="2"><div class="mystatus">'. $time. "</div></td><td></td>\n". "</tr>\n";
if ($extend_this_attempt OR $extend_all) {
foreach ($list1 as $id => $interaction) {
if (($counter % 2) == 0) {
$output .= "<tr class='$oddclass'>\n". '<td></td>'. "\n". '<td></td>'. "\n". '<td> </td>'. "\n". '<td>'. $interaction['order_id']. '</td>'. "\n". '<td>'. $interaction['id']. '</td>'. "\n"
//."<td><font color='$color'><div class='mystatus'>".htmlentities($array_status[$lesson_status],ENT_QUOTES,$charset_lang)."</div></font></td>\n"
. '<td colspan="2">'. $interaction['type']. "</td>\n"
//.'<td>'.$interaction['correct_responses']."</td>\n"
. '<td>'. urldecode($interaction['student_response']). "</td>\n". '<td>'. $interaction['result']. "</td>\n". '<td>'. $interaction['latency']. "</td>\n". '<td>'. $interaction['time']. "</td>\n<td></td>\n</tr>\n";
foreach ($list2 as $id => $interaction) {
if (($counter % 2) == 0) {
$output .= "<tr class='$oddclass'>\n". '<td></td>'. "\n". '<td></td>'. "\n". '<td> </td>'. "\n". '<td>'. $interaction['order_id']. '</td>'. "\n". '<td colspan="2">'. $interaction['objective_id']. '</td>'. "\n"
. '<td colspan="2">'. $interaction['status']. "</td>\n"
. '<td>'. $interaction['score_raw']. "</td>\n". '<td>'. $interaction['score_max']. "</td>\n". '<td>'. $interaction['score_min']. "</td>\n<td></td>\n</tr>\n";
//check if there are interactions below
$extend_attempt_link = '';
$extend_this_attempt = 0;
if (($inter_num > 0 || $objec_num > 0) && !$extend_all) {
if (!empty ($_GET['extend_attempt_id']) && $_GET['extend_attempt_id'] == $row['iv_id']) {
//the extend button for this attempt has been clicked
$extend_this_attempt = 1;
$extend_attempt_link = '<a href="'. api_get_self(). '?action=stats&extend_id='. $my_item_id. '&fold_attempt_id='. $row['iv_id']. $url_suffix. '"><img src="../img/visible.gif" alt="fold_attempt_view" border="0"></a>'. "\n";
} else { //same case if fold_attempt_id is set, so not implemented explicitly
//the extend button for this attempt has not been clicked
$extend_attempt_link = '<a href="'. api_get_self(). '?action=stats&extend_id='. $my_item_id. '&extend_attempt_id='. $row['iv_id']. $url_suffix. '"><img src="../img/invisible.gif" alt="extend_attempt_view" border="0"></a>'. "\n";
if (($counter % 2) == 0) {
//$extend_link = '<img src="../img/invisible.gif" alt="extend_disabled">';
$extend_link = '<a href="'. api_get_self(). '?action=stats&extend_id='. $my_item_id. '&extend_attempt_id='. $row['iv_id']. $url_suffix. '"><img src="../img/invisible.gif" alt="extend_view" border="0"></a>';
if (($counter % 2) == 0) {
$lesson_status = $row['mystatus'];
$score = $row['myscore'];
$subtotal_time = $row['mytime'];
$subtotal_time += $tmp_row['mytime'];
$time_for_total = $subtotal_time;
$scoIdentifier = $row['myid'];
$title = $row['mytitle'];
if($row['item_type'] == 'sco')
if(!empty($row['myviewmaxscore']) and $row['myviewmaxscore']> 0)
$maxscore= $row['myviewmaxscore'];
elseif($row['myviewmaxscore'] === '')
$maxscore = $row['mymaxscore'];
$maxscore = $row['mymaxscore'];
//Remove "NaN" if any (@todo: locate the source of these NaN)
//$time = str_replace('NaN', '00'.$h.'00\'00"', $time);
if (($lesson_status == 'completed') or ($lesson_status == 'passed')) {
$mylanglist = array ('completed' => 'ScormCompstatus', 'incomplete' => 'ScormIncomplete', 'failed' => 'ScormFailed', 'passed' => 'ScormPassed', 'browsed' => 'ScormBrowsed', 'not attempted' => 'ScormNotAttempted',);
$my_lesson_status = htmlentities(get_lang($mylanglist[$lesson_status]), ENT_QUOTES, $dokeos_charset);
if ($row['item_type'] != 'dokeos_chapter') {
if($row['item_type'] == 'quiz'){
if($origin != 'tracking' && $origin != 'tracking_course'){
$sql_last_attempt= 'SELECT exe_id FROM '. $tbl_stats_exercices. ' WHERE exe_exo_id="'. $row['path']. '" AND exe_user_id="'. api_get_user_id(). '" AND exe_cours_id="'. $course_code. '" ORDER BY exe_date ASC';
$sql_last_attempt= 'SELECT exe_id FROM '. $tbl_stats_exercices. ' WHERE exe_exo_id="'. $row['path']. '" AND exe_user_id="'. Database::escape_string($_GET['student_id']). '" AND exe_cours_id="'. Database::escape_string($_GET['course']). '" ORDER BY exe_date';
$resultLastAttempt = api_sql_query($sql_last_attempt,__FILE__ ,__LINE__ );
$laid = $rowLA['exe_id'];
if($origin != 'tracking')
$correct_test_link .= '<a href="../exercice/exercise_show.php?origin=student_progress&id='. $laid. '&cidReq='. $course_code. '" target="_parent" title="'. get_lang('Attempt'). ' '. $i. '"><img src="'. api_get_path(WEB_IMG_PATH). 'quiz.gif"></a> ';
$correct_test_link = substr($correct_test_link,0,- 1);
if($origin != 'tracking'){
$correct_test_link = '<a href="../exercice/exercise_show.php?origin=student_progress&id='. $id_last_attempt. '&cidReq='. $course_code. '" target="_parent"><img src="'. api_get_path(WEB_IMG_PATH). 'quiz.gif"></a>';
$output .= "<tr class='$oddclass'>\n". "<td>$extend_link</td>\n". '<td colspan="4"><div class="mystatus">'. htmlentities($title,ENT_QUOTES,$charset_lang). '</div></td>'. "\n"
//."<td><font color='$color'><div class='mystatus'>".htmlentities($array_status[$lesson_status],ENT_QUOTES,$charset_lang)."</div></font></td>\n"
. '<td colspan="2"><font color="'. $color. '"><div class="mystatus">'. $my_lesson_status. "</div></font></td>\n". '<td colspan="2"><div class="mystatus" align="center">'. ($score == 0 ? '-' : ($maxscore== 0? $score: $score. '/'. $maxscore)). "</div></td>\n". '<td colspan="2"><div class="mystatus">'. $time. "</div></td><td>$correct_test_link</td>\n". "</tr>\n";
$temp[] = ($score == 0 ? '-' : ($maxscore== 0? $score: $score. '/'. $maxscore));
if ($extend_this_attempt OR $extend_all) {
foreach ($list1 as $id => $interaction) {
if (($counter % 2) == 0) {
$output .= "<tr class='$oddclass'>\n". '<td></td>'. "\n". '<td></td>'. "\n". '<td> </td>'. "\n". '<td>'. $interaction['order_id']. '</td>'. "\n". '<td>'. $interaction['id']. '</td>'. "\n"
//."<td><font color='$color'><div class='mystatus'>".htmlentities($array_status[$lesson_status],ENT_QUOTES,$charset_lang)."</div></font></td>\n"
. '<td colspan="2">'. $interaction['type']. "</td>\n"
//.'<td>'.$interaction['correct_responses']."</td>\n"
. '<td>'. urldecode($interaction['student_response']). "</td>\n". '<td>'. $interaction['result']. "</td>\n". '<td>'. $interaction['latency']. "</td>\n". '<td>'. $interaction['time']. "</td>\n<td></td>\n</tr>\n";
foreach ($list2 as $id => $interaction) {
if (($counter % 2) == 0) {
$output .= "<tr class='$oddclass'>\n". '<td></td>'. "\n". '<td></td>'. "\n". '<td> </td>'. "\n". '<td>'. $interaction['order_id']. '</td>'. "\n". '<td colspan="2">'. $interaction['objective_id']. '</td>'. "\n"
. '<td colspan="2">'. $interaction['status']. "</td>\n"
. '<td>'. $interaction['score_raw']. "</td>\n". '<td>'. $interaction['score_max']. "</td>\n". '<td>'. $interaction['score_min']. "</td>\n<td></td>\n</tr>\n";
//only sum up the latest attempt each time
$total_max_score += $maxscore;
$total_time += $time_for_total;
//$total_time = str_replace('NaN','00:00:00',$total_time);
$total_time = str_replace('NaN', '00'. $h. '00\'00"', $total_time);
if($lp_type == 2) //if scorm
if($total_max_score == 0)
$total_percent = number_format((float) $total_score, 1, '.', '');
$final_score = ($total_score == 0 ? '-' : $total_percent);
$total_percent = number_format((((float) $total_score / (float) $total_max_score) * 100), 1, '.', '');
$final_score = ($total_score == 0 ? '-' : $total_percent. '%');
if ($total_max_score == 0) {
$total_percent = number_format((((float) $total_score / (float) $total_max_score) * 100), 1, '.', '');
$final_score = ($total_score == 0 ? '-' : $total_percent. '%');
if (($counter % 2) == 0) {
$output .= "<tr class='$oddclass'>\n". "<td></td>\n". '<td colspan="4"><div class="mystatus"><i>'. htmlentities(get_lang('AccomplishedStepsTotal'), ENT_QUOTES, $dokeos_charset). "</i></div></td>\n"
//."<td><font color='$color'><div class='mystatus'>".htmlentities($array_status[$lesson_status],ENT_QUOTES,$charset_lang)."</div></font></td>\n"
. '<td colspan="2"></td>'. "\n". '<td colspan="2"><div class="mystatus" align="center">'. $final_score. "</div></td>\n". '<td colspan="2"><div class="mystatus">'. $total_time. '</div></td><td></td>'. "\n". "</tr>\n";
$output .= "</table></td></tr></table>";
$temp = array('','','','');
$temp = array(get_lang('AccomplishedStepsTotal'),'',$final_score,$total_time);
if($origin != 'tracking')
$output .= "</body></html>";
|