3 * This is really a little language parser for AICC_SCRIPT
4 * evaluates the expression and returns a boolean answer
5 * see 2.3.2.5.1. Sequencing/Navigation Today - from the SCORM 1.2 spec (CAM).
7 * @param string $prerequisites the aicc_script prerequisites expression
8 * @param array $usertracks the tracked user data of each SCO visited
11 function scorm_eval_prerequisites($prerequisites, $usertracks) {
13 // this is really a little language parser - AICC_SCRIPT is the reference
14 // see 2.3.2.5.1. Sequencing/Navigation Today - from the SCORM 1.2 spec
19 'completed' => 'completed',
21 'incomplete' => 'incomplete',
22 'browsed' => 'browsed',
23 'not attempted' => 'notattempted',
33 // expand the amp entities
34 $prerequisites = preg_replace('/&/', '&', $prerequisites);
35 // find all my parsable tokens
36 $prerequisites = preg_replace('/(&|\||\(|\)|\~)/', '\t$1\t', $prerequisites);
38 $prerequisites = preg_replace('/&/', '&&', $prerequisites);
39 $prerequisites = preg_replace('/\|/', '||', $prerequisites);
40 // now - grab all the tokens
41 $elements = explode('\t', trim($prerequisites));
43 // process each token to build an expression to be evaluated
45 foreach ($elements as $element) {
46 $element = trim($element);
47 if (empty($element)) {
50 if (!preg_match('/^(&&|\|\||\(|\))$/', $element)) {
51 // create each individual expression
52 // search for ~ = <> X*{}
54 // sets like 3*{S34, S36, S37, S39}
55 if (preg_match('/^(\d+)\*\{(.+)\}$/', $element, $matches)) {
56 $repeat = $matches[1];
57 $set = explode(',', $matches[2]);
59 foreach ($set as $setelement) {
60 if (isset($usertracks[$setelement]) &&
61 ($usertracks[$setelement]->status == 'completed' || $usertracks[$setelement]->status == 'passed')) {
65 if ($count >= $repeat) {
72 } else if ($element == '~') {
76 } else if (preg_match('/^(.+)(\=|\<\>)(.+)$/', $element, $matches)) {
77 $element = trim($matches[1]);
78 if (isset($usertracks[$element])) {
79 $value = trim(preg_replace('/(\'|\")/', '', $matches[3]));
80 if (isset($statuses[$value])) {
81 $value = $statuses[$value];
83 if ($matches[2] == '<>') {
88 $element = '(\''.$usertracks[$element]->status.'\' '.$oper.' \''.$value.'\')';
93 // everything else must be an element defined like S45 ...
95 if (isset($usertracks[$element]) &&
96 ($usertracks[$element]->status == 'completed' || $usertracks[$element]->status == 'passed')) {
104 $stack []= ' '.$element.' ';
106 return eval('return '.implode($stack).';');
109 function scorm_get_toc($user,$scorm,$liststyle,$currentorg='',$scoid='',$mode='normal',$attempt='',$play=false, $tocheader=false) {
110 global $CFG, $DB, $PAGE, $OUTPUT;
112 $strexpand = get_string('expcoll','scorm');
114 if ($mode == 'browse') {
115 $modestr = '&mode='.$mode;
118 $result = new stdClass();
119 //$result->toc = "<ul id='s0' class='$liststyle'>\n";
121 $result->toc = '<div id="scorm_layout">';
122 $result->toc .= '<div id="scorm_toc">';
123 $result->toc .= '<div id="scorm_tree">';
125 $result->toc .= '<ul>';
127 $result->prerequisites = true;
131 // Get the current organization infos
133 if (!empty($currentorg)) {
134 if (($organizationtitle = $DB->get_field('scorm_scoes','title', array('scorm'=>$scorm->id,'identifier'=>$currentorg))) != '') {
136 $result->toctitle = "$organizationtitle";
139 $result->toc .= "\t<li>$organizationtitle</li>\n";
141 $tocmenus[] = $organizationtitle;
146 // If not specified retrieve the last attempt number
148 if (empty($attempt)) {
149 $attempt = scorm_get_attempt_count($user->id, $scorm);
151 $result->attemptleft = $scorm->maxattempt == 0 ? 1 : $scorm->maxattempt - $attempt;
152 $conditions['scorm'] = $scorm->id;
153 if ($scoes = scorm_get_scoes($scorm->id, $currentorg)){
155 // Retrieve user tracking data for each learning object
157 $usertracks = array();
158 foreach ($scoes as $sco) {
159 if (!empty($sco->launch)) {
160 if ($usertrack = scorm_get_tracks($sco->id,$user->id,$attempt)) {
161 if ($usertrack->status == '') {
162 $usertrack->status = 'notattempted';
164 $usertracks[$sco->identifier] = $usertrack;
174 $parents[$level]='/';
176 foreach ($scoes as $pos => $sco) {
178 $sco->title = $sco->title;
179 if (!isset($sco->isvisible) || (isset($sco->isvisible) && ($sco->isvisible == 'true'))) {
182 if ($parents[$level] != $sco->parent) {
183 if ($newlevel = array_search($sco->parent,$parents)) {
184 for ($i=0; $i<($level-$newlevel); $i++) {
185 $result->toc .= "\t\t</li></ul></li>\n";
191 while (($i > 0) && ($parents[$level] != $sco->parent)) {
192 $closelist .= "\t\t</li></ul></li>\n";
195 if (($i == 0) && ($sco->parent != $currentorg)) {
197 if (isset($_COOKIE['hide:SCORMitem'.$sco->id])) {
198 $style = ' style="display: none;"';
200 //$result->toc .= "\t\t<li><ul id='s$sublist' class='$liststyle'$style>\n";
201 $result->toc .= "\t\t<ul>\n";
207 $result->toc .= $closelist;
210 $parents[$level] = $sco->parent;
214 $result->toc .= "<li>";
216 if (isset($scoes[$pos+1])) {
217 $nextsco = $scoes[$pos+1];
221 $nextisvisible = false;
222 if (($nextsco !== false) && (!isset($nextsco->isvisible) || (isset($nextsco->isvisible) && ($nextsco->isvisible == 'true')))) {
223 $nextisvisible = true;
225 if ($nextisvisible && ($nextsco !== false) && ($sco->parent != $nextsco->parent) && (($level==0) || (($level>0) && ($nextsco->parent == $sco->identifier)))) {
228 if (isset($_COOKIE['hide:SCORMitem'.$nextsco->id])) {
231 //$result->toc .= '<a href="javascript:expandCollide(\'img'.$sublist.'\',\'s'.$sublist.'\','.$nextsco->id.');"><img id="img'.$sublist.'" src="'.$OUTPUT->pix_url($icon, 'scorm').'" alt="'.$strexpand.'" title="'.$strexpand.'"/></a>';
232 } else if ($isvisible) {
233 //$result->toc .= '<img src="'.$OUTPUT->pix_url('spacer', 'scorm').'" alt="" />';
235 if (empty($sco->title)) {
236 $sco->title = $sco->identifier;
238 if (!empty($sco->launch)) {
243 if (empty($scoid) && ($mode != 'normal')) {
246 if (isset($usertracks[$sco->identifier])) {
247 $usertrack = $usertracks[$sco->identifier];
248 $strstatus = get_string($usertrack->status,'scorm');
249 if ($sco->scormtype == 'sco') {
250 $statusicon = '<img src="'.$OUTPUT->pix_url($usertrack->status, 'scorm').'" alt="'.$strstatus.'" title="'.$strstatus.'" />';
252 $statusicon = '<img src="'.$OUTPUT->pix_url('assetc', 'scorm').'" alt="'.get_string('assetlaunched','scorm').'" title="'.get_string('assetlaunched','scorm').'" />';
255 if (($usertrack->status == 'notattempted') || ($usertrack->status == 'incomplete') || ($usertrack->status == 'browsed')) {
257 if ($play && empty($scoid)) {
261 if ($usertrack->score_raw != '') {
262 $score = '('.get_string('score','scorm').': '.$usertrack->score_raw.')';
264 $strsuspended = get_string('suspended','scorm');
265 if ($incomplete && isset($usertrack->{'cmi.core.exit'}) && ($usertrack->{'cmi.core.exit'} == 'suspend')) {
266 $statusicon = '<img src="'.$OUTPUT->pix_url('suspend', 'scorm').'" alt="'.$strstatus.' - '.$strsuspended.'" title="'.$strstatus.' - '.$strsuspended.'" />';
269 if ($play && empty($scoid)) {
273 if ($sco->scormtype == 'sco') {
274 $statusicon = '<img src="'.$OUTPUT->pix_url('notattempted', 'scorm').'" alt="'.get_string('notattempted','scorm').'" title="'.get_string('notattempted','scorm').'" />';
276 $statusicon = '<img src="'.$OUTPUT->pix_url('asset', 'scorm').'" alt="'.get_string('asset','scorm').'" title="'.get_string('asset','scorm').'" />';
279 if ($sco->id == $scoid) {
283 $shownext = isset($sco->next) ? $sco->next : 0;
284 $showprev = isset($sco->previous) ? $sco->previous : 0;
287 if (($nextid == 0) && (scorm_count_launchable($scorm->id,$currentorg) > 1) && ($nextsco!==false) && (!$findnext)) {
288 if (!empty($sco->launch)) {
292 if (empty($sco->prerequisites) || scorm_eval_prerequisites($sco->prerequisites,$usertracks)) {
293 if ($sco->id == $scoid) {
294 $result->prerequisites = true;
296 // $url = $CFG->wwwroot.'/mod/scorm/player.php?a='.$scorm->id.'&currentorg='.$currentorg.$modestr.'&scoid='.$sco->id;
297 $thisscoidstr = '&scoid='.$sco->id;
298 //$link = $CFG->wwwroot.'/mod/scorm/loadSCO.php?a='.$scorm->id.$thisscoidstr.$modestr;
299 $link = 'a='.$scorm->id.$thisscoidstr.'¤torg='.$currentorg.$modestr.'&attempt='.$attempt;
301 //$result->toc .= $statusicon.' '.$startbold.'<a href="'.$url.'">'.format_string($sco->title).'</a>'.$score.$endbold."\n";
302 //$result->toc .= $startbold.'<a title="'.$link.'">'.$statusicon.' '.format_string($sco->title).' '.$score.'</a>'.$endbold."\n";
304 $result->toc .= '<a title="'.$link.'">'.$statusicon.' '.format_string($sco->title).' '.$score.'</a>';
307 $result->toc .= '<span>'.$statusicon.' '.format_string($sco->title).'</span>';
309 $tocmenus[$sco->id] = scorm_repeater('−',$level) . '>' . format_string($sco->title);
311 if ($sco->id == $scoid) {
312 $result->prerequisites = false;
315 // should be disabled
316 $result->toc .= '<span>'.$statusicon.' '.format_string($sco->title).'</span>';
319 $result->toc .= $statusicon.' '.format_string($sco->title)."\n";
322 if (($nextsco === false) || $nextsco->parent == $sco->parent) {
323 $result->toc .= '</li>';
327 $result->toc .= ' '.format_string($sco->title)."\n";
329 if (($nextsco !== false) && ($nextid == 0) && ($findnext)) {
330 if (!empty($nextsco->launch)) {
331 $nextid = $nextsco->id;
335 for ($i=0;$i<$level;$i++) {
336 $result->toc .= "\t\t</ul></li>\n";
340 // it is possible that scoid is still not set, in this case we dont want an empty object
342 $sco = scorm_get_sco($scoid);
344 $sco->previd = $previd;
345 $sco->nextid = $nextid;
347 $result->incomplete = $incomplete;
349 $result->incomplete = $incomplete;
352 $result->toc .= '</ul>';
356 $result->toc .= '</div></div></div>';
357 $result->toc .= '<div id="scorm_navpanel"></div>';
362 if ($scorm->hidetoc == 0) {
363 $PAGE->requires->data_for_js('scormdata', array(
364 'plusicon' => $OUTPUT->pix_url('plus', 'scorm'),
365 'minusicon' => $OUTPUT->pix_url('minus', 'scorm')));
366 $PAGE->requires->js('/lib/cookies.js');
367 $PAGE->requires->js('/mod/scorm/datamodels/scorm_datamodels.js');
370 $url = new moodle_url('/mod/scorm/player.php?a='.$scorm->id.'¤torg='.$currentorg.$modestr);
371 $result->tocmenu = $OUTPUT->single_select($url, 'scoid', $tocmenus, $sco->id, null, "tocmenu");