. /** * Class definition for mod_attendance_structure * * @package mod_attendance * @copyright 2016 Dan Marsden http://danmarsden.com * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ defined('MOODLE_INTERNAL') || die(); global $CFG; // This class is included inside existing functions. require_once(dirname(__FILE__) . '/calendar_helpers.php'); require_once($CFG->libdir .'/filelib.php'); /** * Main class with all Attendance related info. * * @copyright 2016 Dan Marsden http://danmarsden.com * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class mod_attendance_structure { /** Common sessions */ const SESSION_COMMON = 0; /** Group sessions */ const SESSION_GROUP = 1; /** @var stdclass course module record */ public $cm; /** @var int cmid - needed for calendar internal tests (see Issue #473) */ public $cmid; /** @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; /** @var int last time attendance was modified - used for global search */ public $timemodified; /** @var string required field for activity modules and searching */ public $intro; /** @var int format of the intro (see above) */ public $introformat; /** @var array current page parameters */ public $pageparams; /** @var string subnets (IP range) for student self selection. */ public $subnet; /** @var string subnets (IP range) for student self selection. */ public $automark; /** @var boolean flag set when automarking is complete. */ public $automarkcompleted; /** @var int Define if extra user details should be shown in reports */ public $showextrauserdetails; /** @var int Define if session details should be shown in reports */ public $showsessiondetails; /** @var int Position for the session detail columns related to summary columns.*/ public $sessiondetailspos; /** @var int groupmode */ private $groupmode; /** @var array */ private $statuses; /** @var array Cache list of all statuses (not just one used by current session). */ private $allstatuses; /** @var array of sessionid. */ private $sessioninfo = array(); /** @var float number [0..1], the threshold for student to be shown at low grade report */ private $lowgradethreshold; /** * 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 * @param stdClass $pageparams */ public function __construct(stdClass $dbrecord, stdClass $cm, stdClass $course, stdClass $context=null, $pageparams=null) { global $DB; foreach ($dbrecord as $field => $value) { if (property_exists('mod_attendance_structure', $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; if (empty($this->cmid)) { $this->cmid = $cm->id; } $this->course = $course; if (is_null($context)) { $this->context = context_module::instance($this->cm->id); } else { $this->context = $context; } $this->pageparams = $pageparams; if (isset($pageparams->showextrauserdetails) && $pageparams->showextrauserdetails != $this->showextrauserdetails) { $DB->set_field('attendance', 'showextrauserdetails', $pageparams->showextrauserdetails, array('id' => $this->id)); } if (isset($pageparams->showsessiondetails) && $pageparams->showsessiondetails != $this->showsessiondetails) { $DB->set_field('attendance', 'showsessiondetails', $pageparams->showsessiondetails, array('id' => $this->id)); } if (isset($pageparams->sessiondetailspos) && $pageparams->sessiondetailspos != $this->sessiondetailspos) { $DB->set_field('attendance', 'sessiondetailspos', $pageparams->sessiondetailspos, array('id' => $this->id)); } } /** * Get group mode. * * @return int */ public function get_group_mode() : int { 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() : array { 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() : array { 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} * @param stdClass $sess * @return array of records or an empty array */ public function get_today_sessions_for_copy($sess) : array { 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 int count of hidden sessions */ public function get_hidden_sessions_count() : int { 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 array hidden sessions */ public function get_hidden_sessions() : array { 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); } /** * Get filtered sessions. * * @return array */ public function get_filtered_sessions() : array { 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() > mod_attendance_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'); $statussetmaxpoints = attendance_get_statusset_maxpoints($this->get_statuses(true, true)); 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); } $sess->maxpoints = $statussetmaxpoints[$sess->statusset]; } return $sessions; } /** * Get manage url. * @param array $params * @return moodle_url of manage.php for attendance instance */ public function url_manage($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/manage.php', $params); } /** * Get manage temp users url. * @param array $params optional * @return moodle_url of tempusers.php for attendance instance */ public function url_managetemp($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/tempusers.php', $params); } /** * Get temp delete url. * * @param array $params optional * @return moodle_url of tempdelete.php for attendance instance */ public function url_tempdelete($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id, 'action' => 'delete'), $params); return new moodle_url('/mod/attendance/tempedit.php', $params); } /** * Get temp edit url. * * @param array $params optional * @return moodle_url of tempedit.php for attendance instance */ public function url_tempedit($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/tempedit.php', $params); } /** * Get temp merge url * * @param array $params optional * @return moodle_url of tempedit.php for attendance instance */ public function url_tempmerge($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/tempmerge.php', $params); } /** * Get url for sessions. * @param array $params * @return moodle_url of sessions.php for attendance instance */ public function url_sessions($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/sessions.php', $params); } /** * Get url for report. * @param array $params * @return moodle_url of report.php for attendance instance */ public function url_report($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/report.php', $params); } /** * Get url for report. * @param array $params * @return moodle_url of report.php for attendance instance */ public function url_absentee($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/absentee.php', $params); } /** * Get url for export. * * @return moodle_url of export.php for attendance instance */ public function url_export() : moodle_url { $params = array('id' => $this->cm->id); return new moodle_url('/mod/attendance/export.php', $params); } /** * Get preferences url * @param array $params * @return moodle_url of attsettings.php for attendance instance */ public function url_preferences($params=array()) : moodle_url { // 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); } /** * Get preferences url * @param array $params * @return moodle_url of attsettings.php for attendance instance */ public function url_warnings($params=array()) : moodle_url { // 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/warnings.php', $params); } /** * Get take url. * @param array $params * @return moodle_url of attendances.php for attendance instance */ public function url_take($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/take.php', $params); } /** * Get view url. * @param array $params * @return moodle_url */ public function url_view($params=array()) : moodle_url { $params = array_merge(array('id' => $this->cm->id), $params); return new moodle_url('/mod/attendance/view.php', $params); } /** * Add sessions. * * @param array $sessions */ public function add_sessions($sessions) { foreach ($sessions as $sess) { $this->add_session($sess); } } /** * Add single session. * * @param stdClass $sess * @return int $sessionid */ public function add_session($sess) : int { global $DB; $config = get_config('attendance'); $sess->attendanceid = $this->id; $sess->automarkcompleted = 0; if (!isset($sess->automark)) { $sess->automark = 0; } if (empty($config->enablecalendar)) { // If calendard disabled at site level, don't use it. $sess->calendarevent = 0; } $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)); $sess->caleventid = 0; attendance_create_calendar_event($sess); $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; if (!isset($sess->studentscanmark)) { $sess->studentscanmark = 0; } if (!isset($sess->autoassignstatus)) { $sess->autoassignstatus = 0; } if (!isset($sess->studentpassword)) { $sess->studentpassword = ''; } if (!isset($sess->subnet)) { $sess->subnet = ''; } if (!isset($sess->preventsharedip)) { $sess->preventsharedip = 0; } if (!isset($sess->preventsharediptime)) { $sess->preventsharediptime = ''; } if (!isset($sess->includeqrcode)) { $sess->includeqrcode = 0; } if (!isset($sess->rotateqrcode)) { $sess->rotateqrcode = 0; $sess->rotateqrcodesecret = ''; } $event->add_record_snapshot('attendance_sessions', $sess); $event->trigger(); return $sess->id; } /** * Update session from form. * * @param stdClass $formdata * @param int $sessionid */ 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->calendarevent = empty($formdata->calendarevent) ? 0 : $formdata->calendarevent; $sess->studentscanmark = 0; $sess->autoassignstatus = 0; $sess->studentpassword = ''; $sess->subnet = ''; $sess->automark = 0; $sess->automarkcompleted = 0; $sess->preventsharedip = 0; $sess->preventsharediptime = ''; $sess->includeqrcode = 0; $sess->rotateqrcode = 0; $sess->rotateqrcodesecret = ''; if (!empty(get_config('attendance', 'enablewarnings'))) { $sess->absenteereport = empty($formdata->absenteereport) ? 0 : 1; } if (!empty($formdata->autoassignstatus)) { $sess->autoassignstatus = $formdata->autoassignstatus; } $studentscanmark = get_config('attendance', 'studentscanmark'); if (!empty($studentscanmark) && !empty($formdata->studentscanmark)) { $sess->studentscanmark = $formdata->studentscanmark; $sess->studentpassword = $formdata->studentpassword; $sess->autoassignstatus = $formdata->autoassignstatus; if (!empty($formdata->includeqrcode)) { $sess->includeqrcode = $formdata->includeqrcode; } if (!empty($formdata->rotateqrcode)) { $sess->rotateqrcode = $formdata->rotateqrcode; $sess->studentpassword = attendance_random_string(); $sess->rotateqrcodesecret = attendance_random_string(); } } if (!empty($formdata->usedefaultsubnet)) { $sess->subnet = $this->subnet; } else { $sess->subnet = $formdata->subnet; } if (!empty($formdata->automark)) { $sess->automark = $formdata->automark; } if (!empty($formdata->preventsharedip)) { $sess->preventsharedip = $formdata->preventsharedip; } if (!empty($formdata->preventsharediptime)) { $sess->preventsharediptime = $formdata->preventsharediptime; } $sess->timemodified = time(); $DB->update_record('attendance_sessions', $sess); if (empty($sess->caleventid)) { // This shouldn't really happen, but just in case to prevent fatal error. attendance_create_calendar_event($sess); } else { attendance_update_calendar_event($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' => mod_attendance_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. * * @param stdClass $mformdata * @return boolean */ public function take_from_student($mformdata) : bool { 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; $record->ipaddress = getremoteaddr(null); $existingattendance = $DB->record_exists('attendance_log', array('sessionid' => $mformdata->sessid, 'studentid' => $USER->id)); if ($existingattendance) { // 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; } /** * Take attendance from form data. * * @param stdClass $formdata */ 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. // This whole function could do with a nice clean up. $statuses = implode(',', array_keys( (array)$this->get_statuses() )); $now = time(); $sesslog = array(); $formdata = (array)$formdata; foreach ($formdata as $key => $value) { // Look at Remarks field because the user options may not be passed if empty. if (substr($key, 0, 7) == 'remarks') { $sid = substr($key, 7); if (!(is_numeric($sid))) { // Sanity check on $sid. print_error('nonnumericid', 'attendance'); } $sesslog[$sid] = new stdClass(); $sesslog[$sid]->studentid = $sid; // We check is_numeric on this above. if (array_key_exists('user'.$sid, $formdata) && is_numeric($formdata['user' . $sid])) { $sesslog[$sid]->statusid = $formdata['user' . $sid]; } $sesslog[$sid]->statusset = $statuses; $sesslog[$sid]->remarks = $value; $sesslog[$sid]->sessionid = $this->pageparams->sessionid; $sesslog[$sid]->timetaken = $now; $sesslog[$sid]->takenby = $USER->id; } } // Get existing session log. $dbsesslog = $this->get_session_log($this->pageparams->sessionid); foreach ($sesslog as $log) { // Don't save a record if no statusid or remark. if (!empty($log->statusid) || !empty($log->remarks)) { if (array_key_exists($log->studentid, $dbsesslog)) { // Check if anything important has changed before updating record. // Don't update timetaken/takenby records if nothing has changed. if ($dbsesslog[$log->studentid]->remarks <> $log->remarks || $dbsesslog[$log->studentid]->statusid <> $log->statusid || $dbsesslog[$log->studentid]->statusset <> $log->statusset) { $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(); } /** * Get users with enrolment status (Feature request MDL-27591) * * @param int $groupid * @param int $page * @return array */ public function get_users($groupid = 0, $page = 1) : array { global $DB; $fields = array('username' , 'idnumber' , 'institution' , 'department'); // Get user identity fields if required - doesn't return original $fields array. $extrafields = get_extra_user_fields($this->context, $fields); $fields = array_merge($fields, $extrafields); $userfields = user_picture::fields('u', $fields); if (empty($this->pageparams->sort)) { $this->pageparams->sort = ATT_SORT_DEFAULT; } if ($this->pageparams->sort == ATT_SORT_FIRSTNAME) { $orderby = $DB->sql_fullname('u.firstname', 'u.lastname') . ', u.id'; } else if ($this->pageparams->sort == ATT_SORT_LASTNAME) { $orderby = 'u.lastname, u.firstname, u.id'; } else { list($orderby, $sortparams) = users_order_by_sql('u'); } if ($page) { $usersperpage = $this->pageparams->perpage; if (!empty($this->cm->groupingid)) { $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, $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($this->cm->groupingid)) { 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, $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]->fullname = fullname($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[$tempuser->studentid] = self::tempuser_to_user($tempuser); } return $users; } /** * Convert a tempuser record into a user object. * * @param stdClass $tempuser * @return object */ protected static function tempuser_to_user($tempuser) { global $CFG; $ret = (object)array( 'id' => $tempuser->studentid, 'firstname' => $tempuser->fullname, 'email' => $tempuser->email, 'username' => '', 'enrolmentstatus' => 0, 'enrolmentstart' => 0, 'enrolmentend' => 0, 'picture' => 0, 'type' => 'temporary', ); $allfields = get_all_user_name_fields(); if (!empty($CFG->showuseridentity)) { $allfields = array_merge($allfields, explode(',', $CFG->showuseridentity)); } foreach ($allfields as $namefield) { if (!isset($ret->$namefield)) { $ret->$namefield = ''; } } return $ret; } /** * Get user and include extra info. * * @param int $userid * @return mixed|object */ 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); if (!empty($enrolments)) { $user->enrolmentstatus = $enrolments->status; $user->enrolmentstart = $enrolments->mintime; $user->enrolmentend = $enrolments->maxtime; } else { $user->enrolmentstatus = ''; $user->enrolmentstart = 0; $user->enrolmentend = 0; } return $user; } /** * Get possible statuses. * * @param bool $onlyvisible * @param bool $allsets * @return array */ public function get_statuses($onlyvisible = true, $allsets = false) : array { 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 = attendance_get_statuses($this->id, $onlyvisible, $statusset); $this->allstatuses = attendance_get_statuses($this->id, $onlyvisible); } // Return all sets, if requested. if ($allsets) { return $this->allstatuses; } return $this->statuses; } /** * Get session info. * @param int $sessionid * @return mixed */ 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]; } /** * Get sessions info * * @param array $sessionids * @return array */ public function get_sessions_info($sessionids) : array { 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; } /** * Get log. * * @param int $sessionid * @return array */ public function get_session_log($sessionid) : array { global $DB; return $DB->get_records('attendance_log', array('sessionid' => $sessionid), '', 'studentid,statusid,remarks,id,statusset'); } /** * Update user grade. * @param array $userids */ public function update_users_grade($userids) { attendance_update_users_grade($this, $userids); } /** * Get filtered log. * @param int $userid * @return array */ public function get_user_filtered_sessions_log($userid) : array { 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, ats.preventsharediptime, ats.preventsharedip 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, ats.preventsharediptime, ats.preventsharedip 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; } /** * Get filtered log extended. * @param int $userid * @return array */ public function get_user_filtered_sessions_log_extended($userid) : array { 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, ats.autoassignstatus, ats.preventsharedip, ats.preventsharediptime, ats.rotateqrcode 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, ats.statusset, al.statusid, al.remarks, ats.studentscanmark, ats.autoassignstatus, ats.preventsharedip, ats.preventsharediptime, ats.rotateqrcode 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, ats.statusset, al.statusid, al.remarks, ats.studentscanmark, ats.autoassignstatus, ats.preventsharedip, ats.preventsharediptime, ats.rotateqrcode 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; } /** * Delete sessions. * @param array $sessionsids */ public function delete_sessions($sessionsids) { global $DB; if (attendance_existing_calendar_events_ids($sessionsids)) { attendance_delete_calendar_events($sessionsids); } 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(); } /** * Update duration. * * @param array $sessionsids * @param int $duration */ 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); if ($sess->caleventid) { attendance_update_calendar_event($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(); } /** * 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; } /** * Gets the status to use when auto-marking. * * @param int $time the time the user first accessed the course. * @param int $sessionid the related sessionid to check. * @return int the statusid to assign to this user. */ public function get_automark_status($time, $sessionid) { $statuses = $this->get_statuses(); // Statuses are returned highest grade first, find the first high grade we can assign to this user. // Get status to use when unmarked. $session = $this->sessioninfo[$sessionid]; $duration = $session->duration; if (empty($duration)) { $duration = get_config('attendance', 'studentscanmarksessiontimeend') * 60; } if ($time > $session->sessdate + $duration) { // This session closed after the users access - use the unmarked state. foreach ($statuses as $status) { if (!empty($status->setunmarked)) { return $status->id; } } } else { foreach ($statuses as $status) { if ($status->studentavailability !== '0' && $this->sessioninfo[$sessionid]->sessdate + ($status->studentavailability * 60) > $time) { // Found first status we could set. return $status->id; } } } return; } /** * Gets the lowgrade threshold to use. * */ public function get_lowgrade_threshold() { if (!isset($this->lowgradethreshold)) { $this->lowgradethreshold = 1; if ($this->grade > 0) { $gradeitem = grade_item::fetch(array('courseid' => $this->course->id, 'itemtype' => 'mod', 'itemmodule' => 'attendance', 'iteminstance' => $this->id)); if ($gradeitem->gradepass > 0 && $gradeitem->grademax != $gradeitem->grademin) { $this->lowgradethreshold = ($gradeitem->gradepass - $gradeitem->grademin) / ($gradeitem->grademax - $gradeitem->grademin); } } } return $this->lowgradethreshold; } }