*/ class learnpathItem { const DEBUG = 0; // Logging parameter. public $iId; public $attempt_id; // Also called "objectives" SCORM-wise. public $audio; // The path to an audio file (stored in document/audio/). public $children = []; // Contains the ids of children items. public $condition; // If this item has a special condition embedded. public $current_score; public $current_start_time; public $current_stop_time; public $current_data = ''; public $db_id; public $db_item_view_id = ''; public $description = ''; public $file; /** * At the moment, interactions are just an array of arrays with a structure * of 8 text fields: id(0), type(1), time(2), weighting(3), * correct_responses(4), student_response(5), result(6), latency(7). */ public $interactions = []; public $interactions_count = 0; public $objectives = []; public $objectives_count = 0; public $launch_data = ''; public $lesson_location = ''; public $level = 0; public $core_exit = ''; //var $location; // Only set this for SCORM? public $lp_id; public $max_score; public $mastery_score; public $min_score; public $max_time_allowed = ''; public $name; public $next; public $parent; public $path; // In some cases the exo_id = exercise_id in courseDb exercices table. public $possible_status = [ 'not attempted', 'incomplete', 'completed', 'passed', 'failed', 'browsed', ]; public $prereq_string = ''; public $prereq_alert = ''; public $prereqs = []; public $previous; public $prevent_reinit = 1; // 0 = multiple attempts 1 = one attempt public $seriousgame_mode; public $ref; public $save_on_close = true; public $search_did = null; public $status; public $title; /** * Type attribute can contain one of * link|student_publication|dir|quiz|document|forum|thread. */ public $type; public $view_id; public $oldTotalTime; //var used if absolute session time mode is used private $last_scorm_session_time = 0; private $prerequisiteMaxScore; private $prerequisiteMinScore; /** * Prepares the learning path item for later launch. * Don't forget to use set_lp_view() if applicable after creating the item. * Setting an lp_view will finalise the item_view data collection. * * @param int $id Learning path item ID * @param int $user_id User ID * @param int $course_id Course int id * @param array|null $item_content An array with the contents of the item */ public function __construct( $id, $user_id = 0, $course_id = 0, $item_content = null ) { $items_table = Database::get_course_table(TABLE_LP_ITEM); // Get items table. if (!isset($user_id)) { $user_id = api_get_user_id(); } $id = (int) $id; if (empty($item_content)) { if (empty($course_id)) { $course_id = api_get_course_int_id(); } else { $course_id = (int) $course_id; } $sql = "SELECT * FROM $items_table WHERE iid = $id"; $res = Database::query($sql); if (Database::num_rows($res) < 1) { $this->error = 'Could not find given learnpath item in learnpath_item table'; } $row = Database::fetch_array($res); } else { $row = $item_content; } $this->lp_id = $row['lp_id']; $this->iId = $row['iid']; $this->max_score = $row['max_score']; $this->min_score = $row['min_score']; $this->name = $row['title']; $this->type = $row['item_type']; $this->ref = $row['ref']; $this->title = $row['title']; $this->description = $row['description']; $this->path = $row['path']; $this->mastery_score = $row['mastery_score']; $this->parent = $row['parent_item_id']; $this->next = $row['next_item_id']; $this->previous = $row['previous_item_id']; $this->display_order = $row['display_order']; $this->prereq_string = $row['prerequisite']; $this->max_time_allowed = $row['max_time_allowed']; $this->setPrerequisiteMaxScore($row['prerequisite_max_score']); $this->setPrerequisiteMinScore($row['prerequisite_min_score']); $this->oldTotalTime = 0; if (isset($row['launch_data'])) { $this->launch_data = $row['launch_data']; } $this->save_on_close = true; $this->db_id = $id; // Load children list if (!empty($this->lp_id)) { $sql = "SELECT iid FROM $items_table WHERE c_id = $course_id AND lp_id = ".$this->lp_id." AND parent_item_id = $id"; $res = Database::query($sql); if (Database::num_rows($res) > 0) { while ($row = Database::fetch_assoc($res)) { $this->children[] = $row['iid']; } } // Get search_did. if ('true' == api_get_setting('search_enabled')) { $tbl_se_ref = Database::get_main_table(TABLE_MAIN_SEARCH_ENGINE_REF); $sql = 'SELECT * FROM %s WHERE course_code=\'%s\' AND tool_id=\'%s\' AND ref_id_high_level=%s AND ref_id_second_level=%d LIMIT 1'; // TODO: Verify if it's possible to assume the actual course instead // of getting it from db. $sql = sprintf( $sql, $tbl_se_ref, api_get_course_id(), TOOL_LEARNPATH, $this->lp_id, $id ); $res = Database::query($sql); if (Database::num_rows($res) > 0) { $se_ref = Database::fetch_array($res); $this->search_did = (int) $se_ref['search_did']; } } } $this->seriousgame_mode = 0; //$this->audio = $row['audio']; } /** * Adds a child to the current item. * * @param int $item The child item ID */ public function add_child($item) { if (!empty($item)) { // Do not check in DB as we expect the call to come from the // learnpath class which should be aware of any fake. $this->children[] = $item; } } /** * Adds an interaction to the current item. * * @param int $index Index (order ID) of the interaction inside this item * @param array $params Array of parameters: * id(0), type(1), time(2), weighting(3), correct_responses(4), * student_response(5), result(6), latency(7) */ public function add_interaction($index, $params) { $this->interactions[$index] = $params; // Take the current maximum index to generate the interactions_count. if (($index + 1) > $this->interactions_count) { $this->interactions_count = $index + 1; } } /** * Adds an objective to the current item. * * @param array Array of parameters: * id(0), status(1), score_raw(2), score_max(3), score_min(4) */ public function add_objective($index, $params) { if (empty($params[0])) { return null; } $this->objectives[$index] = $params; // Take the current maximum index to generate the objectives_count. if ((count($this->objectives) + 1) > $this->objectives_count) { $this->objectives_count = (count($this->objectives) + 1); } } /** * Closes/stops the item viewing. Finalises runtime values. * If required, save to DB. * * @return bool True on success, false otherwise */ public function close() { $this->current_stop_time = time(); $type = $this->get_type(); if ('sco' != $type) { if (TOOL_QUIZ == $type || TOOL_HOTPOTATOES == $type) { $this->get_status( true, true ); // Update status (second option forces the update). } else { $this->status = $this->possible_status[2]; } } if ($this->save_on_close) { $this->save(); } return true; } /** * Deletes all traces of this item in the database. * * @return bool true. Doesn't check for errors yet. */ public function delete() { $lp_item_view = Database::get_course_table(TABLE_LP_ITEM_VIEW); $lp_item = Database::get_course_table(TABLE_LP_ITEM); $course_id = api_get_course_int_id(); $sql = "DELETE FROM $lp_item_view WHERE c_id = $course_id AND lp_item_id = ".$this->db_id; Database::query($sql); $sql = "SELECT * FROM $lp_item WHERE iid = ".$this->db_id; $res_sel = Database::query($sql); if (Database::num_rows($res_sel) < 1) { return false; } $sql = "DELETE FROM $lp_item WHERE iid = ".$this->db_id; Database::query($sql); if ('true' == api_get_setting('search_enabled')) { if (!is_null($this->search_did)) { $di = new ChamiloIndexer(); $di->remove_document($this->search_did); } } return true; } /** * Drops a child from the children array. * * @param string $item index of child item to drop */ public function drop_child($item) { if (!empty($item)) { foreach ($this->children as $index => $child) { if ($child == $item) { $this->children[$index] = null; } } } } /** * Gets the current attempt_id for this user on this item. * * @return int attempt_id for this item view by this user or 1 if none defined */ public function get_attempt_id() { $res = 1; if (!empty($this->attempt_id)) { $res = (int) $this->attempt_id; } return $res; } /** * Gets a list of the item's children. * * @return array Array of children items IDs */ public function get_children() { $list = []; foreach ($this->children as $child) { if (!empty($child)) { $list[] = $child; } } return $list; } /** * Gets the core_exit value from the database. */ public function get_core_exit() { return $this->core_exit; } /** * Gets the credit information (rather scorm-stuff) based on current status * and reinit autorization. Credit tells the sco(content) if Chamilo will * record the data it is sent (credit) or not (no-credit). * * @return string 'credit' or 'no-credit'. Defaults to 'credit' * Because if we don't know enough about this item, it's probably because * it was never used before. */ public function get_credit() { if (self::DEBUG > 1) { error_log('learnpathItem::get_credit()', 0); } $credit = 'credit'; // Now check the value of prevent_reinit (if it's 0, return credit as // the default was). // If prevent_reinit == 1 (or more). if (0 != $this->get_prevent_reinit()) { // If status is not attempted or incomplete, credit anyway. // Otherwise: // Check the status in the database rather than in the object, as // checking in the object would always return "no-credit" when we // want to set it to completed. $status = $this->get_status(true); if (self::DEBUG > 2) { error_log( 'learnpathItem::get_credit() - get_prevent_reinit!=0 and '. 'status is '.$status, 0 ); } //0=not attempted - 1 = incomplete if ($status != $this->possible_status[0] && $status != $this->possible_status[1] ) { $credit = 'no-credit'; } } if (self::DEBUG > 1) { error_log("learnpathItem::get_credit() returns: $credit"); } return $credit; } /** * Gets the current start time property. * * @return int Current start time, or current time if none */ public function get_current_start_time() { if (empty($this->current_start_time)) { return time(); } return $this->current_start_time; } /** * Gets the item's description. * * @return string Description */ public function get_description() { if (empty($this->description)) { return ''; } return $this->description; } /** * Gets the file path from the course's root directory, no matter what * tool it is from. * * @param string $path_to_scorm_dir * * @return string The file path, or an empty string if there is no file * attached, or '-1' if the file must be replaced by an error page */ public function get_file_path($path_to_scorm_dir = '') { $course_id = api_get_course_int_id(); $path = $this->get_path(); $type = $this->get_type(); if (empty($path)) { if ('dir' == $type) { return ''; } else { return '-1'; } } elseif ($path == strval(intval($path))) { // The path is numeric, so it is a reference to a Chamilo object. switch ($type) { case 'dir': return ''; case TOOL_DOCUMENT: $table_doc = Database::get_course_table(TABLE_DOCUMENT); $sql = 'SELECT path FROM '.$table_doc.' WHERE c_id = '.$course_id.' AND iid = '.$path; $res = Database::query($sql); $row = Database::fetch_array($res); $real_path = 'document'.$row['path']; return $real_path; case TOOL_STUDENTPUBLICATION: case TOOL_QUIZ: case TOOL_FORUM: case TOOL_THREAD: case TOOL_LINK: default: return '-1'; } } else { if (!empty($path_to_scorm_dir)) { $path = $path_to_scorm_dir.$path; } return $path; } } /** * Gets the DB ID. * * @return int Database ID for the current item */ public function get_id() { if (!empty($this->db_id)) { return $this->db_id; } // TODO: Check this return value is valid for children classes (SCORM?). return 0; } /** * Loads the interactions into the item object, from the database. * If object interactions exist, they will be overwritten by this function, * using the database elements only. */ public function load_interactions() { $this->interactions = []; $course_id = api_get_course_int_id(); $tbl = Database::get_course_table(TABLE_LP_ITEM_VIEW); $sql = "SELECT id FROM $tbl WHERE c_id = $course_id AND lp_item_id = ".$this->db_id." AND lp_view_id = ".$this->view_id." AND view_count = ".$this->get_view_count(); $res = Database::query($sql); if (Database::num_rows($res) > 0) { $row = Database::fetch_array($res); $lp_iv_id = $row[0]; $iva_table = Database::get_course_table(TABLE_LP_IV_INTERACTION); $sql = "SELECT * FROM $iva_table WHERE c_id = $course_id AND lp_iv_id = $lp_iv_id "; $res_sql = Database::query($sql); while ($row = Database::fetch_array($res_sql)) { $this->interactions[$row['interaction_id']] = [ $row['interaction_id'], $row['interaction_type'], $row['weighting'], $row['completion_time'], $row['correct_responses'], $row['student_responses'], $row['result'], $row['latency'], ]; } } } /** * Gets the current count of interactions recorded in the database. * * @param bool $checkdb Whether to count from database or not (defaults to no) * * @return int The current number of interactions recorder */ public function get_interactions_count($checkdb = false) { $return = 0; if (api_is_invitee()) { // If the user is an invitee, we consider there's no interaction return 0; } $course_id = api_get_course_int_id(); if ($checkdb) { $tbl = Database::get_course_table(TABLE_LP_ITEM_VIEW); $sql = "SELECT iid FROM $tbl WHERE c_id = $course_id AND lp_item_id = ".$this->db_id." AND lp_view_id = ".$this->view_id." AND view_count = ".$this->get_attempt_id(); $res = Database::query($sql); if (Database::num_rows($res) > 0) { $row = Database::fetch_array($res); $lp_iv_id = $row[0]; $iva_table = Database::get_course_table( TABLE_LP_IV_INTERACTION ); $sql = "SELECT count(id) as mycount FROM $iva_table WHERE c_id = $course_id AND lp_iv_id = $lp_iv_id "; $res_sql = Database::query($sql); if (Database::num_rows($res_sql) > 0) { $row = Database::fetch_array($res_sql); $return = $row['mycount']; } } } else { if (!empty($this->interactions_count)) { $return = $this->interactions_count; } } return $return; } /** * Gets the JavaScript array content to fill the interactions array. * * @param bool $checkdb Whether to check directly into the database (default no) * * @return string An empty string if no interaction, a JS array definition otherwise */ public function get_interactions_js_array($checkdb = false) { $return = ''; if ($checkdb) { $this->load_interactions(true); } foreach ($this->interactions as $id => $in) { $return .= "[ '$id', '".$in[1]."', '".$in[2]."', '".$in[3]."', '".$in[4]."', '".$in[5]."', '".$in[6]."', '".$in[7]."'],"; } if (!empty($return)) { $return = substr($return, 0, -1); } return $return; } /** * Gets the current count of objectives recorded in the database. * * @return int The current number of objectives recorder */ public function get_objectives_count() { $res = 0; if (!empty($this->objectives_count)) { $res = $this->objectives_count; } return $res; } /** * Gets the launch_data field found in imsmanifests (this is SCORM- or * AICC-related, really). * * @return string Launch data as found in imsmanifest and stored in * Chamilo (read only). Defaults to ''. */ public function get_launch_data() { if (!empty($this->launch_data)) { return str_replace( ["\r", "\n", "'"], ['\r', '\n', "\\'"], $this->launch_data ); } return ''; } /** * Gets the lesson location. * * @return string lesson location as recorded by the SCORM and AICC * elements. Defaults to '' */ public function get_lesson_location() { if (!empty($this->lesson_location)) { return str_replace( ["\r", "\n", "'"], ['\r', '\n', "\\'"], $this->lesson_location ); } return ''; } /** * Gets the lesson_mode (scorm feature, but might be used by aicc as well * as chamilo paths). * * The "browse" mode is not supported yet (because there is no such way of * seeing a sco in Chamilo) * * @return string 'browse','normal' or 'review'. Defaults to 'normal' */ public function get_lesson_mode() { $mode = 'normal'; if (0 != $this->get_prevent_reinit()) { // If prevent_reinit == 0 $my_status = $this->get_status(); if ($my_status != $this->possible_status[0] && $my_status != $this->possible_status[1]) { $mode = 'review'; } } return $mode; } /** * Gets the depth level. * * @return int Level. Defaults to 0 */ public function get_level() { if (empty($this->level)) { return 0; } return $this->level; } /** * Gets the mastery score. */ public function get_mastery_score() { if (isset($this->mastery_score)) { return $this->mastery_score; } return -1; } /** * Gets the maximum (score). * * @return int Maximum score. Defaults to 100 if nothing else is defined */ public function get_max() { if ('sco' == $this->type) { if (isset($this->view_max_score) && !empty($this->view_max_score) && $this->view_max_score > 0 ) { return $this->view_max_score; } elseif (isset($this->view_max_score) && '' === $this->view_max_score ) { return $this->view_max_score; } else { if (!empty($this->max_score)) { return $this->max_score; } else { return 100; } } } else { if (!empty($this->max_score)) { return $this->max_score; } else { return 100; } } } /** * Gets the maximum time allowed for this user in this attempt on this item. * * @return string Time string in SCORM format * (HH:MM:SS or HH:MM:SS.SS or HHHH:MM:SS.SS) */ public function get_max_time_allowed() { if (!empty($this->max_time_allowed)) { return $this->max_time_allowed; } return ''; } /** * Gets the minimum (score). * * @return int Minimum score. Defaults to 0 */ public function get_min() { if (!empty($this->min_score)) { return $this->min_score; } return 0; } /** * Gets the parent ID. * * @return int Parent ID. Defaults to null */ public function get_parent() { if (!empty($this->parent)) { return $this->parent; } // TODO: Check this return value is valid for children classes (SCORM?). return null; } /** * Gets the path attribute. * * @return string Path. Defaults to '' */ public function get_path() { if (empty($this->path)) { return ''; } return $this->path; } /** * Gets the prerequisites string. * * @return string empty string or prerequisites string if defined */ public function get_prereq_string() { if (!empty($this->prereq_string)) { return $this->prereq_string; } return ''; } /** * Gets the prevent_reinit attribute value (and sets it if not set already). * * @return int 1 or 0 (defaults to 1) */ public function get_prevent_reinit() { if (self::DEBUG > 2) { error_log('learnpathItem::get_prevent_reinit()', 0); } if (!isset($this->prevent_reinit)) { if (!empty($this->lp_id)) { $table = Database::get_course_table(TABLE_LP_MAIN); $sql = "SELECT prevent_reinit FROM $table WHERE iid = ".$this->lp_id; $res = Database::query($sql); if (Database::num_rows($res) < 1) { $this->error = "Could not find parent learnpath in lp table"; if (self::DEBUG > 2) { error_log( 'New LP - End of learnpathItem::get_prevent_reinit() - Returning false', 0 ); } return false; } else { $row = Database::fetch_array($res); $this->prevent_reinit = $row['prevent_reinit']; } } else { // Prevent reinit is always 1 by default - see learnpath.class.php $this->prevent_reinit = 1; } } if (self::DEBUG > 2) { error_log( 'New LP - End of learnpathItem::get_prevent_reinit() - Returned '.$this->prevent_reinit, 0 ); } return $this->prevent_reinit; } /** * Returns 1 if seriousgame_mode is activated, 0 otherwise. * * @return int (0 or 1) * * @deprecated seriousgame_mode seems not to be used * * @author ndiechburg */ public function get_seriousgame_mode() { if (!isset($this->seriousgame_mode)) { if (!empty($this->lp_id)) { $table = Database::get_course_table(TABLE_LP_MAIN); $sql = "SELECT seriousgame_mode FROM $table WHERE iid = ".$this->lp_id; $res = Database::query($sql); if (Database::num_rows($res) < 1) { $this->error = 'Could not find parent learnpath in learnpath table'; return false; } else { $row = Database::fetch_array($res); $this->seriousgame_mode = isset($row['seriousgame_mode']) ? $row['seriousgame_mode'] : 0; } } else { $this->seriousgame_mode = 0; //SeriousGame mode is always off by default } } return $this->seriousgame_mode; } /** * Gets the item's reference column. * * @return string The item's reference field (generally used for SCORM identifiers) */ public function get_ref() { return $this->ref; } /** * Gets the list of included resources as a list of absolute or relative * paths of resources included in the current item. This allows for a * better SCORM export. The list will generally include pictures, flash * objects, java applets, or any other stuff included in the source of the * current item. The current item is expected to be an HTML file. If it * is not, then the function will return and empty list. * * @param string $type (one of the Chamilo tools) - optional (otherwise takes the current item's type) * @param string $abs_path absolute file path - optional (otherwise takes the current item's path) * @param int $recursivity level of recursivity we're in * * @return array List of file paths. * An additional field containing 'local' or 'remote' helps determine if * the file should be copied into the zip or just linked */ public function get_resources_from_source( $type = null, $abs_path = null, $recursivity = 1 ) { $max = 5; if ($recursivity > $max) { return []; } $type = empty($type) ? $this->get_type() : $type; if (!isset($abs_path)) { $path = $this->get_file_path(); $abs_path = api_get_path(SYS_COURSE_PATH).api_get_course_path().'/'.$path; } $files_list = []; switch ($type) { case TOOL_DOCUMENT: case TOOL_QUIZ: case 'sco': // Get the document and, if HTML, open it. if (!is_file($abs_path)) { // The file could not be found. return false; } // for now, read the whole file in one go (that's gonna be // a problem when the file is too big). $info = pathinfo($abs_path); $ext = $info['extension']; switch (strtolower($ext)) { case 'html': case 'htm': case 'shtml': case 'css': $wantedAttributes = [ 'src', 'url', '@import', 'href', 'value', ]; // Parse it for included resources. $fileContent = file_get_contents($abs_path); // Get an array of attributes from the HTML source. $attributes = DocumentManager::parse_HTML_attributes( $fileContent, $wantedAttributes ); // Look at 'src' attributes in this file foreach ($wantedAttributes as $attr) { if (isset($attributes[$attr])) { // Find which kind of path these are (local or remote). $sources = $attributes[$attr]; foreach ($sources as $source) { // Skip what is obviously not a resource. if (strpos($source, "+this.")) { continue; } // javascript code - will still work unaltered. if (false === strpos($source, '.')) { continue; } // No dot, should not be an external file anyway. if (strpos($source, 'mailto:')) { continue; } // mailto link. if (strpos($source, ';') && !strpos($source, '&') ) { continue; } // Avoid code - that should help. if ('value' == $attr) { if (strpos($source, 'mp3file')) { $files_list[] = [ substr( $source, 0, strpos( $source, '.swf' ) + 4 ), 'local', 'abs', ]; $mp3file = substr( $source, strpos( $source, 'mp3file=' ) + 8 ); if ('/' == substr($mp3file, 0, 1)) { $files_list[] = [ $mp3file, 'local', 'abs', ]; } else { $files_list[] = [ $mp3file, 'local', 'rel', ]; } } elseif (0 === strpos($source, 'flv=')) { $source = substr($source, 4); if (strpos($source, '&') > 0) { $source = substr( $source, 0, strpos($source, '&') ); } if (strpos($source, '://') > 0) { if (false !== strpos($source, api_get_path(WEB_PATH))) { // We found the current portal url. $files_list[] = [ $source, 'local', 'url', ]; } else { // We didn't find any trace of current portal. $files_list[] = [ $source, 'remote', 'url', ]; } } else { $files_list[] = [ $source, 'local', 'abs', ]; } continue; // Skipping anything else to avoid two entries //(while the others can have sub-files in their url, flv's can't). } } if (strpos($source, '://') > 0) { // Cut at '?' in a URL with params. if (strpos($source, '?') > 0) { $second_part = substr( $source, strpos($source, '?') ); if (strpos($second_part, '://') > 0) { // If the second part of the url contains a url too, // treat the second one before cutting. $pos1 = strpos( $second_part, '=' ); $pos2 = strpos( $second_part, '&' ); $second_part = substr( $second_part, $pos1 + 1, $pos2 - ($pos1 + 1) ); if (false !== strpos($second_part, api_get_path(WEB_PATH))) { // We found the current portal url. $files_list[] = [ $second_part, 'local', 'url', ]; $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $second_part, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } else { // We didn't find any trace of current portal. $files_list[] = [ $second_part, 'remote', 'url', ]; } } elseif (strpos($second_part, '=') > 0) { if ('/' === substr($second_part, 0, 1)) { // Link starts with a /, // making it absolute (relative to DocumentRoot). $files_list[] = [ $second_part, 'local', 'abs', ]; $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $second_part, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } elseif (0 === strstr($second_part, '..')) { // Link is relative but going back in the hierarchy. $files_list[] = [ $second_part, 'local', 'rel', ]; $dir = dirname( $abs_path ); $new_abs_path = realpath( $dir.'/'.$second_part ); $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $new_abs_path, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } else { // No starting '/', making it relative to current document's path. if ('./' == substr($second_part, 0, 2)) { $second_part = substr( $second_part, 2 ); } $files_list[] = [ $second_part, 'local', 'rel', ]; $dir = dirname( $abs_path ); $new_abs_path = realpath( $dir.'/'.$second_part ); $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $new_abs_path, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } } // Leave that second part behind now. $source = substr( $source, 0, strpos($source, '?') ); if (strpos($source, '://') > 0) { if (false !== strpos($source, api_get_path(WEB_PATH))) { // We found the current portal url. $files_list[] = [ $source, 'local', 'url', ]; $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $source, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } else { // We didn't find any trace of current portal. $files_list[] = [ $source, 'remote', 'url', ]; } } else { // No protocol found, make link local. if ('/' === substr($source, 0, 1)) { // Link starts with a /, making it absolute (relative to DocumentRoot). $files_list[] = [ $source, 'local', 'abs', ]; $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $source, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } elseif (0 === strstr($source, '..')) { // Link is relative but going back in the hierarchy. $files_list[] = [ $source, 'local', 'rel', ]; $dir = dirname( $abs_path ); $new_abs_path = realpath( $dir.'/'.$source ); $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $new_abs_path, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } else { // No starting '/', making it relative to current document's path. if ('./' == substr($source, 0, 2)) { $source = substr( $source, 2 ); } $files_list[] = [ $source, 'local', 'rel', ]; $dir = dirname( $abs_path ); $new_abs_path = realpath( $dir.'/'.$source ); $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $new_abs_path, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } } } // Found some protocol there. if (false !== strpos($source, api_get_path(WEB_PATH))) { // We found the current portal url. $files_list[] = [ $source, 'local', 'url', ]; $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $source, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } else { // We didn't find any trace of current portal. $files_list[] = [ $source, 'remote', 'url', ]; } } else { // No protocol found, make link local. if ('/' === substr($source, 0, 1)) { // Link starts with a /, making it absolute (relative to DocumentRoot). $files_list[] = [ $source, 'local', 'abs', ]; $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $source, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } elseif (0 === strstr($source, '..')) { // Link is relative but going back in the hierarchy. $files_list[] = [ $source, 'local', 'rel', ]; $dir = dirname($abs_path); $new_abs_path = realpath( $dir.'/'.$source ); $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $new_abs_path, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } else { // No starting '/', making it relative to current document's path. if (strpos($source, 'width=') || strpos($source, 'autostart=') ) { continue; } if ('./' == substr($source, 0, 2)) { $source = substr( $source, 2 ); } $files_list[] = [ $source, 'local', 'rel', ]; $dir = dirname($abs_path); $new_abs_path = realpath( $dir.'/'.$source ); $in_files_list[] = self::get_resources_from_source( TOOL_DOCUMENT, $new_abs_path, $recursivity + 1 ); if (count($in_files_list) > 0) { $files_list = array_merge( $files_list, $in_files_list ); } } } } } } break; default: break; } break; default: // Ignore. break; } $checked_files_list = []; $checked_array_list = []; foreach ($files_list as $idx => $file) { if (!empty($file[0])) { if (!in_array($file[0], $checked_files_list)) { $checked_files_list[] = $files_list[$idx][0]; $checked_array_list[] = $files_list[$idx]; } } } return $checked_array_list; } /** * Gets the score. * * @return float The current score or 0 if no score set yet */ public function get_score() { $res = 0; if (!empty($this->current_score)) { $res = $this->current_score; } return $res; } /** * Gets the item status. * * @param bool $check_db Do or don't check into the database for the * latest value. Optional. Default is true * @param bool $update_local Do or don't update the local attribute * value with what's been found in DB * * @return string Current status or 'Not attempted' if no status set yet */ public function get_status($check_db = true, $update_local = false) { $course_id = api_get_course_int_id(); $debug = self::DEBUG; if ($debug > 0) { error_log('learnpathItem::get_status() on item '.$this->db_id, 0); } if ($check_db) { if ($debug > 2) { error_log('learnpathItem::get_status(): checking db', 0); } if (!empty($this->db_item_view_id) && !empty($course_id)) { $table = Database::get_course_table(TABLE_LP_ITEM_VIEW); $sql = "SELECT status FROM $table WHERE c_id = $course_id AND iid = '".$this->db_item_view_id."' AND view_count = '".$this->get_attempt_id()."'"; $res = Database::query($sql); if (1 == Database::num_rows($res)) { $row = Database::fetch_array($res); if ($update_local) { $this->set_status($row['status']); } return $row['status']; } } } else { if (!empty($this->status)) { return $this->status; } } return $this->possible_status[0]; } /** * Gets the suspend data. */ public function get_suspend_data() { // TODO: Improve cleaning of breaklines ... it works but is it really // a beautiful way to do it ? if (!empty($this->current_data)) { return str_replace( ["\r", "\n", "'"], ['\r', '\n', "\\'"], $this->current_data ); } return ''; } /** * @param string $origin * @param string $time * * @return string */ public static function getScormTimeFromParameter( $origin = 'php', $time = null ) { $h = get_lang('h'); if (!isset($time)) { if ('js' == $origin) { return '00 : 00: 00'; } return '00 '.$h.' 00 \' 00"'; } return api_format_time($time, $origin); } /** * Gets the total time spent on this item view so far. * * @param string $origin Origin of the request. If coming from PHP, * send formatted as xxhxx'xx", otherwise use scorm format 00:00:00 * @param int|null $given_time Given time is a default time to return formatted * @param bool $query_db Whether to get the value from db or from memory * * @return string A string with the time in SCORM format */ public function get_scorm_time( $origin = 'php', $given_time = null, $query_db = false ) { $time = null; $course_id = api_get_course_int_id(); if (!isset($given_time)) { if (self::DEBUG > 2) { error_log( 'learnpathItem::get_scorm_time(): given time empty, current_start_time = '.$this->current_start_time, 0 ); } if (true === $query_db) { $table = Database::get_course_table(TABLE_LP_ITEM_VIEW); $sql = "SELECT start_time, total_time FROM $table WHERE c_id = $course_id AND iid = '".$this->db_item_view_id."' AND view_count = '".$this->get_attempt_id()."'"; $res = Database::query($sql); $row = Database::fetch_array($res); $start = $row['start_time']; $stop = $start + $row['total_time']; } else { $start = $this->current_start_time; $stop = $this->current_stop_time; } if (!empty($start)) { if (!empty($stop)) { $time = $stop - $start; } else { $time = time() - $start; } } } else { $time = $given_time; } if (self::DEBUG > 2) { error_log( 'learnpathItem::get_scorm_time(): intermediate = '.$time, 0 ); } $time = api_format_time($time, $origin); return $time; } /** * Get the extra terms (tags) that identify this item. * * @return mixed */ public function get_terms() { $table = Database::get_course_table(TABLE_LP_ITEM); $sql = "SELECT * FROM $table WHERE iid = ".intval($this->db_id); $res = Database::query($sql); $row = Database::fetch_array($res); return $row['terms']; } /** * Returns the item's title. * * @return string Title */ public function get_title() { if (empty($this->title)) { return ''; } return $this->title; } /** * Returns the total time used to see that item. * * @return int Total time */ public function get_total_time() { $debug = self::DEBUG; if ($debug) { error_log( 'learnpathItem::get_total_time() for item '.$this->db_id. ' - Initially, current_start_time = '.$this->current_start_time. ' and current_stop_time = '.$this->current_stop_time, 0 ); } if (0 == $this->current_start_time) { // Shouldn't be necessary thanks to the open() method. if ($debug) { error_log( 'learnpathItem::get_total_time() - Current start time was empty', 0 ); } $this->current_start_time = time(); } if (time() < $this->current_stop_time || 0 == $this->current_stop_time ) { if ($debug) { error_log( 'learnpathItem::get_total_time() - Current stop time was ' .'greater than the current time or was empty', 0 ); } // If this case occurs, then we risk to write huge time data in db. // In theory, stop time should be *always* updated here, but it // might be used in some unknown goal. $this->current_stop_time = time(); } $time = $this->current_stop_time - $this->current_start_time; if ($time < 0) { if ($debug) { error_log( 'learnpathItem::get_total_time() - Time smaller than 0. Returning 0', 0 ); } return 0; } else { $time = $this->fixAbusiveTime($time); if ($debug) { error_log( 'Current start time = '.$this->current_start_time.', current stop time = '. $this->current_stop_time.' Returning '.$time."-----------\n" ); } return $time; } } /** * Sometimes time recorded for a learning path item is superior to the maximum allowed duration of the session. * In this case, this session resets the time for that particular learning path item to 5 minutes * (something more realistic, that is also used when leaving the portal without closing one's session). * * @param int $time * * @return int */ public function fixAbusiveTime($time) { // Code based from Event::courseLogout $sessionLifetime = api_get_configuration_value('session_lifetime'); // If session life time too big use 1 hour if (empty($sessionLifetime) || $sessionLifetime > 86400) { $sessionLifetime = 3600; } if (!Tracking::minimumTimeAvailable(api_get_session_id(), api_get_course_int_id())) { $fixedAddedMinute = 5 * 60; // Add only 5 minutes if ($time > $sessionLifetime) { error_log("fixAbusiveTime: Total time is too big: $time replaced with: $fixedAddedMinute"); error_log("item_id : ".$this->db_id." lp_item_view.iid: ".$this->db_item_view_id); $time = $fixedAddedMinute; } return $time; } else { // Calulate minimum and accumulated time $user_id = api_get_user_id(); $myLP = learnpath::getLpFromSession(api_get_course_id(), $this->lp_id, $user_id); $timeLp = $myLP->getAccumulateWorkTime(); $timeTotalCourse = $myLP->getAccumulateWorkTimeTotalCourse(); /* $timeLp = $_SESSION['oLP']->getAccumulateWorkTime(); $timeTotalCourse = $_SESSION['oLP']->getAccumulateWorkTimeTotalCourse(); */ // Minimum connection percentage $perc = 100; // Time from the course $tc = $timeTotalCourse; /*if (!empty($sessionId) && $sessionId != 0) { $sql = "SELECT hours, perc FROM plugin_licences_course_session WHERE session_id = $sessionId"; $res = Database::query($sql); if (Database::num_rows($res) > 0) { $aux = Database::fetch_assoc($res); $perc = $aux['perc']; $tc = $aux['hours'] * 60; } }*/ // Percentage of the learning paths $pl = 0; if (!empty($timeTotalCourse)) { $pl = $timeLp / $timeTotalCourse; } // Minimum time for each learning path $accumulateWorkTime = ($pl * $tc * $perc / 100); $time_seg = intval($accumulateWorkTime * 60); if ($time_seg < $sessionLifetime) { $sessionLifetime = $time_seg; } if ($time > $sessionLifetime) { $fixedAddedMinute = $time_seg + mt_rand(0, 300); if (self::DEBUG > 2) { error_log("Total time is too big: $time replaced with: $fixedAddedMinute"); } $time = $fixedAddedMinute; } return $time; } } /** * Gets the item type. * * @return string The item type (can be doc, dir, sco, asset) */ public function get_type() { $res = 'asset'; if (!empty($this->type)) { $res = $this->type; } return $res; } /** * Gets the view count for this item. * * @return int Number of attempts or 0 */ public function get_view_count() { if (!empty($this->attempt_id)) { return $this->attempt_id; } return 0; } /** * Tells if an item is done ('completed','passed','succeeded') or not. * * @return bool True if the item is done ('completed','passed','succeeded'), * false otherwise */ public function is_done() { $completedStatusList = [ 'completed', 'passed', 'succeeded', 'failed', ]; if ($this->status_is($completedStatusList)) { return true; } return false; } /** * Tells if a restart is allowed (take it from $this->prevent_reinit and $this->status). * * @return int -1 if retaking the sco another time for credit is not allowed, * 0 if it is not allowed but the item has to be finished * 1 if it is allowed. Defaults to 1 */ public function isRestartAllowed() { $restart = 1; $mystatus = $this->get_status(true); if ($this->get_prevent_reinit() > 0) { // If prevent_reinit == 1 (or more) // If status is not attempted or incomplete, authorize retaking (of the same) anyway. Otherwise: if ($mystatus != $this->possible_status[0] && $mystatus != $this->possible_status[1]) { $restart = -1; } else { //status incompleted or not attempted $restart = 0; } } else { if ($mystatus == $this->possible_status[0] || $mystatus == $this->possible_status[1]) { $restart = -1; } } return $restart; } /** * Opens/launches the item. Initialises runtime values. * * @param bool $allow_new_attempt * * @return bool true on success, false on failure */ public function open($allow_new_attempt = false) { if (0 == $this->prevent_reinit) { $this->current_score = 0; $this->current_start_time = time(); // In this case, as we are opening the item, what is important to us // is the database status, in order to know if this item has already // been used in the past (rather than just loaded and modified by // some javascript but not written in the database). // If the database status is different from 'not attempted', we can // consider this item has already been used, and as such we can // open a new attempt. Otherwise, we'll just reuse the current // attempt, which is generally created the first time the item is // loaded (for example as part of the table of contents). $stat = $this->get_status(true); if ($allow_new_attempt && isset($stat) && ($stat != $this->possible_status[0])) { $this->attempt_id = $this->attempt_id + 1; // Open a new attempt. } $this->status = $this->possible_status[1]; } else { /*if ($this->current_start_time == 0) { // Small exception for start time, to avoid amazing values. $this->current_start_time = time(); }*/ // If we don't init start time here, the time is sometimes calculated from the last start time. $this->current_start_time = time(); } } /** * Outputs the item contents. * * @return string HTML file (displayable in an