bolt->update("sex=$sex, birth_year=$birth_year"); } // The user clicked something on a view page. // Make a record of it, and of the time // function finalize_view($view_id, $action) { global $user; if (!$view_id) return null; $view = BoltView::lookup_id($view_id); if (!$view) { error_page("no view"); } if ($view->user_id != $user->id) { error_page("wrong user"); } if (!$view->end_time) { $now = time(); $view->update("end_time=$now, action=$action"); } return $view; } function default_mode($item) { return $item->is_exercise()?BOLT_MODE_SHOW:BOLT_MODE_LESSON; } // A page is being shown to the user; make a record of it // function create_view($iter, $mode, $prev_view_id) { global $user; global $course; $now = time(); $item = $iter->item; if (!$item) { $item = null; $item->name = '--end--'; } $state = $iter->encode_state(); if ($user->bolt->flags&BOLT_FLAGS_DEBUG) { echo "
Ending state: "; print_r($iter->state); echo "
\n"; } return BoltView::insert("(user_id, course_id, item_name, start_time, mode, state, fraction_done, prev_view_id) values ($user->id, $course->id, '$item->name', $now, $mode, '$state', $iter->frac_done, $prev_view_id)"); } function page_header($title) { echo "$title "; if (function_exists('bolt_header')) bolt_header($title); } function page_footer() { if (function_exists('bolt_footer')) bolt_footer(); echo ""; } // show a page saying the course has been completed // function show_finished_page($view_id, $prev_view_id) { global $course; global $url_args; page_header("Course completed"); if ($course->bossa_app_id) { require_once("../inc/bossa_db.inc"); $app = BossaApp::lookup_id($course->bossa_app_id); echo " Congratulations - you have completed the training for $course->name.

You may now bossa_app_id>do work. "; } else { echo "Congratulations - you have completed this course."; $links[] = ""; $up_link = "Course home page"; show_nav($links, $up_link, $view_id); } page_footer(); } function show_refresh_finished() { page_header("Refresh completed"); echo "Refresh finished"; page_footer(); } function show_nav($links, $up_link, $view_id) { global $course; echo "

"; foreach ($links as $link) { echo ""; } echo "
$link


id>

$up_link "; } // show an item (lesson, exercise, answer page) // function show_item($iter, $view_id, $prev_view_id, $mode, $repeat=null) { global $user; global $course; global $bolt_ex_mode; global $bolt_ex_index; global $bolt_ex_score; global $bolt_query_string; global $refresh; global $url_args; $item = $iter->item; page_header($item->title); $bolt_query_string = $item->query_string; $links = array(); if ($prev_view_id) { $links[] = ""; } $next = ""; if ($item->is_exercise()) { $bolt_ex_mode = $mode; $bolt_ex_index = 0; switch ($mode) { case BOLT_MODE_SHOW: echo "

id> "; if ($refresh) { echo " id> "; } srand($view_id); require($item->filename); if (function_exists('bolt_divide')) bolt_divide(); $next = "
"; break; case BOLT_MODE_ANSWER: require($item->filename); if (function_exists('bolt_divide')) bolt_divide(); $score_pct = number_format($bolt_ex_score*100); echo "Score: $score_pct%"; break; } } else { require_once($item->filename); if (function_exists('bolt_divide')) bolt_divide(); } if ($repeat) { $avg = number_format($repeat->avg_score*100, 0); echo "

Score on this exercise set: $avg%"; if ($repeat->flags & REVIEW) { //echo "

";
            //print_r($repeat);
            //echo "
"; $name = urlencode($repeat->unit->name); $r = "Review, then repeat exercises"; $links[] = $r; } if ($repeat->flags & REPEAT) { $r = "Repeat exercises"; $links[] = $r; } if ($repeat->flags & NEXT) { $links[] = $next; } } else { $links[] = $next; } $up_link = "Course home page"; show_nav($links, $up_link, $view_id); page_footer(); if ($refresh) { $refresh->update("last_view_id=$view_id"); } else { $e = new BoltEnrollment(); $e->user_id = $user->id; $e->course_id = $course->id; $e->update("last_view_id=$view_id"); } } // Show the student the results of an old exercise; no navigation items // function show_answer_page($iter, $score) { global $bolt_ex_mode; global $bolt_ex_index; global $bolt_query_string; $bolt_ex_mode = BOLT_MODE_ANSWER; $bolt_ex_index = 0; $item = $iter->item; page_header($item->title); $bolt_query_string = $item->query_string; require_once($item->filename); if (function_exists('bolt_divide')) bolt_divide(); $score_pct = number_format($score*100); echo "Score: $score_pct%"; page_footer(); } function start_course() { global $user; global $course; global $course_doc; BoltEnrollment::delete($user->id, $course->id); $iter = new BoltIter($course_doc); $iter->at(); $now = time(); $mode = default_mode($iter->item); $view_id = create_view($iter, $mode, 0); BoltEnrollment::insert("(create_time, user_id, course_id, last_view_id) values ($now, $user->id, $course->id, $view_id)"); show_item($iter, $view_id, 0, $mode); } function start_repeat() { global $course_doc; global $refresh; $xset_result = BoltXsetResult::lookup_id($refresh->xset_result_id); if (!$xset_result) error_page("result not found"); $view = BoltView::lookup_id($xset_result->view_id); if (!$view) error_page("view not found"); $iter = new BoltIter($course_doc); $iter->decode_state($view->state); $iter->at(); $xset = $iter->xset; if (!$xset || $xset->name != $xset_result->name) { error_page("missing exercise set"); } $xset->start_repeat($iter); $iter->at(); $mode = default_mode($iter->item); $view_id = create_view($iter, $mode, 0); show_item($iter, $view_id, 0, $mode); } $user = get_logged_in_user(); BoltUser::lookup($user); $course_id = get_int('course_id'); $refresh_id = get_int('refresh_id', true); $refresh = null; $url_args = "course_id=$course_id"; if ($refresh_id) { $refresh = BoltRefreshRec::lookup_id($refresh_id); if (!$refresh) error_page("No such refresh"); if ($refresh->user_id != $user->id) error_page("Wrong user"); if ($refresh->course_id != $course_id) error_page("Wrong course"); $url_args .= "&refresh_id=$refresh_id"; } $course = BoltCourse::lookup_id($course_id); if (!$course) { error_page("no such course"); } $view_id = get_int('view_id', true); $action = get_str('action', true); $course_doc = require_once($course->doc_file); switch ($action) { case 'start': if (info_incomplete($user)) { request_info($user, $course); exit(); } if ($refresh) { start_repeat(); exit(); } $e = BoltEnrollment::lookup($user->id, $course_id); if ($e) { page_header("Confirm restart"); echo "You are already enrolled in $course->name.

Are you sure you want to start over from the beginning?

"; show_button( "bolt_sched.php?action=start_confirm&$url_args", "Yes", "Start this course from the beginning" ); show_button( "bolt_sched.php?action=resume&$url_args", "Resume", "Resume course from current position" ); page_footer(); exit(); } // fall through case 'start_confirm': start_course(); break; case 'update_info': update_info(); start_course(); break; case 'prev': $view = finalize_view($view_id, BOLT_ACTION_PREV); if ($view->prev_view_id) { $view = BoltView::lookup_id($view->prev_view_id); $iter = new BoltIter($course_doc); $iter->decode_state($view->state); $iter->at(); $mode = $view->mode; if ($mode == BOLT_MODE_ANSWER) { $v2 = BoltView::lookup_id($view->prev_view_id); $result = BoltResult::lookup_id($v2->result_id); $bolt_ex_query_string = $result->response; } $view_id = create_view($iter, $mode, $view->prev_view_id); show_item($iter, $view_id, $view->prev_view_id, $mode); } else { error_page("At start of course"); } break; case 'next': // "next" button in lesson or exercise answer page $view = finalize_view($view_id, BOLT_ACTION_NEXT); $iter = new BoltIter($course_doc); $iter->decode_state($view->state); if ($user->bolt->flags&BOLT_FLAGS_DEBUG) { echo "

Initial state: "; print_r($iter->state); echo "
\n"; } $iter->next(); if ($user->bolt->flags&BOLT_FLAGS_DEBUG) { echo "
Item: "; print_r($iter->item); echo "
\n"; } if ($iter->item) { $state = $iter->encode_state(); $mode = default_mode($iter->item); $view_id = create_view($iter, $mode, $view->id); show_item($iter, $view_id, $view->id, $mode); } else { if ($refresh) { show_refresh_finished(); } else { $iter->frac_done = 1; $fin_view_id = create_view($iter, BOLT_MODE_FINISHED, $view_id); $e = new BoltEnrollment(); $e->user_id = $user->id; $e->course_id = $course->id; $e->update("last_view_id=$fin_view_id"); show_finished_page($fin_view_id, $view->id); } } break; case 'answer': // submit answer in exercise $view = finalize_view($view_id, BOLT_ACTION_SUBMIT); $iter = new BoltIter($course_doc); $iter->decode_state($view->state); if ($user->bolt->flags&BOLT_FLAGS_DEBUG) { echo "
Initial state:"; print_r($iter->state); echo "
\n"; } $iter->at(); if ($user->bolt->flags&BOLT_FLAGS_DEBUG) { echo "
Item: "; print_r($iter->item); echo "
\n"; } $item = $iter->item; if (!$item->is_exercise()) { print_r($item); error_page("expected an exercise"); } if ($view->item_name != $item->name) { error_page("unexpected name"); } // compute the score $bolt_ex_query_string = $_SERVER['QUERY_STRING']; $bolt_ex_mode = BOLT_MODE_SCORE; $bolt_ex_index = 0; $bolt_ex_score = 0; $bolt_query_string = $item->query_string; srand($view_id); ob_start(); // turn on output buffering require($item->filename); ob_end_clean(); $bolt_ex_score /= $bolt_ex_index; // make a record of the result $qs = BoltDb::escape_string($_SERVER['QUERY_STRING']); $result_id = BoltResult::insert( "(view_id, item_name, score, response) values ($view->id, '$view->item_name', $bolt_ex_score, '$qs')" ); $view->update("result_id=$result_id"); // If this is part of an exercise set, call its callback function // $repeat = null; if ($iter->xset) { $is_last = $iter->xset->xset_callback($iter, $bolt_ex_score, $view->id, $avg_score, $repeat); if ($repeat) $repeat->avg_score = $avg_score; if ($is_last) { // if the exercise set if finished, make or update DB records // $now = time(); $xset = $iter->xset; $id = BoltXsetResult::insert("(create_time, user_id, course_id, name, score, view_id) values ($now, $user->id, $course->id, '$xset->name', $avg_score, $view_id)"); $due_time = $now + 100000; $refresh = BoltRefreshRec::lookup( "user_id=$user->id and course_id=$course->id and name='$xset->name'" ); if ($refresh) { $refresh->update("create_time=$now, xset_result_id=$id, due_time=$due_time"); } else { BoltRefreshRec::insert( "(user_id, course_id, name, create_time, xset_result_id, due_time) values ($user->id, $course->id, '$xset->name', $now, $id, $due_time)" ); } } } // show the answer page srand($view_id); $view_id = create_view($iter, BOLT_MODE_ANSWER, $view->id); show_item($iter, $view_id, $view->id, BOLT_MODE_ANSWER, $repeat); break; case 'answer_page': $view = BoltView::lookup_id($view_id); $iter = new BoltIter($course_doc); $iter->decode_state($view->state); $iter->at(); if ($iter->item->name != $view->item_name) { error_page("Exercise no longer exists in course"); } $result = BoltResult::lookup_id($view->result_id); srand($view_id); $bolt_ex_query_string = $result->response; show_answer_page($iter, $result->score); break; case 'course_home': $view = finalize_view($view_id, BOLT_ACTION_COURSE_HOME); Header("Location: bolt.php"); break; case 'review': $view = finalize_view($view_id, BOLT_ACTION_REVIEW); $iter = new BoltIter($course_doc); $iter->decode_state($view->state); $iter->at(); if (!$iter->xset) { echo "NO XSET"; exit; } $xset = $iter->xset; $unit_name = get_str('unit_name'); $found = $xset->start_review($iter, $unit_name); if (!$found) { echo "REVIEW UNIT MISSING"; exit; } $iter->at(); $mode = default_mode($iter->item); $view_id = create_view($iter, $mode, $view->id); show_item($iter, $view_id, $view->id, $mode); break; case 'repeat': $view = finalize_view($view_id, BOLT_ACTION_REPEAT); $iter = new BoltIter($course_doc); $iter->decode_state($view->state); $iter->at(); if (!$iter->xset) { echo "NO XSET"; exit; } $xset = $iter->xset; $xset->start_repeat($iter); $iter->at(); $mode = default_mode($iter->item); $view_id = create_view($iter, $mode, $view->id); show_item($iter, $view_id, $view->id, $mode); break; case 'refresh': $refresh_id = get_int('refresh_id'); $refresh = BoltRefreshRec::lookup_id($refresh_id); if (!$refresh) error_page("refresh not found"); // fall through case 'resume': if ($refresh) { if ($refresh->last_view_id) { $view = BoltView::lookup_id($refresh->last_view_id); } else { start_repeat(); exit(); } } else { $view = null; $e = BoltEnrollment::lookup($user->id, $course_id); if ($e) { $view = BoltView::lookup_id($e->last_view_id); } if (!$view) { start_course(); break; } } if ($view->mode == BOLT_MODE_FINISHED) { show_finished_page($view->id, $view->prev_view_id); break; } $iter = new BoltIter($course_doc); $iter->decode_state($view->state); $iter->at(); $mode = $view->mode; if ($view->item_name == $iter->item->name && ($mode == BOLT_MODE_ANSWER)) { // if we're returning to an answer page, // we need to look up the user's responses and the score. // $view_orig = BoltView::lookup_id($view->prev_view_id); $result = BoltResult::lookup_id($view_orig->result_id); srand($view_orig->id); $bolt_ex_query_string = $result->response; $bolt_ex_score = $result->score; $bolt_ex_index = 0; $view_id = create_view($iter, $mode, $view_orig->id); show_item($iter, $view_id, $view_orig->id, $mode); } else { $view_id = create_view($iter, $mode, $view->id); show_item($iter, $view_id, $view->id, $mode); } break; case 'question': $view = finalize_view($view_id, BOLT_ACTION_QUESTION); $now = time(); $question = BoltDb::escape_string(get_str('question')); BoltQuestion::insert("(create_time, user_id, course_id, name, question, state) values ($now, $user->id, $course->id, '$view->item_name', '$question', 0)"); page_header("Question recorded"); echo " Thanks; we have recorded your question. Questions help us improve this course. We aren't able to individually respond to all questions. Responses are delivered as private messages.

Resume course "; page_footer(); break; default: error_page("unknown action: $action"); } ?>