. /** * local functions and constants for module attendance * * @package mod_attendance * @copyright 2011 Artem Andreev * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ defined('MOODLE_INTERNAL') || die(); global $CFG; require_once($CFG->libdir . '/gradelib.php'); require_once(dirname(__FILE__).'/renderhelpers.php'); define('ATT_VIEW_DAYS', 1); define('ATT_VIEW_WEEKS', 2); define('ATT_VIEW_MONTHS', 3); define('ATT_VIEW_ALLPAST', 4); define('ATT_VIEW_ALL', 5); define('ATT_VIEW_NOTPRESENT', 6); define('ATT_SORT_LASTNAME', 1); define('ATT_SORT_FIRSTNAME', 2); class att_page_with_filter_controls { const SELECTOR_NONE = 1; const SELECTOR_GROUP = 2; const SELECTOR_SESS_TYPE = 3; const SESSTYPE_COMMON = 0; const SESSTYPE_ALL = -1; const SESSTYPE_NO_VALUE = -2; /** @var int current view mode */ public $view; /** @var int $view and $curdate specify displaed date range */ public $curdate; /** @var int start date of displayed date range */ public $startdate; /** @var int end date of displayed date range */ public $enddate; public $selectortype = self::SELECTOR_NONE; protected $defaultview = ATT_VIEW_WEEKS; private $cm; private $sessgroupslist; private $sesstype; public function init($cm) { $this->cm = $cm; $this->init_view(); $this->init_curdate(); $this->init_start_end_date(); } private function init_view() { global $SESSION; if (isset($this->view)) { $SESSION->attcurrentattview[$this->cm->course] = $this->view; } else if (isset($SESSION->attcurrentattview[$this->cm->course])) { $this->view = $SESSION->attcurrentattview[$this->cm->course]; } else { $this->view = $this->defaultview; } } private function init_curdate() { global $SESSION; if (isset($this->curdate)) { $SESSION->attcurrentattdate[$this->cm->course] = $this->curdate; } else if (isset($SESSION->attcurrentattdate[$this->cm->course])) { $this->curdate = $SESSION->attcurrentattdate[$this->cm->course]; } else { $this->curdate = time(); } } public function init_start_end_date() { global $CFG; // HOURSECS solves issue for weeks view with Daylight saving time and clocks adjusting by one hour backward. $date = usergetdate($this->curdate + HOURSECS); $mday = $date['mday']; $wday = $date['wday'] - $CFG->calendar_startwday; if ($wday < 0) { $wday += 7; } $mon = $date['mon']; $year = $date['year']; switch ($this->view) { case ATT_VIEW_DAYS: $this->startdate = make_timestamp($year, $mon, $mday); $this->enddate = make_timestamp($year, $mon, $mday + 1); break; case ATT_VIEW_WEEKS: $this->startdate = make_timestamp($year, $mon, $mday - $wday); $this->enddate = make_timestamp($year, $mon, $mday + 7 - $wday) - 1; break; case ATT_VIEW_MONTHS: $this->startdate = make_timestamp($year, $mon); $this->enddate = make_timestamp($year, $mon + 1); break; case ATT_VIEW_ALLPAST: $this->startdate = 1; $this->enddate = time(); break; case ATT_VIEW_ALL: $this->startdate = 0; $this->enddate = 0; break; } } private function calc_sessgroupslist_sesstype() { global $SESSION; if (!array_key_exists('attsessiontype', $SESSION)) { $SESSION->attsessiontype = array($this->cm->course => self::SESSTYPE_ALL); } else if (!array_key_exists($this->cm->course, $SESSION->attsessiontype)) { $SESSION->attsessiontype[$this->cm->course] = self::SESSTYPE_ALL; } $group = optional_param('group', self::SESSTYPE_NO_VALUE, PARAM_INT); if ($this->selectortype == self::SELECTOR_SESS_TYPE) { if ($group > self::SESSTYPE_NO_VALUE) { $SESSION->attsessiontype[$this->cm->course] = $group; if ($group > self::SESSTYPE_ALL) { // Set activegroup in $SESSION. groups_get_activity_group($this->cm, true); } else { // Reset activegroup in $SESSION. unset($SESSION->activegroup[$this->cm->course][VISIBLEGROUPS][$this->cm->groupingid]); unset($SESSION->activegroup[$this->cm->course]['aag'][$this->cm->groupingid]); unset($SESSION->activegroup[$this->cm->course][SEPARATEGROUPS][$this->cm->groupingid]); } $this->sesstype = $group; } else { $this->sesstype = $SESSION->attsessiontype[$this->cm->course]; } } else if ($this->selectortype == self::SELECTOR_GROUP) { if ($group == 0) { $SESSION->attsessiontype[$this->cm->course] = self::SESSTYPE_ALL; $this->sesstype = self::SESSTYPE_ALL; } else if ($group > 0) { $SESSION->attsessiontype[$this->cm->course] = $group; $this->sesstype = $group; } else { $this->sesstype = $SESSION->attsessiontype[$this->cm->course]; } } if (is_null($this->sessgroupslist)) { $this->calc_sessgroupslist(); } // For example, we set SESSTYPE_ALL but user can access only to limited set of groups. if (!array_key_exists($this->sesstype, $this->sessgroupslist)) { reset($this->sessgroupslist); $this->sesstype = key($this->sessgroupslist); } } private function calc_sessgroupslist() { global $USER, $PAGE; $this->sessgroupslist = array(); $groupmode = groups_get_activity_groupmode($this->cm); if ($groupmode == NOGROUPS) { return; } if ($groupmode == VISIBLEGROUPS or has_capability('moodle/site:accessallgroups', $PAGE->context)) { $allowedgroups = groups_get_all_groups($this->cm->course, 0, $this->cm->groupingid); } else { $allowedgroups = groups_get_all_groups($this->cm->course, $USER->id, $this->cm->groupingid); } if ($allowedgroups) { if ($groupmode == VISIBLEGROUPS or has_capability('moodle/site:accessallgroups', $PAGE->context)) { $this->sessgroupslist[self::SESSTYPE_ALL] = get_string('all', 'attendance'); } // Show Common groups always. $this->sessgroupslist[self::SESSTYPE_COMMON] = get_string('commonsessions', 'attendance'); foreach ($allowedgroups as $group) { $this->sessgroupslist[$group->id] = get_string('group') . ': ' . format_string($group->name); } } } public function get_sess_groups_list() { if (is_null($this->sessgroupslist)) { $this->calc_sessgroupslist_sesstype(); } return $this->sessgroupslist; } public function get_current_sesstype() { if (is_null($this->sesstype)) { $this->calc_sessgroupslist_sesstype(); } return $this->sesstype; } public function set_current_sesstype($sesstype) { $this->sesstype = $sesstype; } } class att_view_page_params extends att_page_with_filter_controls { const MODE_THIS_COURSE = 0; const MODE_ALL_COURSES = 1; public $studentid; public $mode; public function __construct() { $this->defaultview = ATT_VIEW_MONTHS; } public function get_significant_params() { $params = array(); if (isset($this->studentid)) { $params['studentid'] = $this->studentid; } if ($this->mode != self::MODE_THIS_COURSE) { $params['mode'] = $this->mode; } return $params; } } class att_manage_page_params extends att_page_with_filter_controls { public function __construct() { $this->selectortype = att_page_with_filter_controls::SELECTOR_SESS_TYPE; } public function get_significant_params() { return array(); } } class att_sessions_page_params { const ACTION_ADD = 1; const ACTION_UPDATE = 2; const ACTION_DELETE = 3; const ACTION_DELETE_SELECTED = 4; const ACTION_CHANGE_DURATION = 5; const ACTION_DELETE_HIDDEN = 6; /** @var int view mode of taking attendance page*/ public $action; } class att_take_page_params { const SORTED_LIST = 1; const SORTED_GRID = 2; const DEFAULT_VIEW_MODE = self::SORTED_LIST; public $sessionid; public $grouptype; public $group; public $sort; public $copyfrom; /** @var int view mode of taking attendance page*/ public $viewmode; public $gridcols; public function init() { if (!isset($this->group)) { $this->group = 0; } if (!isset($this->sort)) { $this->sort = ATT_SORT_LASTNAME; } $this->init_view_mode(); $this->init_gridcols(); } private function init_view_mode() { if (isset($this->viewmode)) { set_user_preference("attendance_take_view_mode", $this->viewmode); } else { $this->viewmode = get_user_preferences("attendance_take_view_mode", self::DEFAULT_VIEW_MODE); } } private function init_gridcols() { if (isset($this->gridcols)) { set_user_preference("attendance_gridcolumns", $this->gridcols); } else { $this->gridcols = get_user_preferences("attendance_gridcolumns", 5); } } public function get_significant_params() { $params = array(); $params['sessionid'] = $this->sessionid; $params['grouptype'] = $this->grouptype; if ($this->group) { $params['group'] = $this->group; } if ($this->sort != ATT_SORT_LASTNAME) { $params['sort'] = $this->sort; } if (isset($this->copyfrom)) { $params['copyfrom'] = $this->copyfrom; } return $params; } } class att_report_page_params extends att_page_with_filter_controls { public $group; public $sort; public function __construct() { $this->selectortype = self::SELECTOR_GROUP; } public function init($cm) { parent::init($cm); if (!isset($this->group)) { $this->group = $this->get_current_sesstype() > 0 ? $this->get_current_sesstype() : 0; } if (!isset($this->sort)) { $this->sort = ATT_SORT_LASTNAME; } } public function get_significant_params() { $params = array(); if ($this->sort != ATT_SORT_LASTNAME) { $params['sort'] = $this->sort; } return $params; } } class att_preferences_page_params { const ACTION_ADD = 1; const ACTION_DELETE = 2; const ACTION_HIDE = 3; const ACTION_SHOW = 4; const ACTION_SAVE = 5; /** @var int view mode of taking attendance page*/ public $action; public $statusid; public $statusset; public function get_significant_params() { $params = array(); if (isset($this->action)) { $params['action'] = $this->action; } if (isset($this->statusid)) { $params['statusid'] = $this->statusid; } if (isset($this->statusset)) { $params['statusset'] = $this->statusset; } return $params; } } class attendance { const SESSION_COMMON = 0; const SESSION_GROUP = 1; /** @var stdclass course module record */ public $cm; /** @var stdclass course record */ public $course; /** @var stdclass context object */ public $context; /** @var int attendance instance identifier */ public $id; /** @var string attendance activity name */ public $name; /** @var float number (10, 5) unsigned, the maximum grade for attendance */ public $grade; /** current page parameters */ public $pageparams; private $groupmode; private $statuses; private $allstatuses; // Cache list of all statuses (not just one used by current session). // Array by sessionid. private $sessioninfo = array(); // Arrays by userid. private $usertakensesscount = array(); private $userstatusesstat = array(); /** * Initializes the attendance API instance using the data from DB * * Makes deep copy of all passed records properties. Replaces integer $course attribute * with a full database record (course should not be stored in instances table anyway). * * @param stdClass $dbrecord Attandance instance data from {attendance} table * @param stdClass $cm Course module record as returned by {@link get_coursemodule_from_id()} * @param stdClass $course Course record from {course} table * @param stdClass $context The context of the workshop instance */ public function __construct(stdclass $dbrecord, stdclass $cm, stdclass $course, stdclass $context=null, $pageparams=null) { foreach ($dbrecord as $field => $value) { if (property_exists('attendance', $field)) { $this->{$field} = $value; } else { throw new coding_exception('The attendance table has a field with no property in the attendance class'); } } $this->cm = $cm; $this->course = $course; if (is_null($context)) { $this->context = context_module::instance($this->cm->id); } else { $this->context = $context; } $this->pageparams = $pageparams; } public function get_group_mode() { if (is_null($this->groupmode)) { $this->groupmode = groups_get_activity_groupmode($this->cm, $this->course); } return $this->groupmode; } /** * Returns current sessions for this attendance * * Fetches data from {attendance_sessions} * * @return array of records or an empty array */ public function get_current_sessions() { global $DB; $today = time(); // Because we compare with database, we don't need to use usertime(). $sql = "SELECT * FROM {attendance_sessions} WHERE :time BETWEEN sessdate AND (sessdate + duration) AND attendanceid = :aid"; $params = array( 'time' => $today, 'aid' => $this->id); return $DB->get_records_sql($sql, $params); } /** * Returns today sessions for this attendance * * Fetches data from {attendance_sessions} * * @return array of records or an empty array */ public function get_today_sessions() { global $DB; $start = usergetmidnight(time()); $end = $start + DAYSECS; $sql = "SELECT * FROM {attendance_sessions} WHERE sessdate >= :start AND sessdate < :end AND attendanceid = :aid"; $params = array( 'start' => $start, 'end' => $end, 'aid' => $this->id); return $DB->get_records_sql($sql, $params); } /** * Returns today sessions suitable for copying attendance log * * Fetches data from {attendance_sessions} * * @return array of records or an empty array */ public function get_today_sessions_for_copy($sess) { global $DB; $start = usergetmidnight($sess->sessdate); $sql = "SELECT * FROM {attendance_sessions} WHERE sessdate >= :start AND sessdate <= :end AND (groupid = 0 OR groupid = :groupid) AND lasttaken > 0 AND attendanceid = :aid"; $params = array( 'start' => $start, 'end' => $sess->sessdate, 'groupid' => $sess->groupid, 'aid' => $this->id); return $DB->get_records_sql($sql, $params); } /** * Returns count of hidden sessions for this attendance * * Fetches data from {attendance_sessions} * * @return count of hidden sessions */ public function get_hidden_sessions_count() { global $DB; $where = "attendanceid = :aid AND sessdate < :csdate"; $params = array( 'aid' => $this->id, 'csdate' => $this->course->startdate); return $DB->count_records_select('attendance_sessions', $where, $params); } /** * Returns the hidden sessions for this attendance * * Fetches data from {attendance_sessions} * * @return hidden sessions */ public function get_hidden_sessions() { global $DB; $where = "attendanceid = :aid AND sessdate < :csdate"; $params = array( 'aid' => $this->id, 'csdate' => $this->course->startdate); return $DB->get_records_select('attendance_sessions', $where, $params); } public function get_filtered_sessions() { global $DB; if ($this->pageparams->startdate && $this->pageparams->enddate) { $where = "attendanceid = :aid AND sessdate >= :csdate AND sessdate >= :sdate AND sessdate < :edate"; } else if ($this->pageparams->enddate) { $where = "attendanceid = :aid AND sessdate >= :csdate AND sessdate < :edate"; } else { $where = "attendanceid = :aid AND sessdate >= :csdate"; } if ($this->pageparams->get_current_sesstype() > att_page_with_filter_controls::SESSTYPE_ALL) { $where .= " AND (groupid = :cgroup OR groupid = 0)"; } $params = array( 'aid' => $this->id, 'csdate' => $this->course->startdate, 'sdate' => $this->pageparams->startdate, 'edate' => $this->pageparams->enddate, 'cgroup' => $this->pageparams->get_current_sesstype()); $sessions = $DB->get_records_select('attendance_sessions', $where, $params, 'sessdate asc'); foreach ($sessions as $sess) { if (empty($sess->description)) { $sess->description = get_string('nodescription', 'attendance'); } else { $sess->description = file_rewrite_pluginfile_urls($sess->description, 'pluginfile.php', $this->context->id, 'mod_attendance', 'session', $sess->id); } } return $sessions; } /** * @return moodle_url of manage.php for attendance instance */ public function url_manage($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/manage.php', $params); } /** * @param array $params optional * @return moodle_url of tempusers.php for attendance instance */ public function url_managetemp($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/tempusers.php', $params); } /** * @param array $params optional * @return moodle_url of tempdelete.php for attendance instance */ public function url_tempdelete($params=array()) { $params = array_merge(array('id' => $this->cm->id, 'action' => 'delete'), $params); return new moodle_url('/mod/attendance/tempedit.php', $params); } /** * @param array $params optional * @return moodle_url of tempedit.php for attendance instance */ public function url_tempedit($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/tempedit.php', $params); } /** * @param array $params optional * @return moodle_url of tempedit.php for attendance instance */ public function url_tempmerge($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/tempmerge.php', $params); } /** * @return moodle_url of sessions.php for attendance instance */ public function url_sessions($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/sessions.php', $params); } /** * @return moodle_url of report.php for attendance instance */ public function url_report($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/report.php', $params); } /** * @return moodle_url of export.php for attendance instance */ public function url_export() { $params = array('id' => $this->cm->id); return new moodle_url('/mod/attendance/export.php', $params); } /** * @return moodle_url of attsettings.php for attendance instance */ public function url_preferences($params=array()) { // Add the statusset params. if (isset($this->pageparams->statusset) && !isset($params['statusset'])) { $params['statusset'] = $this->pageparams->statusset; } $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/preferences.php', $params); } /** * @return moodle_url of attendances.php for attendance instance */ public function url_take($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/take.php', $params); } public function url_view($params=array()) { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/view.php', $params); } public function add_sessions($sessions) { global $DB; foreach ($sessions as $sess) { $sess->attendanceid = $this->id; $sess->id = $DB->insert_record('attendance_sessions', $sess); $description = file_save_draft_area_files($sess->descriptionitemid, $this->context->id, 'mod_attendance', 'session', $sess->id, array('subdirs' => false, 'maxfiles' => -1, 'maxbytes' => 0), $sess->description); $DB->set_field('attendance_sessions', 'description', $description, array('id' => $sess->id)); $infoarray = array(); $infoarray[] = construct_session_full_date_time($sess->sessdate, $sess->duration); // Trigger a session added event. $event = \mod_attendance\event\session_added::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => array('info' => implode(',', $infoarray)) )); $event->add_record_snapshot('course_modules', $this->cm); $sess->description = $description; $sess->lasttaken = 0; $sess->lasttakenby = 0; $sess->studentscanmark = 0; $event->add_record_snapshot('attendance_sessions', $sess); $event->trigger(); } } public function update_session_from_form_data($formdata, $sessionid) { global $DB; if (!$sess = $DB->get_record('attendance_sessions', array('id' => $sessionid) )) { print_error('No such session in this course'); } $sesstarttime = $formdata->sestime['starthour'] * HOURSECS + $formdata->sestime['startminute'] * MINSECS; $sesendtime = $formdata->sestime['endhour'] * HOURSECS + $formdata->sestime['endminute'] * MINSECS; $sess->sessdate = $formdata->sessiondate + $sesstarttime; $sess->duration = $sesendtime - $sesstarttime; $description = file_save_draft_area_files($formdata->sdescription['itemid'], $this->context->id, 'mod_attendance', 'session', $sessionid, array('subdirs' => false, 'maxfiles' => -1, 'maxbytes' => 0), $formdata->sdescription['text']); $sess->description = $description; $sess->descriptionformat = $formdata->sdescription['format']; $sess->timemodified = time(); $DB->update_record('attendance_sessions', $sess); $info = construct_session_full_date_time($sess->sessdate, $sess->duration); $event = \mod_attendance\event\session_updated::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => array('info' => $info, 'sessionid' => $sessionid, 'action' => att_sessions_page_params::ACTION_UPDATE))); $event->add_record_snapshot('course_modules', $this->cm); $event->add_record_snapshot('attendance_sessions', $sess); $event->trigger(); } /** * Used to record attendance submitted by the student. * * @global type $DB * @global type $USER * @param type $mformdata * @return boolean */ public function take_from_student($mformdata) { global $DB, $USER; $statuses = implode(',', array_keys( (array)$this->get_statuses() )); $now = time(); $record = new stdClass(); $record->studentid = $USER->id; $record->statusid = $mformdata->status; $record->statusset = $statuses; $record->remarks = get_string('set_by_student', 'mod_attendance'); $record->sessionid = $mformdata->sessid; $record->timetaken = $now; $record->takenby = $USER->id; $dbsesslog = $this->get_session_log($mformdata->sessid); if (array_key_exists($record->studentid, $dbsesslog)) { // Already recorded do not save. return false; } $logid = $DB->insert_record('attendance_log', $record, false); $record->id = $logid; // Update the session to show that a register has been taken, or staff may overwrite records. $session = $this->get_session_info($mformdata->sessid); $session->lasttaken = $now; $session->lasttakenby = $USER->id; $DB->update_record('attendance_sessions', $session); // Update the users grade. $this->update_users_grade(array($USER->id)); /* create url for link in log screen * need to set grouptype to 0 to allow take attendance page to be called * from report/log page */ $params = array( 'sessionid' => $this->pageparams->sessionid, 'grouptype' => 0); // Log the change. $event = \mod_attendance\event\attendance_taken_by_student::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => $params)); $event->add_record_snapshot('course_modules', $this->cm); $event->add_record_snapshot('attendance_sessions', $session); $event->add_record_snapshot('attendance_log', $record); $event->trigger(); return true; } public function take_from_form_data($formdata) { global $DB, $USER; // TODO: WARNING - $formdata is unclean - comes from direct $_POST - ideally needs a rewrite but we do some cleaning below. $statuses = implode(',', array_keys( (array)$this->get_statuses() )); $now = time(); $sesslog = array(); $formdata = (array)$formdata; foreach ($formdata as $key => $value) { if (substr($key, 0, 4) == 'user') { $sid = substr($key, 4); if (!(is_numeric($sid) && is_numeric($value))) { // Sanity check on $sid and $value. print_error('nonnumericid', 'attendance'); } $sesslog[$sid] = new stdClass(); $sesslog[$sid]->studentid = $sid; // We check is_numeric on this above. $sesslog[$sid]->statusid = $value; // We check is_numeric on this above. $sesslog[$sid]->statusset = $statuses; $sesslog[$sid]->remarks = array_key_exists('remarks'.$sid, $formdata) ? clean_param($formdata['remarks'.$sid], PARAM_TEXT) : ''; $sesslog[$sid]->sessionid = $this->pageparams->sessionid; $sesslog[$sid]->timetaken = $now; $sesslog[$sid]->takenby = $USER->id; } } $dbsesslog = $this->get_session_log($this->pageparams->sessionid); foreach ($sesslog as $log) { if ($log->statusid) { if (array_key_exists($log->studentid, $dbsesslog)) { $log->id = $dbsesslog[$log->studentid]->id; $DB->update_record('attendance_log', $log); } else { $DB->insert_record('attendance_log', $log, false); } } } $session = $this->get_session_info($this->pageparams->sessionid); $session->lasttaken = $now; $session->lasttakenby = $USER->id; $DB->update_record('attendance_sessions', $session); if ($this->grade != 0) { $this->update_users_grade(array_keys($sesslog)); } // Create url for link in log screen. $params = array( 'sessionid' => $this->pageparams->sessionid, 'grouptype' => $this->pageparams->grouptype); $event = \mod_attendance\event\attendance_taken::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => $params)); $event->add_record_snapshot('course_modules', $this->cm); $event->add_record_snapshot('attendance_sessions', $session); $event->trigger(); $group = 0; if ($this->pageparams->grouptype != self::SESSION_COMMON) { $group = $this->pageparams->grouptype; } else { if ($this->pageparams->group) { $group = $this->pageparams->group; } } $totalusers = count_enrolled_users(context_module::instance($this->cm->id), 'mod/attendance:canbelisted', $group); $usersperpage = $this->pageparams->perpage; if (!empty($this->pageparams->page) && $this->pageparams->page && $totalusers && $usersperpage) { $numberofpages = ceil($totalusers / $usersperpage); if ($this->pageparams->page < $numberofpages) { $params['page'] = $this->pageparams->page + 1; redirect($this->url_take($params), get_string('moreattendance', 'attendance')); } } redirect($this->url_manage(), get_string('attendancesuccess', 'attendance')); } /** * MDL-27591 made this method obsolete. */ public function get_users($groupid = 0, $page = 1) { global $DB, $CFG; // Fields we need from the user table. $userfields = user_picture::fields('u', array('username' , 'idnumber' , 'institution' , 'department')); if (isset($this->pageparams->sort) and ($this->pageparams->sort == ATT_SORT_FIRSTNAME)) { $orderby = "u.firstname ASC, u.lastname ASC, u.idnumber ASC, u.institution ASC, u.department ASC"; } else { $orderby = "u.lastname ASC, u.firstname ASC, u.idnumber ASC, u.institution ASC, u.department ASC"; } if ($page) { $usersperpage = $this->pageparams->perpage; if (!empty($CFG->enablegroupmembersonly) and $this->cm->groupmembersonly) { $startusers = ($page - 1) * $usersperpage; if ($groupid == 0) { $groups = array_keys(groups_get_all_groups($this->cm->course, 0, $this->cm->groupingid, 'g.id')); } else { $groups = $groupid; } $users = get_users_by_capability($this->context, 'mod/attendance:canbelisted', $userfields.',u.id, u.firstname, u.lastname, u.email', $orderby, $startusers, $usersperpage, $groups, '', false, true); } else { $startusers = ($page - 1) * $usersperpage; $users = get_enrolled_users($this->context, 'mod/attendance:canbelisted', $groupid, $userfields, $orderby, $startusers, $usersperpage); } } else { if (!empty($CFG->enablegroupmembersonly) and $this->cm->groupmembersonly) { if ($groupid == 0) { $groups = array_keys(groups_get_all_groups($this->cm->course, 0, $this->cm->groupingid, 'g.id')); } else { $groups = $groupid; } $users = get_users_by_capability($this->context, 'mod/attendance:canbelisted', $userfields.',u.id, u.firstname, u.lastname, u.email', $orderby, '', '', $groups, '', false, true); } else { $users = get_enrolled_users($this->context, 'mod/attendance:canbelisted', $groupid, $userfields, $orderby); } } // Add a flag to each user indicating whether their enrolment is active. if (!empty($users)) { list($sql, $params) = $DB->get_in_or_equal(array_keys($users), SQL_PARAMS_NAMED, 'usid0'); // See CONTRIB-4868. $mintime = 'MIN(CASE WHEN (ue.timestart > :zerotime) THEN ue.timestart ELSE ue.timecreated END)'; $maxtime = 'CASE WHEN MIN(ue.timeend) = 0 THEN 0 ELSE MAX(ue.timeend) END'; // See CONTRIB-3549. $sql = "SELECT ue.userid, MIN(ue.status) as status, $mintime AS mintime, $maxtime AS maxtime FROM {user_enrolments} ue JOIN {enrol} e ON e.id = ue.enrolid WHERE ue.userid $sql AND e.status = :estatus AND e.courseid = :courseid GROUP BY ue.userid"; $params += array('zerotime' => 0, 'estatus' => ENROL_INSTANCE_ENABLED, 'courseid' => $this->course->id); $enrolments = $DB->get_records_sql($sql, $params); foreach ($users as $user) { $users[$user->id]->enrolmentstatus = $enrolments[$user->id]->status; $users[$user->id]->enrolmentstart = $enrolments[$user->id]->mintime; $users[$user->id]->enrolmentend = $enrolments[$user->id]->maxtime; $users[$user->id]->type = 'standard'; // Mark as a standard (not a temporary) user. } } // Add the 'temporary' users to this list. $tempusers = $DB->get_records('attendance_tempusers', array('courseid' => $this->course->id)); foreach ($tempusers as $tempuser) { $users[] = self::tempuser_to_user($tempuser); } return $users; } // Convert a tempuser record into a user object. protected static function tempuser_to_user($tempuser) { $ret = (object)array( 'id' => $tempuser->studentid, 'firstname' => $tempuser->fullname, 'email' => $tempuser->email, 'username' => '', 'enrolmentstatus' => 0, 'enrolmentstart' => 0, 'enrolmentend' => 0, 'picture' => 0, 'type' => 'temporary', ); foreach (get_all_user_name_fields() as $namefield) { if (!isset($ret->$namefield)) { $ret->$namefield = ''; } } return $ret; } public function get_user($userid) { global $DB; $user = $DB->get_record('user', array('id' => $userid), '*', MUST_EXIST); // Look for 'temporary' users and return their details from the attendance_tempusers table. if ($user->idnumber == 'tempghost') { $tempuser = $DB->get_record('attendance_tempusers', array('studentid' => $userid), '*', MUST_EXIST); return self::tempuser_to_user($tempuser); } $user->type = 'standard'; // See CONTRIB-4868. $mintime = 'MIN(CASE WHEN (ue.timestart > :zerotime) THEN ue.timestart ELSE ue.timecreated END)'; $maxtime = 'CASE WHEN MIN(ue.timeend) = 0 THEN 0 ELSE MAX(ue.timeend) END'; $sql = "SELECT ue.userid, ue.status, $mintime AS mintime, $maxtime AS maxtime FROM {user_enrolments} ue JOIN {enrol} e ON e.id = ue.enrolid WHERE ue.userid = :uid AND e.status = :estatus AND e.courseid = :courseid GROUP BY ue.userid, ue.status"; $params = array('zerotime' => 0, 'uid' => $userid, 'estatus' => ENROL_INSTANCE_ENABLED, 'courseid' => $this->course->id); $enrolments = $DB->get_record_sql($sql, $params); $user->enrolmentstatus = $enrolments->status; $user->enrolmentstart = $enrolments->mintime; $user->enrolmentend = $enrolments->maxtime; return $user; } public function get_statuses($onlyvisible = true, $allsets = false) { if (!isset($this->statuses)) { // Get the statuses for the current set only. $statusset = 0; if (isset($this->pageparams->statusset)) { $statusset = $this->pageparams->statusset; } else if (isset($this->pageparams->sessionid)) { $sessioninfo = $this->get_session_info($this->pageparams->sessionid); $statusset = $sessioninfo->statusset; } $this->statuses = att_get_statuses($this->id, $onlyvisible, $statusset); $this->allstatuses = att_get_statuses($this->id, $onlyvisible); } // Return all sets, if requested. if ($allsets) { return $this->allstatuses; } return $this->statuses; } public function get_session_info($sessionid) { global $DB; if (!array_key_exists($sessionid, $this->sessioninfo)) { $this->sessioninfo[$sessionid] = $DB->get_record('attendance_sessions', array('id' => $sessionid)); } if (empty($this->sessioninfo[$sessionid]->description)) { $this->sessioninfo[$sessionid]->description = get_string('nodescription', 'attendance'); } else { $this->sessioninfo[$sessionid]->description = file_rewrite_pluginfile_urls($this->sessioninfo[$sessionid]->description, 'pluginfile.php', $this->context->id, 'mod_attendance', 'session', $this->sessioninfo[$sessionid]->id); } return $this->sessioninfo[$sessionid]; } public function get_sessions_info($sessionids) { global $DB; list($sql, $params) = $DB->get_in_or_equal($sessionids); $sessions = $DB->get_records_select('attendance_sessions', "id $sql", $params, 'sessdate asc'); foreach ($sessions as $sess) { if (empty($sess->description)) { $sess->description = get_string('nodescription', 'attendance'); } else { $sess->description = file_rewrite_pluginfile_urls($sess->description, 'pluginfile.php', $this->context->id, 'mod_attendance', 'session', $sess->id); } } return $sessions; } public function get_session_log($sessionid) { global $DB; return $DB->get_records('attendance_log', array('sessionid' => $sessionid), '', 'studentid,statusid,remarks,id'); } public function get_user_stat($userid) { $ret = array(); $ret['completed'] = $this->get_user_taken_sessions_count($userid); $ret['statuses'] = $this->get_user_statuses_stat($userid); return $ret; } public function get_user_taken_sessions_count($userid) { if (!array_key_exists($userid, $this->usertakensesscount)) { if (!empty($this->pageparams->startdate) && !empty($this->pageparams->enddate)) { $this->usertakensesscount[$userid] = att_get_user_taken_sessions_count($this->id, $this->course->startdate, $userid, $this->cm, $this->pageparams->startdate, $this->pageparams->enddate); } else { $this->usertakensesscount[$userid] = att_get_user_taken_sessions_count($this->id, $this->course->startdate, $userid, $this->cm); } } return $this->usertakensesscount[$userid]; } /** * * @global type $DB * @param type $userid * @param type $filters - An array things to filter by. For now only enddate is valid. * @return type */ public function get_user_statuses_stat($userid, array $filters = null) { global $DB; $params = array( 'aid' => $this->id, 'cstartdate' => $this->course->startdate, 'uid' => $userid); $processedfilters = array(); // We test for any valid filters sent. if (isset($filters['enddate'])) { $processedfilters[] = 'ats.sessdate <= :enddate'; $params['enddate'] = $filters['enddate']; } // Make the filter array into a SQL string. if (!empty($processedfilters)) { $processedfilters = ' AND '.implode(' AND ', $processedfilters); } else { $processedfilters = ''; } $period = ''; if (!empty($this->pageparams->startdate) && !empty($this->pageparams->enddate)) { $period = ' AND ats.sessdate >= :sdate AND ats.sessdate < :edate '; $params['sdate'] = $this->pageparams->startdate; $params['edate'] = $this->pageparams->enddate; } if ($this->get_group_mode()) { $qry = "SELECT al.statusid, count(al.statusid) AS stcnt FROM {attendance_log} al JOIN {attendance_sessions} ats ON al.sessionid = ats.id LEFT JOIN {groups_members} gm ON gm.userid = al.studentid AND gm.groupid = ats.groupid WHERE ats.attendanceid = :aid AND ats.sessdate >= :cstartdate AND al.studentid = :uid AND (ats.groupid = 0 or gm.id is NOT NULL)".$period.$processedfilters." GROUP BY al.statusid"; } else { $qry = "SELECT al.statusid, count(al.statusid) AS stcnt FROM {attendance_log} al JOIN {attendance_sessions} ats ON al.sessionid = ats.id WHERE ats.attendanceid = :aid AND ats.sessdate >= :cstartdate AND al.studentid = :uid".$period.$processedfilters." GROUP BY al.statusid"; } // We do not want to cache, or use a cached version of the results when a filter is set. if ($filters !== null) { return $DB->get_records_sql($qry, $params); } else if (!array_key_exists($userid, $this->userstatusesstat)) { // Not filtered so if we do not already have them do the query. $this->userstatusesstat[$userid] = $DB->get_records_sql($qry, $params); } // Return the cached stats. return $this->userstatusesstat[$userid]; } /** * * @param type $userid * @param type $filters - An array things to filter by. For now only enddate is valid. * @return type */ public function get_user_grade($userid, array $filters = null) { return att_get_user_grade($this->get_user_statuses_stat($userid, $filters), $this->get_statuses(true, true)); } // For getting sessions count implemented simplest method - taken sessions. // It can have error if users don't have attendance info for some sessions. // In the future we can implement another methods: // * all sessions between user start enrolment date and now; // * all sessions between user start and end enrolment date. // While implementing those methods we need recalculate grades of all users // on session adding. public function get_user_max_grade($userid) { return att_get_user_max_grade($this->get_user_taken_sessions_count($userid), $this->get_statuses(true, true)); } public function update_users_grade($userids) { $grades = array(); foreach ($userids as $userid) { $grades[$userid] = new stdClass(); $grades[$userid]->userid = $userid; $grades[$userid]->rawgrade = attendance_calc_user_grade_fraction($this->get_user_grade($userid), $this->get_user_max_grade($userid)) * $this->grade; } return grade_update('mod/attendance', $this->course->id, 'mod', 'attendance', $this->id, 0, $grades); } public function get_user_filtered_sessions_log($userid) { global $DB; if ($this->pageparams->startdate && $this->pageparams->enddate) { $where = "ats.attendanceid = :aid AND ats.sessdate >= :csdate AND ats.sessdate >= :sdate AND ats.sessdate < :edate"; } else { $where = "ats.attendanceid = :aid AND ats.sessdate >= :csdate"; } if ($this->get_group_mode()) { $sql = "SELECT ats.id, ats.sessdate, ats.groupid, al.statusid, al.remarks FROM {attendance_sessions} ats JOIN {attendance_log} al ON ats.id = al.sessionid AND al.studentid = :uid LEFT JOIN {groups_members} gm ON gm.userid = al.studentid AND gm.groupid = ats.groupid WHERE $where AND (ats.groupid = 0 or gm.id is NOT NULL) ORDER BY ats.sessdate ASC"; $params = array( 'uid' => $userid, 'aid' => $this->id, 'csdate' => $this->course->startdate, 'sdate' => $this->pageparams->startdate, 'edate' => $this->pageparams->enddate); } else { $sql = "SELECT ats.id, ats.sessdate, ats.groupid, al.statusid, al.remarks FROM {attendance_sessions} ats JOIN {attendance_log} al ON ats.id = al.sessionid AND al.studentid = :uid WHERE $where ORDER BY ats.sessdate ASC"; $params = array( 'uid' => $userid, 'aid' => $this->id, 'csdate' => $this->course->startdate, 'sdate' => $this->pageparams->startdate, 'edate' => $this->pageparams->enddate); } $sessions = $DB->get_records_sql($sql, $params); return $sessions; } public function get_user_filtered_sessions_log_extended($userid) { global $DB; // All taked sessions (including previous groups). if ($this->pageparams->startdate && $this->pageparams->enddate) { $where = "ats.attendanceid = :aid AND ats.sessdate >= :csdate AND ats.sessdate >= :sdate AND ats.sessdate < :edate"; } else { $where = "ats.attendanceid = :aid AND ats.sessdate >= :csdate"; } // We need to add this concatination so that moodle will use it as the array index that is a string. // If the array's index is a number it will not merge entries. // It would be better as a UNION query but unfortunatly MS SQL does not seem to support doing a // DISTINCT on a the description field. $id = $DB->sql_concat(':value', 'ats.id'); if ($this->get_group_mode()) { $sql = "SELECT $id, ats.id, ats.groupid, ats.sessdate, ats.duration, ats.description, al.statusid, al.remarks, ats.studentscanmark FROM {attendance_sessions} ats RIGHT JOIN {attendance_log} al ON ats.id = al.sessionid AND al.studentid = :uid LEFT JOIN {groups_members} gm ON gm.userid = al.studentid AND gm.groupid = ats.groupid WHERE $where AND (ats.groupid = 0 or gm.id is NOT NULL) ORDER BY ats.sessdate ASC"; } else { $sql = "SELECT $id, ats.id, ats.groupid, ats.sessdate, ats.duration, ats.description, al.statusid, al.remarks, ats.studentscanmark FROM {attendance_sessions} ats RIGHT JOIN {attendance_log} al ON ats.id = al.sessionid AND al.studentid = :uid WHERE $where ORDER BY ats.sessdate ASC"; } $params = array( 'uid' => $userid, 'aid' => $this->id, 'csdate' => $this->course->startdate, 'sdate' => $this->pageparams->startdate, 'edate' => $this->pageparams->enddate, 'value' => 'c'); $sessions = $DB->get_records_sql($sql, $params); // All sessions for current groups. $groups = array_keys(groups_get_all_groups($this->course->id, $userid)); $groups[] = 0; list($gsql, $gparams) = $DB->get_in_or_equal($groups, SQL_PARAMS_NAMED, 'gid0'); if ($this->pageparams->startdate && $this->pageparams->enddate) { $where = "ats.attendanceid = :aid AND ats.sessdate >= :csdate AND ats.sessdate >= :sdate AND ats.sessdate < :edate AND ats.groupid $gsql"; } else { $where = "ats.attendanceid = :aid AND ats.sessdate >= :csdate AND ats.groupid $gsql"; } $sql = "SELECT $id, ats.id, ats.groupid, ats.sessdate, ats.duration, ats.description, al.statusid, al.remarks, ats.studentscanmark FROM {attendance_sessions} ats LEFT JOIN {attendance_log} al ON ats.id = al.sessionid AND al.studentid = :uid WHERE $where ORDER BY ats.sessdate ASC"; $params = array_merge($params, $gparams); $sessions = array_merge($sessions, $DB->get_records_sql($sql, $params)); foreach ($sessions as $sess) { if (empty($sess->description)) { $sess->description = get_string('nodescription', 'attendance'); } else { $sess->description = file_rewrite_pluginfile_urls($sess->description, 'pluginfile.php', $this->context->id, 'mod_attendance', 'session', $sess->id); } } return $sessions; } public function delete_sessions($sessionsids) { global $DB; list($sql, $params) = $DB->get_in_or_equal($sessionsids); $DB->delete_records_select('attendance_log', "sessionid $sql", $params); $DB->delete_records_list('attendance_sessions', 'id', $sessionsids); $event = \mod_attendance\event\session_deleted::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => array('info' => implode(', ', $sessionsids)))); $event->add_record_snapshot('course_modules', $this->cm); $event->trigger(); } public function update_sessions_duration($sessionsids, $duration) { global $DB; $now = time(); $sessions = $DB->get_recordset_list('attendance_sessions', 'id', $sessionsids); foreach ($sessions as $sess) { $sess->duration = $duration; $sess->timemodified = $now; $DB->update_record('attendance_sessions', $sess); $event = \mod_attendance\event\session_duration_updated::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => array('info' => implode(', ', $sessionsids)))); $event->add_record_snapshot('course_modules', $this->cm); $event->add_record_snapshot('attendance_sessions', $sess); $event->trigger(); } $sessions->close(); } /** * Remove a status variable from an attendance instance * * @global moodle_database $DB * @param stdClass $status */ public function remove_status($status) { global $DB; $DB->set_field('attendance_statuses', 'deleted', 1, array('id' => $status->id)); $event = \mod_attendance\event\status_removed::create(array( 'objectid' => $status->id, 'context' => $this->context, 'other' => array( 'acronym' => $status->acronym, 'description' => $status->description ))); $event->add_record_snapshot('course_modules', $this->cm); $event->add_record_snapshot('attendance_statuses', $status); $event->trigger(); } /** * Add an attendance status variable * * @global moodle_database $DB * @param string $acronym * @param string $description * @param int $grade */ public function add_status($acronym, $description, $grade) { global $DB; if ($acronym && $description) { $rec = new stdClass(); $rec->courseid = $this->course->id; $rec->attendanceid = $this->id; $rec->acronym = $acronym; $rec->description = $description; $rec->grade = $grade; $rec->setnumber = $this->pageparams->statusset; // Save which set it is part of. $rec->deleted = 0; $rec->visible = 1; $id = $DB->insert_record('attendance_statuses', $rec); $rec->id = $id; $event = \mod_attendance\event\status_added::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => array('acronym' => $acronym, 'description' => $description, 'grade' => $grade))); $event->add_record_snapshot('course_modules', $this->cm); $event->add_record_snapshot('attendance_statuses', $rec); $event->trigger(); } else { print_error('cantaddstatus', 'attendance', $this->url_preferences()); } } /** * Update status variable for a particular Attendance module instance * * @global moodle_database $DB * @param stdClass $status * @param string $acronym * @param string $description * @param int $grade * @param bool $visible */ public function update_status($status, $acronym, $description, $grade, $visible) { global $DB; if (isset($visible)) { $status->visible = $visible; $updated[] = $visible ? get_string('show') : get_string('hide'); } else if (empty($acronym) || empty($description)) { return array('acronym' => $acronym, 'description' => $description); } $updated = array(); if ($acronym) { $status->acronym = $acronym; $updated[] = $acronym; } if ($description) { $status->description = $description; $updated[] = $description; } if (isset($grade)) { $status->grade = $grade; $updated[] = $grade; } $DB->update_record('attendance_statuses', $status); $event = \mod_attendance\event\status_updated::create(array( 'objectid' => $this->id, 'context' => $this->context, 'other' => array('acronym' => $acronym, 'description' => $description, 'grade' => $grade, 'updated' => implode(' ', $updated)))); $event->add_record_snapshot('course_modules', $this->cm); $event->add_record_snapshot('attendance_statuses', $status); $event->trigger(); } /** * Check if the email address is already in use by either another temporary user, * or a real user. * * @param string $email the address to check for * @param int $tempuserid optional the ID of the temporary user (to avoid matching against themself) * @return null|string the error message to display, null if there is no error */ public static function check_existing_email($email, $tempuserid = 0) { global $DB; if (empty($email)) { return null; // Fine to create temporary users without an email address. } if ($tempuser = $DB->get_record('attendance_tempusers', array('email' => $email), 'id')) { if ($tempuser->id != $tempuserid) { return get_string('tempexists', 'attendance'); } } if ($DB->record_exists('user', array('email' => $email))) { return get_string('userexists', 'attendance'); } return null; } } function att_get_statuses($attid, $onlyvisible=true, $statusset = -1) { global $DB; // Set selector. $params = array('aid' => $attid); $setsql = ''; if ($statusset >= 0) { $params['statusset'] = $statusset; $setsql = ' AND setnumber = :statusset '; } if ($onlyvisible) { $statuses = $DB->get_records_select('attendance_statuses', "attendanceid = :aid AND visible = 1 AND deleted = 0 $setsql", $params, 'setnumber ASC, grade DESC'); } else { $statuses = $DB->get_records_select('attendance_statuses', "attendanceid = :aid AND deleted = 0 $setsql", $params, 'setnumber ASC, grade DESC'); } return $statuses; } /** * Get the name of the status set. * * @param int $attid * @param int $statusset * @param bool $includevalues * @return string */ function att_get_setname($attid, $statusset, $includevalues = true) { $statusname = get_string('statusset', 'mod_attendance', $statusset + 1); if ($includevalues) { $statuses = att_get_statuses($attid, true, $statusset); $statusesout = array(); foreach ($statuses as $status) { $statusesout[] = $status->acronym; } if ($statusesout) { if (count($statusesout) > 6) { $statusesout = array_slice($statusesout, 0, 6); $statusesout[] = '&helip;'; } $statusesout = implode(' ', $statusesout); $statusname .= ' ('.$statusesout.')'; } } return $statusname; } function att_get_user_taken_sessions_count($attid, $coursestartdate, $userid, $coursemodule, $startdate = '', $enddate = '') { global $DB, $COURSE; $groupmode = groups_get_activity_groupmode($coursemodule, $COURSE); if (!empty($groupmode)) { $qry = "SELECT count(*) as cnt FROM {attendance_log} al JOIN {attendance_sessions} ats ON al.sessionid = ats.id LEFT JOIN {groups_members} gm ON gm.userid = al.studentid AND gm.groupid = ats.groupid WHERE ats.attendanceid = :aid AND ats.sessdate >= :cstartdate AND al.studentid = :uid AND (ats.groupid = 0 or gm.id is NOT NULL)"; } else { $qry = "SELECT count(*) as cnt FROM {attendance_log} al JOIN {attendance_sessions} ats ON al.sessionid = ats.id WHERE ats.attendanceid = :aid AND ats.sessdate >= :cstartdate AND al.studentid = :uid"; } $params = array( 'aid' => $attid, 'cstartdate' => $coursestartdate, 'uid' => $userid); if (!empty($startdate) && !empty($enddate)) { $qry .= ' AND sessdate >= :sdate AND sessdate < :edate '; $params['sdate'] = $startdate; $params['edate'] = $enddate; } return $DB->count_records_sql($qry, $params); } function att_get_user_statuses_stat($attid, $coursestartdate, $userid, $coursemodule) { global $DB, $COURSE; $groupmode = groups_get_activity_groupmode($coursemodule, $COURSE); if (!empty($groupmode)) { $qry = "SELECT al.statusid, count(al.statusid) AS stcnt FROM {attendance_log} al JOIN {attendance_sessions} ats ON al.sessionid = ats.id LEFT JOIN {groups_members} gm ON gm.userid = al.studentid AND gm.groupid = ats.groupid WHERE ats.attendanceid = :aid AND ats.sessdate >= :cstartdate AND al.studentid = :uid AND (ats.groupid = 0 or gm.id is NOT NULL) GROUP BY al.statusid"; } else { $qry = "SELECT al.statusid, count(al.statusid) AS stcnt FROM {attendance_log} al JOIN {attendance_sessions} ats ON al.sessionid = ats.id WHERE ats.attendanceid = :aid AND ats.sessdate >= :cstartdate AND al.studentid = :uid GROUP BY al.statusid"; } $params = array( 'aid' => $attid, 'cstartdate' => $coursestartdate, 'uid' => $userid); return $DB->get_records_sql($qry, $params); } function att_get_user_grade($userstatusesstat, $statuses) { $sum = 0; foreach ($userstatusesstat as $stat) { $sum += $stat->stcnt * $statuses[$stat->statusid]->grade; } return $sum; } function att_get_user_max_grade($sesscount, $statuses) { reset($statuses); return current($statuses)->grade * $sesscount; } function att_get_user_courses_attendances($userid) { global $DB; $usercourses = enrol_get_users_courses($userid); list($usql, $uparams) = $DB->get_in_or_equal(array_keys($usercourses), SQL_PARAMS_NAMED, 'cid0'); $sql = "SELECT att.id as attid, att.course as courseid, course.fullname as coursefullname, course.startdate as coursestartdate, att.name as attname, att.grade as attgrade FROM {attendance} att JOIN {course} course ON att.course = course.id WHERE att.course $usql ORDER BY coursefullname ASC, attname ASC"; $params = array_merge($uparams, array('uid' => $userid)); return $DB->get_records_sql($sql, $params); } /** * Used to caclulate usergrade based on rawgrade and max grade. * * @param $grade - raw grade for user * @param $maxgrade - maxgrade for this session. */ function attendance_calc_user_grade_fraction($grade, $maxgrade) { if ($maxgrade == 0) { return 0; } else { return $grade / $maxgrade; } } /** * Update all user grades - used when settings have changed. * * @param $attid * @param $course * @param $context * @param $coursemodule */ function attendance_update_all_users_grades($attid, $course, $context, $coursemodule) { global $DB; $grades = array(); $userids = array_keys(get_enrolled_users($context, 'mod/attendance:canbelisted', 0, 'u.id')); $attgrades = grade_get_grades($course->id, 'mod', 'attendance', $attid, $userids); $usergrades = []; if (!empty($attgrades->items[0]) and !empty($attgrades->items[0]->grades)) { $usergrades = $attgrades->items[0]->grades; } $statuses = att_get_statuses($attid); // TODO: we should be able to pass full $attendance record into this function and avoid this db call. $gradebookmaxgrade = $DB->get_field('attendance', 'grade', array('id' => $attid)); foreach ($usergrades as $userid => $existinggrade) { if (is_null($existinggrade->grade)) { // Don't update grades where one doesn't exist yet. continue; } $grade = new stdClass; $grade->userid = $userid; $userstatusesstat = att_get_user_statuses_stat($attid, $course->startdate, $userid, $coursemodule); $usertakensesscount = att_get_user_taken_sessions_count($attid, $course->startdate, $userid, $coursemodule); $usergrade = att_get_user_grade($userstatusesstat, $statuses); $usermaxgrade = att_get_user_max_grade($usertakensesscount, $statuses); $grade->rawgrade = attendance_calc_user_grade_fraction($usergrade, $usermaxgrade) * $gradebookmaxgrade; $grades[$userid] = $grade; } if (!empty($grades)) { $result = grade_update('mod/attendance', $course->id, 'mod', 'attendance', $attid, 0, $grades); } else { $result = true; } return $result; } /** * Check to see if statusid in use to help prevent deletion etc. * * @param integer $statusid */ function attendance_has_logs_for_status($statusid) { global $DB; return $DB->record_exists('attendance_log', array('statusid' => $statusid)); } /** * Helper function to add sessiondate_selector to add/update forms. * * @param MoodleQuickForm $mform */ function attendance_form_sessiondate_selector (MoodleQuickForm $mform) { $mform->addElement('date_selector', 'sessiondate', get_string('sessiondate', 'attendance')); for ($i = 0; $i <= 23; $i++) { $hours[$i] = sprintf("%02d", $i); } for ($i = 0; $i < 60; $i += 5) { $minutes[$i] = sprintf("%02d", $i); } $sesendtime = array(); $sesendtime[] =& $mform->createElement('static', 'from', '', get_string('from', 'attendance')); $sesendtime[] =& $mform->createElement('select', 'starthour', get_string('hour', 'form'), $hours, false, true); $sesendtime[] =& $mform->createElement('select', 'startminute', get_string('minute', 'form'), $minutes, false, true); $sesendtime[] =& $mform->createElement('static', 'to', '', get_string('to', 'attendance')); $sesendtime[] =& $mform->createElement('select', 'endhour', get_string('hour', 'form'), $hours, false, true); $sesendtime[] =& $mform->createElement('select', 'endminute', get_string('minute', 'form'), $minutes, false, true); $mform->addGroup($sesendtime, 'sestime', get_string('time', 'attendance'), array(' '), true); }