* * @todo: add backup/restore methods * * Contributors: * Peter R. Bloomfield * Julio Lopez (SL: Julio Solo) * Michael Callaghan (SL: HarmonyHill Allen) * Kerri McCusker (SL: Kerri Macchi) * Edmund Edgar (SL: Edmund Earp) * * A project developed by the Serious Games and Virtual Worlds Group. * Intelligent Systems Research Centre. * University of Ulster, Magee */ /** The Sloodle module base. */ require_once(SLOODLE_LIBROOT.'/modules/module_base.php'); /** General Sloodle functions. */ require_once(SLOODLE_LIBROOT.'/general.php'); /** * The Sloodle SLOODLE Tracker module class. * @package sloodle */ class SloodleModuleTracker extends SloodleModule { // DATA // /** * Internal for Moodle only - course module instance. * Corresponds to one record from the Moodle 'course_modules' table. * @var object * @access private */ var $cm = null; /** * Internal only - Sloodle module instance database object. * Corresponds to one record from the Moodle 'mdl_sloodle' table. * @var object * @access private */ var $sloodle_instance = null; /** * Secondary data about this instance. * Corresponds to one record from the Moodle 'mdl_sloodle_tracker' table. * @var object * @access private */ var $tracker = null; // FUNCTIONS // /** * Constructor */ function SloodleModuleTracker(&$_session = null) { $constructor = get_parent_class($this); parent::$constructor($_session); } /** * Loads data from the database. * Note: even if the function fails, it may still have overwritten some or all existing data in the object. * @param mixed $id The site-wide unique identifier for all modules. Type depends on VLE. On Moodle, it is an integer course module identifier ('id' field of 'course_modules' table) * @return bool True if successful, or false otherwise */ function load($id) { // Make sure the ID is valid $id = (int)$id; if ($id <= 0) return false; // Fetch the course module data if (!($this->cm = get_coursemodule_from_id('sloodle', $id))) { sloodle_debug("Failed to load course module instance #$id.
"); return false; } // Make sure the module is visible if ($this->cm->visible == 0) { sloodle_debug("Error: course module instance #$id not visible.
"); return false; } // Load from the primary table: sloodle instance if (!($this->sloodle_instance = sloodle_get_record('sloodle', 'id', $this->cm->instance))) { sloodle_debug("Failed to load Sloodle module with instance ID #{$this->cm->instance}.
"); return false; } // Load from the secondary table: sloodle_tracker if (!($this->tracker = sloodle_get_record('sloodle_tracker', 'sloodleid', $this->cm->instance))) { sloodle_debug("Failed to load Tracker record with sloodleid #{$this->cm->instance}.
"); return false; } return true; } /** * Gets a list of all tools for this SLOODLE Tracker. * @return an array of strings, each string containing the name of one tool. */ function get_objects() { // Get all tool record entries for this SLOODLE Tracker, sorted by name $recs = sloodle_get_records('sloodle_activity_tool', 'trackerid', $this->sloodle_instance->id, 'name'); if (!$recs) return array(); // Convert it to an array of strings $entries = array(); foreach ($recs as $r) { $entries[] = stripslashes($recs->name); } return $entries; } /** * Sets the list of tools/tasks for this SLOODLE Tracker. * @return bool True if successful, or false if not */ function record_object($uuid,$name,$type,$trackerid,$description,$taskname) { $timestamp = time(); $result = true; $entry = sloodle_get_record('sloodle_activity_tool', 'uuid', $uuid); // The tool doesn't exist in the database if (!$entry) { // Construct the new record $entry = new stdClass(); $entry->uuid = $uuid; $entry->name = $name; $entry->type = $type; $entry->trackerid = $trackerid; $entry->description = $description; $entry->taskname = $taskname; $entry->timeupdated = $timestamp; // Insert it $entry->id = sloodle_insert_record('sloodle_activity_tool', $entry); if (!$entry->id) $result = false; } // The tool already exists, it has to be updated else{ $entry->name = $name; $entry->type = $type; $entry->trackerid = $trackerid; $entry->description = $description; $entry->taskname = $taskname; $entry->timeupdated = $timestamp; // Update it if (!sloodle_update_record('sloodle_activity_tool', $entry)) $result = false; } return $result; } /** * Sets the actions completed by an avatar in Second Life. An avatar interacts with an object in SL, and this action is recorded * @param $trackerid: The site-wide unique identifier for this Second Life Tracker module * @param $objuudi: The SL unique identifier for the object/tool (task) * @param $avuuid: The SL unique identifier for the avatar * @return bool True if successful, or false if not */ function record_action($trackerid,$objuuid,$avuuid) { $timestamp = time(); $result = true; // Has the avatar already interact with this object? $entry = sloodle_get_record('sloodle_activity_tracker', 'objuuid', $objuuid, 'avuuid',$avuuid); // If not, the new action is recorded if (!$entry) { // Construct the new record $entry = new stdClass(); $entry->objuuid = $objuuid; $entry->avuuid = $avuuid; $entry->trackerid = $trackerid; $entry->timeupdated = $timestamp; // Insert it $entry->id = sloodle_insert_record('sloodle_activity_tracker', $entry); if (!$entry->id) $result = false; } // If yes, "old" interaction is updated else{ $entry->trackerid = $trackerid; $entry->timeupdated = $timestamp; if (!sloodle_update_record('sloodle_activity_tracker', $entry)) $result = false; } return $result; } // BACKUP AND RESTORE // /** * Backs-up secondary data regarding this module. * That includes everything except the main 'sloodle' database table for this instance. * @param $bf Handle to the file which backup data should be written to. * @param bool $includeuserdata Indicates whether or not to backup 'user' data, i.e. any content. Most SLOODLE tools don't have any user data. * @return bool True if successful, or false on failure. */ function backup($bf, $includeuserdata) { /* //EXAMPLE CODE FROM PRESENTER // Data about the Presenter itself fwrite($bf, full_tag('ID', 5, false, $this->presenter->id)); fwrite($bf, full_tag('FRAMEWIDTH', 5, false, $this->presenter->framewidth)); fwrite($bf, full_tag('FRAMEHEIGHT', 5, false, $this->presenter->frameheight)); // Attempt to fetch all the slides in the presentation $slides = $this->get_slides(); if (!$slides) return false; // Data about the slides in the presenter. // Currently this will only backup the raw URLs, and won't transfer any files. // In future, it should backup any files which are on the same server. fwrite($bf, start_tag('SLIDES', 5, true)); foreach ($slides as $slide) { fwrite($bf, start_tag('SLIDE', 6, true)); // Convert plugin class names back to simple slide types switch ($slide->type) { case 'SloodlePluginPresenterSlideImage': case 'PresenterSlideImage': $slide->type = 'image'; break; case 'SloodlePluginPresenterSlideVideo': case 'PresenterSlideVideo': $slide->type = 'video'; break; case 'SloodlePluginPresenterSlideWeb': case 'PresenterSlideWeb': $slide->type = 'web'; break; } fwrite($bf, full_tag('ID', 7, false, $slide->id)); fwrite($bf, full_tag('NAME', 7, false, $slide->name)); fwrite($bf, full_tag('SOURCE', 7, false, $slide->source)); fwrite($bf, full_tag('TYPE', 7, false, $slide->type)); fwrite($bf, full_tag('ORDERING', 7, false, $slide->ordering)); fwrite($bf, end_tag('SLIDE', 6, true)); } fwrite($bf, end_tag('SLIDES', 5, true)); return true;*/ return false; } /** * Restore this module's secondary data into the database. * This ignores any member data, so can be called statically. * @param int $sloodleid The ID of the primary SLOODLE entry this restore belongs to (i.e. the ID of the record in the "sloodle" table) * @param array $info An associative array representing the XML backup information for the secondary module data * @param bool $includeuserdata Indicates whether or not to restore user data * @return bool True if successful, or false on failure. */ function restore($sloodleid, $info, $includeuserdata) { /* // EXAMPLE CODE FROM PRESENTER // Construct the database record for the Presenter itself $presenter = new object(); $presenter->sloodleid = $sloodleid; $presenter->framewidth = $info['FRAMEWIDTH']['0']['#']; $presenter->frameheight = $info['FRAMEHEIGHT']['0']['#']; $presenter->id = sloodle_insert_record('sloodle_presenter', $presenter); // Go through each slide in the presenter backup $numslides = count($info['SLIDES']['0']['#']['SLIDE']); $curslide = null; for ($slidenum = 0; $slidenum < $numslides; $slidenum++) { // Get the current slide data $curslide = $info['SLIDES']['0']['#']['SLIDE'][$slidenum]['#']; // Construct a new Presenter slide database object $slide = new object(); $slide->sloodleid = $sloodleid; $slide->name = $curslide['NAME']['0']['#']; $slide->source = $curslide['SOURCE']['0']['#']; $slide->type = $curslide['TYPE']['0']['#']; $slide->ordering = $curslide['ORDERING']['0']['#']; $slide->id = sloodle_insert_record('sloodle_presenter_entry', $slide); } return true;*/ return false; } /** * Gets the name of the user data required by this type, or an empty string if none is required. * For example, a chatroom would use the name "Messages" for user data. * Note that this should respect current language settings in Moodle. * @return string Localised name of the user data. */ function get_user_data_name() { return ''; } /** * Gets the number of user data records to be backed-up. * @return int A count of the number of user data records which can be backed-up. */ function get_user_data_count() { return 0; } // ACCESSORS // /** * Gets the name of this module instance. * @return string The name of this module */ function get_name() { return $this->sloodle_instance->name; } /** * Gets the intro description of this module instance, if available. * @return string The intro description of this controller */ function get_intro() { return $this->sloodle_instance->intro; } /** * Gets the identifier of the course this controller belongs to. * @return mixed Course identifier. Type depends on VLE. (In Moodle, it will be an integer). */ function get_course_id() { return (int)$this->sloodle_instance->course; } /** * Gets the time at which this instance was created, or 0 if unknown. * @return int Timestamp */ function get_creation_time() { return (int)$this->sloodle_instance->timecreated; } /** * Gets the time at which this instance was last modified, or 0 if unknown. * @return int Timestamp */ function get_modification_time() { return (int)$this->sloodle_instance->timemodified; } /** * Gets the short type name of this instance. * @return string */ function get_type() { return SLOODLE_TYPE_TRACKER; } /** * Gets the full type name of this instance, according to the current language pack, if available. * Note: should be overridden by sub-classes. * @return string Full type name if possible, or the short name otherwise. */ function get_type_full() { return get_string('moduletype:'.SLOODLE_TYPE_TRACKER, 'sloodle'); } /* Returns an array of error messages for requirements that haven't been satisfied. ...eg. If an object has been configured to require 3 gold coins, and the user doesn't have enough, it'll return a message saying you don't have enough gold coins. */ function RequirementFailures( $relevant_configs, $controllerid, $multiplier, $userid, $useruuid, $objectuuid ) { global $CFG; if ( isset($relevant_configs['sloodletrackerrequire_taskcompleted']) ) { $required_task = $relevant_configs['sloodletrackerrequire_taskcompleted']; if (!$required_task){ return false; } } if ( !SloodleModuleTracker::UserHasCompleted($useruuid, $controllerid, $task) ) { if (isset($relevant_configs['sloodletrackerrequire_tasknotcompletedmessage']) && $relevant_configs['sloodletrackerrequire_tasknotcompletedmessage'] != '') { return $relevant_configs['sloodletrackerrequire_tasknotcompletedmessage']; } else { return get_string('tracker:requiredtasknotcompleted', 'sloodle'); } } return ''; } /* An array of the names of config parameters that are understood by this module to mean it should do something. Will have the name of the specific interaction appended to it. eg. awards makes available an interaction config called "sloodleawardsdeposit_numpoints". The quiz would then have a config name=>value pair like sloodleawards_deposit_numpoints_answerquestion => 3 Via the ActiveObject, the quiz will tell the awards module that answerquestion has happened to a particular user ...and the awards module will give them the points. */ function ActionConfigNames() { /* return array( 'sloodletrackersatisfy_taskcompleted' ); */ } /* An array of the names of config parameters that are understood by this module to check something before doing whatever it would normally do. Will have the name of the specific interaction appended to it. */ function RequirementConfigNames() { /* return array( 'sloodletrackerrequire_taskcompleted', 'sloodletrackerrequire_tasknotcompletedmessage' ); */ } /* Not yet implemented When we authorize an object, we should run this function for each module that has it allowing us to do whatever setup tasks the module defines. */ function HandleObjectInitializationSteps( $relevant_configs, $active_object ) { $taskname = isset($relevant_configs['sloodletrackersatisfy_taskcompleted']) ? $relevant_configs['sloodletrackersatisfy_taskcompleted'] : ''; $description = isset($relevant_configs['sloodletrackersatisfy_taskcompleteddescription']) ? $relevant_configs['sloodletrackersatisfy_taskcompleteddescription'] : ''; if ( isset($relevant_configs['sloodletrackersatisfy_taskcompleted']) ) { } SloodleModuleTracker::record_object($active_object->uuid,$active_object->name,$active_object->type,$active_object->controllerid,$description,$taskname); } function ProcessActions( $relevant_configs, $controllerid, $multiplier, $userid, $useruuid, $objectuuid) { return true; global $CFG; $controller = new SloodleController(); if (!$controller->load_by_course_module_id($controllerid)) { return false; } $time = time(); if ( isset($relevant_configs['sloodletrackersatisfy_taskcompleted']) ) { } return true; } } ?>