|
|
|
<?php
|
|
|
|
// This file is part of Moodle - http://moodle.org/
|
|
|
|
//
|
|
|
|
// Moodle is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// Moodle is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
/**
|
|
|
|
* local functions and constants for module attendance
|
|
|
|
*
|
|
|
|
* @package mod_attendance
|
|
|
|
* @copyright 2011 Artem Andreev <andreev.artem@gmail.com>
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
|
*/
|
|
|
|
|
|
|
|
defined('MOODLE_INTERNAL') || die();
|
|
|
|
|
|
|
|
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_VIEW_SUMMARY', 7);
|
|
|
|
|
|
|
|
define('ATT_SORT_DEFAULT', 0);
|
|
|
|
define('ATT_SORT_LASTNAME', 1);
|
|
|
|
define('ATT_SORT_FIRSTNAME', 2);
|
|
|
|
|
|
|
|
define('ATTENDANCE_AUTOMARK_DISABLED', 0);
|
|
|
|
define('ATTENDANCE_AUTOMARK_ALL', 1);
|
|
|
|
define('ATTENDANCE_AUTOMARK_CLOSE', 2);
|
|
|
|
|
|
|
|
define('ATTENDANCE_SHAREDIP_DISABLED', 0);
|
|
|
|
define('ATTENDANCE_SHAREDIP_MINUTES', 1);
|
|
|
|
define('ATTENDANCE_SHAREDIP_FORCE', 2);
|
|
|
|
|
|
|
|
// Max number of sessions available in the warnings set form to trigger warnings.
|
|
|
|
define('ATTENDANCE_MAXWARNAFTER', 100);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get statuses,
|
|
|
|
*
|
|
|
|
* @param int $attid
|
|
|
|
* @param bool $onlyvisible
|
|
|
|
* @param int $statusset
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function attendance_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 attendance_get_setname($attid, $statusset, $includevalues = true) {
|
|
|
|
$statusname = get_string('statusset', 'mod_attendance', $statusset + 1);
|
|
|
|
if ($includevalues) {
|
|
|
|
$statuses = attendance_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[] = '...';
|
|
|
|
}
|
|
|
|
$statusesout = implode(' ', $statusesout);
|
|
|
|
$statusname .= ' ('.$statusesout.')';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $statusname;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get full filtered log.
|
|
|
|
* @param int $userid
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function get_user_sessions_log_full($userid, $pageparams) {
|
|
|
|
global $DB;
|
|
|
|
// All taken sessions (including previous groups).
|
|
|
|
|
|
|
|
$usercourses = enrol_get_users_courses($userid);
|
|
|
|
list($usql, $uparams) = $DB->get_in_or_equal(array_keys($usercourses), SQL_PARAMS_NAMED, 'cid0');
|
|
|
|
|
|
|
|
// If we base on structure->get_user_filtered_sessions_log_extended:
|
|
|
|
/* if ($this->pageparams->startdate && $this->pageparams->enddate) { */
|
|
|
|
/* $where = "ats.sessdate >= :csdate AND */
|
|
|
|
/* ats.sessdate >= :sdate AND ats.sessdate < :edate"; */
|
|
|
|
/* } else { */
|
|
|
|
/* $where = "ats.attendanceid = :aid AND ats.sessdate >= :csdate"; */
|
|
|
|
/* } */
|
|
|
|
|
|
|
|
$coursesql = "(1 = 1)";
|
|
|
|
$courseparams = array();
|
|
|
|
$now = time();
|
|
|
|
if ($pageparams->sesscourses === 'current') {
|
|
|
|
$coursesql = "(c.startdate = 0 OR c.startdate <= :now1) AND (c.enddate = 0 OR c.enddate >= :now2)";
|
|
|
|
$courseparams = array(
|
|
|
|
'now1' => $now,
|
|
|
|
'now2' => $now,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$datesql = "(1 = 1)";
|
|
|
|
$dateparams = array();
|
|
|
|
if ($pageparams->startdate && $pageparams->enddate) {
|
|
|
|
$datesql = "ats.sessdate >= :sdate AND ats.sessdate < :edate";
|
|
|
|
$dateparams = array(
|
|
|
|
'sdate' => $pageparams->startdate,
|
|
|
|
'edate' => $pageparams->enddate,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($pageparams->groupby === 'date') {
|
|
|
|
$ordersql = "ats.sessdate ASC, c.fullname ASC, att.name ASC, att.id ASC";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$ordersql = "c.fullname ASC, att.name ASC, att.id ASC, ats.sessdate ASC";
|
|
|
|
}
|
|
|
|
|
|
|
|
// WHERE clause is important:
|
|
|
|
// gm.userid not null => get unmarked attendances for user's current groups
|
|
|
|
// ats.groupid 0 => get all sessions that are for all students enrolled in course
|
|
|
|
// al.id not null => get all marked sessions whether or not user currently still in group
|
|
|
|
//
|
|
|
|
$sql = "SELECT 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.attendanceid, att.name AS attname, att.course AS courseid, c.fullname AS cname
|
|
|
|
FROM {attendance_sessions} ats
|
|
|
|
JOIN {attendance} att
|
|
|
|
ON att.id = ats.attendanceid
|
|
|
|
JOIN {course} c
|
|
|
|
ON att.course = c.id
|
|
|
|
LEFT JOIN {attendance_log} al
|
|
|
|
ON ats.id = al.sessionid AND al.studentid = :uid
|
|
|
|
LEFT JOIN {groups_members} gm
|
|
|
|
ON (ats.groupid = gm.groupid AND gm.userid = :uid1)
|
|
|
|
WHERE (gm.userid IS NOT NULL OR ats.groupid = 0 OR al.id IS NOT NULL)
|
|
|
|
AND att.course $usql
|
|
|
|
AND $datesql
|
|
|
|
AND $coursesql
|
|
|
|
ORDER BY $ordersql";
|
|
|
|
|
|
|
|
$params = array(
|
|
|
|
'uid' => $userid,
|
|
|
|
'uid1' => $userid,
|
|
|
|
);
|
|
|
|
$params = array_merge($params, $uparams);
|
|
|
|
$params = array_merge($params, $dateparams);
|
|
|
|
$params = array_merge($params, $courseparams);
|
|
|
|
$sessions = $DB->get_records_sql($sql, $params);
|
|
|
|
|
|
|
|
foreach ($sessions as $sess) {
|
|
|
|
if (empty($sess->description)) {
|
|
|
|
$sess->description = get_string('nodescription', 'attendance');
|
|
|
|
} else {
|
|
|
|
$modinfo = get_fast_modinfo($sess->courseid);
|
|
|
|
$cmid = $modinfo->instances['attendance'][$sess->attendanceid]->get_course_module_record()->id;
|
|
|
|
$ctx = context_module::instance($cmid);
|
|
|
|
$sess->description = file_rewrite_pluginfile_urls($sess->description,
|
|
|
|
'pluginfile.php', $ctx->id, 'mod_attendance', 'session', $sess->id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sessions;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get users courses and the relevant attendances.
|
|
|
|
*
|
|
|
|
* @param int $userid
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function attendance_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 calculate a fraction based on the part and total values
|
|
|
|
*
|
|
|
|
* @param float $part - part of the total value
|
|
|
|
* @param float $total - total value.
|
|
|
|
* @return float the calculated fraction.
|
|
|
|
*/
|
|
|
|
function attendance_calc_fraction($part, $total) {
|
|
|
|
if ($total == 0) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return $part / $total;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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();
|
|
|
|
if (!right_to_left()) {
|
|
|
|
$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);
|
|
|
|
} else {
|
|
|
|
$sesendtime[] =& $mform->createElement('static', 'from', '', get_string('from', 'attendance'));
|
|
|
|
$sesendtime[] =& $mform->createElement('select', 'startminute', get_string('minute', 'form'), $minutes, false, true);
|
|
|
|
$sesendtime[] =& $mform->createElement('select', 'starthour', get_string('hour', 'form'), $hours, false, true);
|
|
|
|
$sesendtime[] =& $mform->createElement('static', 'to', '', get_string('to', 'attendance'));
|
|
|
|
$sesendtime[] =& $mform->createElement('select', 'endminute', get_string('minute', 'form'), $minutes, false, true);
|
|
|
|
$sesendtime[] =& $mform->createElement('select', 'endhour', get_string('hour', 'form'), $hours, false, true);
|
|
|
|
}
|
|
|
|
$mform->addGroup($sesendtime, 'sestime', get_string('time', 'attendance'), array(' '), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Count the number of status sets that exist for this instance.
|
|
|
|
*
|
|
|
|
* @param int $attendanceid
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
function attendance_get_max_statusset($attendanceid) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$max = $DB->get_field_sql('SELECT MAX(setnumber) FROM {attendance_statuses} WHERE attendanceid = ? AND deleted = 0',
|
|
|
|
array($attendanceid));
|
|
|
|
if ($max) {
|
|
|
|
return $max;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the maxpoints for each statusset
|
|
|
|
*
|
|
|
|
* @param array $statuses
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function attendance_get_statusset_maxpoints($statuses) {
|
|
|
|
$statussetmaxpoints = array();
|
|
|
|
foreach ($statuses as $st) {
|
|
|
|
if (!isset($statussetmaxpoints[$st->setnumber])) {
|
|
|
|
$statussetmaxpoints[$st->setnumber] = $st->grade;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $statussetmaxpoints;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update user grades
|
|
|
|
*
|
|
|
|
* @param mod_attendance_structure|stdClass $attendance
|
|
|
|
* @param array $userids
|
|
|
|
*/
|
|
|
|
function attendance_update_users_grade($attendance, $userids=array()) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if (empty($attendance->grade)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
list($course, $cm) = get_course_and_cm_from_instance($attendance->id, 'attendance');
|
|
|
|
|
|
|
|
$summary = new mod_attendance_summary($attendance->id, $userids);
|
|
|
|
|
|
|
|
if (empty($userids)) {
|
|
|
|
$context = context_module::instance($cm->id);
|
|
|
|
$userids = array_keys(get_enrolled_users($context, 'mod/attendance:canbelisted', 0, 'u.id'));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($attendance->grade < 0) {
|
|
|
|
$dbparams = array('id' => -($attendance->grade));
|
|
|
|
$scale = $DB->get_record('scale', $dbparams);
|
|
|
|
$scalearray = explode(',', $scale->scale);
|
|
|
|
$attendancegrade = count($scalearray);
|
|
|
|
} else {
|
|
|
|
$attendancegrade = $attendance->grade;
|
|
|
|
}
|
|
|
|
|
|
|
|
$grades = array();
|
|
|
|
foreach ($userids as $userid) {
|
|
|
|
$grades[$userid] = new stdClass();
|
|
|
|
$grades[$userid]->userid = $userid;
|
|
|
|
|
|
|
|
if ($summary->has_taken_sessions($userid)) {
|
|
|
|
$usersummary = $summary->get_taken_sessions_summary_for($userid);
|
|
|
|
$grades[$userid]->rawgrade = $usersummary->takensessionspercentage * $attendancegrade;
|
|
|
|
} else {
|
|
|
|
$grades[$userid]->rawgrade = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return grade_update('mod/attendance', $course->id, 'mod', 'attendance', $attendance->id, 0, $grades);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add an attendance status variable
|
|
|
|
*
|
|
|
|
* @param stdClass $status
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
function attendance_add_status($status) {
|
|
|
|
global $DB;
|
|
|
|
if (empty($status->context)) {
|
|
|
|
$status->context = context_system::instance();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($status->acronym) && !empty($status->description)) {
|
|
|
|
$status->deleted = 0;
|
|
|
|
$status->visible = 1;
|
|
|
|
$status->setunmarked = 0;
|
|
|
|
|
|
|
|
$id = $DB->insert_record('attendance_statuses', $status);
|
|
|
|
$status->id = $id;
|
|
|
|
|
|
|
|
$event = \mod_attendance\event\status_added::create(array(
|
|
|
|
'objectid' => $status->attendanceid,
|
|
|
|
'context' => $status->context,
|
|
|
|
'other' => array('acronym' => $status->acronym,
|
|
|
|
'description' => $status->description,
|
|
|
|
'grade' => $status->grade)));
|
|
|
|
if (!empty($status->cm)) {
|
|
|
|
$event->add_record_snapshot('course_modules', $status->cm);
|
|
|
|
}
|
|
|
|
$event->add_record_snapshot('attendance_statuses', $status);
|
|
|
|
$event->trigger();
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove a status variable from an attendance instance
|
|
|
|
*
|
|
|
|
* @param stdClass $status
|
|
|
|
* @param stdClass $context
|
|
|
|
* @param stdClass $cm
|
|
|
|
*/
|
|
|
|
function attendance_remove_status($status, $context = null, $cm = null) {
|
|
|
|
global $DB;
|
|
|
|
if (empty($context)) {
|
|
|
|
$context = context_system::instance();
|
|
|
|
}
|
|
|
|
$DB->set_field('attendance_statuses', 'deleted', 1, array('id' => $status->id));
|
|
|
|
$event = \mod_attendance\event\status_removed::create(array(
|
|
|
|
'objectid' => $status->id,
|
|
|
|
'context' => $context,
|
|
|
|
'other' => array(
|
|
|
|
'acronym' => $status->acronym,
|
|
|
|
'description' => $status->description
|
|
|
|
)));
|
|
|
|
if (!empty($cm)) {
|
|
|
|
$event->add_record_snapshot('course_modules', $cm);
|
|
|
|
}
|
|
|
|
$event->add_record_snapshot('attendance_statuses', $status);
|
|
|
|
$event->trigger();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update status variable for a particular Attendance module instance
|
|
|
|
*
|
|
|
|
* @param stdClass $status
|
|
|
|
* @param string $acronym
|
|
|
|
* @param string $description
|
|
|
|
* @param int $grade
|
|
|
|
* @param bool $visible
|
|
|
|
* @param stdClass $context
|
|
|
|
* @param stdClass $cm
|
|
|
|
* @param int $studentavailability
|
|
|
|
* @param bool $setunmarked
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function attendance_update_status($status, $acronym, $description, $grade, $visible,
|
|
|
|
$context = null, $cm = null, $studentavailability = null, $setunmarked = false) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if (empty($context)) {
|
|
|
|
$context = context_system::instance();
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
if (isset($studentavailability)) {
|
|
|
|
if (empty($studentavailability)) {
|
|
|
|
if ($studentavailability !== '0') {
|
|
|
|
$studentavailability = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$status->studentavailability = $studentavailability;
|
|
|
|
$updated[] = $studentavailability;
|
|
|
|
}
|
|
|
|
if ($setunmarked) {
|
|
|
|
$status->setunmarked = 1;
|
|
|
|
} else {
|
|
|
|
$status->setunmarked = 0;
|
|
|
|
}
|
|
|
|
$DB->update_record('attendance_statuses', $status);
|
|
|
|
|
|
|
|
$event = \mod_attendance\event\status_updated::create(array(
|
|
|
|
'objectid' => $status->attendanceid,
|
|
|
|
'context' => $context,
|
|
|
|
'other' => array('acronym' => $acronym, 'description' => $description, 'grade' => $grade,
|
|
|
|
'updated' => implode(' ', $updated))));
|
|
|
|
if (!empty($cm)) {
|
|
|
|
$event->add_record_snapshot('course_modules', $cm);
|
|
|
|
}
|
|
|
|
$event->add_record_snapshot('attendance_statuses', $status);
|
|
|
|
$event->trigger();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Similar to core random_string function but only lowercase letters.
|
|
|
|
* designed to make it relatively easy to provide a simple password in class.
|
|
|
|
*
|
|
|
|
* @param int $length The length of the string to be created.
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
function attendance_random_string($length=6) {
|
|
|
|
$randombytes = random_bytes_emulate($length);
|
|
|
|
$pool = 'abcdefghijklmnopqrstuvwxyz';
|
|
|
|
$pool .= '0123456789';
|
|
|
|
$poollen = strlen($pool);
|
|
|
|
$string = '';
|
|
|
|
for ($i = 0; $i < $length; $i++) {
|
|
|
|
$rand = ord($randombytes[$i]);
|
|
|
|
$string .= substr($pool, ($rand % ($poollen)), 1);
|
|
|
|
}
|
|
|
|
return $string;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check to see if this session is open for student marking.
|
|
|
|
*
|
|
|
|
* @param stdclass $sess the session record from attendance_sessions.
|
|
|
|
* @param boolean $log - if student cannot mark, generate log event.
|
|
|
|
* @return array (boolean, string reason for failure)
|
|
|
|
*/
|
|
|
|
function attendance_can_student_mark($sess, $log = true) {
|
|
|
|
global $DB, $USER, $OUTPUT;
|
|
|
|
$canmark = false;
|
|
|
|
$reason = 'closed';
|
|
|
|
$attconfig = get_config('attendance');
|
|
|
|
if (!empty($attconfig->studentscanmark) && !empty($sess->studentscanmark)) {
|
|
|
|
if (empty($attconfig->studentscanmarksessiontime)) {
|
|
|
|
$canmark = true;
|
|
|
|
$reason = '';
|
|
|
|
} else {
|
|
|
|
$duration = $sess->duration;
|
|
|
|
if (empty($duration)) {
|
|
|
|
$duration = $attconfig->studentscanmarksessiontimeend * 60;
|
|
|
|
}
|
|
|
|
if ($sess->sessdate < time() && time() < ($sess->sessdate + $duration)) {
|
|
|
|
$canmark = true;
|
|
|
|
$reason = '';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Check if another student has marked attendance from this IP address recently.
|
|
|
|
if ($canmark && !empty($sess->preventsharedip)) {
|
|
|
|
if ($sess->preventsharedip == ATTENDANCE_SHAREDIP_MINUTES) {
|
|
|
|
$time = time() - ($sess->preventsharediptime * 60);
|
|
|
|
$sql = 'sessionid = ? AND studentid <> ? AND timetaken > ? AND ipaddress = ?';
|
|
|
|
$params = array($sess->id, $USER->id, $time, getremoteaddr());
|
|
|
|
$record = $DB->get_record_select('attendance_log', $sql, $params);
|
|
|
|
} else {
|
|
|
|
// Assume ATTENDANCE_SHAREDIP_FORCED.
|
|
|
|
$sql = 'sessionid = ? AND studentid <> ? AND ipaddress = ?';
|
|
|
|
$params = array($sess->id, $USER->id, getremoteaddr());
|
|
|
|
$record = $DB->get_record_select('attendance_log', $sql, $params);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($record)) {
|
|
|
|
$canmark = false;
|
|
|
|
$reason = 'preventsharederror';
|
|
|
|
if ($log) {
|
|
|
|
// Trigger an ip_shared event.
|
|
|
|
$attendanceid = $DB->get_field('attendance_sessions', 'attendanceid', array('id' => $record->sessionid));
|
|
|
|
$cm = get_coursemodule_from_instance('attendance', $attendanceid);
|
|
|
|
$event = \mod_attendance\event\session_ip_shared::create(array(
|
|
|
|
'objectid' => 0,
|
|
|
|
'context' => \context_module::instance($cm->id),
|
|
|
|
'other' => array(
|
|
|
|
'sessionid' => $record->sessionid,
|
|
|
|
'otheruser' => $record->studentid
|
|
|
|
)
|
|
|
|
));
|
|
|
|
|
|
|
|
$event->trigger();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return array($canmark, $reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate worksheet for Attendance export
|
|
|
|
*
|
|
|
|
* @param stdclass $data The data for the report
|
|
|
|
* @param string $filename The name of the file
|
|
|
|
* @param string $format excel|ods
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
function attendance_exporttotableed($data, $filename, $format) {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
if ($format === 'excel') {
|
|
|
|
require_once("$CFG->libdir/excellib.class.php");
|
|
|
|
$filename .= ".xls";
|
|
|
|
$workbook = new MoodleExcelWorkbook("-");
|
|
|
|
} else {
|
|
|
|
require_once("$CFG->libdir/odslib.class.php");
|
|
|
|
$filename .= ".ods";
|
|
|
|
$workbook = new MoodleODSWorkbook("-");
|
|
|
|
}
|
|
|
|
// Sending HTTP headers.
|
|
|
|
$workbook->send($filename);
|
|
|
|
// Creating the first worksheet.
|
|
|
|
$myxls = $workbook->add_worksheet(get_string('modulenameplural', 'attendance'));
|
|
|
|
// Format types.
|
|
|
|
$formatbc = $workbook->add_format();
|
|
|
|
$formatbc->set_bold(1);
|
|
|
|
|
|
|
|
$myxls->write(0, 0, get_string('course'), $formatbc);
|
|
|
|
$myxls->write(0, 1, $data->course);
|
|
|
|
$myxls->write(1, 0, get_string('group'), $formatbc);
|
|
|
|
$myxls->write(1, 1, $data->group);
|
|
|
|
|
|
|
|
$i = 3;
|
|
|
|
$j = 0;
|
|
|
|
foreach ($data->tabhead as $cell) {
|
|
|
|
// Merge cells if the heading would be empty (remarks column).
|
|
|
|
if (empty($cell)) {
|
|
|
|
$myxls->merge_cells($i, $j - 1, $i, $j);
|
|
|
|
} else {
|
|
|
|
$myxls->write($i, $j, $cell, $formatbc);
|
|
|
|
}
|
|
|
|
$j++;
|
|
|
|
}
|
|
|
|
$i++;
|
|
|
|
$j = 0;
|
|
|
|
foreach ($data->table as $row) {
|
|
|
|
foreach ($row as $cell) {
|
|
|
|
$myxls->write($i, $j++, $cell);
|
|
|
|
}
|
|
|
|
$i++;
|
|
|
|
$j = 0;
|
|
|
|
}
|
|
|
|
$workbook->close();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate csv for Attendance export
|
|
|
|
*
|
|
|
|
* @param stdclass $data The data for the report
|
|
|
|
* @param string $filename The name of the file
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
function attendance_exporttocsv($data, $filename) {
|
|
|
|
$filename .= ".txt";
|
|
|
|
|
|
|
|
header("Content-Type: application/download\n");
|
|
|
|
header("Content-Disposition: attachment; filename=\"$filename\"");
|
|
|
|
header("Expires: 0");
|
|
|
|
header("Cache-Control: must-revalidate,post-check=0,pre-check=0");
|
|
|
|
header("Pragma: public");
|
|
|
|
|
|
|
|
echo get_string('course')."\t".$data->course."\n";
|
|
|
|
echo get_string('group')."\t".$data->group."\n\n";
|
|
|
|
|
|
|
|
echo implode("\t", $data->tabhead)."\n";
|
|
|
|
foreach ($data->table as $row) {
|
|
|
|
echo implode("\t", $row)."\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get session data for form.
|
|
|
|
* @param stdClass $formdata moodleform - attendance form.
|
|
|
|
* @param mod_attendance_structure $att - used to get attendance level subnet.
|
|
|
|
* @return array.
|
|
|
|
*/
|
|
|
|
function attendance_construct_sessions_data_for_add($formdata, mod_attendance_structure $att) {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
$sesstarttime = $formdata->sestime['starthour'] * HOURSECS + $formdata->sestime['startminute'] * MINSECS;
|
|
|
|
$sesendtime = $formdata->sestime['endhour'] * HOURSECS + $formdata->sestime['endminute'] * MINSECS;
|
|
|
|
$sessiondate = $formdata->sessiondate + $sesstarttime;
|
|
|
|
$duration = $sesendtime - $sesstarttime;
|
|
|
|
if (empty(get_config('attendance', 'enablewarnings'))) {
|
|
|
|
$absenteereport = get_config('attendance', 'absenteereport_default');
|
|
|
|
} else {
|
|
|
|
$absenteereport = empty($formdata->absenteereport) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
$now = time();
|
|
|
|
|
|
|
|
if (empty(get_config('attendance', 'studentscanmark'))) {
|
|
|
|
$formdata->studentscanmark = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
$calendarevent = 0;
|
|
|
|
if (isset($formdata->calendarevent)) { // Calendar event should be created.
|
|
|
|
$calendarevent = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
$sessions = array();
|
|
|
|
if (isset($formdata->addmultiply)) {
|
|
|
|
$startdate = $sessiondate;
|
|
|
|
$enddate = $formdata->sessionenddate + DAYSECS; // Because enddate in 0:0am.
|
|
|
|
|
|
|
|
if ($enddate < $startdate) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Getting first day of week.
|
|
|
|
$sdate = $startdate;
|
|
|
|
$dinfo = usergetdate($sdate);
|
|
|
|
if ($CFG->calendar_startwday === '0') { // Week start from sunday.
|
|
|
|
$startweek = $startdate - $dinfo['wday'] * DAYSECS; // Call new variable.
|
|
|
|
} else {
|
|
|
|
$wday = $dinfo['wday'] === 0 ? 7 : $dinfo['wday'];
|
|
|
|
$startweek = $startdate - ($wday - 1) * DAYSECS;
|
|
|
|
}
|
|
|
|
|
|
|
|
$wdaydesc = array(0 => 'Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat');
|
|
|
|
|
|
|
|
while ($sdate < $enddate) {
|
|
|
|
if ($sdate < $startweek + WEEKSECS) {
|
|
|
|
$dinfo = usergetdate($sdate);
|
|
|
|
if (isset($formdata->sdays) && array_key_exists($wdaydesc[$dinfo['wday']], $formdata->sdays)) {
|
|
|
|
$sess = new stdClass();
|
|
|
|
$sess->sessdate = make_timestamp($dinfo['year'], $dinfo['mon'], $dinfo['mday'],
|
|
|
|
$formdata->sestime['starthour'], $formdata->sestime['startminute']);
|
|
|
|
$sess->duration = $duration;
|
|
|
|
$sess->descriptionitemid = $formdata->sdescription['itemid'];
|
|
|
|
$sess->description = $formdata->sdescription['text'];
|
|
|
|
$sess->descriptionformat = $formdata->sdescription['format'];
|
|
|
|
$sess->calendarevent = $calendarevent;
|
|
|
|
$sess->timemodified = $now;
|
|
|
|
$sess->absenteereport = $absenteereport;
|
|
|
|
$sess->studentpassword = '';
|
|
|
|
$sess->includeqrcode = 0;
|
|
|
|
$sess->rotateqrcode = 0;
|
|
|
|
$sess->rotateqrcodesecret = '';
|
|
|
|
|
|
|
|
if (!empty($formdata->usedefaultsubnet)) {
|
|
|
|
$sess->subnet = $att->subnet;
|
|
|
|
} else {
|
|
|
|
$sess->subnet = $formdata->subnet;
|
|
|
|
}
|
|
|
|
$sess->automark = $formdata->automark;
|
|
|
|
$sess->automarkcompleted = 0;
|
|
|
|
if (!empty($formdata->preventsharedip)) {
|
|
|
|
$sess->preventsharedip = $formdata->preventsharedip;
|
|
|
|
}
|
|
|
|
if (!empty($formdata->preventsharediptime)) {
|
|
|
|
$sess->preventsharediptime = $formdata->preventsharediptime;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($formdata->studentscanmark)) { // Students will be able to mark their own attendance.
|
|
|
|
$sess->studentscanmark = 1;
|
|
|
|
if (isset($formdata->autoassignstatus)) {
|
|
|
|
$sess->autoassignstatus = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($formdata->randompassword)) {
|
|
|
|
$sess->studentpassword = attendance_random_string();
|
|
|
|
} else if (!empty($formdata->studentpassword)) {
|
|
|
|
$sess->studentpassword = $formdata->studentpassword;
|
|
|
|
}
|
|
|
|
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->preventsharedip)) {
|
|
|
|
$sess->preventsharedip = $formdata->preventsharedip;
|
|
|
|
}
|
|
|
|
if (!empty($formdata->preventsharediptime)) {
|
|
|
|
$sess->preventsharediptime = $formdata->preventsharediptime;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$sess->subnet = '';
|
|
|
|
$sess->automark = 0;
|
|
|
|
$sess->automarkcompleted = 0;
|
|
|
|
$sess->preventsharedip = 0;
|
|
|
|
$sess->preventsharediptime = '';
|
|
|
|
}
|
|
|
|
$sess->statusset = $formdata->statusset;
|
|
|
|
|
|
|
|
attendance_fill_groupid($formdata, $sessions, $sess);
|
|
|
|
}
|
|
|
|
$sdate += DAYSECS;
|
|
|
|
} else {
|
|
|
|
$startweek += WEEKSECS * $formdata->period;
|
|
|
|
$sdate = $startweek;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$sess = new stdClass();
|
|
|
|
$sess->sessdate = $sessiondate;
|
|
|
|
$sess->duration = $duration;
|
|
|
|
$sess->descriptionitemid = $formdata->sdescription['itemid'];
|
|
|
|
$sess->description = $formdata->sdescription['text'];
|
|
|
|
$sess->descriptionformat = $formdata->sdescription['format'];
|
|
|
|
$sess->calendarevent = $calendarevent;
|
|
|
|
$sess->timemodified = $now;
|
|
|
|
$sess->studentscanmark = 0;
|
|
|
|
$sess->autoassignstatus = 0;
|
|
|
|
$sess->subnet = '';
|
|
|
|
$sess->studentpassword = '';
|
|
|
|
$sess->automark = 0;
|
|
|
|
$sess->automarkcompleted = 0;
|
|
|
|
$sess->absenteereport = $absenteereport;
|
|
|
|
$sess->includeqrcode = 0;
|
|
|
|
$sess->rotateqrcode = 0;
|
|
|
|
$sess->rotateqrcodesecret = '';
|
|
|
|
|
|
|
|
if (!empty($formdata->usedefaultsubnet)) {
|
|
|
|
$sess->subnet = $att->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;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($formdata->studentscanmark) && !empty($formdata->studentscanmark)) {
|
|
|
|
// Students will be able to mark their own attendance.
|
|
|
|
$sess->studentscanmark = 1;
|
|
|
|
if (isset($formdata->autoassignstatus) && !empty($formdata->autoassignstatus)) {
|
|
|
|
$sess->autoassignstatus = 1;
|
|
|
|
}
|
|
|
|
if (!empty($formdata->randompassword)) {
|
|
|
|
$sess->studentpassword = attendance_random_string();
|
|
|
|
} else if (!empty($formdata->studentpassword)) {
|
|
|
|
$sess->studentpassword = $formdata->studentpassword;
|
|
|
|
}
|
|
|
|
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 = $att->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->statusset = $formdata->statusset;
|
|
|
|
|
|
|
|
attendance_fill_groupid($formdata, $sessions, $sess);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sessions;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper function for attendance_construct_sessions_data_for_add().
|
|
|
|
*
|
|
|
|
* @param stdClass $formdata
|
|
|
|
* @param stdClass $sessions
|
|
|
|
* @param stdClass $sess
|
|
|
|
*/
|
|
|
|
function attendance_fill_groupid($formdata, &$sessions, $sess) {
|
|
|
|
if ($formdata->sessiontype == mod_attendance_structure::SESSION_COMMON) {
|
|
|
|
$sess = clone $sess;
|
|
|
|
$sess->groupid = 0;
|
|
|
|
$sessions[] = $sess;
|
|
|
|
} else {
|
|
|
|
foreach ($formdata->groups as $groupid) {
|
|
|
|
$sess = clone $sess;
|
|
|
|
$sess->groupid = $groupid;
|
|
|
|
$sessions[] = $sess;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates a summary of points for the courses selected.
|
|
|
|
*
|
|
|
|
* @param array $courseids optional list of courses to return
|
|
|
|
* @param string $orderby - optional order by param
|
|
|
|
* @return stdClass
|
|
|
|
*/
|
|
|
|
function attendance_course_users_points($courseids = array(), $orderby = '') {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$where = '';
|
|
|
|
$params = array();
|
|
|
|
$where .= ' AND ats.sessdate < :enddate ';
|
|
|
|
$params['enddate'] = time();
|
|
|
|
|
|
|
|
$joingroup = 'LEFT JOIN {groups_members} gm ON (gm.userid = atl.studentid AND gm.groupid = ats.groupid)';
|
|
|
|
$where .= ' AND (ats.groupid = 0 or gm.id is NOT NULL)';
|
|
|
|
|
|
|
|
if (!empty($courseids)) {
|
|
|
|
list($insql, $inparams) = $DB->get_in_or_equal($courseids, SQL_PARAMS_NAMED);
|
|
|
|
$where .= ' AND c.id ' . $insql;
|
|
|
|
$params = array_merge($params, $inparams);
|
|
|
|
}
|
|
|
|
|
|
|
|
$sql = "SELECT courseid, coursename, sum(points) / sum(maxpoints) as percentage FROM (
|
|
|
|
SELECT a.id, a.course as courseid, c.fullname as coursename, atl.studentid AS userid, COUNT(DISTINCT ats.id) AS numtakensessions,
|
|
|
|
SUM(stg.grade) AS points, SUM(stm.maxgrade) AS maxpoints
|
|
|
|
FROM {attendance_sessions} ats
|
|
|
|
JOIN {attendance} a ON a.id = ats.attendanceid
|
|
|
|
JOIN {course} c ON c.id = a.course
|
|
|
|
JOIN {attendance_log} atl ON (atl.sessionid = ats.id)
|
|
|
|
JOIN {attendance_statuses} stg ON (stg.id = atl.statusid AND stg.deleted = 0 AND stg.visible = 1)
|
|
|
|
JOIN (SELECT attendanceid, setnumber, MAX(grade) AS maxgrade
|
|
|
|
FROM {attendance_statuses}
|
|
|
|
WHERE deleted = 0
|
|
|
|
AND visible = 1
|
|
|
|
GROUP BY attendanceid, setnumber) stm
|
|
|
|
ON (stm.setnumber = ats.statusset AND stm.attendanceid = ats.attendanceid)
|
|
|
|
{$joingroup}
|
|
|
|
WHERE ats.sessdate >= c.startdate
|
|
|
|
AND ats.lasttaken != 0
|
|
|
|
{$where}
|
|
|
|
GROUP BY a.id, a.course, c.fullname, atl.studentid
|
|
|
|
) p GROUP by courseid, coursename {$orderby}";
|
|
|
|
|
|
|
|
return $DB->get_records_sql($sql, $params);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates a list of users flagged absent.
|
|
|
|
*
|
|
|
|
* @param array $courseids optional list of courses to return
|
|
|
|
* @param string $orderby how to order results.
|
|
|
|
* @param bool $allfornotify get notification list for scheduled task.
|
|
|
|
* @return stdClass
|
|
|
|
*/
|
|
|
|
function attendance_get_users_to_notify($courseids = array(), $orderby = '', $allfornotify = false) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$joingroup = 'LEFT JOIN {groups_members} gm ON (gm.userid = atl.studentid AND gm.groupid = ats.groupid)';
|
|
|
|
$where = ' AND (ats.groupid = 0 or gm.id is NOT NULL)';
|
|
|
|
$having = '';
|
|
|
|
$params = array();
|
|
|
|
|
|
|
|
if (!empty($courseids)) {
|
|
|
|
list($insql, $inparams) = $DB->get_in_or_equal($courseids, SQL_PARAMS_NAMED);
|
|
|
|
$where .= ' AND c.id ' . $insql;
|
|
|
|
$params = array_merge($params, $inparams);
|
|
|
|
}
|
|
|
|
if ($allfornotify) {
|
|
|
|
// Exclude warnings that have already sent the max num.
|
|
|
|
$having .= ' AND n.maxwarn > COUNT(DISTINCT ns.id) ';
|
|
|
|
}
|
|
|
|
|
|
|
|
$unames = get_all_user_name_fields(true);
|
|
|
|
$unames2 = get_all_user_name_fields(true, 'u');
|
|
|
|
|
|
|
|
$idfield = $DB->sql_concat('cm.id', 'atl.studentid', 'n.id');
|
|
|
|
$sql = "SELECT {$idfield} as uniqueid, a.id as aid, {$unames2}, a.name as aname, cm.id as cmid, c.id as courseid,
|
|
|
|
c.fullname as coursename, atl.studentid AS userid, n.id as notifyid, n.warningpercent, n.emailsubject,
|
|
|
|
n.emailcontent, n.emailcontentformat, n.emailuser, n.thirdpartyemails, n.warnafter, n.maxwarn,
|
|
|
|
COUNT(DISTINCT ats.id) AS numtakensessions, SUM(stg.grade) AS points, SUM(stm.maxgrade) AS maxpoints,
|
|
|
|
COUNT(DISTINCT ns.id) as nscount, MAX(ns.timesent) as timesent,
|
|
|
|
SUM(stg.grade) / SUM(stm.maxgrade) AS percent
|
|
|
|
FROM {attendance_sessions} ats
|
|
|
|
JOIN {attendance} a ON a.id = ats.attendanceid
|
|
|
|
JOIN {course_modules} cm ON cm.instance = a.id
|
|
|
|
JOIN {course} c on c.id = cm.course
|
|
|
|
JOIN {modules} md ON md.id = cm.module AND md.name = 'attendance'
|
|
|
|
JOIN {attendance_log} atl ON (atl.sessionid = ats.id)
|
|
|
|
JOIN {user} u ON (u.id = atl.studentid)
|
|
|
|
JOIN {attendance_statuses} stg ON (stg.id = atl.statusid AND stg.deleted = 0 AND stg.visible = 1)
|
|
|
|
JOIN {attendance_warning} n ON n.idnumber = a.id
|
|
|
|
LEFT JOIN {attendance_warning_done} ns ON ns.notifyid = n.id AND ns.userid = atl.studentid
|
|
|
|
JOIN (SELECT attendanceid, setnumber, MAX(grade) AS maxgrade
|
|
|
|
FROM {attendance_statuses}
|
|
|
|
WHERE deleted = 0
|
|
|
|
AND visible = 1
|
|
|
|
GROUP BY attendanceid, setnumber) stm
|
|
|
|
ON (stm.setnumber = ats.statusset AND stm.attendanceid = ats.attendanceid)
|
|
|
|
{$joingroup}
|
|
|
|
WHERE ats.absenteereport = 1 {$where}
|
|
|
|
GROUP BY uniqueid, a.id, a.name, a.course, c.fullname, atl.studentid, n.id, n.warningpercent,
|
|
|
|
n.emailsubject, n.emailcontent, n.emailcontentformat, n.warnafter, n.maxwarn,
|
|
|
|
n.emailuser, n.thirdpartyemails, cm.id, c.id, {$unames2}, ns.userid
|
|
|
|
HAVING n.warnafter <= COUNT(DISTINCT ats.id) AND n.warningpercent > ((SUM(stg.grade) / SUM(stm.maxgrade)) * 100)
|
|
|
|
{$having}
|
|
|
|
{$orderby}";
|
|
|
|
|
|
|
|
if (!$allfornotify) {
|
|
|
|
$idfield = $DB->sql_concat('cmid', 'userid');
|
|
|
|
// Only show one record per attendance for teacher reports.
|
|
|
|
$sql = "SELECT DISTINCT {$idfield} as id, {$unames}, aid, cmid, courseid, aname, coursename, userid,
|
|
|
|
numtakensessions, percent, MAX(timesent) as timesent
|
|
|
|
FROM ({$sql}) as m
|
|
|
|
GROUP BY id, aid, cmid, courseid, aname, userid, numtakensessions,
|
|
|
|
percent, coursename, {$unames} {$orderby}";
|
|
|
|
}
|
|
|
|
|
|
|
|
return $DB->get_records_sql($sql, $params);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Template variables into place in supplied email content.
|
|
|
|
*
|
|
|
|
* @param object $record db record of details
|
|
|
|
* @return array - the content of the fields after templating.
|
|
|
|
*/
|
|
|
|
function attendance_template_variables($record) {
|
|
|
|
$templatevars = array(
|
|
|
|
'/%coursename%/' => $record->coursename,
|
|
|
|
'/%courseid%/' => $record->courseid,
|
|
|
|
'/%userfirstname%/' => $record->firstname,
|
|
|
|
'/%userlastname%/' => $record->lastname,
|
|
|
|
'/%userid%/' => $record->userid,
|
|
|
|
'/%warningpercent%/' => $record->warningpercent,
|
|
|
|
'/%attendancename%/' => $record->aname,
|
|
|
|
'/%cmid%/' => $record->cmid,
|
|
|
|
'/%numtakensessions%/' => $record->numtakensessions,
|
|
|
|
'/%points%/' => $record->points,
|
|
|
|
'/%maxpoints%/' => $record->maxpoints,
|
|
|
|
'/%percent%/' => $record->percent,
|
|
|
|
);
|
|
|
|
$extrauserfields = get_all_user_name_fields();
|
|
|
|
foreach ($extrauserfields as $extra) {
|
|
|
|
$templatevars['/%'.$extra.'%/'] = $record->$extra;
|
|
|
|
}
|
|
|
|
$patterns = array_keys($templatevars); // The placeholders which are to be replaced.
|
|
|
|
$replacements = array_values($templatevars); // The values which are to be templated in for the placeholders.
|
|
|
|
// Array to describe which fields in reengagement object should have a template replacement.
|
|
|
|
$replacementfields = array('emailsubject', 'emailcontent');
|
|
|
|
|
|
|
|
// Replace %variable% with relevant value everywhere it occurs in reengagement->field.
|
|
|
|
foreach ($replacementfields as $field) {
|
|
|
|
$record->$field = preg_replace($patterns, $replacements, $record->$field);
|
|
|
|
}
|
|
|
|
return $record;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find highest available status for a user.
|
|
|
|
*
|
|
|
|
* @param mod_attendance_structure $att attendance structure
|
|
|
|
* @param stdclass $attforsession attendance_session record.
|
|
|
|
* @param int $scantime - time that session should be recorded against.
|
|
|
|
* @return bool/int
|
|
|
|
*/
|
|
|
|
function attendance_session_get_highest_status(mod_attendance_structure $att, $attforsession, $scantime = null) {
|
|
|
|
// Find the status to set here.
|
|
|
|
$statuses = $att->get_statuses();
|
|
|
|
$highestavailablegrade = 0;
|
|
|
|
$highestavailablestatus = new stdClass();
|
|
|
|
// Override time used in status recording.
|
|
|
|
$scantime = empty($scantime) ? time() : $scantime;
|
|
|
|
foreach ($statuses as $status) {
|
|
|
|
if ($status->studentavailability === '0') {
|
|
|
|
// This status is never available to students.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!empty($status->studentavailability)) {
|
|
|
|
$toolateforstatus = (($attforsession->sessdate + ($status->studentavailability * 60)) < $scantime);
|
|
|
|
if ($toolateforstatus) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This status is available to the student.
|
|
|
|
if ($status->grade >= $highestavailablegrade) {
|
|
|
|
// This is the most favourable grade so far; save it.
|
|
|
|
$highestavailablegrade = $status->grade;
|
|
|
|
$highestavailablestatus = $status;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (empty($highestavailablestatus)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return $highestavailablestatus->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get available automark options.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function attendance_get_automarkoptions() {
|
|
|
|
$options = array();
|
|
|
|
$options[ATTENDANCE_AUTOMARK_DISABLED] = get_string('noautomark', 'attendance');
|
|
|
|
if (strpos(get_config('tool_log', 'enabled_stores'), 'logstore_standard') !== false) {
|
|
|
|
$options[ATTENDANCE_AUTOMARK_ALL] = get_string('automarkall', 'attendance');
|
|
|
|
}
|
|
|
|
$options[ATTENDANCE_AUTOMARK_CLOSE] = get_string('automarkclose', 'attendance');
|
|
|
|
return $options;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get available sharedip options.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function attendance_get_sharedipoptions() {
|
|
|
|
$options = array();
|
|
|
|
$options[ATTENDANCE_SHAREDIP_DISABLED] = get_string('no');
|
|
|
|
$options[ATTENDANCE_SHAREDIP_FORCE] = get_string('yes');
|
|
|
|
$options[ATTENDANCE_SHAREDIP_MINUTES] = get_string('setperiod', 'attendance');
|
|
|
|
|
|
|
|
return $options;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used to print simple time - 1am instead of 1:00am.
|
|
|
|
*
|
|
|
|
* @param int $time - unix timestamp.
|
|
|
|
*/
|
|
|
|
function attendance_strftimehm($time) {
|
|
|
|
$mins = userdate($time, '%M');
|
|
|
|
|
|
|
|
if ($mins == '00') {
|
|
|
|
$format = get_string('strftimeh', 'attendance');
|
|
|
|
} else {
|
|
|
|
$format = get_string('strftimehm', 'attendance');
|
|
|
|
}
|
|
|
|
|
|
|
|
$userdate = userdate($time, $format);
|
|
|
|
|
|
|
|
// Some Lang packs use %p to suffix with AM/PM but not all strftime support this.
|
|
|
|
// Check if %p is in use and make sure it's being respected.
|
|
|
|
if (stripos($format, '%p')) {
|
|
|
|
// Check if $userdate did something with %p by checking userdate against the same format without %p.
|
|
|
|
$formatwithoutp = str_ireplace('%p', '', $format);
|
|
|
|
if (userdate($time, $formatwithoutp) == $userdate) {
|
|
|
|
// The date is the same with and without %p - we have a problem.
|
|
|
|
if (userdate($time, '%H') > 11) {
|
|
|
|
$userdate .= 'pm';
|
|
|
|
} else {
|
|
|
|
$userdate .= 'am';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Some locales and O/S don't respect correct intended case of %p vs %P
|
|
|
|
// This can cause problems with behat which expects AM vs am.
|
|
|
|
if (strpos($format, '%p')) { // Should be upper case according to PHP spec.
|
|
|
|
$userdate = str_replace('am', 'AM', $userdate);
|
|
|
|
$userdate = str_replace('pm', 'PM', $userdate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $userdate;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used to print simple time - 1am instead of 1:00am.
|
|
|
|
*
|
|
|
|
* @param int $datetime - unix timestamp.
|
|
|
|
* @param int $duration - number of seconds.
|
|
|
|
*/
|
|
|
|
function attendance_construct_session_time($datetime, $duration) {
|
|
|
|
$starttime = attendance_strftimehm($datetime);
|
|
|
|
$endtime = attendance_strftimehm($datetime + $duration);
|
|
|
|
|
|
|
|
return $starttime . ($duration > 0 ? ' - ' . $endtime : '');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used to print session time.
|
|
|
|
*
|
|
|
|
* @param int $datetime - unix timestamp.
|
|
|
|
* @param int $duration - number of seconds duration.
|
|
|
|
* @return string.
|
|
|
|
*/
|
|
|
|
function construct_session_full_date_time($datetime, $duration) {
|
|
|
|
$sessinfo = userdate($datetime, get_string('strftimedmyw', 'attendance'));
|
|
|
|
$sessinfo .= ' '.attendance_construct_session_time($datetime, $duration);
|
|
|
|
|
|
|
|
return $sessinfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render the session password.
|
|
|
|
*
|
|
|
|
* @param stdClass $session
|
|
|
|
*/
|
|
|
|
function attendance_renderpassword($session) {
|
|
|
|
echo html_writer::tag('h2', get_string('passwordgrp', 'attendance'));
|
|
|
|
echo html_writer::span($session->studentpassword, 'student-password');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render the session QR code.
|
|
|
|
*
|
|
|
|
* @param stdClass $session
|
|
|
|
*/
|
|
|
|
function attendance_renderqrcode($session) {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
if (strlen($session->studentpassword) > 0) {
|
|
|
|
$qrcodeurl = $CFG->wwwroot . '/mod/attendance/attendance.php?qrpass=' .
|
|
|
|
$session->studentpassword . '&sessid=' . $session->id;
|
|
|
|
} else {
|
|
|
|
$qrcodeurl = $CFG->wwwroot . '/mod/attendance/attendance.php?sessid=' . $session->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
echo html_writer::tag('h3', get_string('qrcode', 'attendance'));
|
|
|
|
|
|
|
|
$barcode = new TCPDF2DBarcode($qrcodeurl, 'QRCODE');
|
|
|
|
$image = $barcode->getBarcodePngData(15, 15);
|
|
|
|
echo html_writer::img('data:image/png;base64,' . base64_encode($image), get_string('qrcode', 'attendance'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate QR code passwords.
|
|
|
|
*
|
|
|
|
* @param stdClass $session
|
|
|
|
*/
|
|
|
|
function attendance_generate_passwords($session) {
|
|
|
|
global $DB;
|
|
|
|
$attconfig = get_config('attendance');
|
|
|
|
$password = array();
|
|
|
|
|
|
|
|
for ($i = 0; $i < 30; $i++) {
|
|
|
|
array_push($password, array("attendanceid" => $session->id,
|
|
|
|
"password" => mt_rand(1000, 10000), "expirytime" => time() + ($attconfig->rotateqrcodeinterval * $i)));
|
|
|
|
}
|
|
|
|
|
|
|
|
$DB->insert_records('attendance_rotate_passwords', $password);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render JS for rotate QR code passwords.
|
|
|
|
*
|
|
|
|
* @param stdClass $session
|
|
|
|
*/
|
|
|
|
function attendance_renderqrcoderotate($session) {
|
|
|
|
// Load required js.
|
|
|
|
echo html_writer::tag('script', '',
|
|
|
|
[
|
|
|
|
'src' => 'js/qrcode/qrcode.min.js',
|
|
|
|
'type' => 'text/javascript'
|
|
|
|
]
|
|
|
|
);
|
|
|
|
echo html_writer::tag('script', '',
|
|
|
|
[
|
|
|
|
'src' => 'js/password/attendance_QRCodeRotate.js',
|
|
|
|
'type' => 'text/javascript'
|
|
|
|
]
|
|
|
|
);
|
|
|
|
echo html_writer::tag('div', '', ['id' => 'rotate-time']); // Div to display timer.
|
|
|
|
echo html_writer::tag('h3', get_string('passwordgrp', 'attendance'));
|
|
|
|
echo html_writer::tag('div', '', ['id' => 'text-password']); // Div to display password.
|
|
|
|
echo html_writer::tag('h3', get_string('qrcode', 'attendance'));
|
|
|
|
echo html_writer::tag('div', '', ['id' => 'qrcode']); // Div to display qr code.
|
|
|
|
// Js to start the password manager.
|
|
|
|
echo '
|
|
|
|
<script type="text/javascript">
|
|
|
|
let qrCodeRotate = new attendance_QRCodeRotate();
|
|
|
|
qrCodeRotate.start(' . $session->id . ', document.getElementById("qrcode"), document.getElementById("text-password"),
|
|
|
|
document.getElementById("rotate-time"));
|
|
|
|
</script>';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return QR code passwords.
|
|
|
|
*
|
|
|
|
* @param stdClass $session
|
|
|
|
*/
|
|
|
|
function attendance_return_passwords($session) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$sql = 'SELECT * FROM {attendance_rotate_passwords} WHERE attendanceid = ? AND expirytime > ? ORDER BY expirytime ASC';
|
|
|
|
return json_encode($DB->get_records_sql($sql, ['attendanceid' => $session->id, time()], $strictness = IGNORE_MISSING));
|
|
|
|
}
|