Source for file aicc_api.php
Documentation is available at aicc_api.php
==============================================================================
* API event handler functions for AICC / CMIv4 in API communication mode
* @author Denes Nagy <darkden@freemail.hu>
* @author Yannick Warnier <ywarnier@beeznest.org>
* @package dokeos.learnpath
* @license GNU/GPL - See Dokeos license directory for details
==============================================================================
* This script is divided into three sections.
* The first section (below) is the initialisation part.
* The second section is the AICC object part
* The third section defines the event handlers for Dokeos' internal messaging
* This script implements the API messaging for AICC. The HACP messaging is
* made by another set of scripts.
==============================================================================
==============================================================================
//flag to allow for anonymous user - needs to be set before global.inc.php
//Load common libraries using a compatibility script to bridge between 1.6 and 1.8
require_once('back_compat.inc.php');
//Load learning path libraries so we can use the objects to define the initial values
require_once('learnpath.class.php');
require_once('learnpathItem.class.php');
require_once('aicc.class.php');
// Is this needed? This is probabaly done in the header file
// $_user = $_SESSION['_user'];
$file = $_SESSION['file'];
$oItem = $oLP->items[$oLP->current];
error_log('New LP - scorm_api - Could not load oItem item',0);
$autocomplete_when_80pct = 0;
==============================================================================
==============================================================================
?>var scorm_logs= <?php echo (empty($oLP->scorm_debug)? '0': '3');?>; //debug log level for SCORM. 0 = none, 1=light, 2=a lot, 3=all - displays logs in log frame
var lms_logs=0; //debug log level for LMS actions. 0=none, 1=light, 2=a lot, 3=all
//logit_lms('scormfunctions.php included',0);
this.LMSInitialize=LMSInitialize;
this.LMSGetValue=LMSGetValue;
this.LMSSetValue=LMSSetValue;
this.LMSCommit=LMSCommit;
this.LMSFinish=LMSFinish;
this.LMSGetLastError=LMSGetLastError;
this.LMSGetErrorString=LMSGetErrorString;
this.LMSGetDiagnostic=LMSGetDiagnostic;
//it is not sure that the scos use the above declarations
API = new APIobject(); //for scorm 1.2
var G_GeneralException = 101;
var G_InvalidArgumentError = 201;
var G_ElementCannotHaveChildren = 202;
var G_ElementIsNotAnArray = 203;
var G_NotInitialized = 301;
var G_NotImplementedError = 401;
var G_InvalidSetValue = 402;
var G_ElementIsReadOnly = 403;
var G_ElementIsWriteOnly = 404;
var G_IncorrectDataType = 405;
var G_LastError = G_NoError ;
//Strictly scorm variables
var score= <?php echo $oItem->get_score();?>;
var max= <?php echo $oItem->get_max();?>;
var min= <?php echo $oItem->get_min();?>;
var lesson_status=' <?php echo $oItem->get_status();?>';
var session_time=' <?php echo $oItem->get_scorm_time('js');?>';
var suspend_data = ' <?php echo $oItem->get_suspend_data();?>';
var lesson_location = ' <?php echo $oItem->get_lesson_location();?>';
var total_time = ' <?php echo $oItem->get_scorm_time('js');?>';
//Dokeos internal variables
var saved_lesson_status = 'not attempted';
var lms_lp_id = <?php echo $oLP->get_id();?>;
var lms_item_id = <?php echo $oItem->get_id();?>;
//var lms_new_item_id = 0; //temporary value (only there between a load_item() and a LMSInitialize())
var lms_been_synchronized = 0;
var lms_total_lessons = <?php echo $oLP->get_total_items_count(); ?>;
var lms_complete_lessons = <?php echo $oLP->get_complete_items_count();?>;
var lms_progress_bar_mode = ' <?php echo $oLP->progress_bar_mode;?>';
if(lms_progress_bar_mode == ''){lms_progress_bar_mode='%';}
var lms_view_id = ' <?php echo $oLP->get_view();?>';
if(lms_view_id == ''){ lms_view_id = 1;}
var lms_user_id = ' <?php echo $_user['user_id'];?>';
var lms_next_item = ' <?php echo $oLP->get_next_item_id();?>';
var lms_previous_item = ' <?php echo $oLP->get_previous_item_id();?>';
var lms_lp_type = ' <?php echo $oLP->get_type();?>';
var lms_item_type = ' <?php echo $oItem->get_type();?>';
var old_lesson_status = '';
var old_session_time = '';
var old_suspend_data = '';
function LMSInitialize() { //this is the initialize function of all APIobjects
logit_scorm('LMSInitialise()',0);
function LMSGetValue(param) {
//logit_scorm("LMSGetValue('"+param+"')",1);
if(param=='cmi.core._children' || param=='cmi.core_children'){
result='entry, exit, lesson_status, student_id, student_name, lesson_location, total_time, credit, lesson_mode, score, session_time';
}else if(param == 'cmi.core.entry'){
}else if(param == 'cmi.core.exit'){
}else if(param == 'cmi.core.lesson_status'){
if(lesson_status != '') {
}else if(param == 'cmi.core.student_id'){
result=' <?php echo $_user['user_id']; ?>';
}else if(param == 'cmi.core.student_name'){
$who= addslashes($_user['lastName']. ", ". $_user['firstName']);
}else if(param == 'cmi.core.lesson_location'){
}else if(param == 'cmi.core.total_time'){
}else if(param == 'cmi.core.score._children'){
}else if(param == 'cmi.core.score.raw'){
}else if(param == 'cmi.core.score.max'){
}else if(param == 'cmi.core.score.min'){
}else if(param == 'cmi.core.score'){
}else if(param == 'cmi.core.credit'){
}else if(param == 'cmi.core.lesson_mode'){
}else if(param == 'cmi.suspend_data'){
result=' <?php echo $oItem->get_suspend_data();?>';
}else if(param == 'cmi.launch_data'){
}else if(param == 'cmi.objectives._count'){
result=' <?php echo $oItem->get_view_count();?>';
//Switch not working??? WTF???
case 'cmi.core._children' :
result='entry, exit, lesson_status, student_id, student_name, lesson_location, total_time, credit, lesson_mode, score, session_time';
case 'cmi.core_children' :
result='entry, exit, lesson_status, student_id, student_name, lesson_location, total_time, credit, lesson_mode, score, session_time';
case 'cmi.core.lesson_status' :
if(lesson_status != '') {
case 'cmi.core.student_id' :
result=' <?php echo $_user['user_id']; ?>';
case 'cmi.core.student_name' :
$who= addslashes($_user['firstName']. ",". $_user['lastName']);
case 'cmi.core.lesson_location' :
case 'cmi.core.total_time' :
case 'cmi.core.score._children' :
case 'cmi.core.score.raw' :
case 'cmi.core.score.max' :
case 'cmi.core.score.min' :
case 'cmi.core.lesson_mode' :
case 'cmi.suspend_data' :
result=' <?php echo $oItem->get_suspend_data();?>';
case 'cmi.objectives._count':
result=' <?php echo $oItem->get_view_count();?>';
logit_scorm("LMSGetValue('"+param+"') returned '"+result+"'",1);
function LMSSetValue(param, val) {
logit_scorm("LMSSetValue('"+param+"','"+val+"')",0);
case 'cmi.core.score.raw' : score= val ; break;
case 'cmi.core.score.max' : max = val; break;
case 'cmi.core.score.min' : min = val; break;
case 'cmi.core.lesson_location' : lesson_location = val;break;
case 'cmi.core.lesson_status' :
saved_lesson_status = lesson_status;
<?php if($oLP->mode != 'fullscreen'){ ?>
//var update = update_toc(lesson_status,lms_item_id);
case 'cmi.completion_status' : lesson_status = val; break; //1.3
case 'cmi.core.session_time' : session_time = val; break;
case 'cmi.score.scaled' : score = val ; break; //1.3
case 'cmi.success_status' : success_status = val; break; //1.3
case 'cmi.suspend_data' : suspend_data = val; break;
//var update = update_toc();
//var update_progress = update_progress_bar();
if ($oLP->force_commit == 1){
echo " var mycommit = LMSCommit('force');";
function savedata(origin) { //origin can be 'commit', 'finish' or 'terminate'
<?php if ($autocomplete_when_80pct){ ?>
if( ( lesson_status == 'incomplete') && (score >= (0.8*max) ) ){
lesson_status = 'completed';
param = 'id='+lms_item_id+'&origin='+origin+'&score='+score+'&max='+max+'&min='+min+'&lesson_status='+lesson_status+'&time='+session_time+'&suspend_data='+suspend_data;
$url= $_SERVER['HTTP_HOST']. $self;
$url= substr($url,0,- 14);//14 is the length of this file's name (/scorm_api.php)
?>/lp_controller.php?cidReq= <?php echo api_get_course_id();?>&action=save&lp_id= <?php echo $oLP->get_id();?>&" + param + "";
logit_lms('saving data (status='+lesson_status+')',1);
xajax_save_item(lms_lp_id, lms_user_id, lms_view_id, lms_item_id, score, max, min, lesson_status, session_time, suspend_data, lesson_location);
function LMSCommit(val) {
logit_scorm('LMSCommit()',0);
function LMSFinish(val) {
if (( commit == false )) {
logit_scorm('LMSFinish() (no LMSCommit())',1);
logit_scorm('LMSFinish() called',1);
function LMSGetLastError() {
logit_scorm('LMSGetLastError()',1);
function LMSGetErrorString(errCode){
logit_scorm('LMSGetErrorString()',1);
function LMSGetDiagnostic(errCode){
logit_scorm('LMSGetDiagnostic()',1);
return(API.LMSGetLastError());
//--------------------------------------------------------------------//
* Dokeos-specific code that deals with event handling and inter-frames
* Note that from now on, the Dokeos JS code in this library will act as
* a controller, of the MVC pattern, and receive all requests for frame
* updates, then redispatch to any frame concerned.
* Defining the AJAX-object class to be made available from other frames
this.xajax_switch_item_details=xajax_switch_item_details;
this.switch_item=switch_item;
//it is not sure that the scos use the above declarations
oXAJAX = new XAJAXobject();
oxajax = new XAJAXobject();
* Cross-browser event handling by Scott Andrew
* @param element Element that needs an event attached
* @param string Event type (load, unload, click, keyDown, ...)
* @param string Function name (the event handler)
* @param string used in addEventListener
function addEvent(elm, evType, fn, useCapture){
if(elm.addEventListener){
elm.addEventListener(evType, fn, useCapture);
}else if (elm.attachEvent){
var r = elm.attachEvent('on' + evType, fn);
* Add listeners to the page objects. This has to be defined for
* the current context as it acts on objects that should exist
//exit if the browser doesn't support ID or tag retrieval
logit_lms('Entering addListeners()',2);
if(!document.getElementsByTagName){
logit_lms("getElementsByTagName not available",2);
if(!document.getElementById){
logit_lms("getElementById not available",2);
//assign event handlers to objects
if(lms_lp_type==1 || lms_item_type=='asset'){
logit_lms('Dokeos LP or asset',2);
//if this path is a Dokeos learnpath, then start manual save
//when something is loaded in there
var myelem = document.getElementById('content_id');
if(!myelem){logit_lms("Impossible to find content_id element in document",2);}
addEvent(myelem,'unload',dokeos_save_asset,false);
logit_lms('Added event listener on content_id for unload',2);
logit_lms('Quitting addListeners()',2);
* Load an item into the content frame:
* - making sure the previous item status have been saved
* - first updating the current item ID (to save the right item)
* - updating the frame src
function load_item(item_id,url){
if(document.getElementById('content_id'))
logit_lms('Loading item '+item_id,2);
var cont_f = document.getElementById('content_id');
lms_old_item_id = lms_item_id;
var lms_new_item_id = item_id;
//load new content page into content frame
if(lms_lp_type==1 || lms_item_type=='asset'){
update_toc('unhighlight',lms_old_item_id);
update_toc('highlight',item_id);
lms_been_synchronized = 0;
if(lms_lp_type==1 || lms_item_type=='asset'){
lms_item_id = lms_new_item_id;
logit_lms('cont_f.src has no properties',0);
logit_lms('content_id has no properties',0);
* Save a Dokeos learnpath item's time and mark as completed upon
function dokeos_save_asset(){
//var linkparams = 'id='+lms_item_id+'&score='+score+'&max='+max+'&min='+min+'&lesson_status='+lesson_status+'&time='+session_time+'&suspend_data='+suspend_data;
logit_lms('dokeos_save_asset: '+url,0);
//frames["message_name"].src = url;
xajax_save_item(lms_lp_id, lms_user_id, lms_view_id, lms_item_id, score, max, min, lesson_status, session_time, suspend_data, lesson_location);
* Logs information about SCORM messages into the log frame
* @param string Message to log
* @param integer Priority (0 for top priority, 3 for lowest)
function logit_scorm(message,priority){
if(frames["lp_log_name"] && scorm_logs>priority){
frames["lp_log_name"].document.getElementById("log_content").innerHTML += "AICC: " + message + "<br/>";
* Logs information about LMS activity into the log frame
* @param string Message to log
* @param integer Priority (0 for top priority, 3 for lowest)
function logit_lms(message,priority){
if(frames["lp_log_name"] && lms_logs>priority){
frames["lp_log_name"].document.getElementById("log_content").innerHTML += "LMS: " + message + "<br/>";
* update the Table Of Contents frame, by changing CSS styles, mostly
* @param string Action to be taken
* @param integer Item id to update
function update_toc(update_action,update_id)
<?php if($oLP->mode != 'fullscreen'){ ?>
var myframe = frames["toc_name"];
var myelem = myframe.document.getElementById("toc_"+update_id);
var myelemimg = myframe.document.getElementById("toc_img_"+update_id);
logit_lms('update_toc('+update_action+','+update_id+')',2);
myelem.className = "scorm_item";
myelem.className = "scorm_item_highlight";
if(myelemimg.src != '../img/notattempted.gif'){
myelemimg.src = "../img/notattempted.gif";
myelemimg.alt = "not attempted";
if(myelemimg.src != '../img/incomplete.gif'){
myelemimg.src = "../img/incomplete.gif";
myelemimg.alt = "incomplete";
if(myelemimg.src != '../img/completed.gif'){
myelemimg.src = "../img/completed.gif";
myelemimg.alt = "completed";
if(myelemimg.src != '../img/failed.gif'){
myelemimg.src = "../img/failed.gif";
myelemimg.alt = "failed";
if(myelemimg.src != '../img/completed.gif' && myelemimg.alt != 'passed'){
myelemimg.src = "../img/completed.gif";
myelemimg.alt = "passed";
if(myelemimg.src != '../img/completed.gif' && myelemimg.alt != 'browsed'){
myelemimg.src = "../img/completed.gif";
myelemimg.alt = "browsed";
logit_lms('Update action unknown',2);
* Updates the progress bar with the new status. Prevents the need of a page refresh and flickering
* @param integer Number of completed items
* @param integer Number of items in total
* @param string Display mode (absolute 'abs' or percentage '%').Defaults to %
function update_progress_bar(nbr_complete, nbr_total, mode)
logit_lms('update_progress_bar('+nbr_complete+','+nbr_total+','+mode+')',2);
logit_lms('could update with data: '+lms_lp_id+','+lms_view_id+','+lms_user_id,2);
var myframe = frames["nav_name"];
if(mode == ''){mode='%';}
if(nbr_total == 0){nbr_total=1;}
var percentage = (nbr_complete/nbr_total)*100;
percentage = Math.round(percentage);
var pr_text = myframe.document.getElementById('progress_text');
var pr_full = myframe.document.getElementById('progress_img_full');
var pr_empty = myframe.document.getElementById('progress_img_empty');
pr_full.width = percentage;
pr_empty.width = 100-percentage;
mytext = nbr_complete + '/' + nbr_total;
mytext = percentage + '%';
pr_text.innerHTML = mytext;
function update_stats_page()
var myframe = document.getElementById('content_id');
var mysrc = myframe.location.href;
if(mysrc == 'lp_controller.php?action=stats'){
if(myframe && myframe.src){
// = mysrc; //refresh page
* Updates the message frame with the given string
function update_message_frame(msg_msg)
if(msg_msg==null){msg_msg='';}
var msg_f = frames["message_name"];
if(!msg_f.document || !msg_f.document.getElementById('msg_div_id')){
logit_lms('In update_message_frame() - message frame has no document property',0);
logit_lms('In update_message_frame() - updating frame',0);
msg_f.document.getElementById('msg_div_id').innerHTML= msg_msg;
* Function that handles the saving of an item and switching from an item to another.
* Once called, this function should be able to do the whole process of (1) saving the
* current item, (2) refresh all the values inside the SCORM API object, (3) open the
* new item into the content_id frame, (4) refresh the table of contents, (5) refresh
* the progress bar (completion), (6) refresh the message frame
* @param integer Dokeos ID for the current item
* @param string This parameter can be a string specifying the next
* item (like 'next', 'previous', 'first' or 'last') or the id to the next item
function switch_item(current_item, next_item){
logit_lms('In switch - no current_item defined',0);
logit_lms('In switch - no next_item defined',0);
if(lms_item_id == next_item){
return; //nothing to switch
//(1) save the current item
logit_lms('Called switch_item with params '+lms_item_id+' and '+next_item+'',0);
xajax_save_item(lms_lp_id, lms_user_id, lms_view_id, lms_item_id, score, max, min, lesson_status, session_time, suspend_data, lesson_location);
//(2) Refresh all the values inside this SCORM API object - use AJAX
//xajax_backup_item_details(lms_lp_id, lms_user_id, lms_view_id, lms_item_id, score, max, min, lesson_status, session_time, suspend_data);
xajax_switch_item_details(lms_lp_id,lms_user_id,lms_view_id,lms_item_id,next_item);
//(3) open the new item in the content_id frame
var cont_f = document.getElementById('content_id');
if(!cont_f){logit_lms('In switch - content frame not found',0);return false;}
next_item = lms_next_item;
next_item = lms_previous_item;
cont_f.src = 'lp_controller.php?action=content&lp_id='+lms_lp_id+'&item_id='+next_item;
//(4) refresh table of contents
var toc_f = document.getElementById('toc_id');
if(!toc_f){logit_lms('In switch - toc frame not found',0);return false;}
var myrefresh = toc_f.src;
//(5) refresh the progress bar
var prg_f = document.getElementById('nav_id');
if(!prg_f){logit_lms('In switch - navigation frame not found',0);return false;}
var myrefresh = prg_f.src;
//(6) refresh the message box (included in switch_item_details)
addEvent(window,'load',addListeners,false);
|