2009-09-29 02:51:00 +00:00
|
|
|
<?php
|
2012-01-13 17:45:55 +08:00
|
|
|
// 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/>.
|
2004-03-29 15:28:15 +00:00
|
|
|
|
2012-01-13 17:45:55 +08:00
|
|
|
/**
|
|
|
|
* Calendar extension
|
|
|
|
*
|
|
|
|
* @package core_calendar
|
|
|
|
* @copyright 2004 Greek School Network (http://www.sch.gr), Jon Papaioannou,
|
|
|
|
* Avgoustos Tsinakos
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
|
*/
|
|
|
|
|
2012-10-15 16:31:12 +08:00
|
|
|
if (!defined('MOODLE_INTERNAL')) {
|
|
|
|
die('Direct access to this script is forbidden.'); /// It must be included from a Moodle page
|
|
|
|
}
|
MDL-16660: Add calendar import.
Adds an extra import section alongside the existing export in the
calendar views. Allows the user to import from either a file or a
URL as a subscription, with an optional polling interval of hourly,
daily, weekly, monthly, or annually. This subscription may be added
to the user, group, global or course calendars. These subscriptions
are tracked in a separate database table, and an extra column added
to the events table to relate them to the subscription. The event
uuid field is also expanded to allow for the RFC-2445 UID property.
Subscriptions are listed on the calendar view page, and can be
added and removed, manually polled, and the polling interval
adjusted. Subscription events are updated on cron.
2012-07-23 13:49:43 +12:00
|
|
|
|
2012-01-13 17:45:55 +08:00
|
|
|
/**
|
|
|
|
* These are read by the administration component to provide default values
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_DEFAULT_UPCOMING_LOOKAHEAD - default value of upcoming event preference
|
|
|
|
*/
|
2004-12-16 18:49:57 +00:00
|
|
|
define('CALENDAR_DEFAULT_UPCOMING_LOOKAHEAD', 21);
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_DEFAULT_UPCOMING_MAXEVENTS - default value to display the maximum number of upcoming event
|
|
|
|
*/
|
2004-12-16 18:49:57 +00:00
|
|
|
define('CALENDAR_DEFAULT_UPCOMING_MAXEVENTS', 10);
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_DEFAULT_STARTING_WEEKDAY - default value to display the starting weekday
|
|
|
|
*/
|
|
|
|
define('CALENDAR_DEFAULT_STARTING_WEEKDAY', 1);
|
|
|
|
|
2004-12-16 18:49:57 +00:00
|
|
|
// This is a packed bitfield: day X is "weekend" if $field & (1 << X) is true
|
|
|
|
// Default value = 65 = 64 + 1 = 2^6 + 2^0 = Saturday & Sunday
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_DEFAULT_WEEKEND - default value for weekend (Saturday & Sunday)
|
|
|
|
*/
|
|
|
|
define('CALENDAR_DEFAULT_WEEKEND', 65);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_URL - path to calendar's folder
|
|
|
|
*/
|
2009-11-02 03:50:56 +00:00
|
|
|
define('CALENDAR_URL', $CFG->wwwroot.'/calendar/');
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_TF_24 - Calendar time in 24 hours format
|
|
|
|
*/
|
2009-11-02 03:50:56 +00:00
|
|
|
define('CALENDAR_TF_24', '%H:%M');
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_TF_12 - Calendar time in 12 hours format
|
|
|
|
*/
|
2009-11-02 03:50:56 +00:00
|
|
|
define('CALENDAR_TF_12', '%I:%M %p');
|
2004-12-16 18:49:57 +00:00
|
|
|
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_SITE - Site calendar event types
|
|
|
|
*/
|
|
|
|
define('CALENDAR_EVENT_SITE', 1);
|
|
|
|
|
2012-01-13 17:45:55 +08:00
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_COURSE - Course calendar event types
|
|
|
|
*/
|
2011-06-29 14:50:38 +08:00
|
|
|
define('CALENDAR_EVENT_COURSE', 2);
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_GROUP - group calendar event types
|
|
|
|
*/
|
2011-06-29 14:50:38 +08:00
|
|
|
define('CALENDAR_EVENT_GROUP', 4);
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_USER - user calendar event types
|
|
|
|
*/
|
2011-06-29 14:50:38 +08:00
|
|
|
define('CALENDAR_EVENT_USER', 8);
|
|
|
|
|
2017-09-18 12:47:37 +08:00
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_COURSECAT - Course category calendar event types
|
|
|
|
*/
|
|
|
|
define('CALENDAR_EVENT_COURSECAT', 16);
|
2004-03-29 15:28:15 +00:00
|
|
|
|
MDL-16660: Add calendar import.
Adds an extra import section alongside the existing export in the
calendar views. Allows the user to import from either a file or a
URL as a subscription, with an optional polling interval of hourly,
daily, weekly, monthly, or annually. This subscription may be added
to the user, group, global or course calendars. These subscriptions
are tracked in a separate database table, and an extra column added
to the events table to relate them to the subscription. The event
uuid field is also expanded to allow for the RFC-2445 UID property.
Subscriptions are listed on the calendar view page, and can be
added and removed, manually polled, and the polling interval
adjusted. Subscription events are updated on cron.
2012-07-23 13:49:43 +12:00
|
|
|
/**
|
|
|
|
* CALENDAR_IMPORT_FROM_FILE - import the calendar from a file
|
|
|
|
*/
|
|
|
|
define('CALENDAR_IMPORT_FROM_FILE', 0);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_IMPORT_FROM_URL - import the calendar from a URL
|
|
|
|
*/
|
|
|
|
define('CALENDAR_IMPORT_FROM_URL', 1);
|
|
|
|
|
2019-08-14 09:02:08 +08:00
|
|
|
/**
|
|
|
|
* CALENDAR_IMPORT_EVENT_UPDATED_SKIPPED - imported event was skipped
|
|
|
|
*/
|
|
|
|
define('CALENDAR_IMPORT_EVENT_SKIPPED', -1);
|
|
|
|
|
MDL-16660: Add calendar import.
Adds an extra import section alongside the existing export in the
calendar views. Allows the user to import from either a file or a
URL as a subscription, with an optional polling interval of hourly,
daily, weekly, monthly, or annually. This subscription may be added
to the user, group, global or course calendars. These subscriptions
are tracked in a separate database table, and an extra column added
to the events table to relate them to the subscription. The event
uuid field is also expanded to allow for the RFC-2445 UID property.
Subscriptions are listed on the calendar view page, and can be
added and removed, manually polled, and the polling interval
adjusted. Subscription events are updated on cron.
2012-07-23 13:49:43 +12:00
|
|
|
/**
|
|
|
|
* CALENDAR_IMPORT_EVENT_UPDATED - imported event was updated
|
|
|
|
*/
|
|
|
|
define('CALENDAR_IMPORT_EVENT_UPDATED', 1);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_IMPORT_EVENT_INSERTED - imported event was added by insert
|
|
|
|
*/
|
|
|
|
define('CALENDAR_IMPORT_EVENT_INSERTED', 2);
|
|
|
|
|
2013-08-29 14:54:26 +08:00
|
|
|
/**
|
|
|
|
* CALENDAR_SUBSCRIPTION_UPDATE - Used to represent update action for subscriptions in various forms.
|
|
|
|
*/
|
|
|
|
define('CALENDAR_SUBSCRIPTION_UPDATE', 1);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_SUBSCRIPTION_REMOVE - Used to represent remove action for subscriptions in various forms.
|
|
|
|
*/
|
|
|
|
define('CALENDAR_SUBSCRIPTION_REMOVE', 2);
|
|
|
|
|
2017-01-11 11:25:02 +08:00
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_USER_OVERRIDE_PRIORITY - Constant for the user override priority.
|
|
|
|
*/
|
2017-04-13 12:44:27 +08:00
|
|
|
define('CALENDAR_EVENT_USER_OVERRIDE_PRIORITY', 0);
|
2017-01-11 11:25:02 +08:00
|
|
|
|
2017-02-15 03:15:06 +08:00
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_TYPE_STANDARD - Standard events.
|
|
|
|
*/
|
|
|
|
define('CALENDAR_EVENT_TYPE_STANDARD', 0);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CALENDAR_EVENT_TYPE_ACTION - Action events.
|
|
|
|
*/
|
|
|
|
define('CALENDAR_EVENT_TYPE_ACTION', 1);
|
|
|
|
|
2017-03-29 13:33:11 +08:00
|
|
|
/**
|
|
|
|
* Manage calendar events.
|
|
|
|
*
|
|
|
|
* This class provides the required functionality in order to manage calendar events.
|
|
|
|
* It was introduced as part of Moodle 2.0 and was created in order to provide a
|
|
|
|
* better framework for dealing with calendar events in particular regard to file
|
|
|
|
* handling through the new file API.
|
|
|
|
*
|
|
|
|
* @package core_calendar
|
|
|
|
* @category calendar
|
|
|
|
* @copyright 2009 Sam Hemelryk
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
|
*
|
|
|
|
* @property int $id The id within the event table
|
|
|
|
* @property string $name The name of the event
|
|
|
|
* @property string $description The description of the event
|
|
|
|
* @property int $format The format of the description FORMAT_?
|
|
|
|
* @property int $courseid The course the event is associated with (0 if none)
|
|
|
|
* @property int $groupid The group the event is associated with (0 if none)
|
|
|
|
* @property int $userid The user the event is associated with (0 if none)
|
|
|
|
* @property int $repeatid If this is a repeated event this will be set to the
|
|
|
|
* id of the original
|
2020-04-29 22:22:10 +02:00
|
|
|
* @property string $component If created by a plugin/component (other than module), the full frankenstyle name of a component
|
2017-03-29 13:33:11 +08:00
|
|
|
* @property string $modulename If added by a module this will be the module name
|
|
|
|
* @property int $instance If added by a module this will be the module instance
|
|
|
|
* @property string $eventtype The event type
|
|
|
|
* @property int $timestart The start time as a timestamp
|
|
|
|
* @property int $timeduration The duration of the event in seconds
|
2020-07-23 14:53:49 +10:00
|
|
|
* @property int $timeusermidnight User midnight for the event
|
2017-03-29 13:33:11 +08:00
|
|
|
* @property int $visible 1 if the event is visible
|
|
|
|
* @property int $uuid ?
|
|
|
|
* @property int $sequence ?
|
|
|
|
* @property int $timemodified The time last modified as a timestamp
|
|
|
|
*/
|
|
|
|
class calendar_event {
|
|
|
|
|
|
|
|
/** @var array An object containing the event properties can be accessed via the magic __get/set methods */
|
|
|
|
protected $properties = null;
|
|
|
|
|
|
|
|
/** @var string The converted event discription with file paths resolved.
|
|
|
|
* This gets populated when someone requests description for the first time */
|
|
|
|
protected $_description = null;
|
|
|
|
|
|
|
|
/** @var array The options to use with this description editor */
|
|
|
|
protected $editoroptions = array(
|
|
|
|
'subdirs' => false,
|
|
|
|
'forcehttps' => false,
|
|
|
|
'maxfiles' => -1,
|
|
|
|
'maxbytes' => null,
|
|
|
|
'trusttext' => false);
|
|
|
|
|
|
|
|
/** @var object The context to use with the description editor */
|
|
|
|
protected $editorcontext = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Instantiates a new event and optionally populates its properties with the data provided.
|
|
|
|
*
|
|
|
|
* @param \stdClass $data Optional. An object containing the properties to for
|
|
|
|
* an event
|
|
|
|
*/
|
|
|
|
public function __construct($data = null) {
|
|
|
|
global $CFG, $USER;
|
|
|
|
|
|
|
|
// First convert to object if it is not already (should either be object or assoc array).
|
|
|
|
if (!is_object($data)) {
|
|
|
|
$data = (object) $data;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->editoroptions['maxbytes'] = $CFG->maxbytes;
|
|
|
|
|
|
|
|
$data->eventrepeats = 0;
|
|
|
|
|
|
|
|
if (empty($data->id)) {
|
|
|
|
$data->id = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($data->subscriptionid)) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$data->subscription = calendar_get_subscription($data->subscriptionid);
|
2017-03-29 13:33:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Default to a user event.
|
|
|
|
if (empty($data->eventtype)) {
|
|
|
|
$data->eventtype = 'user';
|
|
|
|
}
|
|
|
|
|
2021-03-23 19:26:50 +08:00
|
|
|
// Default to the current user.
|
2017-03-29 13:33:11 +08:00
|
|
|
if (empty($data->userid)) {
|
2021-03-23 19:26:50 +08:00
|
|
|
$data->userid = $USER->id;
|
2017-03-29 13:33:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($data->timeduration) && is_array($data->timeduration)) {
|
|
|
|
$data->timeduration = make_timestamp(
|
|
|
|
$data->timeduration['year'], $data->timeduration['month'], $data->timeduration['day'],
|
|
|
|
$data->timeduration['hour'], $data->timeduration['minute']) - $data->timestart;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($data->description) && is_array($data->description)) {
|
|
|
|
$data->format = $data->description['format'];
|
|
|
|
$data->description = $data->description['text'];
|
|
|
|
} else if (empty($data->description)) {
|
|
|
|
$data->description = '';
|
|
|
|
$data->format = editors_get_preferred_format();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure form is defaulted correctly.
|
|
|
|
if (empty($data->format)) {
|
|
|
|
$data->format = editors_get_preferred_format();
|
|
|
|
}
|
|
|
|
|
2020-04-29 22:22:10 +02:00
|
|
|
if (empty($data->component)) {
|
|
|
|
$data->component = null;
|
|
|
|
}
|
|
|
|
|
2017-03-29 13:33:11 +08:00
|
|
|
$this->properties = $data;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Magic set method.
|
|
|
|
*
|
|
|
|
* Attempts to call a set_$key method if one exists otherwise falls back
|
|
|
|
* to simply set the property.
|
|
|
|
*
|
|
|
|
* @param string $key property name
|
|
|
|
* @param mixed $value value of the property
|
|
|
|
*/
|
|
|
|
public function __set($key, $value) {
|
|
|
|
if (method_exists($this, 'set_'.$key)) {
|
|
|
|
$this->{'set_'.$key}($value);
|
|
|
|
}
|
|
|
|
$this->properties->{$key} = $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Magic get method.
|
|
|
|
*
|
|
|
|
* Attempts to call a get_$key method to return the property and ralls over
|
|
|
|
* to return the raw property.
|
|
|
|
*
|
|
|
|
* @param string $key property name
|
|
|
|
* @return mixed property value
|
|
|
|
* @throws \coding_exception
|
|
|
|
*/
|
|
|
|
public function __get($key) {
|
|
|
|
if (method_exists($this, 'get_'.$key)) {
|
|
|
|
return $this->{'get_'.$key}();
|
|
|
|
}
|
2017-03-29 17:19:51 +08:00
|
|
|
if (!property_exists($this->properties, $key)) {
|
2017-03-29 13:33:11 +08:00
|
|
|
throw new \coding_exception('Undefined property requested');
|
|
|
|
}
|
|
|
|
return $this->properties->{$key};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Magic isset method.
|
|
|
|
*
|
|
|
|
* PHP needs an isset magic method if you use the get magic method and
|
|
|
|
* still want empty calls to work.
|
|
|
|
*
|
|
|
|
* @param string $key $key property name
|
|
|
|
* @return bool|mixed property value, false if property is not exist
|
|
|
|
*/
|
|
|
|
public function __isset($key) {
|
|
|
|
return !empty($this->properties->{$key});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate the context value needed for an event.
|
|
|
|
*
|
|
|
|
* Event's type can be determine by the available value store in $data
|
|
|
|
* It is important to check for the existence of course/courseid to determine
|
|
|
|
* the course event.
|
|
|
|
* Default value is set to CONTEXT_USER
|
|
|
|
*
|
|
|
|
* @return \stdClass The context object.
|
|
|
|
*/
|
|
|
|
protected function calculate_context() {
|
|
|
|
global $USER, $DB;
|
|
|
|
|
|
|
|
$context = null;
|
2017-09-22 12:01:00 +08:00
|
|
|
if (isset($this->properties->categoryid) && $this->properties->categoryid > 0) {
|
|
|
|
$context = \context_coursecat::instance($this->properties->categoryid);
|
|
|
|
} else if (isset($this->properties->courseid) && $this->properties->courseid > 0) {
|
2017-03-29 13:33:11 +08:00
|
|
|
$context = \context_course::instance($this->properties->courseid);
|
|
|
|
} else if (isset($this->properties->course) && $this->properties->course > 0) {
|
|
|
|
$context = \context_course::instance($this->properties->course);
|
|
|
|
} else if (isset($this->properties->groupid) && $this->properties->groupid > 0) {
|
|
|
|
$group = $DB->get_record('groups', array('id' => $this->properties->groupid));
|
|
|
|
$context = \context_course::instance($group->courseid);
|
|
|
|
} else if (isset($this->properties->userid) && $this->properties->userid > 0
|
|
|
|
&& $this->properties->userid == $USER->id) {
|
|
|
|
$context = \context_user::instance($this->properties->userid);
|
|
|
|
} else if (isset($this->properties->userid) && $this->properties->userid > 0
|
|
|
|
&& $this->properties->userid != $USER->id &&
|
2020-04-29 22:22:10 +02:00
|
|
|
!empty($this->properties->modulename) &&
|
2017-03-29 13:33:11 +08:00
|
|
|
isset($this->properties->instance) && $this->properties->instance > 0) {
|
|
|
|
$cm = get_coursemodule_from_instance($this->properties->modulename, $this->properties->instance, 0,
|
|
|
|
false, MUST_EXIST);
|
|
|
|
$context = \context_course::instance($cm->course);
|
|
|
|
} else {
|
|
|
|
$context = \context_user::instance($this->properties->userid);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $context;
|
|
|
|
}
|
|
|
|
|
2017-11-24 06:33:41 +00:00
|
|
|
/**
|
|
|
|
* Returns the context for this event. The context is calculated
|
|
|
|
* the first time is is requested and then stored in a member
|
|
|
|
* variable to be returned each subsequent time.
|
|
|
|
*
|
|
|
|
* This is a magical getter function that will be called when
|
|
|
|
* ever the context property is accessed, e.g. $event->context.
|
|
|
|
*
|
|
|
|
* @return context
|
|
|
|
*/
|
|
|
|
protected function get_context() {
|
|
|
|
if (!isset($this->properties->context)) {
|
|
|
|
$this->properties->context = $this->calculate_context();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->properties->context;
|
|
|
|
}
|
|
|
|
|
2017-03-29 13:33:11 +08:00
|
|
|
/**
|
|
|
|
* Returns an array of editoroptions for this event.
|
|
|
|
*
|
|
|
|
* @return array event editor options
|
|
|
|
*/
|
|
|
|
protected function get_editoroptions() {
|
|
|
|
return $this->editoroptions;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an event description: Called by __get
|
|
|
|
* Please use $blah = $event->description;
|
|
|
|
*
|
|
|
|
* @return string event description
|
|
|
|
*/
|
|
|
|
protected function get_description() {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
require_once($CFG->libdir . '/filelib.php');
|
|
|
|
|
|
|
|
if ($this->_description === null) {
|
|
|
|
// Check if we have already resolved the context for this event.
|
|
|
|
if ($this->editorcontext === null) {
|
|
|
|
// Switch on the event type to decide upon the appropriate context to use for this event.
|
2017-11-24 06:33:41 +00:00
|
|
|
$this->editorcontext = $this->get_context();
|
2017-09-18 12:47:37 +08:00
|
|
|
if (!calendar_is_valid_eventtype($this->properties->eventtype)) {
|
2017-03-29 13:33:11 +08:00
|
|
|
return clean_text($this->properties->description, $this->properties->format);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Work out the item id for the editor, if this is a repeated event
|
|
|
|
// then the files will be associated with the original.
|
|
|
|
if (!empty($this->properties->repeatid) && $this->properties->repeatid > 0) {
|
|
|
|
$itemid = $this->properties->repeatid;
|
|
|
|
} else {
|
|
|
|
$itemid = $this->properties->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert file paths in the description so that things display correctly.
|
|
|
|
$this->_description = file_rewrite_pluginfile_urls($this->properties->description, 'pluginfile.php',
|
|
|
|
$this->editorcontext->id, 'calendar', 'event_description', $itemid);
|
|
|
|
// Clean the text so no nasties get through.
|
|
|
|
$this->_description = clean_text($this->_description, $this->properties->format);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finally return the description.
|
|
|
|
return $this->_description;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the number of repeat events there are in this events series.
|
|
|
|
*
|
|
|
|
* @return int number of event repeated
|
|
|
|
*/
|
|
|
|
public function count_repeats() {
|
|
|
|
global $DB;
|
|
|
|
if (!empty($this->properties->repeatid)) {
|
|
|
|
$this->properties->eventrepeats = $DB->count_records('event',
|
|
|
|
array('repeatid' => $this->properties->repeatid));
|
|
|
|
// We don't want to count ourselves.
|
|
|
|
$this->properties->eventrepeats--;
|
|
|
|
}
|
|
|
|
return $this->properties->eventrepeats;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update or create an event within the database
|
|
|
|
*
|
|
|
|
* Pass in a object containing the event properties and this function will
|
|
|
|
* insert it into the database and deal with any associated files
|
|
|
|
*
|
2018-08-24 10:06:11 +01:00
|
|
|
* Capability checking should be performed if the user is directly manipulating the event
|
|
|
|
* and no other capability has been tested. However if the event is not being manipulated
|
|
|
|
* directly by the user and another capability has been checked for them to do this then
|
|
|
|
* capabilites should not be checked.
|
|
|
|
*
|
|
|
|
* For example if a user is editing an event in the calendar the check should be true,
|
|
|
|
* but if you are updating an event in an activities settings are changed then the calendar
|
|
|
|
* capabilites should not be checked.
|
|
|
|
*
|
2017-03-29 13:33:11 +08:00
|
|
|
* @see self::create()
|
|
|
|
* @see self::update()
|
|
|
|
*
|
|
|
|
* @param \stdClass $data object of event
|
2018-08-24 10:06:11 +01:00
|
|
|
* @param bool $checkcapability If Moodle should check the user can manage the calendar events for this call or not.
|
2017-03-29 13:33:11 +08:00
|
|
|
* @return bool event updated
|
|
|
|
*/
|
|
|
|
public function update($data, $checkcapability=true) {
|
|
|
|
global $DB, $USER;
|
|
|
|
|
|
|
|
foreach ($data as $key => $value) {
|
|
|
|
$this->properties->$key = $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->properties->timemodified = time();
|
|
|
|
$usingeditor = (!empty($this->properties->description) && is_array($this->properties->description));
|
|
|
|
|
|
|
|
// Prepare event data.
|
|
|
|
$eventargs = array(
|
2017-11-24 06:33:41 +00:00
|
|
|
'context' => $this->get_context(),
|
2017-03-29 13:33:11 +08:00
|
|
|
'objectid' => $this->properties->id,
|
|
|
|
'other' => array(
|
|
|
|
'repeatid' => empty($this->properties->repeatid) ? 0 : $this->properties->repeatid,
|
|
|
|
'timestart' => $this->properties->timestart,
|
|
|
|
'name' => $this->properties->name
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (empty($this->properties->id) || $this->properties->id < 1) {
|
|
|
|
if ($checkcapability) {
|
2017-03-30 12:18:32 +08:00
|
|
|
if (!calendar_add_event_allowed($this->properties)) {
|
2017-03-29 13:33:11 +08:00
|
|
|
print_error('nopermissiontoupdatecalendar');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($usingeditor) {
|
2021-03-23 19:26:50 +08:00
|
|
|
switch ($this->properties->eventtype) {
|
|
|
|
case 'user':
|
|
|
|
$this->properties->courseid = 0;
|
|
|
|
$this->properties->course = 0;
|
|
|
|
$this->properties->groupid = 0;
|
|
|
|
$this->properties->userid = $USER->id;
|
|
|
|
break;
|
|
|
|
case 'site':
|
|
|
|
$this->properties->courseid = SITEID;
|
|
|
|
$this->properties->course = SITEID;
|
|
|
|
$this->properties->groupid = 0;
|
|
|
|
$this->properties->userid = $USER->id;
|
|
|
|
break;
|
|
|
|
case 'course':
|
|
|
|
$this->properties->groupid = 0;
|
|
|
|
$this->properties->userid = $USER->id;
|
|
|
|
break;
|
|
|
|
case 'category':
|
|
|
|
$this->properties->groupid = 0;
|
|
|
|
$this->properties->category = 0;
|
|
|
|
$this->properties->userid = $USER->id;
|
|
|
|
break;
|
|
|
|
case 'group':
|
|
|
|
$this->properties->userid = $USER->id;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// We should NEVER get here, but just incase we do lets fail gracefully.
|
|
|
|
$usingeditor = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-03-29 13:33:11 +08:00
|
|
|
// If we are actually using the editor, we recalculate the context because some default values
|
|
|
|
// were set when calculate_context() was called from the constructor.
|
2021-03-23 19:26:50 +08:00
|
|
|
if ($usingeditor) {
|
|
|
|
$this->properties->context = $this->calculate_context();
|
|
|
|
$this->editorcontext = $this->get_context();
|
|
|
|
}
|
2017-03-29 13:33:11 +08:00
|
|
|
|
|
|
|
$editor = $this->properties->description;
|
|
|
|
$this->properties->format = $this->properties->description['format'];
|
|
|
|
$this->properties->description = $this->properties->description['text'];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert the event into the database.
|
|
|
|
$this->properties->id = $DB->insert_record('event', $this->properties);
|
|
|
|
|
|
|
|
if ($usingeditor) {
|
|
|
|
$this->properties->description = file_save_draft_area_files(
|
|
|
|
$editor['itemid'],
|
|
|
|
$this->editorcontext->id,
|
|
|
|
'calendar',
|
|
|
|
'event_description',
|
|
|
|
$this->properties->id,
|
|
|
|
$this->editoroptions,
|
|
|
|
$editor['text'],
|
|
|
|
$this->editoroptions['forcehttps']);
|
|
|
|
$DB->set_field('event', 'description', $this->properties->description,
|
|
|
|
array('id' => $this->properties->id));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Log the event entry.
|
|
|
|
$eventargs['objectid'] = $this->properties->id;
|
2017-11-24 06:33:41 +00:00
|
|
|
$eventargs['context'] = $this->get_context();
|
2017-03-29 13:33:11 +08:00
|
|
|
$event = \core\event\calendar_event_created::create($eventargs);
|
|
|
|
$event->trigger();
|
|
|
|
|
|
|
|
$repeatedids = array();
|
|
|
|
|
|
|
|
if (!empty($this->properties->repeat)) {
|
|
|
|
$this->properties->repeatid = $this->properties->id;
|
|
|
|
$DB->set_field('event', 'repeatid', $this->properties->repeatid, array('id' => $this->properties->id));
|
|
|
|
|
|
|
|
$eventcopy = clone($this->properties);
|
|
|
|
unset($eventcopy->id);
|
|
|
|
|
|
|
|
$timestart = new \DateTime('@' . $eventcopy->timestart);
|
|
|
|
$timestart->setTimezone(\core_date::get_user_timezone_object());
|
|
|
|
|
|
|
|
for ($i = 1; $i < $eventcopy->repeats; $i++) {
|
|
|
|
|
|
|
|
$timestart->add(new \DateInterval('P7D'));
|
|
|
|
$eventcopy->timestart = $timestart->getTimestamp();
|
|
|
|
|
|
|
|
// Get the event id for the log record.
|
|
|
|
$eventcopyid = $DB->insert_record('event', $eventcopy);
|
|
|
|
|
|
|
|
// If the context has been set delete all associated files.
|
|
|
|
if ($usingeditor) {
|
|
|
|
$fs = get_file_storage();
|
|
|
|
$files = $fs->get_area_files($this->editorcontext->id, 'calendar', 'event_description',
|
|
|
|
$this->properties->id);
|
|
|
|
foreach ($files as $file) {
|
|
|
|
$fs->create_file_from_storedfile(array('itemid' => $eventcopyid), $file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$repeatedids[] = $eventcopyid;
|
|
|
|
|
|
|
|
// Trigger an event.
|
|
|
|
$eventargs['objectid'] = $eventcopyid;
|
|
|
|
$eventargs['other']['timestart'] = $eventcopy->timestart;
|
|
|
|
$event = \core\event\calendar_event_created::create($eventargs);
|
|
|
|
$event->trigger();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
|
|
|
|
if ($checkcapability) {
|
2017-03-30 12:18:32 +08:00
|
|
|
if (!calendar_edit_event_allowed($this->properties)) {
|
2017-03-29 13:33:11 +08:00
|
|
|
print_error('nopermissiontoupdatecalendar');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($usingeditor) {
|
|
|
|
if ($this->editorcontext !== null) {
|
|
|
|
$this->properties->description = file_save_draft_area_files(
|
|
|
|
$this->properties->description['itemid'],
|
|
|
|
$this->editorcontext->id,
|
|
|
|
'calendar',
|
|
|
|
'event_description',
|
|
|
|
$this->properties->id,
|
|
|
|
$this->editoroptions,
|
|
|
|
$this->properties->description['text'],
|
|
|
|
$this->editoroptions['forcehttps']);
|
|
|
|
} else {
|
|
|
|
$this->properties->format = $this->properties->description['format'];
|
|
|
|
$this->properties->description = $this->properties->description['text'];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$event = $DB->get_record('event', array('id' => $this->properties->id));
|
|
|
|
|
|
|
|
$updaterepeated = (!empty($this->properties->repeatid) && !empty($this->properties->repeateditall));
|
|
|
|
|
|
|
|
if ($updaterepeated) {
|
2019-03-28 13:17:32 +01:00
|
|
|
|
|
|
|
$sqlset = 'name = ?,
|
|
|
|
description = ?,
|
|
|
|
timeduration = ?,
|
|
|
|
timemodified = ?,
|
|
|
|
groupid = ?,
|
|
|
|
courseid = ?';
|
|
|
|
|
|
|
|
// Note: Group and course id may not be set. If not, keep their current values.
|
|
|
|
$params = [
|
|
|
|
$this->properties->name,
|
|
|
|
$this->properties->description,
|
|
|
|
$this->properties->timeduration,
|
|
|
|
time(),
|
|
|
|
isset($this->properties->groupid) ? $this->properties->groupid : $event->groupid,
|
|
|
|
isset($this->properties->courseid) ? $this->properties->courseid : $event->courseid,
|
|
|
|
];
|
|
|
|
|
|
|
|
// Note: Only update start date, if it was changed by the user.
|
2017-03-29 13:33:11 +08:00
|
|
|
if ($this->properties->timestart != $event->timestart) {
|
|
|
|
$timestartoffset = $this->properties->timestart - $event->timestart;
|
2019-03-28 13:17:32 +01:00
|
|
|
$sqlset .= ', timestart = timestart + ?';
|
|
|
|
$params[] = $timestartoffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note: Only update location, if it was changed by the user.
|
|
|
|
$updatelocation = (!empty($this->properties->location) && $this->properties->location !== $event->location);
|
|
|
|
if ($updatelocation) {
|
|
|
|
$sqlset .= ', location = ?';
|
|
|
|
$params[] = $this->properties->location;
|
2017-03-29 13:33:11 +08:00
|
|
|
}
|
2019-03-28 13:17:32 +01:00
|
|
|
|
|
|
|
// Update all.
|
|
|
|
$sql = "UPDATE {event}
|
|
|
|
SET $sqlset
|
|
|
|
WHERE repeatid = ?";
|
|
|
|
|
|
|
|
$params[] = $event->repeatid;
|
2017-03-29 13:33:11 +08:00
|
|
|
$DB->execute($sql, $params);
|
|
|
|
|
|
|
|
// Trigger an update event for each of the calendar event.
|
|
|
|
$events = $DB->get_records('event', array('repeatid' => $event->repeatid), '', '*');
|
|
|
|
foreach ($events as $calendarevent) {
|
|
|
|
$eventargs['objectid'] = $calendarevent->id;
|
|
|
|
$eventargs['other']['timestart'] = $calendarevent->timestart;
|
|
|
|
$event = \core\event\calendar_event_updated::create($eventargs);
|
|
|
|
$event->add_record_snapshot('event', $calendarevent);
|
|
|
|
$event->trigger();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$DB->update_record('event', $this->properties);
|
|
|
|
$event = self::load($this->properties->id);
|
|
|
|
$this->properties = $event->properties();
|
|
|
|
|
|
|
|
// Trigger an update event.
|
|
|
|
$event = \core\event\calendar_event_updated::create($eventargs);
|
|
|
|
$event->add_record_snapshot('event', $this->properties);
|
|
|
|
$event->trigger();
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Deletes an event and if selected an repeated events in the same series
|
|
|
|
*
|
|
|
|
* This function deletes an event, any associated events if $deleterepeated=true,
|
|
|
|
* and cleans up any files associated with the events.
|
|
|
|
*
|
|
|
|
* @see self::delete()
|
|
|
|
*
|
|
|
|
* @param bool $deleterepeated delete event repeatedly
|
|
|
|
* @return bool succession of deleting event
|
|
|
|
*/
|
|
|
|
public function delete($deleterepeated = false) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
// If $this->properties->id is not set then something is wrong.
|
|
|
|
if (empty($this->properties->id)) {
|
|
|
|
debugging('Attempting to delete an event before it has been loaded', DEBUG_DEVELOPER);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$calevent = $DB->get_record('event', array('id' => $this->properties->id), '*', MUST_EXIST);
|
|
|
|
// Delete the event.
|
|
|
|
$DB->delete_records('event', array('id' => $this->properties->id));
|
|
|
|
|
|
|
|
// Trigger an event for the delete action.
|
|
|
|
$eventargs = array(
|
2017-11-24 06:33:41 +00:00
|
|
|
'context' => $this->get_context(),
|
2017-03-29 13:33:11 +08:00
|
|
|
'objectid' => $this->properties->id,
|
|
|
|
'other' => array(
|
|
|
|
'repeatid' => empty($this->properties->repeatid) ? 0 : $this->properties->repeatid,
|
|
|
|
'timestart' => $this->properties->timestart,
|
|
|
|
'name' => $this->properties->name
|
|
|
|
));
|
|
|
|
$event = \core\event\calendar_event_deleted::create($eventargs);
|
|
|
|
$event->add_record_snapshot('event', $calevent);
|
|
|
|
$event->trigger();
|
|
|
|
|
|
|
|
// If we are deleting parent of a repeated event series, promote the next event in the series as parent.
|
|
|
|
if (($this->properties->id == $this->properties->repeatid) && !$deleterepeated) {
|
|
|
|
$newparent = $DB->get_field_sql("SELECT id from {event} where repeatid = ? order by id ASC",
|
|
|
|
array($this->properties->id), IGNORE_MULTIPLE);
|
|
|
|
if (!empty($newparent)) {
|
|
|
|
$DB->execute("UPDATE {event} SET repeatid = ? WHERE repeatid = ?",
|
|
|
|
array($newparent, $this->properties->id));
|
|
|
|
// Get all records where the repeatid is the same as the event being removed.
|
|
|
|
$events = $DB->get_records('event', array('repeatid' => $newparent));
|
|
|
|
// For each of the returned events trigger an update event.
|
|
|
|
foreach ($events as $calendarevent) {
|
|
|
|
// Trigger an event for the update.
|
|
|
|
$eventargs['objectid'] = $calendarevent->id;
|
|
|
|
$eventargs['other']['timestart'] = $calendarevent->timestart;
|
|
|
|
$event = \core\event\calendar_event_updated::create($eventargs);
|
|
|
|
$event->add_record_snapshot('event', $calendarevent);
|
|
|
|
$event->trigger();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the editor context hasn't already been set then set it now.
|
|
|
|
if ($this->editorcontext === null) {
|
2017-11-24 06:33:41 +00:00
|
|
|
$this->editorcontext = $this->get_context();
|
2017-03-29 13:33:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the context has been set delete all associated files.
|
|
|
|
if ($this->editorcontext !== null) {
|
|
|
|
$fs = get_file_storage();
|
|
|
|
$files = $fs->get_area_files($this->editorcontext->id, 'calendar', 'event_description', $this->properties->id);
|
|
|
|
foreach ($files as $file) {
|
|
|
|
$file->delete();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we need to delete repeated events then we will fetch them all and delete one by one.
|
|
|
|
if ($deleterepeated && !empty($this->properties->repeatid) && $this->properties->repeatid > 0) {
|
|
|
|
// Get all records where the repeatid is the same as the event being removed.
|
|
|
|
$events = $DB->get_records('event', array('repeatid' => $this->properties->repeatid));
|
|
|
|
// For each of the returned events populate an event object and call delete.
|
|
|
|
// make sure the arg passed is false as we are already deleting all repeats.
|
|
|
|
foreach ($events as $event) {
|
|
|
|
$event = new calendar_event($event);
|
|
|
|
$event->delete(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch all event properties.
|
|
|
|
*
|
|
|
|
* This function returns all of the events properties as an object and optionally
|
|
|
|
* can prepare an editor for the description field at the same time. This is
|
|
|
|
* designed to work when the properties are going to be used to set the default
|
|
|
|
* values of a moodle forms form.
|
|
|
|
*
|
|
|
|
* @param bool $prepareeditor If set to true a editor is prepared for use with
|
|
|
|
* the mforms editor element. (for description)
|
|
|
|
* @return \stdClass Object containing event properties
|
|
|
|
*/
|
|
|
|
public function properties($prepareeditor = false) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
// First take a copy of the properties. We don't want to actually change the
|
|
|
|
// properties or we'd forever be converting back and forwards between an
|
|
|
|
// editor formatted description and not.
|
|
|
|
$properties = clone($this->properties);
|
|
|
|
// Clean the description here.
|
|
|
|
$properties->description = clean_text($properties->description, $properties->format);
|
|
|
|
|
|
|
|
// If set to true we need to prepare the properties for use with an editor
|
|
|
|
// and prepare the file area.
|
|
|
|
if ($prepareeditor) {
|
|
|
|
|
|
|
|
// We may or may not have a property id. If we do then we need to work
|
|
|
|
// out the context so we can copy the existing files to the draft area.
|
|
|
|
if (!empty($properties->id)) {
|
|
|
|
|
|
|
|
if ($properties->eventtype === 'site') {
|
|
|
|
// Site context.
|
2017-11-24 06:33:41 +00:00
|
|
|
$this->editorcontext = $this->get_context();
|
2017-03-29 13:33:11 +08:00
|
|
|
} else if ($properties->eventtype === 'user') {
|
|
|
|
// User context.
|
2017-11-24 06:33:41 +00:00
|
|
|
$this->editorcontext = $this->get_context();
|
2017-03-29 13:33:11 +08:00
|
|
|
} else if ($properties->eventtype === 'group' || $properties->eventtype === 'course') {
|
|
|
|
// First check the course is valid.
|
|
|
|
$course = $DB->get_record('course', array('id' => $properties->courseid));
|
|
|
|
if (!$course) {
|
|
|
|
print_error('invalidcourse');
|
|
|
|
}
|
|
|
|
// Course context.
|
2017-11-24 06:33:41 +00:00
|
|
|
$this->editorcontext = $this->get_context();
|
2017-03-29 13:33:11 +08:00
|
|
|
// We have a course and are within the course context so we had
|
|
|
|
// better use the courses max bytes value.
|
|
|
|
$this->editoroptions['maxbytes'] = $course->maxbytes;
|
2017-09-18 12:47:37 +08:00
|
|
|
} else if ($properties->eventtype === 'category') {
|
|
|
|
// First check the course is valid.
|
2018-06-16 11:57:27 +08:00
|
|
|
\core_course_category::get($properties->categoryid, MUST_EXIST, true);
|
2017-09-18 12:47:37 +08:00
|
|
|
// Course context.
|
2017-11-24 06:33:41 +00:00
|
|
|
$this->editorcontext = $this->get_context();
|
2017-03-29 13:33:11 +08:00
|
|
|
} else {
|
|
|
|
// If we get here we have a custom event type as used by some
|
|
|
|
// modules. In this case the event will have been added by
|
|
|
|
// code and we won't need the editor.
|
|
|
|
$this->editoroptions['maxbytes'] = 0;
|
|
|
|
$this->editoroptions['maxfiles'] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($this->editorcontext) || empty($this->editorcontext->id)) {
|
|
|
|
$contextid = false;
|
|
|
|
} else {
|
|
|
|
// Get the context id that is what we really want.
|
|
|
|
$contextid = $this->editorcontext->id;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
|
|
|
|
// If we get here then this is a new event in which case we don't need a
|
|
|
|
// context as there is no existing files to copy to the draft area.
|
|
|
|
$contextid = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the contextid === false we don't support files so no preparing
|
|
|
|
// a draft area.
|
|
|
|
if ($contextid !== false) {
|
|
|
|
// Just encase it has already been submitted.
|
|
|
|
$draftiddescription = file_get_submitted_draft_itemid('description');
|
|
|
|
// Prepare the draft area, this copies existing files to the draft area as well.
|
|
|
|
$properties->description = file_prepare_draft_area($draftiddescription, $contextid, 'calendar',
|
|
|
|
'event_description', $properties->id, $this->editoroptions, $properties->description);
|
|
|
|
} else {
|
|
|
|
$draftiddescription = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Structure the description field as the editor requires.
|
|
|
|
$properties->description = array('text' => $properties->description, 'format' => $properties->format,
|
|
|
|
'itemid' => $draftiddescription);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finally return the properties.
|
|
|
|
return $properties;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Toggles the visibility of an event
|
|
|
|
*
|
|
|
|
* @param null|bool $force If it is left null the events visibility is flipped,
|
|
|
|
* If it is false the event is made hidden, if it is true it
|
|
|
|
* is made visible.
|
|
|
|
* @return bool if event is successfully updated, toggle will be visible
|
|
|
|
*/
|
|
|
|
public function toggle_visibility($force = null) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
// Set visible to the default if it is not already set.
|
|
|
|
if (empty($this->properties->visible)) {
|
|
|
|
$this->properties->visible = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($force === true || ($force !== false && $this->properties->visible == 0)) {
|
|
|
|
// Make this event visible.
|
|
|
|
$this->properties->visible = 1;
|
|
|
|
} else {
|
|
|
|
// Make this event hidden.
|
|
|
|
$this->properties->visible = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the database to reflect this change.
|
|
|
|
$success = $DB->set_field('event', 'visible', $this->properties->visible, array('id' => $this->properties->id));
|
|
|
|
$calendarevent = $DB->get_record('event', array('id' => $this->properties->id), '*', MUST_EXIST);
|
|
|
|
|
|
|
|
// Prepare event data.
|
|
|
|
$eventargs = array(
|
2017-11-24 06:33:41 +00:00
|
|
|
'context' => $this->get_context(),
|
2017-03-29 13:33:11 +08:00
|
|
|
'objectid' => $this->properties->id,
|
|
|
|
'other' => array(
|
|
|
|
'repeatid' => empty($this->properties->repeatid) ? 0 : $this->properties->repeatid,
|
|
|
|
'timestart' => $this->properties->timestart,
|
|
|
|
'name' => $this->properties->name
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$event = \core\event\calendar_event_updated::create($eventargs);
|
|
|
|
$event->add_record_snapshot('event', $calendarevent);
|
|
|
|
$event->trigger();
|
|
|
|
|
|
|
|
return $success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an event object when provided with an event id.
|
|
|
|
*
|
|
|
|
* This function makes use of MUST_EXIST, if the event id passed in is invalid
|
|
|
|
* it will result in an exception being thrown.
|
|
|
|
*
|
|
|
|
* @param int|object $param event object or event id
|
|
|
|
* @return calendar_event
|
|
|
|
*/
|
|
|
|
public static function load($param) {
|
|
|
|
global $DB;
|
|
|
|
if (is_object($param)) {
|
|
|
|
$event = new calendar_event($param);
|
|
|
|
} else {
|
|
|
|
$event = $DB->get_record('event', array('id' => (int)$param), '*', MUST_EXIST);
|
|
|
|
$event = new calendar_event($event);
|
|
|
|
}
|
|
|
|
return $event;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-08-24 10:06:11 +01:00
|
|
|
* Creates a new event and returns an event object.
|
|
|
|
*
|
|
|
|
* Capability checking should be performed if the user is directly creating the event
|
|
|
|
* and no other capability has been tested. However if the event is not being created
|
|
|
|
* directly by the user and another capability has been checked for them to do this then
|
|
|
|
* capabilites should not be checked.
|
|
|
|
*
|
|
|
|
* For example if a user is creating an event in the calendar the check should be true,
|
|
|
|
* but if you are creating an event in an activity when it is created then the calendar
|
|
|
|
* capabilites should not be checked.
|
2017-03-29 13:33:11 +08:00
|
|
|
*
|
|
|
|
* @param \stdClass|array $properties An object containing event properties
|
2018-08-24 10:06:11 +01:00
|
|
|
* @param bool $checkcapability If Moodle should check the user can manage the calendar events for this call or not.
|
2017-03-29 13:33:11 +08:00
|
|
|
* @throws \coding_exception
|
|
|
|
*
|
|
|
|
* @return calendar_event|bool The event object or false if it failed
|
|
|
|
*/
|
|
|
|
public static function create($properties, $checkcapability = true) {
|
|
|
|
if (is_array($properties)) {
|
|
|
|
$properties = (object)$properties;
|
|
|
|
}
|
|
|
|
if (!is_object($properties)) {
|
|
|
|
throw new \coding_exception('When creating an event properties should be either an object or an assoc array');
|
|
|
|
}
|
|
|
|
$event = new calendar_event($properties);
|
|
|
|
if ($event->update($properties, $checkcapability)) {
|
|
|
|
return $event;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-04 09:21:45 +02:00
|
|
|
/**
|
|
|
|
* Format the event name using the external API.
|
|
|
|
*
|
|
|
|
* This function should we used when text formatting is required in external functions.
|
|
|
|
*
|
|
|
|
* @return string Formatted name.
|
|
|
|
*/
|
|
|
|
public function format_external_name() {
|
|
|
|
if ($this->editorcontext === null) {
|
|
|
|
// Switch on the event type to decide upon the appropriate context to use for this event.
|
|
|
|
$this->editorcontext = $this->get_context();
|
|
|
|
}
|
|
|
|
|
|
|
|
return external_format_string($this->properties->name, $this->editorcontext->id);
|
|
|
|
}
|
|
|
|
|
2017-03-29 13:33:11 +08:00
|
|
|
/**
|
|
|
|
* Format the text using the external API.
|
|
|
|
*
|
|
|
|
* This function should we used when text formatting is required in external functions.
|
|
|
|
*
|
|
|
|
* @return array an array containing the text formatted and the text format
|
|
|
|
*/
|
|
|
|
public function format_external_text() {
|
|
|
|
|
|
|
|
if ($this->editorcontext === null) {
|
|
|
|
// Switch on the event type to decide upon the appropriate context to use for this event.
|
2017-11-24 06:33:41 +00:00
|
|
|
$this->editorcontext = $this->get_context();
|
2017-03-29 13:33:11 +08:00
|
|
|
|
2017-09-18 12:47:37 +08:00
|
|
|
if (!calendar_is_valid_eventtype($this->properties->eventtype)) {
|
2017-03-29 13:33:11 +08:00
|
|
|
// We don't have a context here, do a normal format_text.
|
|
|
|
return external_format_text($this->properties->description, $this->properties->format, $this->editorcontext->id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Work out the item id for the editor, if this is a repeated event then the files will be associated with the original.
|
|
|
|
if (!empty($this->properties->repeatid) && $this->properties->repeatid > 0) {
|
|
|
|
$itemid = $this->properties->repeatid;
|
|
|
|
} else {
|
|
|
|
$itemid = $this->properties->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
return external_format_text($this->properties->description, $this->properties->format, $this->editorcontext->id,
|
|
|
|
'calendar', 'event_description', $itemid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-22 04:03:44 +00:00
|
|
|
/**
|
|
|
|
* Calendar information class
|
|
|
|
*
|
|
|
|
* This class is used simply to organise the information pertaining to a calendar
|
|
|
|
* and is used primarily to make information easily available.
|
2012-01-13 17:45:55 +08:00
|
|
|
*
|
|
|
|
* @package core_calendar
|
|
|
|
* @category calendar
|
|
|
|
* @copyright 2010 Sam Hemelryk
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
2010-06-22 04:03:44 +00:00
|
|
|
*/
|
|
|
|
class calendar_information {
|
2012-01-13 17:45:55 +08:00
|
|
|
|
2013-09-16 17:30:59 +08:00
|
|
|
/**
|
|
|
|
* @var int The timestamp
|
|
|
|
*
|
|
|
|
* Rather than setting the day, month and year we will set a timestamp which will be able
|
|
|
|
* to be used by multiple calendars.
|
|
|
|
*/
|
|
|
|
public $time;
|
2010-06-22 04:03:44 +00:00
|
|
|
|
2012-01-13 17:45:55 +08:00
|
|
|
/** @var int A course id */
|
2010-06-22 04:03:44 +00:00
|
|
|
public $courseid = null;
|
2012-01-13 17:45:55 +08:00
|
|
|
|
2017-09-19 11:12:29 +08:00
|
|
|
/** @var array An array of categories */
|
|
|
|
public $categories = array();
|
|
|
|
|
|
|
|
/** @var int The current category */
|
|
|
|
public $categoryid = null;
|
|
|
|
|
2012-01-13 17:45:55 +08:00
|
|
|
/** @var array An array of courses */
|
2010-06-22 04:03:44 +00:00
|
|
|
public $courses = array();
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/** @var array An array of groups */
|
2010-06-22 04:03:44 +00:00
|
|
|
public $groups = array();
|
2012-01-13 17:45:55 +08:00
|
|
|
|
|
|
|
/** @var array An array of users */
|
2010-06-22 04:03:44 +00:00
|
|
|
public $users = array();
|
|
|
|
|
2017-10-27 10:24:51 +08:00
|
|
|
/** @var context The anticipated context that the calendar is viewed in */
|
|
|
|
public $context = null;
|
|
|
|
|
2021-07-21 23:49:23 +08:00
|
|
|
/** @var string The calendar's view mode. */
|
|
|
|
protected $viewmode;
|
|
|
|
|
2010-06-22 04:03:44 +00:00
|
|
|
/**
|
|
|
|
* Creates a new instance
|
|
|
|
*
|
2012-01-13 17:45:55 +08:00
|
|
|
* @param int $day the number of the day
|
|
|
|
* @param int $month the number of the month
|
|
|
|
* @param int $year the number of the year
|
2013-09-16 17:30:59 +08:00
|
|
|
* @param int $time the unixtimestamp representing the date we want to view, this is used instead of $calmonth
|
|
|
|
* and $calyear to support multiple calendars
|
2010-06-22 04:03:44 +00:00
|
|
|
*/
|
2013-09-16 17:30:59 +08:00
|
|
|
public function __construct($day = 0, $month = 0, $year = 0, $time = 0) {
|
|
|
|
// If a day, month and year were passed then convert it to a timestamp. If these were passed
|
|
|
|
// then we can assume the day, month and year are passed as Gregorian, as no where in core
|
|
|
|
// should we be passing these values rather than the time. This is done for BC.
|
|
|
|
if (!empty($day) || !empty($month) || !empty($year)) {
|
|
|
|
$date = usergetdate(time());
|
|
|
|
if (empty($day)) {
|
|
|
|
$day = $date['mday'];
|
|
|
|
}
|
|
|
|
if (empty($month)) {
|
|
|
|
$month = $date['mon'];
|
|
|
|
}
|
|
|
|
if (empty($year)) {
|
|
|
|
$year = $date['year'];
|
|
|
|
}
|
|
|
|
if (checkdate($month, $day, $year)) {
|
2017-09-10 13:21:33 +08:00
|
|
|
$time = make_timestamp($year, $month, $day);
|
2013-09-16 17:30:59 +08:00
|
|
|
} else {
|
2017-09-10 13:21:33 +08:00
|
|
|
$time = time();
|
2013-09-16 17:30:59 +08:00
|
|
|
}
|
2017-09-10 13:21:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->set_time($time);
|
|
|
|
}
|
|
|
|
|
2017-10-27 10:24:51 +08:00
|
|
|
/**
|
|
|
|
* Creates and set up a instance.
|
|
|
|
*
|
|
|
|
* @param int $time the unixtimestamp representing the date we want to view.
|
|
|
|
* @param int $courseid The ID of the course the user wishes to view.
|
|
|
|
* @param int $categoryid The ID of the category the user wishes to view
|
|
|
|
* If a courseid is specified, this value is ignored.
|
|
|
|
* @return calendar_information
|
|
|
|
*/
|
|
|
|
public static function create($time, int $courseid, int $categoryid = null) : calendar_information {
|
|
|
|
$calendar = new static(0, 0, 0, $time);
|
|
|
|
if ($courseid != SITEID && !empty($courseid)) {
|
|
|
|
// Course ID must be valid and existing.
|
|
|
|
$course = get_course($courseid);
|
|
|
|
$calendar->context = context_course::instance($course->id);
|
|
|
|
|
2018-04-25 14:49:15 +02:00
|
|
|
if (!$course->visible && !is_role_switched($course->id)) {
|
2017-10-27 10:24:51 +08:00
|
|
|
require_capability('moodle/course:viewhiddencourses', $calendar->context);
|
|
|
|
}
|
|
|
|
|
|
|
|
$courses = [$course->id => $course];
|
2018-06-16 11:57:27 +08:00
|
|
|
$category = (\core_course_category::get($course->category, MUST_EXIST, true))->get_db_record();
|
2017-10-27 10:24:51 +08:00
|
|
|
} else if (!empty($categoryid)) {
|
|
|
|
$course = get_site();
|
2018-07-16 11:39:22 +08:00
|
|
|
$courses = calendar_get_default_courses(null, 'id, category, groupmode, groupmodeforce');
|
2017-10-27 10:24:51 +08:00
|
|
|
|
|
|
|
// Filter available courses to those within this category or it's children.
|
|
|
|
$ids = [$categoryid];
|
2018-06-16 11:57:27 +08:00
|
|
|
$category = \core_course_category::get($categoryid);
|
2017-10-27 10:24:51 +08:00
|
|
|
$ids = array_merge($ids, array_keys($category->get_children()));
|
|
|
|
$courses = array_filter($courses, function($course) use ($ids) {
|
|
|
|
return array_search($course->category, $ids) !== false;
|
|
|
|
});
|
|
|
|
$category = $category->get_db_record();
|
|
|
|
|
|
|
|
$calendar->context = context_coursecat::instance($categoryid);
|
|
|
|
} else {
|
|
|
|
$course = get_site();
|
2018-07-16 11:39:22 +08:00
|
|
|
$courses = calendar_get_default_courses(null, 'id, category, groupmode, groupmodeforce');
|
2017-10-27 10:24:51 +08:00
|
|
|
$category = null;
|
|
|
|
|
|
|
|
$calendar->context = context_system::instance();
|
|
|
|
}
|
|
|
|
|
|
|
|
$calendar->set_sources($course, $courses, $category);
|
|
|
|
|
|
|
|
return $calendar;
|
|
|
|
}
|
|
|
|
|
2017-09-10 13:21:33 +08:00
|
|
|
/**
|
|
|
|
* Set the time period of this instance.
|
|
|
|
*
|
|
|
|
* @param int $time the unixtimestamp representing the date we want to view.
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function set_time($time = null) {
|
2017-09-19 11:16:30 +08:00
|
|
|
if (empty($time)) {
|
2013-09-16 17:30:59 +08:00
|
|
|
$this->time = time();
|
2017-09-10 13:21:33 +08:00
|
|
|
} else {
|
|
|
|
$this->time = $time;
|
2010-09-28 02:02:04 +00:00
|
|
|
}
|
2017-09-10 13:21:33 +08:00
|
|
|
|
|
|
|
return $this;
|
2010-06-22 04:03:44 +00:00
|
|
|
}
|
|
|
|
|
2011-06-29 14:50:38 +08:00
|
|
|
/**
|
2012-01-13 17:45:55 +08:00
|
|
|
* Initialize calendar information
|
2011-06-29 14:50:38 +08:00
|
|
|
*
|
2017-09-19 11:12:29 +08:00
|
|
|
* @deprecated 3.4
|
2012-01-13 17:45:55 +08:00
|
|
|
* @param stdClass $course object
|
2011-06-29 14:50:38 +08:00
|
|
|
* @param array $coursestoload An array of courses [$course->id => $course]
|
2012-01-13 17:45:55 +08:00
|
|
|
* @param bool $ignorefilters options to use filter
|
2011-06-29 14:50:38 +08:00
|
|
|
*/
|
|
|
|
public function prepare_for_view(stdClass $course, array $coursestoload, $ignorefilters = false) {
|
2017-09-19 11:12:29 +08:00
|
|
|
debugging('The prepare_for_view() function has been deprecated. Please update your code to use set_sources()',
|
|
|
|
DEBUG_DEVELOPER);
|
|
|
|
$this->set_sources($course, $coursestoload);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the sources for events within the calendar.
|
|
|
|
*
|
|
|
|
* If no category is provided, then the category path for the current
|
|
|
|
* course will be used.
|
|
|
|
*
|
|
|
|
* @param stdClass $course The current course being viewed.
|
2018-03-05 16:16:41 +00:00
|
|
|
* @param stdClass[] $courses The list of all courses currently accessible.
|
2017-09-19 11:12:29 +08:00
|
|
|
* @param stdClass $category The current category to show.
|
|
|
|
*/
|
|
|
|
public function set_sources(stdClass $course, array $courses, stdClass $category = null) {
|
2017-10-27 10:25:20 +08:00
|
|
|
global $USER;
|
|
|
|
|
2017-09-19 11:12:29 +08:00
|
|
|
// A cousre must always be specified.
|
2011-06-29 14:50:38 +08:00
|
|
|
$this->course = $course;
|
2017-09-19 11:12:29 +08:00
|
|
|
$this->courseid = $course->id;
|
|
|
|
|
|
|
|
list($courseids, $group, $user) = calendar_set_filters($courses);
|
|
|
|
$this->courses = $courseids;
|
2011-06-29 14:50:38 +08:00
|
|
|
$this->groups = $group;
|
|
|
|
$this->users = $user;
|
2017-09-19 11:12:29 +08:00
|
|
|
|
|
|
|
// Do not show category events by default.
|
|
|
|
$this->categoryid = null;
|
|
|
|
$this->categories = null;
|
|
|
|
|
2017-10-27 10:25:20 +08:00
|
|
|
// Determine the correct category information to show.
|
|
|
|
// When called with a course, the category of that course is usually included too.
|
|
|
|
// When a category was specifically requested, it should be requested with the site id.
|
|
|
|
if (SITEID !== $this->courseid) {
|
|
|
|
// A specific course was requested.
|
|
|
|
// Fetch the category that this course is in, along with all parents.
|
|
|
|
// Do not include child categories of this category, as the user many not have enrolments in those siblings or children.
|
2018-06-16 11:57:27 +08:00
|
|
|
$category = \core_course_category::get($course->category, MUST_EXIST, true);
|
2017-09-19 11:12:29 +08:00
|
|
|
$this->categoryid = $category->id;
|
|
|
|
|
|
|
|
$this->categories = $category->get_parents();
|
|
|
|
$this->categories[] = $category->id;
|
2017-10-27 10:25:20 +08:00
|
|
|
} else if (null !== $category && $category->id > 0) {
|
|
|
|
// A specific category was requested.
|
|
|
|
// Fetch all parents of this category, along with all children too.
|
2018-06-16 11:57:27 +08:00
|
|
|
$category = \core_course_category::get($category->id);
|
2017-10-27 10:25:20 +08:00
|
|
|
$this->categoryid = $category->id;
|
|
|
|
|
|
|
|
// Build the category list.
|
|
|
|
// This includes the current category.
|
2018-03-05 16:16:41 +00:00
|
|
|
$this->categories = $category->get_parents();
|
|
|
|
$this->categories[] = $category->id;
|
|
|
|
$this->categories = array_merge($this->categories, $category->get_all_children_ids());
|
2017-10-27 10:25:20 +08:00
|
|
|
} else if (SITEID === $this->courseid) {
|
|
|
|
// The site was requested.
|
|
|
|
// Fetch all categories where this user has any enrolment, and all categories that this user can manage.
|
|
|
|
|
|
|
|
// Grab the list of categories that this user has courses in.
|
|
|
|
$coursecategories = array_flip(array_map(function($course) {
|
|
|
|
return $course->category;
|
|
|
|
}, $courses));
|
|
|
|
|
2018-03-05 16:16:41 +00:00
|
|
|
$calcatcache = cache::make('core', 'calendar_categories');
|
|
|
|
$this->categories = $calcatcache->get('site');
|
|
|
|
if ($this->categories === false) {
|
|
|
|
// Use the category id as the key in the following array. That way we do not have to remove duplicates.
|
|
|
|
$categories = [];
|
2018-06-16 11:57:27 +08:00
|
|
|
foreach (\core_course_category::get_all() as $category) {
|
2018-03-05 16:16:41 +00:00
|
|
|
if (isset($coursecategories[$category->id]) ||
|
|
|
|
has_capability('moodle/category:manage', $category->get_context(), $USER, false)) {
|
|
|
|
// If the user has access to a course in this category or can manage the category,
|
|
|
|
// then they can see all parent categories too.
|
|
|
|
$categories[$category->id] = true;
|
|
|
|
foreach ($category->get_parents() as $catid) {
|
|
|
|
$categories[$catid] = true;
|
2017-10-27 10:25:20 +08:00
|
|
|
}
|
|
|
|
}
|
2017-09-19 11:12:29 +08:00
|
|
|
}
|
2018-03-05 16:16:41 +00:00
|
|
|
$this->categories = array_keys($categories);
|
|
|
|
$calcatcache->set('site', $this->categories);
|
2017-09-19 11:12:29 +08:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 14:50:38 +08:00
|
|
|
}
|
|
|
|
|
2010-06-22 04:03:44 +00:00
|
|
|
/**
|
|
|
|
* Ensures the date for the calendar is correct and either sets it to now
|
|
|
|
* or throws a moodle_exception if not
|
|
|
|
*
|
2012-01-13 17:45:55 +08:00
|
|
|
* @param bool $defaultonow use current time
|
|
|
|
* @throws moodle_exception
|
|
|
|
* @return bool validation of checkdate
|
2010-06-22 04:03:44 +00:00
|
|
|
*/
|
|
|
|
public function checkdate($defaultonow = true) {
|
|
|
|
if (!checkdate($this->month, $this->day, $this->year)) {
|
|
|
|
if ($defaultonow) {
|
|
|
|
$now = usergetdate(time());
|
|
|
|
$this->day = intval($now['mday']);
|
|
|
|
$this->month = intval($now['mon']);
|
|
|
|
$this->year = intval($now['year']);
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
throw new moodle_exception('invaliddate');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2013-09-16 17:30:59 +08:00
|
|
|
|
2010-06-22 04:03:44 +00:00
|
|
|
/**
|
|
|
|
* Gets todays timestamp for the calendar
|
2012-01-13 17:45:55 +08:00
|
|
|
*
|
|
|
|
* @return int today timestamp
|
2010-06-22 04:03:44 +00:00
|
|
|
*/
|
|
|
|
public function timestamp_today() {
|
2013-09-16 17:30:59 +08:00
|
|
|
return $this->time;
|
2010-06-22 04:03:44 +00:00
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Gets tomorrows timestamp for the calendar
|
2012-01-13 17:45:55 +08:00
|
|
|
*
|
|
|
|
* @return int tomorrow timestamp
|
2010-06-22 04:03:44 +00:00
|
|
|
*/
|
|
|
|
public function timestamp_tomorrow() {
|
2014-10-29 10:30:54 +08:00
|
|
|
return strtotime('+1 day', $this->time);
|
2010-06-22 04:03:44 +00:00
|
|
|
}
|
|
|
|
/**
|
2012-10-15 16:31:12 +08:00
|
|
|
* Adds the pretend blocks for the calendar
|
2010-06-22 04:03:44 +00:00
|
|
|
*
|
|
|
|
* @param core_calendar_renderer $renderer
|
2012-01-13 17:45:55 +08:00
|
|
|
* @param bool $showfilters display filters, false is set as default
|
|
|
|
* @param string|null $view preference view options (eg: day, month, upcoming)
|
2010-06-22 04:03:44 +00:00
|
|
|
*/
|
|
|
|
public function add_sidecalendar_blocks(core_calendar_renderer $renderer, $showfilters=false, $view=null) {
|
2020-08-12 08:02:42 +08:00
|
|
|
global $PAGE;
|
|
|
|
|
|
|
|
if (!has_capability('moodle/block:view', $PAGE->context) ) {
|
2020-08-10 10:08:17 +08:00
|
|
|
return;
|
|
|
|
}
|
2020-08-12 08:02:42 +08:00
|
|
|
|
2010-06-22 04:03:44 +00:00
|
|
|
if ($showfilters) {
|
|
|
|
$filters = new block_contents();
|
2017-09-08 13:45:38 +08:00
|
|
|
$filters->content = $renderer->event_filter();
|
2010-06-22 04:03:44 +00:00
|
|
|
$filters->footer = '';
|
|
|
|
$filters->title = get_string('eventskey', 'calendar');
|
|
|
|
$renderer->add_pretend_calendar_block($filters, BLOCK_POS_RIGHT);
|
|
|
|
}
|
|
|
|
$block = new block_contents;
|
|
|
|
$block->content = $renderer->fake_block_threemonths($this);
|
|
|
|
$block->footer = '';
|
|
|
|
$block->title = get_string('monthlyview', 'calendar');
|
|
|
|
$renderer->add_pretend_calendar_block($block, BLOCK_POS_RIGHT);
|
|
|
|
}
|
2021-07-21 23:49:23 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Getter method for the calendar's view mode.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function get_viewmode(): string {
|
|
|
|
return $this->viewmode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Setter method for the calendar's view mode.
|
|
|
|
*
|
|
|
|
* @param string $viewmode
|
|
|
|
*/
|
|
|
|
public function set_viewmode(string $viewmode): void {
|
|
|
|
$this->viewmode = $viewmode;
|
|
|
|
}
|
2010-11-17 07:56:40 +00:00
|
|
|
}
|
MDL-16660: Add calendar import.
Adds an extra import section alongside the existing export in the
calendar views. Allows the user to import from either a file or a
URL as a subscription, with an optional polling interval of hourly,
daily, weekly, monthly, or annually. This subscription may be added
to the user, group, global or course calendars. These subscriptions
are tracked in a separate database table, and an extra column added
to the events table to relate them to the subscription. The event
uuid field is also expanded to allow for the RFC-2445 UID property.
Subscriptions are listed on the calendar view page, and can be
added and removed, manually polled, and the polling interval
adjusted. Subscription events are updated on cron.
2012-07-23 13:49:43 +12:00
|
|
|
|
2017-03-30 11:23:08 +08:00
|
|
|
/**
|
|
|
|
* Get calendar events.
|
|
|
|
*
|
|
|
|
* @param int $tstart Start time of time range for events
|
|
|
|
* @param int $tend End time of time range for events
|
|
|
|
* @param array|int|boolean $users array of users, user id or boolean for all/no user events
|
|
|
|
* @param array|int|boolean $groups array of groups, group id or boolean for all/no group events
|
|
|
|
* @param array|int|boolean $courses array of courses, course id or boolean for all/no course events
|
|
|
|
* @param boolean $withduration whether only events starting within time range selected
|
|
|
|
* or events in progress/already started selected as well
|
|
|
|
* @param boolean $ignorehidden whether to select only visible events or all events
|
2017-09-19 11:15:13 +08:00
|
|
|
* @param array|int|boolean $categories array of categories, category id or boolean for all/no course events
|
2017-03-30 11:23:08 +08:00
|
|
|
* @return array $events of selected events or an empty array if there aren't any (or there was an error)
|
|
|
|
*/
|
2017-09-19 11:15:13 +08:00
|
|
|
function calendar_get_events($tstart, $tend, $users, $groups, $courses,
|
|
|
|
$withduration = true, $ignorehidden = true, $categories = []) {
|
2017-03-30 11:23:08 +08:00
|
|
|
global $DB;
|
|
|
|
|
2017-05-04 12:09:21 +08:00
|
|
|
$whereclause = '';
|
|
|
|
$params = array();
|
2017-03-30 11:23:08 +08:00
|
|
|
// Quick test.
|
2017-09-19 11:15:13 +08:00
|
|
|
if (empty($users) && empty($groups) && empty($courses) && empty($categories)) {
|
2017-03-30 11:23:08 +08:00
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((is_array($users) && !empty($users)) or is_numeric($users)) {
|
2017-05-04 12:09:21 +08:00
|
|
|
// Events from a number of users
|
|
|
|
if(!empty($whereclause)) $whereclause .= ' OR';
|
2017-03-30 11:23:08 +08:00
|
|
|
list($insqlusers, $inparamsusers) = $DB->get_in_or_equal($users, SQL_PARAMS_NAMED);
|
2017-09-19 11:15:13 +08:00
|
|
|
$whereclause .= " (e.userid $insqlusers AND e.courseid = 0 AND e.groupid = 0 AND e.categoryid = 0)";
|
2017-03-30 11:23:08 +08:00
|
|
|
$params = array_merge($params, $inparamsusers);
|
2017-05-04 12:09:21 +08:00
|
|
|
} else if($users === true) {
|
|
|
|
// Events from ALL users
|
|
|
|
if(!empty($whereclause)) $whereclause .= ' OR';
|
2017-09-19 11:15:13 +08:00
|
|
|
$whereclause .= ' (e.userid != 0 AND e.courseid = 0 AND e.groupid = 0 AND e.categoryid = 0)';
|
2017-05-04 12:09:21 +08:00
|
|
|
} else if($users === false) {
|
|
|
|
// No user at all, do nothing
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((is_array($groups) && !empty($groups)) or is_numeric($groups)) {
|
2017-05-04 12:09:21 +08:00
|
|
|
// Events from a number of groups
|
|
|
|
if(!empty($whereclause)) $whereclause .= ' OR';
|
2017-03-30 11:23:08 +08:00
|
|
|
list($insqlgroups, $inparamsgroups) = $DB->get_in_or_equal($groups, SQL_PARAMS_NAMED);
|
2017-05-04 12:09:21 +08:00
|
|
|
$whereclause .= " e.groupid $insqlgroups ";
|
2017-03-30 11:23:08 +08:00
|
|
|
$params = array_merge($params, $inparamsgroups);
|
2017-05-04 12:09:21 +08:00
|
|
|
} else if($groups === true) {
|
|
|
|
// Events from ALL groups
|
|
|
|
if(!empty($whereclause)) $whereclause .= ' OR ';
|
|
|
|
$whereclause .= ' e.groupid != 0';
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
2017-05-04 12:09:21 +08:00
|
|
|
// boolean false (no groups at all): we don't need to do anything
|
2017-03-30 11:23:08 +08:00
|
|
|
|
|
|
|
if ((is_array($courses) && !empty($courses)) or is_numeric($courses)) {
|
2017-05-04 12:09:21 +08:00
|
|
|
if(!empty($whereclause)) $whereclause .= ' OR';
|
2017-03-30 11:23:08 +08:00
|
|
|
list($insqlcourses, $inparamscourses) = $DB->get_in_or_equal($courses, SQL_PARAMS_NAMED);
|
2017-05-04 12:09:21 +08:00
|
|
|
$whereclause .= " (e.groupid = 0 AND e.courseid $insqlcourses)";
|
2017-03-30 11:23:08 +08:00
|
|
|
$params = array_merge($params, $inparamscourses);
|
|
|
|
} else if ($courses === true) {
|
2017-05-04 12:09:21 +08:00
|
|
|
// Events from ALL courses
|
|
|
|
if(!empty($whereclause)) $whereclause .= ' OR';
|
|
|
|
$whereclause .= ' (e.groupid = 0 AND e.courseid != 0)';
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
2017-09-19 11:15:13 +08:00
|
|
|
if ((is_array($categories) && !empty($categories)) || is_numeric($categories)) {
|
2017-09-22 09:56:18 +08:00
|
|
|
if (!empty($whereclause)) {
|
|
|
|
$whereclause .= ' OR';
|
|
|
|
}
|
2017-09-19 11:15:13 +08:00
|
|
|
list($insqlcategories, $inparamscategories) = $DB->get_in_or_equal($categories, SQL_PARAMS_NAMED);
|
|
|
|
$whereclause .= " (e.groupid = 0 AND e.courseid = 0 AND e.categoryid $insqlcategories)";
|
|
|
|
$params = array_merge($params, $inparamscategories);
|
|
|
|
} else if ($categories === true) {
|
|
|
|
// Events from ALL categories.
|
2017-09-22 09:56:18 +08:00
|
|
|
if (!empty($whereclause)) {
|
|
|
|
$whereclause .= ' OR';
|
|
|
|
}
|
2017-09-19 11:15:13 +08:00
|
|
|
$whereclause .= ' (e.groupid = 0 AND e.courseid = 0 AND e.categoryid != 0)';
|
|
|
|
}
|
|
|
|
|
2017-03-30 11:23:08 +08:00
|
|
|
// Security check: if, by now, we have NOTHING in $whereclause, then it means
|
|
|
|
// that NO event-selecting clauses were defined. Thus, we won't be returning ANY
|
|
|
|
// events no matter what. Allowing the code to proceed might return a completely
|
|
|
|
// valid query with only time constraints, thus selecting ALL events in that time frame!
|
2017-05-04 12:09:21 +08:00
|
|
|
if(empty($whereclause)) {
|
2017-03-30 11:23:08 +08:00
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
2017-05-04 12:09:21 +08:00
|
|
|
if($withduration) {
|
|
|
|
$timeclause = '(e.timestart >= '.$tstart.' OR e.timestart + e.timeduration > '.$tstart.') AND e.timestart <= '.$tend;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
2017-05-04 12:09:21 +08:00
|
|
|
else {
|
|
|
|
$timeclause = 'e.timestart >= '.$tstart.' AND e.timestart <= '.$tend;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
2017-05-04 12:09:21 +08:00
|
|
|
if(!empty($whereclause)) {
|
|
|
|
// We have additional constraints
|
|
|
|
$whereclause = $timeclause.' AND ('.$whereclause.')';
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
2017-05-04 12:09:21 +08:00
|
|
|
else {
|
|
|
|
// Just basic time filtering
|
|
|
|
$whereclause = $timeclause;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
2017-05-04 12:09:21 +08:00
|
|
|
if ($ignorehidden) {
|
|
|
|
$whereclause .= ' AND e.visible = 1';
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
$sql = "SELECT e.*
|
2017-05-04 12:09:21 +08:00
|
|
|
FROM {event} e
|
|
|
|
LEFT JOIN {modules} m ON e.modulename = m.name
|
|
|
|
-- Non visible modules will have a value of 0.
|
|
|
|
WHERE (m.visible = 1 OR m.visible IS NULL) AND $whereclause
|
|
|
|
ORDER BY e.timestart";
|
2017-03-30 11:23:08 +08:00
|
|
|
$events = $DB->get_records_sql($sql, $params);
|
|
|
|
|
|
|
|
if ($events === false) {
|
|
|
|
$events = array();
|
|
|
|
}
|
|
|
|
return $events;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the days of the week.
|
|
|
|
*
|
|
|
|
* @return array array of days
|
|
|
|
*/
|
|
|
|
function calendar_get_days() {
|
2017-03-30 12:18:32 +08:00
|
|
|
$calendartype = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
return $calendartype->get_weekdays();
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the subscription from a given id.
|
|
|
|
*
|
|
|
|
* @since Moodle 2.5
|
|
|
|
* @param int $id id of the subscription
|
|
|
|
* @return stdClass Subscription record from DB
|
|
|
|
* @throws moodle_exception for an invalid id
|
|
|
|
*/
|
|
|
|
function calendar_get_subscription($id) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$cache = \cache::make('core', 'calendar_subscriptions');
|
|
|
|
$subscription = $cache->get($id);
|
|
|
|
if (empty($subscription)) {
|
|
|
|
$subscription = $DB->get_record('event_subscriptions', array('id' => $id), '*', MUST_EXIST);
|
|
|
|
$cache->set($id, $subscription);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $subscription;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the first day of the week.
|
|
|
|
*
|
|
|
|
* Used to be define('CALENDAR_STARTING_WEEKDAY', blah);
|
|
|
|
*
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
function calendar_get_starting_weekday() {
|
2017-03-30 12:18:32 +08:00
|
|
|
$calendartype = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
return $calendartype->get_starting_weekday();
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a HTML link to a course.
|
|
|
|
*
|
2017-04-24 13:40:54 +08:00
|
|
|
* @param int|stdClass $course the course id or course object
|
2017-03-30 11:23:08 +08:00
|
|
|
* @return string a link to the course (as HTML); empty if the course id is invalid
|
|
|
|
*/
|
2017-04-24 13:40:54 +08:00
|
|
|
function calendar_get_courselink($course) {
|
|
|
|
if (!$course) {
|
2017-03-30 12:18:32 +08:00
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2017-04-24 13:40:54 +08:00
|
|
|
if (!is_object($course)) {
|
|
|
|
$course = calendar_get_course_cached($coursecache, $course);
|
|
|
|
}
|
|
|
|
$context = \context_course::instance($course->id);
|
|
|
|
$fullname = format_string($course->fullname, true, array('context' => $context));
|
|
|
|
$url = new \moodle_url('/course/view.php', array('id' => $course->id));
|
2017-03-30 12:18:32 +08:00
|
|
|
$link = \html_writer::link($url, $fullname);
|
|
|
|
|
|
|
|
return $link;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get current module cache.
|
|
|
|
*
|
2017-04-24 14:09:09 +08:00
|
|
|
* Only use this method if you do not know courseid. Otherwise use:
|
2017-04-24 13:40:54 +08:00
|
|
|
* get_fast_modinfo($courseid)->instances[$modulename][$instance]
|
|
|
|
*
|
2017-03-30 12:18:32 +08:00
|
|
|
* @param array $modulecache in memory module cache
|
2017-03-30 11:23:08 +08:00
|
|
|
* @param string $modulename name of the module
|
|
|
|
* @param int $instance module instance number
|
|
|
|
* @return stdClass|bool $module information
|
|
|
|
*/
|
2017-03-30 12:18:32 +08:00
|
|
|
function calendar_get_module_cached(&$modulecache, $modulename, $instance) {
|
|
|
|
if (!isset($modulecache[$modulename . '_' . $instance])) {
|
|
|
|
$modulecache[$modulename . '_' . $instance] = get_coursemodule_from_instance($modulename, $instance);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
return $modulecache[$modulename . '_' . $instance];
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get current course cache.
|
|
|
|
*
|
|
|
|
* @param array $coursecache list of course cache
|
|
|
|
* @param int $courseid id of the course
|
|
|
|
* @return stdClass $coursecache[$courseid] return the specific course cache
|
|
|
|
*/
|
|
|
|
function calendar_get_course_cached(&$coursecache, $courseid) {
|
2017-03-30 12:18:32 +08:00
|
|
|
if (!isset($coursecache[$courseid])) {
|
|
|
|
$coursecache[$courseid] = get_course($courseid);
|
|
|
|
}
|
|
|
|
return $coursecache[$courseid];
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get group from groupid for calendar display
|
|
|
|
*
|
|
|
|
* @param int $groupid
|
|
|
|
* @return stdClass group object with fields 'id', 'name' and 'courseid'
|
|
|
|
*/
|
|
|
|
function calendar_get_group_cached($groupid) {
|
2017-03-30 12:18:32 +08:00
|
|
|
static $groupscache = array();
|
|
|
|
if (!isset($groupscache[$groupid])) {
|
|
|
|
$groupscache[$groupid] = groups_get_group($groupid, 'id,name,courseid');
|
|
|
|
}
|
|
|
|
return $groupscache[$groupid];
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add calendar event metadata
|
|
|
|
*
|
2020-04-29 22:22:10 +02:00
|
|
|
* @deprecated since 3.9
|
|
|
|
*
|
2017-03-30 11:23:08 +08:00
|
|
|
* @param stdClass $event event info
|
|
|
|
* @return stdClass $event metadata
|
|
|
|
*/
|
|
|
|
function calendar_add_event_metadata($event) {
|
2020-04-29 22:22:10 +02:00
|
|
|
debugging('This function is no longer used', DEBUG_DEVELOPER);
|
2017-03-30 12:18:32 +08:00
|
|
|
global $CFG, $OUTPUT;
|
|
|
|
|
|
|
|
// Support multilang in event->name.
|
|
|
|
$event->name = format_string($event->name, true);
|
|
|
|
|
|
|
|
if (!empty($event->modulename)) { // Activity event.
|
|
|
|
// The module name is set. I will assume that it has to be displayed, and
|
|
|
|
// also that it is an automatically-generated event. And of course that the
|
2017-04-24 13:40:54 +08:00
|
|
|
// instace id and modulename are set correctly.
|
|
|
|
$instances = get_fast_modinfo($event->courseid)->get_instances_of($event->modulename);
|
|
|
|
if (!array_key_exists($event->instance, $instances)) {
|
2017-03-30 12:18:32 +08:00
|
|
|
return;
|
|
|
|
}
|
2017-04-24 13:40:54 +08:00
|
|
|
$module = $instances[$event->instance];
|
2017-03-30 12:18:32 +08:00
|
|
|
|
2017-04-24 13:40:54 +08:00
|
|
|
$modulename = $module->get_module_type_name(false);
|
2017-03-30 12:18:32 +08:00
|
|
|
if (get_string_manager()->string_exists($event->eventtype, $event->modulename)) {
|
|
|
|
// Will be used as alt text if the event icon.
|
|
|
|
$eventtype = get_string($event->eventtype, $event->modulename);
|
|
|
|
} else {
|
|
|
|
$eventtype = '';
|
|
|
|
}
|
|
|
|
|
2017-04-24 13:40:54 +08:00
|
|
|
$event->icon = '<img src="' . s($module->get_icon_url()) . '" alt="' . s($eventtype) .
|
|
|
|
'" title="' . s($modulename) . '" class="icon" />';
|
|
|
|
$event->referer = html_writer::link($module->url, $event->name);
|
|
|
|
$event->courselink = calendar_get_courselink($module->get_course());
|
2017-03-30 12:18:32 +08:00
|
|
|
$event->cmid = $module->id;
|
|
|
|
} else if ($event->courseid == SITEID) { // Site event.
|
|
|
|
$event->icon = '<img src="' . $OUTPUT->image_url('i/siteevent') . '" alt="' .
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
get_string('siteevent', 'calendar') . '" class="icon" />';
|
|
|
|
$event->cssclass = 'calendar_event_site';
|
2017-03-30 12:18:32 +08:00
|
|
|
} else if ($event->courseid != 0 && $event->courseid != SITEID && $event->groupid == 0) { // Course event.
|
|
|
|
$event->icon = '<img src="' . $OUTPUT->image_url('i/courseevent') . '" alt="' .
|
|
|
|
get_string('courseevent', 'calendar') . '" class="icon" />';
|
|
|
|
$event->courselink = calendar_get_courselink($event->courseid);
|
|
|
|
$event->cssclass = 'calendar_event_course';
|
|
|
|
} else if ($event->groupid) { // Group event.
|
|
|
|
if ($group = calendar_get_group_cached($event->groupid)) {
|
|
|
|
$groupname = format_string($group->name, true, \context_course::instance($group->courseid));
|
|
|
|
} else {
|
|
|
|
$groupname = '';
|
|
|
|
}
|
|
|
|
$event->icon = \html_writer::empty_tag('image', array('src' => $OUTPUT->image_url('i/groupevent'),
|
|
|
|
'alt' => get_string('groupevent', 'calendar'), 'title' => $groupname, 'class' => 'icon'));
|
|
|
|
$event->courselink = calendar_get_courselink($event->courseid) . ', ' . $groupname;
|
|
|
|
$event->cssclass = 'calendar_event_group';
|
|
|
|
} else if ($event->userid) { // User event.
|
|
|
|
$event->icon = '<img src="' . $OUTPUT->image_url('i/userevent') . '" alt="' .
|
|
|
|
get_string('userevent', 'calendar') . '" class="icon" />';
|
|
|
|
$event->cssclass = 'calendar_event_user';
|
|
|
|
}
|
|
|
|
|
|
|
|
return $event;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get calendar events by id.
|
|
|
|
*
|
|
|
|
* @since Moodle 2.5
|
|
|
|
* @param array $eventids list of event ids
|
|
|
|
* @return array Array of event entries, empty array if nothing found
|
|
|
|
*/
|
|
|
|
function calendar_get_events_by_id($eventids) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if (!is_array($eventids) || empty($eventids)) {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
|
|
|
list($wheresql, $params) = $DB->get_in_or_equal($eventids);
|
|
|
|
$wheresql = "id $wheresql";
|
|
|
|
|
|
|
|
return $DB->get_records_select('event', $wheresql, $params);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get control options for calendar.
|
|
|
|
*
|
|
|
|
* @param string $type of calendar
|
|
|
|
* @param array $data calendar information
|
|
|
|
* @return string $content return available control for the calender in html
|
|
|
|
*/
|
|
|
|
function calendar_top_controls($type, $data) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $PAGE, $OUTPUT;
|
|
|
|
|
|
|
|
// Get the calendar type we are using.
|
|
|
|
$calendartype = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
|
|
|
|
$content = '';
|
|
|
|
|
|
|
|
// Ensure course id passed if relevant.
|
|
|
|
$courseid = '';
|
|
|
|
if (!empty($data['id'])) {
|
|
|
|
$courseid = '&course=' . $data['id'];
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we are passing a month and year then we need to convert this to a timestamp to
|
|
|
|
// support multiple calendars. No where in core should these be passed, this logic
|
|
|
|
// here is for third party plugins that may use this function.
|
|
|
|
if (!empty($data['m']) && !empty($date['y'])) {
|
|
|
|
if (!isset($data['d'])) {
|
|
|
|
$data['d'] = 1;
|
|
|
|
}
|
|
|
|
if (!checkdate($data['m'], $data['d'], $data['y'])) {
|
|
|
|
$time = time();
|
|
|
|
} else {
|
|
|
|
$time = make_timestamp($data['y'], $data['m'], $data['d']);
|
|
|
|
}
|
|
|
|
} else if (!empty($data['time'])) {
|
|
|
|
$time = $data['time'];
|
|
|
|
} else {
|
|
|
|
$time = time();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the date for the calendar type.
|
|
|
|
$date = $calendartype->timestamp_to_date_array($time);
|
|
|
|
|
|
|
|
$urlbase = $PAGE->url;
|
|
|
|
|
|
|
|
// We need to get the previous and next months in certain cases.
|
|
|
|
if ($type == 'frontpage' || $type == 'course' || $type == 'month') {
|
|
|
|
$prevmonth = calendar_sub_month($date['mon'], $date['year']);
|
|
|
|
$prevmonthtime = $calendartype->convert_to_gregorian($prevmonth[1], $prevmonth[0], 1);
|
|
|
|
$prevmonthtime = make_timestamp($prevmonthtime['year'], $prevmonthtime['month'], $prevmonthtime['day'],
|
|
|
|
$prevmonthtime['hour'], $prevmonthtime['minute']);
|
|
|
|
|
|
|
|
$nextmonth = calendar_add_month($date['mon'], $date['year']);
|
|
|
|
$nextmonthtime = $calendartype->convert_to_gregorian($nextmonth[1], $nextmonth[0], 1);
|
|
|
|
$nextmonthtime = make_timestamp($nextmonthtime['year'], $nextmonthtime['month'], $nextmonthtime['day'],
|
|
|
|
$nextmonthtime['hour'], $nextmonthtime['minute']);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ($type) {
|
|
|
|
case 'frontpage':
|
2019-08-12 00:13:08 +02:00
|
|
|
$prevlink = calendar_get_link_previous(get_string('monthprev', 'calendar'), $urlbase, false, false, false,
|
2017-03-30 12:18:32 +08:00
|
|
|
true, $prevmonthtime);
|
2019-08-12 00:13:08 +02:00
|
|
|
$nextlink = calendar_get_link_next(get_string('monthnext', 'calendar'), $urlbase, false, false, false, true,
|
2017-03-30 12:18:32 +08:00
|
|
|
$nextmonthtime);
|
|
|
|
$calendarlink = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', array('view' => 'month')),
|
|
|
|
false, false, false, $time);
|
|
|
|
|
|
|
|
if (!empty($data['id'])) {
|
|
|
|
$calendarlink->param('course', $data['id']);
|
|
|
|
}
|
|
|
|
|
|
|
|
$right = $nextlink;
|
|
|
|
|
|
|
|
$content .= \html_writer::start_tag('div', array('class' => 'calendar-controls'));
|
|
|
|
$content .= $prevlink . '<span class="hide"> | </span>';
|
|
|
|
$content .= \html_writer::tag('span', \html_writer::link($calendarlink,
|
|
|
|
userdate($time, get_string('strftimemonthyear')), array('title' => get_string('monththis', 'calendar'))
|
|
|
|
), array('class' => 'current'));
|
|
|
|
$content .= '<span class="hide"> | </span>' . $right;
|
|
|
|
$content .= "<span class=\"clearer\"><!-- --></span>\n";
|
|
|
|
$content .= \html_writer::end_tag('div');
|
|
|
|
|
|
|
|
break;
|
|
|
|
case 'course':
|
2019-08-12 00:13:08 +02:00
|
|
|
$prevlink = calendar_get_link_previous(get_string('monthprev', 'calendar'), $urlbase, false, false, false,
|
2017-03-30 12:18:32 +08:00
|
|
|
true, $prevmonthtime);
|
2019-08-12 00:13:08 +02:00
|
|
|
$nextlink = calendar_get_link_next(get_string('monthnext', 'calendar'), $urlbase, false, false, false,
|
2017-03-30 12:18:32 +08:00
|
|
|
true, $nextmonthtime);
|
|
|
|
$calendarlink = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', array('view' => 'month')),
|
|
|
|
false, false, false, $time);
|
|
|
|
|
|
|
|
if (!empty($data['id'])) {
|
|
|
|
$calendarlink->param('course', $data['id']);
|
|
|
|
}
|
|
|
|
|
|
|
|
$content .= \html_writer::start_tag('div', array('class' => 'calendar-controls'));
|
|
|
|
$content .= $prevlink . '<span class="hide"> | </span>';
|
|
|
|
$content .= \html_writer::tag('span', \html_writer::link($calendarlink,
|
|
|
|
userdate($time, get_string('strftimemonthyear')), array('title' => get_string('monththis', 'calendar'))
|
|
|
|
), array('class' => 'current'));
|
|
|
|
$content .= '<span class="hide"> | </span>' . $nextlink;
|
|
|
|
$content .= "<span class=\"clearer\"><!-- --></span>";
|
|
|
|
$content .= \html_writer::end_tag('div');
|
|
|
|
break;
|
|
|
|
case 'upcoming':
|
|
|
|
$calendarlink = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', array('view' => 'upcoming')),
|
|
|
|
false, false, false, $time);
|
|
|
|
if (!empty($data['id'])) {
|
|
|
|
$calendarlink->param('course', $data['id']);
|
|
|
|
}
|
|
|
|
$calendarlink = \html_writer::link($calendarlink, userdate($time, get_string('strftimemonthyear')));
|
|
|
|
$content .= \html_writer::tag('div', $calendarlink, array('class' => 'centered'));
|
|
|
|
break;
|
|
|
|
case 'display':
|
|
|
|
$calendarlink = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', array('view' => 'month')),
|
|
|
|
false, false, false, $time);
|
|
|
|
if (!empty($data['id'])) {
|
|
|
|
$calendarlink->param('course', $data['id']);
|
|
|
|
}
|
|
|
|
$calendarlink = \html_writer::link($calendarlink, userdate($time, get_string('strftimemonthyear')));
|
|
|
|
$content .= \html_writer::tag('h3', $calendarlink);
|
|
|
|
break;
|
|
|
|
case 'month':
|
|
|
|
$prevlink = calendar_get_link_previous(userdate($prevmonthtime, get_string('strftimemonthyear')),
|
|
|
|
'view.php?view=month' . $courseid . '&', false, false, false, false, $prevmonthtime);
|
|
|
|
$nextlink = calendar_get_link_next(userdate($nextmonthtime, get_string('strftimemonthyear')),
|
|
|
|
'view.php?view=month' . $courseid . '&', false, false, false, false, $nextmonthtime);
|
|
|
|
|
|
|
|
$content .= \html_writer::start_tag('div', array('class' => 'calendar-controls'));
|
|
|
|
$content .= $prevlink . '<span class="hide"> | </span>';
|
|
|
|
$content .= $OUTPUT->heading(userdate($time, get_string('strftimemonthyear')), 2, 'current');
|
|
|
|
$content .= '<span class="hide"> | </span>' . $nextlink;
|
|
|
|
$content .= '<span class="clearer"><!-- --></span>';
|
|
|
|
$content .= \html_writer::end_tag('div')."\n";
|
|
|
|
break;
|
|
|
|
case 'day':
|
|
|
|
$days = calendar_get_days();
|
|
|
|
|
|
|
|
$prevtimestamp = strtotime('-1 day', $time);
|
|
|
|
$nexttimestamp = strtotime('+1 day', $time);
|
|
|
|
|
|
|
|
$prevdate = $calendartype->timestamp_to_date_array($prevtimestamp);
|
|
|
|
$nextdate = $calendartype->timestamp_to_date_array($nexttimestamp);
|
|
|
|
|
|
|
|
$prevname = $days[$prevdate['wday']]['fullname'];
|
|
|
|
$nextname = $days[$nextdate['wday']]['fullname'];
|
|
|
|
$prevlink = calendar_get_link_previous($prevname, 'view.php?view=day' . $courseid . '&', false, false,
|
|
|
|
false, false, $prevtimestamp);
|
|
|
|
$nextlink = calendar_get_link_next($nextname, 'view.php?view=day' . $courseid . '&', false, false, false,
|
|
|
|
false, $nexttimestamp);
|
|
|
|
|
|
|
|
$content .= \html_writer::start_tag('div', array('class' => 'calendar-controls'));
|
|
|
|
$content .= $prevlink;
|
|
|
|
$content .= '<span class="hide"> | </span><span class="current">' .userdate($time,
|
|
|
|
get_string('strftimedaydate')) . '</span>';
|
|
|
|
$content .= '<span class="hide"> | </span>' . $nextlink;
|
|
|
|
$content .= "<span class=\"clearer\"><!-- --></span>";
|
|
|
|
$content .= \html_writer::end_tag('div') . "\n";
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $content;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the representation day.
|
|
|
|
*
|
|
|
|
* @param int $tstamp Timestamp in GMT
|
|
|
|
* @param int|bool $now current Unix timestamp
|
|
|
|
* @param bool $usecommonwords
|
|
|
|
* @return string the formatted date/time
|
|
|
|
*/
|
|
|
|
function calendar_day_representation($tstamp, $now = false, $usecommonwords = true) {
|
2017-03-30 12:18:32 +08:00
|
|
|
static $shortformat;
|
|
|
|
|
|
|
|
if (empty($shortformat)) {
|
|
|
|
$shortformat = get_string('strftimedayshort');
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($now === false) {
|
|
|
|
$now = time();
|
|
|
|
}
|
|
|
|
|
|
|
|
// To have it in one place, if a change is needed.
|
|
|
|
$formal = userdate($tstamp, $shortformat);
|
|
|
|
|
|
|
|
$datestamp = usergetdate($tstamp);
|
|
|
|
$datenow = usergetdate($now);
|
|
|
|
|
|
|
|
if ($usecommonwords == false) {
|
|
|
|
// We don't want words, just a date.
|
|
|
|
return $formal;
|
|
|
|
} else if ($datestamp['year'] == $datenow['year'] && $datestamp['yday'] == $datenow['yday']) {
|
|
|
|
return get_string('today', 'calendar');
|
|
|
|
} else if (($datestamp['year'] == $datenow['year'] && $datestamp['yday'] == $datenow['yday'] - 1 ) ||
|
|
|
|
($datestamp['year'] == $datenow['year'] - 1 && $datestamp['mday'] == 31 && $datestamp['mon'] == 12
|
|
|
|
&& $datenow['yday'] == 1)) {
|
|
|
|
return get_string('yesterday', 'calendar');
|
|
|
|
} else if (($datestamp['year'] == $datenow['year'] && $datestamp['yday'] == $datenow['yday'] + 1 ) ||
|
|
|
|
($datestamp['year'] == $datenow['year'] + 1 && $datenow['mday'] == 31 && $datenow['mon'] == 12
|
|
|
|
&& $datestamp['yday'] == 1)) {
|
|
|
|
return get_string('tomorrow', 'calendar');
|
|
|
|
} else {
|
|
|
|
return $formal;
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* return the formatted representation time.
|
|
|
|
*
|
|
|
|
|
|
|
|
* @param int $time the timestamp in UTC, as obtained from the database
|
|
|
|
* @return string the formatted date/time
|
|
|
|
*/
|
|
|
|
function calendar_time_representation($time) {
|
2017-03-30 12:18:32 +08:00
|
|
|
static $langtimeformat = null;
|
|
|
|
|
|
|
|
if ($langtimeformat === null) {
|
|
|
|
$langtimeformat = get_string('strftimetime');
|
|
|
|
}
|
|
|
|
|
|
|
|
$timeformat = get_user_preferences('calendar_timeformat');
|
|
|
|
if (empty($timeformat)) {
|
|
|
|
$timeformat = get_config(null, 'calendar_site_timeformat');
|
|
|
|
}
|
|
|
|
|
2018-01-04 14:52:57 +08:00
|
|
|
// Allow language customization of selected time format.
|
|
|
|
if ($timeformat === CALENDAR_TF_12) {
|
|
|
|
$timeformat = get_string('strftimetime12', 'langconfig');
|
|
|
|
} else if ($timeformat === CALENDAR_TF_24) {
|
|
|
|
$timeformat = get_string('strftimetime24', 'langconfig');
|
|
|
|
}
|
|
|
|
|
2017-03-30 12:18:32 +08:00
|
|
|
return userdate($time, empty($timeformat) ? $langtimeformat : $timeformat);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds day, month, year arguments to a URL and returns a moodle_url object.
|
|
|
|
*
|
|
|
|
* @param string|moodle_url $linkbase
|
|
|
|
* @param int $d The number of the day.
|
|
|
|
* @param int $m The number of the month.
|
|
|
|
* @param int $y The number of the year.
|
|
|
|
* @param int $time the unixtime, used for multiple calendar support. The values $d,
|
|
|
|
* $m and $y are kept for backwards compatibility.
|
|
|
|
* @return moodle_url|null $linkbase
|
|
|
|
*/
|
|
|
|
function calendar_get_link_href($linkbase, $d, $m, $y, $time = 0) {
|
2017-03-30 12:18:32 +08:00
|
|
|
if (empty($linkbase)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!($linkbase instanceof \moodle_url)) {
|
|
|
|
$linkbase = new \moodle_url($linkbase);
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:27:35 +08:00
|
|
|
$linkbase->param('time', calendar_get_timestamp($d, $m, $y, $time));
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
return $linkbase;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build and return a previous month HTML link, with an arrow.
|
|
|
|
*
|
|
|
|
* @param string $text The text label.
|
|
|
|
* @param string|moodle_url $linkbase The URL stub.
|
|
|
|
* @param int $d The number of the date.
|
|
|
|
* @param int $m The number of the month.
|
|
|
|
* @param int $y year The number of the year.
|
|
|
|
* @param bool $accesshide Default visible, or hide from all except screenreaders.
|
|
|
|
* @param int $time the unixtime, used for multiple calendar support. The values $d,
|
|
|
|
* $m and $y are kept for backwards compatibility.
|
|
|
|
* @return string HTML string.
|
|
|
|
*/
|
|
|
|
function calendar_get_link_previous($text, $linkbase, $d, $m, $y, $accesshide = false, $time = 0) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$href = calendar_get_link_href(new \moodle_url($linkbase), $d, $m, $y, $time);
|
|
|
|
|
|
|
|
if (empty($href)) {
|
|
|
|
return $text;
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:30:03 +08:00
|
|
|
$attrs = [
|
|
|
|
'data-time' => calendar_get_timestamp($d, $m, $y, $time),
|
2017-08-24 08:20:35 +00:00
|
|
|
'data-drop-zone' => 'nav-link',
|
2017-07-19 09:30:03 +08:00
|
|
|
];
|
|
|
|
|
|
|
|
return link_arrow_left($text, $href->out(false), $accesshide, 'previous', $attrs);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build and return a next month HTML link, with an arrow.
|
|
|
|
*
|
|
|
|
* @param string $text The text label.
|
|
|
|
* @param string|moodle_url $linkbase The URL stub.
|
|
|
|
* @param int $d the number of the Day
|
|
|
|
* @param int $m The number of the month.
|
|
|
|
* @param int $y The number of the year.
|
|
|
|
* @param bool $accesshide Default visible, or hide from all except screenreaders.
|
|
|
|
* @param int $time the unixtime, used for multiple calendar support. The values $d,
|
|
|
|
* $m and $y are kept for backwards compatibility.
|
|
|
|
* @return string HTML string.
|
|
|
|
*/
|
|
|
|
function calendar_get_link_next($text, $linkbase, $d, $m, $y, $accesshide = false, $time = 0) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$href = calendar_get_link_href(new \moodle_url($linkbase), $d, $m, $y, $time);
|
|
|
|
|
|
|
|
if (empty($href)) {
|
|
|
|
return $text;
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:30:03 +08:00
|
|
|
$attrs = [
|
|
|
|
'data-time' => calendar_get_timestamp($d, $m, $y, $time),
|
2017-08-24 08:20:35 +00:00
|
|
|
'data-drop-zone' => 'nav-link',
|
2017-07-19 09:30:03 +08:00
|
|
|
];
|
|
|
|
|
|
|
|
return link_arrow_right($text, $href->out(false), $accesshide, 'next', $attrs);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the number of days in month.
|
|
|
|
*
|
|
|
|
* @param int $month the number of the month.
|
|
|
|
* @param int $year the number of the year
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
function calendar_days_in_month($month, $year) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$calendartype = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
return $calendartype->get_num_days_in_month($year, $month);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the next following month.
|
|
|
|
*
|
|
|
|
* @param int $month the number of the month.
|
|
|
|
* @param int $year the number of the year.
|
|
|
|
* @return array the following month
|
|
|
|
*/
|
|
|
|
function calendar_add_month($month, $year) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$calendartype = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
return $calendartype->get_next_month($year, $month);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the previous month.
|
|
|
|
*
|
|
|
|
* @param int $month the number of the month.
|
|
|
|
* @param int $year the number of the year.
|
|
|
|
* @return array previous month
|
|
|
|
*/
|
|
|
|
function calendar_sub_month($month, $year) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$calendartype = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
return $calendartype->get_prev_month($year, $month);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get per-day basis events
|
|
|
|
*
|
|
|
|
* @param array $events list of events
|
|
|
|
* @param int $month the number of the month
|
|
|
|
* @param int $year the number of the year
|
|
|
|
* @param array $eventsbyday event on specific day
|
|
|
|
* @param array $durationbyday duration of the event in days
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
* @param array $typesbyday event type (eg: site, course, user, or group)
|
2017-03-30 11:23:08 +08:00
|
|
|
* @param array $courses list of courses
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function calendar_events_by_day($events, $month, $year, &$eventsbyday, &$durationbyday, &$typesbyday, &$courses) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$calendartype = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
|
|
|
|
$eventsbyday = array();
|
|
|
|
$typesbyday = array();
|
|
|
|
$durationbyday = array();
|
|
|
|
|
|
|
|
if ($events === false) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($events as $event) {
|
|
|
|
$startdate = $calendartype->timestamp_to_date_array($event->timestart);
|
|
|
|
if ($event->timeduration) {
|
|
|
|
$enddate = $calendartype->timestamp_to_date_array($event->timestart + $event->timeduration - 1);
|
|
|
|
} else {
|
|
|
|
$enddate = $startdate;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simple arithmetic: $year * 13 + $month is a distinct integer for each distinct ($year, $month) pair.
|
|
|
|
if (!($startdate['year'] * 13 + $startdate['mon'] <= $year * 13 + $month) &&
|
|
|
|
($enddate['year'] * 13 + $enddate['mon'] >= $year * 13 + $month)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$eventdaystart = intval($startdate['mday']);
|
|
|
|
|
|
|
|
if ($startdate['mon'] == $month && $startdate['year'] == $year) {
|
|
|
|
// Give the event to its day.
|
|
|
|
$eventsbyday[$eventdaystart][] = $event->id;
|
|
|
|
|
|
|
|
// Mark the day as having such an event.
|
|
|
|
if ($event->courseid == SITEID && $event->groupid == 0) {
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
$typesbyday[$eventdaystart]['startsite'] = true;
|
|
|
|
// Set event class for site event.
|
|
|
|
$events[$event->id]->class = 'calendar_event_site';
|
2017-03-30 12:18:32 +08:00
|
|
|
} else if ($event->courseid != 0 && $event->courseid != SITEID && $event->groupid == 0) {
|
|
|
|
$typesbyday[$eventdaystart]['startcourse'] = true;
|
|
|
|
// Set event class for course event.
|
|
|
|
$events[$event->id]->class = 'calendar_event_course';
|
|
|
|
} else if ($event->groupid) {
|
|
|
|
$typesbyday[$eventdaystart]['startgroup'] = true;
|
|
|
|
// Set event class for group event.
|
|
|
|
$events[$event->id]->class = 'calendar_event_group';
|
|
|
|
} else if ($event->userid) {
|
|
|
|
$typesbyday[$eventdaystart]['startuser'] = true;
|
|
|
|
// Set event class for user event.
|
|
|
|
$events[$event->id]->class = 'calendar_event_user';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($event->timeduration == 0) {
|
|
|
|
// Proceed with the next.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The event starts on $month $year or before.
|
|
|
|
if ($startdate['mon'] == $month && $startdate['year'] == $year) {
|
|
|
|
$lowerbound = intval($startdate['mday']);
|
|
|
|
} else {
|
|
|
|
$lowerbound = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Also, it ends on $month $year or later.
|
|
|
|
if ($enddate['mon'] == $month && $enddate['year'] == $year) {
|
|
|
|
$upperbound = intval($enddate['mday']);
|
|
|
|
} else {
|
|
|
|
$upperbound = calendar_days_in_month($month, $year);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark all days between $lowerbound and $upperbound (inclusive) as duration.
|
|
|
|
for ($i = $lowerbound + 1; $i <= $upperbound; ++$i) {
|
|
|
|
$durationbyday[$i][] = $event->id;
|
|
|
|
if ($event->courseid == SITEID && $event->groupid == 0) {
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
$typesbyday[$i]['durationsite'] = true;
|
2017-03-30 12:18:32 +08:00
|
|
|
} else if ($event->courseid != 0 && $event->courseid != SITEID && $event->groupid == 0) {
|
|
|
|
$typesbyday[$i]['durationcourse'] = true;
|
|
|
|
} else if ($event->groupid) {
|
|
|
|
$typesbyday[$i]['durationgroup'] = true;
|
|
|
|
} else if ($event->userid) {
|
|
|
|
$typesbyday[$i]['durationuser'] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
return;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the courses to load events for.
|
|
|
|
*
|
|
|
|
* @param array $courseeventsfrom An array of courses to load calendar events for
|
|
|
|
* @param bool $ignorefilters specify the use of filters, false is set as default
|
2018-03-28 00:19:10 +11:00
|
|
|
* @param stdClass $user The user object. This defaults to the global $USER object.
|
2017-03-30 11:23:08 +08:00
|
|
|
* @return array An array of courses, groups, and user to load calendar events for based upon filters
|
|
|
|
*/
|
2018-03-28 00:19:10 +11:00
|
|
|
function calendar_set_filters(array $courseeventsfrom, $ignorefilters = false, stdClass $user = null) {
|
|
|
|
global $CFG, $USER;
|
|
|
|
|
|
|
|
if (is_null($user)) {
|
|
|
|
$user = $USER;
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
$courses = array();
|
2018-03-28 00:19:10 +11:00
|
|
|
$userid = false;
|
2017-03-30 12:18:32 +08:00
|
|
|
$group = false;
|
|
|
|
|
|
|
|
// Get the capabilities that allow seeing group events from all groups.
|
|
|
|
$allgroupscaps = array('moodle/site:accessallgroups', 'moodle/calendar:manageentries');
|
|
|
|
|
2018-03-28 00:19:10 +11:00
|
|
|
$isvaliduser = !empty($user->id);
|
2017-03-30 12:18:32 +08:00
|
|
|
|
2018-03-28 00:19:10 +11:00
|
|
|
if ($ignorefilters || calendar_show_event_type(CALENDAR_EVENT_COURSE, $user)) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$courses = array_keys($courseeventsfrom);
|
|
|
|
}
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
if ($ignorefilters || calendar_show_event_type(CALENDAR_EVENT_SITE, $user)) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$courses[] = SITEID;
|
|
|
|
}
|
|
|
|
$courses = array_unique($courses);
|
|
|
|
sort($courses);
|
|
|
|
|
|
|
|
if (!empty($courses) && in_array(SITEID, $courses)) {
|
|
|
|
// Sort courses for consistent colour highlighting.
|
|
|
|
// Effectively ignoring SITEID as setting as last course id.
|
|
|
|
$key = array_search(SITEID, $courses);
|
|
|
|
unset($courses[$key]);
|
|
|
|
$courses[] = SITEID;
|
|
|
|
}
|
|
|
|
|
2018-03-28 00:19:10 +11:00
|
|
|
if ($ignorefilters || ($isvaliduser && calendar_show_event_type(CALENDAR_EVENT_USER, $user))) {
|
|
|
|
$userid = $user->id;
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
|
2018-03-28 00:19:10 +11:00
|
|
|
if (!empty($courseeventsfrom) && (calendar_show_event_type(CALENDAR_EVENT_GROUP, $user) || $ignorefilters)) {
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
if (count($courseeventsfrom) == 1) {
|
|
|
|
$course = reset($courseeventsfrom);
|
|
|
|
if (has_any_capability($allgroupscaps, \context_course::instance($course->id))) {
|
|
|
|
$coursegroups = groups_get_all_groups($course->id, 0, 0, 'g.id');
|
|
|
|
$group = array_keys($coursegroups);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($group === false) {
|
|
|
|
if (!empty($CFG->calendar_adminseesall) && has_any_capability($allgroupscaps, \context_system::instance())) {
|
|
|
|
$group = true;
|
2018-03-28 00:19:10 +11:00
|
|
|
} else if ($isvaliduser) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$groupids = array();
|
|
|
|
foreach ($courseeventsfrom as $courseid => $course) {
|
|
|
|
// If the user is an editing teacher in there.
|
2018-03-28 00:19:10 +11:00
|
|
|
if (!empty($user->groupmember[$course->id])) {
|
2017-03-30 12:18:32 +08:00
|
|
|
// We've already cached the users groups for this course so we can just use that.
|
2018-03-28 00:19:10 +11:00
|
|
|
$groupids = array_merge($groupids, $user->groupmember[$course->id]);
|
2017-03-30 12:18:32 +08:00
|
|
|
} else if ($course->groupmode != NOGROUPS || !$course->groupmodeforce) {
|
|
|
|
// If this course has groups, show events from all of those related to the current user.
|
2018-03-28 00:19:10 +11:00
|
|
|
$coursegroups = groups_get_user_groups($course->id, $user->id);
|
2017-03-30 12:18:32 +08:00
|
|
|
$groupids = array_merge($groupids, $coursegroups['0']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!empty($groupids)) {
|
|
|
|
$group = $groupids;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (empty($courses)) {
|
|
|
|
$courses = false;
|
|
|
|
}
|
|
|
|
|
2018-03-28 00:19:10 +11:00
|
|
|
return array($courses, $group, $userid);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
2017-10-31 11:25:48 +08:00
|
|
|
/**
|
|
|
|
* Return the capability for viewing a calendar event.
|
|
|
|
*
|
|
|
|
* @param calendar_event $event event object
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
function calendar_view_event_allowed(calendar_event $event) {
|
|
|
|
global $USER;
|
|
|
|
|
|
|
|
// Anyone can see site events.
|
|
|
|
if ($event->courseid && $event->courseid == SITEID) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If a user can manage events at the site level they can see any event.
|
|
|
|
$sitecontext = \context_system::instance();
|
|
|
|
// If user has manageentries at site level, return true.
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $sitecontext)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($event->groupid)) {
|
|
|
|
// If it is a group event we need to be able to manage events in the course, or be in the group.
|
2017-11-06 12:21:27 +08:00
|
|
|
if (has_capability('moodle/calendar:manageentries', $event->context) ||
|
|
|
|
has_capability('moodle/calendar:managegroupentries', $event->context)) {
|
2017-10-31 11:25:48 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$mycourses = enrol_get_my_courses('id');
|
|
|
|
return isset($mycourses[$event->courseid]) && groups_is_member($event->groupid);
|
|
|
|
} else if ($event->modulename) {
|
|
|
|
// If this is a module event we need to be able to see the module.
|
2017-11-06 12:21:27 +08:00
|
|
|
$coursemodules = [];
|
|
|
|
$courseid = 0;
|
|
|
|
// Override events do not have the courseid set.
|
|
|
|
if ($event->courseid) {
|
|
|
|
$courseid = $event->courseid;
|
|
|
|
$coursemodules = get_fast_modinfo($event->courseid)->instances;
|
|
|
|
} else {
|
|
|
|
$cmraw = get_coursemodule_from_instance($event->modulename, $event->instance, 0, false, MUST_EXIST);
|
|
|
|
$courseid = $cmraw->course;
|
|
|
|
$coursemodules = get_fast_modinfo($cmraw->course)->instances;
|
|
|
|
}
|
2017-10-31 11:25:48 +08:00
|
|
|
$hasmodule = isset($coursemodules[$event->modulename]);
|
|
|
|
$hasinstance = isset($coursemodules[$event->modulename][$event->instance]);
|
|
|
|
|
|
|
|
// If modinfo doesn't know about the module, return false to be safe.
|
|
|
|
if (!$hasmodule || !$hasinstance) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Must be able to see the course and the module - MDL-59304.
|
|
|
|
$cm = $coursemodules[$event->modulename][$event->instance];
|
|
|
|
if (!$cm->uservisible) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$mycourses = enrol_get_my_courses('id');
|
2017-11-06 12:21:27 +08:00
|
|
|
return isset($mycourses[$courseid]);
|
2017-10-31 11:25:48 +08:00
|
|
|
} else if ($event->categoryid) {
|
|
|
|
// If this is a category we need to be able to see the category.
|
2018-06-16 11:57:27 +08:00
|
|
|
$cat = \core_course_category::get($event->categoryid, IGNORE_MISSING);
|
2017-10-31 11:25:48 +08:00
|
|
|
if (!$cat) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else if (!empty($event->courseid)) {
|
|
|
|
// If it is a course event we need to be able to manage events in the course, or be in the course.
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $event->context)) {
|
|
|
|
return true;
|
|
|
|
}
|
2019-09-17 13:58:56 +08:00
|
|
|
|
|
|
|
return can_access_course(get_course($event->courseid));
|
2017-10-31 11:25:48 +08:00
|
|
|
} else if ($event->userid) {
|
|
|
|
if ($event->userid != $USER->id) {
|
|
|
|
// No-one can ever see another users events.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
throw new moodle_exception('unknown event type');
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the capability for editing calendar event.
|
|
|
|
*
|
|
|
|
* @param calendar_event $event event object
|
2017-08-18 05:29:07 +00:00
|
|
|
* @param bool $manualedit is the event being edited manually by the user
|
2017-03-30 11:23:08 +08:00
|
|
|
* @return bool capability to edit event
|
|
|
|
*/
|
2017-08-18 05:29:07 +00:00
|
|
|
function calendar_edit_event_allowed($event, $manualedit = false) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $USER, $DB;
|
|
|
|
|
|
|
|
// Must be logged in.
|
|
|
|
if (!isloggedin()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Can not be using guest account.
|
|
|
|
if (isguestuser()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-08-18 05:29:07 +00:00
|
|
|
if ($manualedit && !empty($event->modulename)) {
|
2017-09-05 07:45:24 +00:00
|
|
|
$hascallback = component_callback_exists(
|
|
|
|
'mod_' . $event->modulename,
|
|
|
|
'core_calendar_event_timestart_updated'
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!$hascallback) {
|
|
|
|
// If the activity hasn't implemented the correct callback
|
|
|
|
// to handle changes to it's events then don't allow any
|
|
|
|
// manual changes to them.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$coursemodules = get_fast_modinfo($event->courseid)->instances;
|
|
|
|
$hasmodule = isset($coursemodules[$event->modulename]);
|
|
|
|
$hasinstance = isset($coursemodules[$event->modulename][$event->instance]);
|
|
|
|
|
|
|
|
// If modinfo doesn't know about the module, return false to be safe.
|
|
|
|
if (!$hasmodule || !$hasinstance) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$coursemodule = $coursemodules[$event->modulename][$event->instance];
|
|
|
|
$context = context_module::instance($coursemodule->id);
|
|
|
|
// This is the capability that allows a user to modify the activity
|
|
|
|
// settings. Since the activity generated this event we need to check
|
|
|
|
// that the current user has the same capability before allowing them
|
|
|
|
// to update the event because the changes to the event will be
|
|
|
|
// reflected within the activity.
|
|
|
|
return has_capability('moodle/course:manageactivities', $context);
|
2017-08-18 05:29:07 +00:00
|
|
|
}
|
|
|
|
|
2020-04-29 22:22:10 +02:00
|
|
|
if ($manualedit && !empty($event->component)) {
|
2020-04-30 18:55:36 +02:00
|
|
|
// TODO possibly we can later add a callback similar to core_calendar_event_timestart_updated in the modules.
|
2020-04-29 22:22:10 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-30 12:18:32 +08:00
|
|
|
// You cannot edit URL based calendar subscription events presently.
|
|
|
|
if (!empty($event->subscriptionid)) {
|
|
|
|
if (!empty($event->subscription->url)) {
|
|
|
|
// This event can be updated externally, so it cannot be edited.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$sitecontext = \context_system::instance();
|
|
|
|
|
|
|
|
// If user has manageentries at site level, return true.
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $sitecontext)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If groupid is set, it's definitely a group event.
|
|
|
|
if (!empty($event->groupid)) {
|
|
|
|
// Allow users to add/edit group events if -
|
|
|
|
// 1) They have manageentries for the course OR
|
|
|
|
// 2) They have managegroupentries AND are in the group.
|
|
|
|
$group = $DB->get_record('groups', array('id' => $event->groupid));
|
|
|
|
return $group && (
|
2017-10-05 10:14:35 +08:00
|
|
|
has_capability('moodle/calendar:manageentries', $event->context) ||
|
|
|
|
(has_capability('moodle/calendar:managegroupentries', $event->context)
|
2017-03-30 12:18:32 +08:00
|
|
|
&& groups_is_member($event->groupid)));
|
|
|
|
} else if (!empty($event->courseid)) {
|
2017-09-22 12:01:00 +08:00
|
|
|
// If groupid is not set, but course is set, it's definitely a course event.
|
2017-10-05 10:14:35 +08:00
|
|
|
return has_capability('moodle/calendar:manageentries', $event->context);
|
2017-09-22 12:01:00 +08:00
|
|
|
} else if (!empty($event->categoryid)) {
|
|
|
|
// If groupid is not set, but category is set, it's definitely a category event.
|
2017-10-05 10:14:35 +08:00
|
|
|
return has_capability('moodle/calendar:manageentries', $event->context);
|
2017-03-30 12:18:32 +08:00
|
|
|
} else if (!empty($event->userid) && $event->userid == $USER->id) {
|
|
|
|
// If course is not set, but userid id set, it's a user event.
|
2017-10-05 10:14:35 +08:00
|
|
|
return (has_capability('moodle/calendar:manageownentries', $event->context));
|
2017-03-30 12:18:32 +08:00
|
|
|
} else if (!empty($event->userid)) {
|
2017-10-05 10:14:35 +08:00
|
|
|
return (has_capability('moodle/calendar:manageentries', $event->context));
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
2017-08-09 14:47:35 +08:00
|
|
|
/**
|
|
|
|
* Return the capability for deleting a calendar event.
|
|
|
|
*
|
|
|
|
* @param calendar_event $event The event object
|
|
|
|
* @return bool Whether the user has permission to delete the event or not.
|
|
|
|
*/
|
|
|
|
function calendar_delete_event_allowed($event) {
|
2020-04-29 22:22:10 +02:00
|
|
|
// Only allow delete if you have capabilities and it is not an module or component event.
|
|
|
|
return (calendar_edit_event_allowed($event) && empty($event->modulename) && empty($event->component));
|
2017-08-09 14:47:35 +08:00
|
|
|
}
|
|
|
|
|
2017-03-30 11:23:08 +08:00
|
|
|
/**
|
|
|
|
* Returns the default courses to display on the calendar when there isn't a specific
|
|
|
|
* course to display.
|
|
|
|
*
|
2017-10-24 15:18:18 +08:00
|
|
|
* @param int $courseid (optional) If passed, an additional course can be returned for admins (the current course).
|
2017-10-26 14:21:29 +08:00
|
|
|
* @param string $fields Comma separated list of course fields to return.
|
2018-03-06 15:22:11 +11:00
|
|
|
* @param bool $canmanage If true, this will return the list of courses the user can create events in, rather
|
2017-10-26 14:39:35 +08:00
|
|
|
* than the list of courses they see events from (an admin can always add events in a course
|
|
|
|
* calendar, even if they are not enrolled in the course).
|
2018-03-06 15:22:11 +11:00
|
|
|
* @param int $userid (optional) The user which this function returns the default courses for.
|
|
|
|
* By default the current user.
|
2017-03-30 11:23:08 +08:00
|
|
|
* @return array $courses Array of courses to display
|
|
|
|
*/
|
2018-03-06 15:22:11 +11:00
|
|
|
function calendar_get_default_courses($courseid = null, $fields = '*', $canmanage = false, int $userid = null) {
|
|
|
|
global $CFG, $USER;
|
2017-03-30 12:18:32 +08:00
|
|
|
|
2018-03-06 15:22:11 +11:00
|
|
|
if (!$userid) {
|
|
|
|
if (!isloggedin()) {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
$userid = $USER->id;
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
|
2018-03-06 15:22:11 +11:00
|
|
|
if ((!empty($CFG->calendar_adminseesall) || $canmanage) &&
|
|
|
|
has_capability('moodle/calendar:manageentries', context_system::instance(), $userid)) {
|
2017-10-26 14:39:35 +08:00
|
|
|
|
2017-10-26 14:21:29 +08:00
|
|
|
// Add a c. prefix to every field as expected by get_courses function.
|
|
|
|
$fieldlist = explode(',', $fields);
|
|
|
|
|
|
|
|
$prefixedfields = array_map(function($value) {
|
2018-06-21 13:34:59 +08:00
|
|
|
return 'c.' . trim(strtolower($value));
|
2017-10-26 14:21:29 +08:00
|
|
|
}, $fieldlist);
|
2021-02-17 09:54:24 +00:00
|
|
|
|
2017-10-26 14:21:29 +08:00
|
|
|
$courses = get_courses('all', 'c.shortname', implode(',', $prefixedfields));
|
2017-10-24 15:18:18 +08:00
|
|
|
} else {
|
2018-03-06 15:22:11 +11:00
|
|
|
$courses = enrol_get_users_courses($userid, true, $fields);
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
|
2017-10-24 15:18:18 +08:00
|
|
|
if ($courseid && $courseid != SITEID) {
|
2018-03-06 15:22:11 +11:00
|
|
|
if (empty($courses[$courseid]) && has_capability('moodle/calendar:manageentries', context_system::instance(), $userid)) {
|
2017-10-24 15:18:18 +08:00
|
|
|
// Allow a site admin to see calendars from courses he is not enrolled in.
|
|
|
|
// This will come from $COURSE.
|
|
|
|
$courses[$courseid] = get_course($courseid);
|
|
|
|
}
|
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
return $courses;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get event format time.
|
|
|
|
*
|
|
|
|
* @param calendar_event $event event object
|
|
|
|
* @param int $now current time in gmt
|
|
|
|
* @param array $linkparams list of params for event link
|
|
|
|
* @param bool $usecommonwords the words as formatted date/time.
|
|
|
|
* @param int $showtime determine the show time GMT timestamp
|
|
|
|
* @return string $eventtime link/string for event time
|
|
|
|
*/
|
|
|
|
function calendar_format_event_time($event, $now, $linkparams = null, $usecommonwords = true, $showtime = 0) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$starttime = $event->timestart;
|
|
|
|
$endtime = $event->timestart + $event->timeduration;
|
|
|
|
|
|
|
|
if (empty($linkparams) || !is_array($linkparams)) {
|
|
|
|
$linkparams = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
$linkparams['view'] = 'day';
|
|
|
|
|
|
|
|
// OK, now to get a meaningful display.
|
|
|
|
// Check if there is a duration for this event.
|
|
|
|
if ($event->timeduration) {
|
|
|
|
// Get the midnight of the day the event will start.
|
|
|
|
$usermidnightstart = usergetmidnight($starttime);
|
|
|
|
// Get the midnight of the day the event will end.
|
|
|
|
$usermidnightend = usergetmidnight($endtime);
|
|
|
|
// Check if we will still be on the same day.
|
|
|
|
if ($usermidnightstart == $usermidnightend) {
|
|
|
|
// Check if we are running all day.
|
|
|
|
if ($event->timeduration == DAYSECS) {
|
|
|
|
$time = get_string('allday', 'calendar');
|
|
|
|
} else { // Specify the time we will be running this from.
|
|
|
|
$datestart = calendar_time_representation($starttime);
|
|
|
|
$dateend = calendar_time_representation($endtime);
|
|
|
|
$time = $datestart . ' <strong>»</strong> ' . $dateend;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set printable representation.
|
|
|
|
if (!$showtime) {
|
|
|
|
$day = calendar_day_representation($event->timestart, $now, $usecommonwords);
|
|
|
|
$url = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', $linkparams), 0, 0, 0, $endtime);
|
|
|
|
$eventtime = \html_writer::link($url, $day) . ', ' . $time;
|
|
|
|
} else {
|
|
|
|
$eventtime = $time;
|
|
|
|
}
|
|
|
|
} else { // It must spans two or more days.
|
|
|
|
$daystart = calendar_day_representation($event->timestart, $now, $usecommonwords) . ', ';
|
|
|
|
if ($showtime == $usermidnightstart) {
|
|
|
|
$daystart = '';
|
|
|
|
}
|
|
|
|
$timestart = calendar_time_representation($event->timestart);
|
|
|
|
$dayend = calendar_day_representation($event->timestart + $event->timeduration, $now, $usecommonwords) . ', ';
|
|
|
|
if ($showtime == $usermidnightend) {
|
|
|
|
$dayend = '';
|
|
|
|
}
|
|
|
|
$timeend = calendar_time_representation($event->timestart + $event->timeduration);
|
|
|
|
|
|
|
|
// Set printable representation.
|
|
|
|
if ($now >= $usermidnightstart && $now < strtotime('+1 day', $usermidnightstart)) {
|
|
|
|
$url = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', $linkparams), 0, 0, 0, $endtime);
|
|
|
|
$eventtime = $timestart . ' <strong>»</strong> ' . \html_writer::link($url, $dayend) . $timeend;
|
|
|
|
} else {
|
|
|
|
// The event is in the future, print start and end links.
|
|
|
|
$url = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', $linkparams), 0, 0, 0, $starttime);
|
|
|
|
$eventtime = \html_writer::link($url, $daystart) . $timestart . ' <strong>»</strong> ';
|
|
|
|
|
|
|
|
$url = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', $linkparams), 0, 0, 0, $endtime);
|
|
|
|
$eventtime .= \html_writer::link($url, $dayend) . $timeend;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else { // There is no time duration.
|
|
|
|
$time = calendar_time_representation($event->timestart);
|
|
|
|
// Set printable representation.
|
|
|
|
if (!$showtime) {
|
|
|
|
$day = calendar_day_representation($event->timestart, $now, $usecommonwords);
|
|
|
|
$url = calendar_get_link_href(new \moodle_url(CALENDAR_URL . 'view.php', $linkparams), 0, 0, 0, $starttime);
|
|
|
|
$eventtime = \html_writer::link($url, $day) . ', ' . trim($time);
|
|
|
|
} else {
|
|
|
|
$eventtime = $time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if It has expired.
|
|
|
|
if ($event->timestart + $event->timeduration < $now) {
|
|
|
|
$eventtime = '<span class="dimmed_text">' . str_replace(' href=', ' class="dimmed" href=', $eventtime) . '</span>';
|
|
|
|
}
|
|
|
|
|
|
|
|
return $eventtime;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks to see if the requested type of event should be shown for the given user.
|
|
|
|
*
|
|
|
|
* @param int $type The type to check the display for (default is to display all)
|
|
|
|
* @param stdClass|int|null $user The user to check for - by default the current user
|
|
|
|
* @return bool True if the tyep should be displayed false otherwise
|
|
|
|
*/
|
|
|
|
function calendar_show_event_type($type, $user = null) {
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
$default = CALENDAR_EVENT_SITE + CALENDAR_EVENT_COURSE + CALENDAR_EVENT_GROUP + CALENDAR_EVENT_USER;
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
if (get_user_preferences('calendar_persistflt', 0, $user) === 0) {
|
|
|
|
global $SESSION;
|
|
|
|
if (!isset($SESSION->calendarshoweventtype)) {
|
|
|
|
$SESSION->calendarshoweventtype = $default;
|
|
|
|
}
|
|
|
|
return $SESSION->calendarshoweventtype & $type;
|
|
|
|
} else {
|
|
|
|
return get_user_preferences('calendar_savedflt', $default, $user) & $type;
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the display of the event type given $display.
|
|
|
|
*
|
|
|
|
* If $display = true the event type will be shown.
|
|
|
|
* If $display = false the event type will NOT be shown.
|
|
|
|
* If $display = null the current value will be toggled and saved.
|
|
|
|
*
|
|
|
|
* @param int $type object of CALENDAR_EVENT_XXX
|
|
|
|
* @param bool $display option to display event type
|
|
|
|
* @param stdClass|int $user moodle user object or id, null means current user
|
|
|
|
*/
|
|
|
|
function calendar_set_event_type_display($type, $display = null, $user = null) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$persist = get_user_preferences('calendar_persistflt', 0, $user);
|
MDL-66612 calendar: Change global events to site events
AMOS BEGIN
CPY [category,moodle],[category,calendar]
CPY [site,notes],[eventtypesite,calendar]
CPY [site,moodle],[site,calendar]
MOV [typesite,moodle],[siteevent,calendar]
AMOS END
2019-09-06 08:46:28 +02:00
|
|
|
$default = CALENDAR_EVENT_SITE + CALENDAR_EVENT_COURSE + CALENDAR_EVENT_GROUP
|
2017-09-18 12:47:37 +08:00
|
|
|
+ CALENDAR_EVENT_USER + CALENDAR_EVENT_COURSECAT;
|
2017-03-30 12:18:32 +08:00
|
|
|
if ($persist === 0) {
|
|
|
|
global $SESSION;
|
|
|
|
if (!isset($SESSION->calendarshoweventtype)) {
|
|
|
|
$SESSION->calendarshoweventtype = $default;
|
|
|
|
}
|
|
|
|
$preference = $SESSION->calendarshoweventtype;
|
|
|
|
} else {
|
|
|
|
$preference = get_user_preferences('calendar_savedflt', $default, $user);
|
|
|
|
}
|
|
|
|
$current = $preference & $type;
|
|
|
|
if ($display === null) {
|
|
|
|
$display = !$current;
|
|
|
|
}
|
|
|
|
if ($display && !$current) {
|
|
|
|
$preference += $type;
|
|
|
|
} else if (!$display && $current) {
|
|
|
|
$preference -= $type;
|
|
|
|
}
|
|
|
|
if ($persist === 0) {
|
|
|
|
$SESSION->calendarshoweventtype = $preference;
|
|
|
|
} else {
|
|
|
|
if ($preference == $default) {
|
|
|
|
unset_user_preference('calendar_savedflt', $user);
|
|
|
|
} else {
|
|
|
|
set_user_preference('calendar_savedflt', $preference, $user);
|
|
|
|
}
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get calendar's allowed types.
|
|
|
|
*
|
|
|
|
* @param stdClass $allowed list of allowed edit for event type
|
|
|
|
* @param stdClass|int $course object of a course or course id
|
2017-07-24 08:01:14 +00:00
|
|
|
* @param array $groups array of groups for the given course
|
2017-09-18 12:47:37 +08:00
|
|
|
* @param stdClass|int $category object of a category
|
2017-03-30 11:23:08 +08:00
|
|
|
*/
|
2017-09-18 12:47:37 +08:00
|
|
|
function calendar_get_allowed_types(&$allowed, $course = null, $groups = null, $category = null) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $USER, $DB;
|
|
|
|
|
|
|
|
$allowed = new \stdClass();
|
|
|
|
$allowed->user = has_capability('moodle/calendar:manageownentries', \context_system::instance());
|
|
|
|
$allowed->groups = false;
|
|
|
|
$allowed->courses = false;
|
2017-09-18 12:47:37 +08:00
|
|
|
$allowed->categories = false;
|
2017-03-30 12:18:32 +08:00
|
|
|
$allowed->site = has_capability('moodle/calendar:manageentries', \context_course::instance(SITEID));
|
2017-07-24 08:01:14 +00:00
|
|
|
$getgroupsfunc = function($course, $context, $user) use ($groups) {
|
|
|
|
if ($course->groupmode != NOGROUPS || !$course->groupmodeforce) {
|
|
|
|
if (has_capability('moodle/site:accessallgroups', $context)) {
|
|
|
|
return is_null($groups) ? groups_get_all_groups($course->id) : $groups;
|
|
|
|
} else {
|
|
|
|
if (is_null($groups)) {
|
|
|
|
return groups_get_all_groups($course->id, $user->id);
|
|
|
|
} else {
|
|
|
|
return array_filter($groups, function($group) use ($user) {
|
|
|
|
return isset($group->members[$user->id]);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
if (!empty($course)) {
|
|
|
|
if (!is_object($course)) {
|
2017-11-17 11:39:27 +08:00
|
|
|
$course = $DB->get_record('course', array('id' => $course), 'id, groupmode, groupmodeforce', MUST_EXIST);
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
if ($course->id != SITEID) {
|
|
|
|
$coursecontext = \context_course::instance($course->id);
|
|
|
|
$allowed->user = has_capability('moodle/calendar:manageownentries', $coursecontext);
|
|
|
|
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $coursecontext)) {
|
|
|
|
$allowed->courses = array($course->id => 1);
|
2017-07-24 08:01:14 +00:00
|
|
|
$allowed->groups = $getgroupsfunc($course, $coursecontext, $USER);
|
2017-03-30 12:18:32 +08:00
|
|
|
} else if (has_capability('moodle/calendar:managegroupentries', $coursecontext)) {
|
2017-07-24 08:01:14 +00:00
|
|
|
$allowed->groups = $getgroupsfunc($course, $coursecontext, $USER);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-09-18 12:47:37 +08:00
|
|
|
|
|
|
|
if (!empty($category)) {
|
|
|
|
$catcontext = \context_coursecat::instance($category->id);
|
|
|
|
if (has_capability('moodle/category:manage', $catcontext)) {
|
|
|
|
$allowed->categories = [$category->id => 1];
|
|
|
|
}
|
|
|
|
}
|
2017-07-24 08:01:14 +00:00
|
|
|
}
|
|
|
|
|
2017-03-30 11:23:08 +08:00
|
|
|
/**
|
|
|
|
* See if user can add calendar entries at all used to print the "New Event" button.
|
|
|
|
*
|
|
|
|
* @param stdClass $course object of a course or course id
|
|
|
|
* @return bool has the capability to add at least one event type
|
|
|
|
*/
|
|
|
|
function calendar_user_can_add_event($course) {
|
2017-03-30 12:18:32 +08:00
|
|
|
if (!isloggedin() || isguestuser()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
calendar_get_allowed_types($allowed, $course);
|
|
|
|
|
2017-10-16 10:19:14 +08:00
|
|
|
return (bool)($allowed->user || $allowed->groups || $allowed->courses || $allowed->categories || $allowed->site);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check wether the current user is permitted to add events.
|
|
|
|
*
|
|
|
|
* @param stdClass $event object of event
|
|
|
|
* @return bool has the capability to add event
|
|
|
|
*/
|
|
|
|
function calendar_add_event_allowed($event) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $USER, $DB;
|
|
|
|
|
|
|
|
// Can not be using guest account.
|
|
|
|
if (!isloggedin() or isguestuser()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$sitecontext = \context_system::instance();
|
|
|
|
|
|
|
|
// If user has manageentries at site level, always return true.
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $sitecontext)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ($event->eventtype) {
|
2017-09-18 12:47:37 +08:00
|
|
|
case 'category':
|
|
|
|
return has_capability('moodle/category:manage', $event->context);
|
2017-03-30 12:18:32 +08:00
|
|
|
case 'course':
|
|
|
|
return has_capability('moodle/calendar:manageentries', $event->context);
|
|
|
|
case 'group':
|
|
|
|
// Allow users to add/edit group events if -
|
|
|
|
// 1) They have manageentries (= entries for whole course).
|
|
|
|
// 2) They have managegroupentries AND are in the group.
|
|
|
|
$group = $DB->get_record('groups', array('id' => $event->groupid));
|
|
|
|
return $group && (
|
|
|
|
has_capability('moodle/calendar:manageentries', $event->context) ||
|
|
|
|
(has_capability('moodle/calendar:managegroupentries', $event->context)
|
|
|
|
&& groups_is_member($event->groupid)));
|
|
|
|
case 'user':
|
|
|
|
if ($event->userid == $USER->id) {
|
|
|
|
return (has_capability('moodle/calendar:manageownentries', $event->context));
|
|
|
|
}
|
|
|
|
// There is intentionally no 'break'.
|
|
|
|
case 'site':
|
|
|
|
return has_capability('moodle/calendar:manageentries', $event->context);
|
|
|
|
default:
|
|
|
|
return has_capability('moodle/calendar:manageentries', $event->context);
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns option list for the poll interval setting.
|
|
|
|
*
|
|
|
|
* @return array An array of poll interval options. Interval => description.
|
|
|
|
*/
|
|
|
|
function calendar_get_pollinterval_choices() {
|
2017-03-30 12:18:32 +08:00
|
|
|
return array(
|
2021-08-01 21:18:51 +07:00
|
|
|
'0' => get_string('never', 'calendar'),
|
|
|
|
HOURSECS => get_string('hourly', 'calendar'),
|
|
|
|
DAYSECS => get_string('daily', 'calendar'),
|
|
|
|
WEEKSECS => get_string('weekly', 'calendar'),
|
|
|
|
'2628000' => get_string('monthly', 'calendar'),
|
|
|
|
YEARSECS => get_string('annually', 'calendar')
|
2017-03-30 12:18:32 +08:00
|
|
|
);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns option list of available options for the calendar event type, given the current user and course.
|
|
|
|
*
|
|
|
|
* @param int $courseid The id of the course
|
|
|
|
* @return array An array containing the event types the user can create.
|
|
|
|
*/
|
|
|
|
function calendar_get_eventtype_choices($courseid) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$choices = array();
|
|
|
|
$allowed = new \stdClass;
|
|
|
|
calendar_get_allowed_types($allowed, $courseid);
|
|
|
|
|
|
|
|
if ($allowed->user) {
|
|
|
|
$choices['user'] = get_string('userevents', 'calendar');
|
|
|
|
}
|
|
|
|
if ($allowed->site) {
|
|
|
|
$choices['site'] = get_string('siteevents', 'calendar');
|
|
|
|
}
|
|
|
|
if (!empty($allowed->courses)) {
|
|
|
|
$choices['course'] = get_string('courseevents', 'calendar');
|
|
|
|
}
|
2017-09-18 12:47:37 +08:00
|
|
|
if (!empty($allowed->categories)) {
|
|
|
|
$choices['category'] = get_string('categoryevents', 'calendar');
|
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
if (!empty($allowed->groups) and is_array($allowed->groups)) {
|
|
|
|
$choices['group'] = get_string('group');
|
|
|
|
}
|
|
|
|
|
|
|
|
return array($choices, $allowed->groups);
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add an iCalendar subscription to the database.
|
|
|
|
*
|
|
|
|
* @param stdClass $sub The subscription object (e.g. from the form)
|
|
|
|
* @return int The insert ID, if any.
|
|
|
|
*/
|
|
|
|
function calendar_add_subscription($sub) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $DB, $USER, $SITE;
|
|
|
|
|
2017-10-06 09:05:41 +08:00
|
|
|
// Undo the form definition work around to allow us to have two different
|
|
|
|
// course selectors present depending on which event type the user selects.
|
|
|
|
if (!empty($sub->groupcourseid)) {
|
|
|
|
$sub->courseid = $sub->groupcourseid;
|
|
|
|
unset($sub->groupcourseid);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Default course id if none is set.
|
|
|
|
if (empty($sub->courseid)) {
|
|
|
|
if ($sub->eventtype === 'site') {
|
|
|
|
$sub->courseid = SITEID;
|
|
|
|
} else {
|
|
|
|
$sub->courseid = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-30 12:18:32 +08:00
|
|
|
if ($sub->eventtype === 'site') {
|
|
|
|
$sub->courseid = $SITE->id;
|
|
|
|
} else if ($sub->eventtype === 'group' || $sub->eventtype === 'course') {
|
2017-10-06 09:05:41 +08:00
|
|
|
$sub->courseid = $sub->courseid;
|
2017-09-18 12:47:37 +08:00
|
|
|
} else if ($sub->eventtype === 'category') {
|
2017-10-06 09:05:41 +08:00
|
|
|
$sub->categoryid = $sub->categoryid;
|
2017-03-30 12:18:32 +08:00
|
|
|
} else {
|
|
|
|
// User events.
|
|
|
|
$sub->courseid = 0;
|
|
|
|
}
|
2021-03-23 19:26:50 +08:00
|
|
|
$sub->userid = $USER->id;
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
// File subscriptions never update.
|
|
|
|
if (empty($sub->url)) {
|
|
|
|
$sub->pollinterval = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($sub->name)) {
|
|
|
|
if (empty($sub->id)) {
|
|
|
|
$id = $DB->insert_record('event_subscriptions', $sub);
|
|
|
|
// We cannot cache the data here because $sub is not complete.
|
|
|
|
$sub->id = $id;
|
|
|
|
// Trigger event, calendar subscription added.
|
|
|
|
$eventparams = array('objectid' => $sub->id,
|
|
|
|
'context' => calendar_get_calendar_context($sub),
|
2017-10-06 09:05:41 +08:00
|
|
|
'other' => array(
|
|
|
|
'eventtype' => $sub->eventtype,
|
|
|
|
)
|
2017-03-30 12:18:32 +08:00
|
|
|
);
|
2017-10-06 09:05:41 +08:00
|
|
|
switch ($sub->eventtype) {
|
|
|
|
case 'category':
|
|
|
|
$eventparams['other']['categoryid'] = $sub->categoryid;
|
|
|
|
break;
|
|
|
|
case 'course':
|
|
|
|
$eventparams['other']['courseid'] = $sub->courseid;
|
|
|
|
break;
|
|
|
|
case 'group':
|
|
|
|
$eventparams['other']['courseid'] = $sub->courseid;
|
|
|
|
$eventparams['other']['groupid'] = $sub->groupid;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$eventparams['other']['courseid'] = $sub->courseid;
|
|
|
|
}
|
|
|
|
|
2017-03-30 12:18:32 +08:00
|
|
|
$event = \core\event\calendar_subscription_created::create($eventparams);
|
|
|
|
$event->trigger();
|
|
|
|
return $id;
|
|
|
|
} else {
|
|
|
|
// Why are we doing an update here?
|
|
|
|
calendar_update_subscription($sub);
|
|
|
|
return $sub->id;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
print_error('errorbadsubscription', 'importcalendar');
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add an iCalendar event to the Moodle calendar.
|
|
|
|
*
|
|
|
|
* @param stdClass $event The RFC-2445 iCalendar event
|
2017-10-06 09:05:41 +08:00
|
|
|
* @param int $unused Deprecated
|
2017-03-30 11:23:08 +08:00
|
|
|
* @param int $subscriptionid The iCalendar subscription ID
|
|
|
|
* @param string $timezone The X-WR-TIMEZONE iCalendar property if provided
|
|
|
|
* @throws dml_exception A DML specific exception is thrown for invalid subscriptionids.
|
|
|
|
* @return int Code: CALENDAR_IMPORT_EVENT_UPDATED = updated, CALENDAR_IMPORT_EVENT_INSERTED = inserted, 0 = error
|
|
|
|
*/
|
2021-02-15 15:17:38 +01:00
|
|
|
function calendar_add_icalendar_event($event, $unused, $subscriptionid, $timezone='UTC') {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $DB;
|
|
|
|
|
|
|
|
// Probably an unsupported X-MICROSOFT-CDO-BUSYSTATUS event.
|
|
|
|
if (empty($event->properties['SUMMARY'])) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
$name = $event->properties['SUMMARY'][0]->value;
|
|
|
|
$name = str_replace('\n', '<br />', $name);
|
|
|
|
$name = str_replace('\\', '', $name);
|
|
|
|
$name = preg_replace('/\s+/u', ' ', $name);
|
|
|
|
|
|
|
|
$eventrecord = new \stdClass;
|
|
|
|
$eventrecord->name = clean_param($name, PARAM_NOTAGS);
|
|
|
|
|
|
|
|
if (empty($event->properties['DESCRIPTION'][0]->value)) {
|
|
|
|
$description = '';
|
|
|
|
} else {
|
|
|
|
$description = $event->properties['DESCRIPTION'][0]->value;
|
|
|
|
$description = clean_param($description, PARAM_NOTAGS);
|
|
|
|
$description = str_replace('\n', '<br />', $description);
|
|
|
|
$description = str_replace('\\', '', $description);
|
|
|
|
$description = preg_replace('/\s+/u', ' ', $description);
|
|
|
|
}
|
|
|
|
$eventrecord->description = $description;
|
|
|
|
|
|
|
|
// Probably a repeating event with RRULE etc. TODO: skip for now.
|
|
|
|
if (empty($event->properties['DTSTART'][0]->value)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($event->properties['DTSTART'][0]->parameters['TZID'])) {
|
|
|
|
$tz = $event->properties['DTSTART'][0]->parameters['TZID'];
|
|
|
|
} else {
|
|
|
|
$tz = $timezone;
|
|
|
|
}
|
|
|
|
$tz = \core_date::normalise_timezone($tz);
|
|
|
|
$eventrecord->timestart = strtotime($event->properties['DTSTART'][0]->value . ' ' . $tz);
|
|
|
|
if (empty($event->properties['DTEND'])) {
|
|
|
|
$eventrecord->timeduration = 0; // No duration if no end time specified.
|
|
|
|
} else {
|
|
|
|
if (isset($event->properties['DTEND'][0]->parameters['TZID'])) {
|
|
|
|
$endtz = $event->properties['DTEND'][0]->parameters['TZID'];
|
|
|
|
} else {
|
|
|
|
$endtz = $timezone;
|
|
|
|
}
|
|
|
|
$endtz = \core_date::normalise_timezone($endtz);
|
|
|
|
$eventrecord->timeduration = strtotime($event->properties['DTEND'][0]->value . ' ' . $endtz) - $eventrecord->timestart;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check to see if it should be treated as an all day event.
|
|
|
|
if ($eventrecord->timeduration == DAYSECS) {
|
|
|
|
// Check to see if the event started at Midnight on the imported calendar.
|
|
|
|
date_default_timezone_set($timezone);
|
|
|
|
if (date('H:i:s', $eventrecord->timestart) === "00:00:00") {
|
2017-07-14 09:06:56 +05:30
|
|
|
// This event should be an all day event. This is not correct, we don't do anything differently for all day events.
|
|
|
|
// See MDL-56227.
|
2017-03-30 12:18:32 +08:00
|
|
|
$eventrecord->timeduration = 0;
|
|
|
|
}
|
|
|
|
\core_date::set_default_server_timezone();
|
|
|
|
}
|
|
|
|
|
2018-05-24 09:44:54 +08:00
|
|
|
$eventrecord->location = empty($event->properties['LOCATION'][0]->value) ? '' :
|
2019-04-05 07:17:51 +08:00
|
|
|
trim(str_replace('\\', '', $event->properties['LOCATION'][0]->value));
|
2017-03-30 12:18:32 +08:00
|
|
|
$eventrecord->uuid = $event->properties['UID'][0]->value;
|
|
|
|
$eventrecord->timemodified = time();
|
|
|
|
|
|
|
|
// Add the iCal subscription details if required.
|
|
|
|
// We should never do anything with an event without a subscription reference.
|
|
|
|
$sub = calendar_get_subscription($subscriptionid);
|
|
|
|
$eventrecord->subscriptionid = $subscriptionid;
|
2021-03-23 19:26:50 +08:00
|
|
|
$eventrecord->userid = $sub->userid;
|
2017-03-30 12:18:32 +08:00
|
|
|
$eventrecord->groupid = $sub->groupid;
|
|
|
|
$eventrecord->courseid = $sub->courseid;
|
2017-10-06 09:05:41 +08:00
|
|
|
$eventrecord->categoryid = $sub->categoryid;
|
2017-03-30 12:18:32 +08:00
|
|
|
$eventrecord->eventtype = $sub->eventtype;
|
|
|
|
|
|
|
|
if ($updaterecord = $DB->get_record('event', array('uuid' => $eventrecord->uuid,
|
|
|
|
'subscriptionid' => $eventrecord->subscriptionid))) {
|
2019-08-14 09:02:08 +08:00
|
|
|
|
|
|
|
// Compare iCal event data against the moodle event to see if something has changed.
|
|
|
|
$result = array_diff((array) $eventrecord, (array) $updaterecord);
|
|
|
|
|
|
|
|
// Unset timemodified field because it's always going to be different.
|
|
|
|
unset($result['timemodified']);
|
|
|
|
|
|
|
|
if (count($result)) {
|
|
|
|
$eventrecord->id = $updaterecord->id;
|
|
|
|
$return = CALENDAR_IMPORT_EVENT_UPDATED; // Update.
|
|
|
|
} else {
|
|
|
|
return CALENDAR_IMPORT_EVENT_SKIPPED;
|
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
} else {
|
|
|
|
$return = CALENDAR_IMPORT_EVENT_INSERTED; // Insert.
|
|
|
|
}
|
2019-08-14 09:02:08 +08:00
|
|
|
|
2017-03-30 12:18:32 +08:00
|
|
|
if ($createdevent = \calendar_event::create($eventrecord, false)) {
|
|
|
|
if (!empty($event->properties['RRULE'])) {
|
|
|
|
// Repeating events.
|
|
|
|
date_default_timezone_set($tz); // Change time zone to parse all events.
|
2017-05-04 08:58:11 +08:00
|
|
|
$rrule = new \core_calendar\rrule_manager($event->properties['RRULE'][0]->value);
|
2017-03-30 12:18:32 +08:00
|
|
|
$rrule->parse_rrule();
|
|
|
|
$rrule->create_events($createdevent);
|
|
|
|
\core_date::set_default_server_timezone(); // Change time zone back to what it was.
|
|
|
|
}
|
|
|
|
return $return;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete subscription and all related events.
|
|
|
|
*
|
|
|
|
* @param int|stdClass $subscription subscription or it's id, which needs to be deleted.
|
|
|
|
*/
|
|
|
|
function calendar_delete_subscription($subscription) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if (!is_object($subscription)) {
|
|
|
|
$subscription = $DB->get_record('event_subscriptions', array('id' => $subscription), '*', MUST_EXIST);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete subscription and related events.
|
|
|
|
$DB->delete_records('event', array('subscriptionid' => $subscription->id));
|
|
|
|
$DB->delete_records('event_subscriptions', array('id' => $subscription->id));
|
|
|
|
\cache_helper::invalidate_by_definition('core', 'calendar_subscriptions', array(), array($subscription->id));
|
|
|
|
|
|
|
|
// Trigger event, calendar subscription deleted.
|
|
|
|
$eventparams = array('objectid' => $subscription->id,
|
|
|
|
'context' => calendar_get_calendar_context($subscription),
|
2017-10-06 09:05:41 +08:00
|
|
|
'other' => array(
|
|
|
|
'eventtype' => $subscription->eventtype,
|
|
|
|
)
|
2017-03-30 12:18:32 +08:00
|
|
|
);
|
2017-10-06 09:05:41 +08:00
|
|
|
switch ($subscription->eventtype) {
|
|
|
|
case 'category':
|
|
|
|
$eventparams['other']['categoryid'] = $subscription->categoryid;
|
|
|
|
break;
|
|
|
|
case 'course':
|
|
|
|
$eventparams['other']['courseid'] = $subscription->courseid;
|
|
|
|
break;
|
|
|
|
case 'group':
|
|
|
|
$eventparams['other']['courseid'] = $subscription->courseid;
|
|
|
|
$eventparams['other']['groupid'] = $subscription->groupid;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$eventparams['other']['courseid'] = $subscription->courseid;
|
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
$event = \core\event\calendar_subscription_deleted::create($eventparams);
|
|
|
|
$event->trigger();
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* From a URL, fetch the calendar and return an iCalendar object.
|
|
|
|
*
|
|
|
|
* @param string $url The iCalendar URL
|
|
|
|
* @return iCalendar The iCalendar object
|
|
|
|
*/
|
|
|
|
function calendar_get_icalendar($url) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
require_once($CFG->libdir . '/filelib.php');
|
2021-08-01 21:18:51 +07:00
|
|
|
require_once($CFG->libdir . '/bennu/bennu.inc.php');
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
$curl = new \curl();
|
|
|
|
$curl->setopt(array('CURLOPT_FOLLOWLOCATION' => 1, 'CURLOPT_MAXREDIRS' => 5));
|
|
|
|
$calendar = $curl->get($url);
|
|
|
|
|
|
|
|
// Http code validation should actually be the job of curl class.
|
|
|
|
if (!$calendar || $curl->info['http_code'] != 200 || !empty($curl->errorno)) {
|
|
|
|
throw new \moodle_exception('errorinvalidicalurl', 'calendar');
|
|
|
|
}
|
|
|
|
|
|
|
|
$ical = new \iCalendar();
|
|
|
|
$ical->unserialize($calendar);
|
|
|
|
|
|
|
|
return $ical;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Import events from an iCalendar object into a course calendar.
|
|
|
|
*
|
|
|
|
* @param iCalendar $ical The iCalendar object.
|
2021-08-01 21:18:51 +07:00
|
|
|
* @param int|null $subscriptionid The subscription ID.
|
|
|
|
* @return array A log of the import progress, including errors.
|
2017-03-30 11:23:08 +08:00
|
|
|
*/
|
2021-08-01 21:18:51 +07:00
|
|
|
function calendar_import_events_from_ical(iCalendar $ical, int $subscriptionid = null): array {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $DB;
|
|
|
|
|
2021-08-01 21:18:51 +07:00
|
|
|
$errors = [];
|
2017-03-30 12:18:32 +08:00
|
|
|
$eventcount = 0;
|
|
|
|
$updatecount = 0;
|
2019-08-14 09:02:08 +08:00
|
|
|
$skippedcount = 0;
|
2021-08-01 21:18:51 +07:00
|
|
|
$deletedcount = 0;
|
2017-03-30 12:18:32 +08:00
|
|
|
|
|
|
|
// Large calendars take a while...
|
|
|
|
if (!CLI_SCRIPT) {
|
|
|
|
\core_php_time_limit::raise(300);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Grab the timezone from the iCalendar file to be used later.
|
|
|
|
if (isset($ical->properties['X-WR-TIMEZONE'][0]->value)) {
|
|
|
|
$timezone = $ical->properties['X-WR-TIMEZONE'][0]->value;
|
|
|
|
} else {
|
|
|
|
$timezone = 'UTC';
|
|
|
|
}
|
|
|
|
|
2019-08-14 09:02:08 +08:00
|
|
|
$icaluuids = [];
|
2017-03-30 12:18:32 +08:00
|
|
|
foreach ($ical->components['VEVENT'] as $event) {
|
2019-08-14 09:02:08 +08:00
|
|
|
$icaluuids[] = $event->properties['UID'][0]->value;
|
2017-10-06 09:05:41 +08:00
|
|
|
$res = calendar_add_icalendar_event($event, null, $subscriptionid, $timezone);
|
2017-03-30 12:18:32 +08:00
|
|
|
switch ($res) {
|
|
|
|
case CALENDAR_IMPORT_EVENT_UPDATED:
|
|
|
|
$updatecount++;
|
|
|
|
break;
|
|
|
|
case CALENDAR_IMPORT_EVENT_INSERTED:
|
|
|
|
$eventcount++;
|
|
|
|
break;
|
2019-08-14 09:02:08 +08:00
|
|
|
case CALENDAR_IMPORT_EVENT_SKIPPED:
|
|
|
|
$skippedcount++;
|
|
|
|
break;
|
2017-03-30 12:18:32 +08:00
|
|
|
case 0:
|
|
|
|
if (empty($event->properties['SUMMARY'])) {
|
2021-08-01 21:18:51 +07:00
|
|
|
$errors[] = '(' . get_string('notitle', 'calendar') . ')';
|
2017-03-30 12:18:32 +08:00
|
|
|
} else {
|
2021-08-01 21:18:51 +07:00
|
|
|
$errors[] = $event->properties['SUMMARY'][0]->value;
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-16 08:36:26 +08:00
|
|
|
$existing = $DB->get_field('event_subscriptions', 'lastupdated', ['id' => $subscriptionid]);
|
|
|
|
if (!empty($existing)) {
|
2019-08-14 09:02:08 +08:00
|
|
|
$eventsuuids = $DB->get_records_menu('event', ['subscriptionid' => $subscriptionid], '', 'id, uuid');
|
|
|
|
|
|
|
|
$icaleventscount = count($icaluuids);
|
|
|
|
$tobedeleted = [];
|
|
|
|
if (count($eventsuuids) > $icaleventscount) {
|
|
|
|
foreach ($eventsuuids as $eventid => $eventuuid) {
|
|
|
|
if (!in_array($eventuuid, $icaluuids)) {
|
|
|
|
$tobedeleted[] = $eventid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!empty($tobedeleted)) {
|
|
|
|
$DB->delete_records_list('event', 'id', $tobedeleted);
|
2021-08-01 21:18:51 +07:00
|
|
|
$deletedcount = count($tobedeleted);
|
2019-08-14 09:02:08 +08:00
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-01 21:18:51 +07:00
|
|
|
$result = [
|
|
|
|
'eventsimported' => $eventcount,
|
|
|
|
'eventsskipped' => $skippedcount,
|
|
|
|
'eventsupdated' => $updatecount,
|
|
|
|
'eventsdeleted' => $deletedcount,
|
|
|
|
'haserror' => !empty($errors),
|
|
|
|
'errors' => $errors,
|
|
|
|
];
|
|
|
|
|
|
|
|
return $result;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch a calendar subscription and update the events in the calendar.
|
|
|
|
*
|
|
|
|
* @param int $subscriptionid The course ID for the calendar.
|
|
|
|
* @return string A log of the import progress, including errors.
|
|
|
|
*/
|
|
|
|
function calendar_update_subscription_events($subscriptionid) {
|
2017-03-30 12:18:32 +08:00
|
|
|
$sub = calendar_get_subscription($subscriptionid);
|
|
|
|
|
|
|
|
// Don't update a file subscription.
|
|
|
|
if (empty($sub->url)) {
|
|
|
|
return 'File subscription not updated.';
|
|
|
|
}
|
|
|
|
|
|
|
|
$ical = calendar_get_icalendar($sub->url);
|
2021-08-01 21:18:51 +07:00
|
|
|
$return = calendar_import_events_from_ical($ical, $subscriptionid);
|
2017-03-30 12:18:32 +08:00
|
|
|
$sub->lastupdated = time();
|
|
|
|
|
|
|
|
calendar_update_subscription($sub);
|
|
|
|
|
|
|
|
return $return;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update a calendar subscription. Also updates the associated cache.
|
|
|
|
*
|
|
|
|
* @param stdClass|array $subscription Subscription record.
|
|
|
|
* @throws coding_exception If something goes wrong
|
|
|
|
* @since Moodle 2.5
|
|
|
|
*/
|
|
|
|
function calendar_update_subscription($subscription) {
|
2017-03-30 12:18:32 +08:00
|
|
|
global $DB;
|
|
|
|
|
|
|
|
if (is_array($subscription)) {
|
|
|
|
$subscription = (object)$subscription;
|
|
|
|
}
|
|
|
|
if (empty($subscription->id) || !$DB->record_exists('event_subscriptions', array('id' => $subscription->id))) {
|
|
|
|
throw new \coding_exception('Cannot update a subscription without a valid id');
|
|
|
|
}
|
|
|
|
|
|
|
|
$DB->update_record('event_subscriptions', $subscription);
|
|
|
|
|
|
|
|
// Update cache.
|
|
|
|
$cache = \cache::make('core', 'calendar_subscriptions');
|
|
|
|
$cache->set($subscription->id, $subscription);
|
|
|
|
|
|
|
|
// Trigger event, calendar subscription updated.
|
|
|
|
$eventparams = array('userid' => $subscription->userid,
|
|
|
|
'objectid' => $subscription->id,
|
|
|
|
'context' => calendar_get_calendar_context($subscription),
|
2017-10-06 09:05:41 +08:00
|
|
|
'other' => array(
|
|
|
|
'eventtype' => $subscription->eventtype,
|
|
|
|
)
|
2017-03-30 12:18:32 +08:00
|
|
|
);
|
2017-10-06 09:05:41 +08:00
|
|
|
switch ($subscription->eventtype) {
|
|
|
|
case 'category':
|
|
|
|
$eventparams['other']['categoryid'] = $subscription->categoryid;
|
|
|
|
break;
|
|
|
|
case 'course':
|
|
|
|
$eventparams['other']['courseid'] = $subscription->courseid;
|
|
|
|
break;
|
|
|
|
case 'group':
|
|
|
|
$eventparams['other']['courseid'] = $subscription->courseid;
|
|
|
|
$eventparams['other']['groupid'] = $subscription->groupid;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$eventparams['other']['courseid'] = $subscription->courseid;
|
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
$event = \core\event\calendar_subscription_updated::create($eventparams);
|
|
|
|
$event->trigger();
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks to see if the user can edit a given subscription feed.
|
|
|
|
*
|
|
|
|
* @param mixed $subscriptionorid Subscription object or id
|
|
|
|
* @return bool true if current user can edit the subscription else false
|
|
|
|
*/
|
|
|
|
function calendar_can_edit_subscription($subscriptionorid) {
|
2021-07-06 16:55:21 +10:00
|
|
|
global $USER;
|
2017-03-30 12:18:32 +08:00
|
|
|
if (is_array($subscriptionorid)) {
|
|
|
|
$subscription = (object)$subscriptionorid;
|
|
|
|
} else if (is_object($subscriptionorid)) {
|
|
|
|
$subscription = $subscriptionorid;
|
|
|
|
} else {
|
|
|
|
$subscription = calendar_get_subscription($subscriptionorid);
|
|
|
|
}
|
|
|
|
|
|
|
|
$allowed = new \stdClass;
|
|
|
|
$courseid = $subscription->courseid;
|
2017-10-06 09:05:41 +08:00
|
|
|
$categoryid = $subscription->categoryid;
|
2017-03-30 12:18:32 +08:00
|
|
|
$groupid = $subscription->groupid;
|
2017-10-06 09:05:41 +08:00
|
|
|
$category = null;
|
2017-03-30 12:18:32 +08:00
|
|
|
|
2017-10-06 09:05:41 +08:00
|
|
|
if (!empty($categoryid)) {
|
2018-06-16 11:57:27 +08:00
|
|
|
$category = \core_course_category::get($categoryid);
|
2017-10-06 09:05:41 +08:00
|
|
|
}
|
|
|
|
calendar_get_allowed_types($allowed, $courseid, null, $category);
|
2017-03-30 12:18:32 +08:00
|
|
|
switch ($subscription->eventtype) {
|
|
|
|
case 'user':
|
2021-07-06 16:55:21 +10:00
|
|
|
return ($USER->id == $subscription->userid && $allowed->user);
|
2017-03-30 12:18:32 +08:00
|
|
|
case 'course':
|
|
|
|
if (isset($allowed->courses[$courseid])) {
|
|
|
|
return $allowed->courses[$courseid];
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
2017-10-06 09:05:41 +08:00
|
|
|
case 'category':
|
|
|
|
if (isset($allowed->categories[$categoryid])) {
|
|
|
|
return $allowed->categories[$categoryid];
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-30 12:18:32 +08:00
|
|
|
case 'site':
|
|
|
|
return $allowed->site;
|
|
|
|
case 'group':
|
|
|
|
if (isset($allowed->groups[$groupid])) {
|
|
|
|
return $allowed->groups[$groupid];
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper function to determine the context of a calendar subscription.
|
|
|
|
* Subscriptions can be created in two contexts COURSE, or USER.
|
|
|
|
*
|
|
|
|
* @param stdClass $subscription
|
|
|
|
* @return context instance
|
|
|
|
*/
|
|
|
|
function calendar_get_calendar_context($subscription) {
|
2017-03-30 12:18:32 +08:00
|
|
|
// Determine context based on calendar type.
|
|
|
|
if ($subscription->eventtype === 'site') {
|
|
|
|
$context = \context_course::instance(SITEID);
|
|
|
|
} else if ($subscription->eventtype === 'group' || $subscription->eventtype === 'course') {
|
|
|
|
$context = \context_course::instance($subscription->courseid);
|
|
|
|
} else {
|
|
|
|
$context = \context_user::instance($subscription->userid);
|
|
|
|
}
|
|
|
|
return $context;
|
2017-03-30 11:23:08 +08:00
|
|
|
}
|
|
|
|
|
2017-03-05 13:20:03 +08:00
|
|
|
/**
|
2020-09-16 08:45:46 +02:00
|
|
|
* Implements callback user_preferences, lists preferences that users are allowed to update directly
|
2017-03-05 13:20:03 +08:00
|
|
|
*
|
|
|
|
* Used in {@see core_user::fill_preferences_cache()}, see also {@see useredit_update_user_preference()}
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function core_calendar_user_preferences() {
|
|
|
|
$preferences = [];
|
|
|
|
$preferences['calendar_timeformat'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED, 'default' => '0',
|
|
|
|
'choices' => array('0', CALENDAR_TF_12, CALENDAR_TF_24)
|
|
|
|
);
|
|
|
|
$preferences['calendar_startwday'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 0,
|
|
|
|
'choices' => array(0, 1, 2, 3, 4, 5, 6));
|
|
|
|
$preferences['calendar_maxevents'] = array('type' => PARAM_INT, 'choices' => range(1, 20));
|
|
|
|
$preferences['calendar_lookahead'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 365,
|
|
|
|
'choices' => array(365, 270, 180, 150, 120, 90, 60, 30, 21, 14, 7, 6, 5, 4, 3, 2, 1));
|
|
|
|
$preferences['calendar_persistflt'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 0,
|
|
|
|
'choices' => array(0, 1));
|
|
|
|
return $preferences;
|
2017-01-19 16:20:27 +08:00
|
|
|
}
|
2017-04-12 14:01:10 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get legacy calendar events
|
|
|
|
*
|
|
|
|
* @param int $tstart Start time of time range for events
|
|
|
|
* @param int $tend End time of time range for events
|
|
|
|
* @param array|int|boolean $users array of users, user id or boolean for all/no user events
|
|
|
|
* @param array|int|boolean $groups array of groups, group id or boolean for all/no group events
|
|
|
|
* @param array|int|boolean $courses array of courses, course id or boolean for all/no course events
|
|
|
|
* @param boolean $withduration whether only events starting within time range selected
|
|
|
|
* or events in progress/already started selected as well
|
|
|
|
* @param boolean $ignorehidden whether to select only visible events or all events
|
2017-11-07 08:39:03 +08:00
|
|
|
* @param array $categories array of category ids and/or objects.
|
2019-05-21 09:59:39 +08:00
|
|
|
* @param int $limitnum Number of events to fetch or zero to fetch all.
|
|
|
|
*
|
2017-04-12 14:01:10 +08:00
|
|
|
* @return array $events of selected events or an empty array if there aren't any (or there was an error)
|
|
|
|
*/
|
2017-09-19 11:39:39 +08:00
|
|
|
function calendar_get_legacy_events($tstart, $tend, $users, $groups, $courses,
|
2019-05-21 09:59:39 +08:00
|
|
|
$withduration = true, $ignorehidden = true, $categories = [], $limitnum = 0) {
|
2017-04-12 14:01:10 +08:00
|
|
|
// Normalise the users, groups and courses parameters so that they are compliant with \core_calendar\local\api::get_events().
|
|
|
|
// Existing functions that were using the old calendar_get_events() were passing a mixture of array, int, boolean for these
|
|
|
|
// parameters, but with the new API method, only null and arrays are accepted.
|
2017-09-19 11:39:39 +08:00
|
|
|
list($userparam, $groupparam, $courseparam, $categoryparam) = array_map(function($param) {
|
2017-04-12 14:01:10 +08:00
|
|
|
// If parameter is true, return null.
|
|
|
|
if ($param === true) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If parameter is false, return an empty array.
|
|
|
|
if ($param === false) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the parameter is a scalar value, enclose it in an array.
|
|
|
|
if (!is_array($param)) {
|
|
|
|
return [$param];
|
|
|
|
}
|
|
|
|
|
|
|
|
// No normalisation required.
|
|
|
|
return $param;
|
2017-09-19 11:39:39 +08:00
|
|
|
}, [$users, $groups, $courses, $categories]);
|
2017-04-12 14:01:10 +08:00
|
|
|
|
2018-04-05 00:37:56 +10:00
|
|
|
// If a single user is provided, we can use that for capability checks.
|
|
|
|
// Otherwise current logged in user is used - See MDL-58768.
|
|
|
|
if (is_array($userparam) && count($userparam) == 1) {
|
|
|
|
\core_calendar\local\event\container::set_requesting_user($userparam[0]);
|
|
|
|
}
|
2017-04-12 14:01:10 +08:00
|
|
|
$mapper = \core_calendar\local\event\container::get_event_mapper();
|
|
|
|
$events = \core_calendar\local\api::get_events(
|
|
|
|
$tstart,
|
|
|
|
$tend,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
2019-05-21 09:59:39 +08:00
|
|
|
$limitnum,
|
2017-04-12 14:01:10 +08:00
|
|
|
null,
|
|
|
|
$userparam,
|
|
|
|
$groupparam,
|
|
|
|
$courseparam,
|
2017-09-19 11:39:39 +08:00
|
|
|
$categoryparam,
|
2017-04-12 14:01:10 +08:00
|
|
|
$withduration,
|
|
|
|
$ignorehidden
|
|
|
|
);
|
|
|
|
|
|
|
|
return array_reduce($events, function($carry, $event) use ($mapper) {
|
|
|
|
return $carry + [$event->get_id() => $mapper->from_event_to_stdclass($event)];
|
|
|
|
}, []);
|
|
|
|
}
|
2017-07-24 08:01:14 +00:00
|
|
|
|
2017-07-19 09:34:34 +08:00
|
|
|
|
|
|
|
/**
|
2017-08-08 13:11:42 +08:00
|
|
|
* Get the calendar view output.
|
2017-07-19 09:34:34 +08:00
|
|
|
*
|
|
|
|
* @param \calendar_information $calendar The calendar being represented
|
2017-09-09 17:16:32 +08:00
|
|
|
* @param string $view The type of calendar to have displayed
|
|
|
|
* @param bool $includenavigation Whether to include navigation
|
2017-11-10 03:41:06 +00:00
|
|
|
* @param bool $skipevents Whether to load the events or not
|
2019-04-05 16:40:50 +02:00
|
|
|
* @param int $lookahead Overwrites site and users's lookahead setting.
|
2017-07-19 09:34:34 +08:00
|
|
|
* @return array[array, string]
|
|
|
|
*/
|
2019-04-05 16:40:50 +02:00
|
|
|
function calendar_get_view(\calendar_information $calendar, $view, $includenavigation = true, bool $skipevents = false,
|
|
|
|
?int $lookahead = null) {
|
2017-08-08 08:11:31 +08:00
|
|
|
global $PAGE, $CFG;
|
2017-07-19 09:34:34 +08:00
|
|
|
|
|
|
|
$renderer = $PAGE->get_renderer('core_calendar');
|
|
|
|
$type = \core_calendar\type_factory::get_calendar_instance();
|
|
|
|
|
|
|
|
// Calculate the bounds of the month.
|
2017-10-17 09:19:14 +08:00
|
|
|
$calendardate = $type->timestamp_to_date_array($calendar->time);
|
|
|
|
|
|
|
|
$date = new \DateTime('now', core_date::get_user_timezone_object(99));
|
2019-05-21 10:46:47 +08:00
|
|
|
$eventlimit = 0;
|
2017-07-19 09:34:34 +08:00
|
|
|
|
|
|
|
if ($view === 'day') {
|
2017-10-17 09:19:14 +08:00
|
|
|
$tstart = $type->convert_to_timestamp($calendardate['year'], $calendardate['mon'], $calendardate['mday']);
|
|
|
|
$date->setTimestamp($tstart);
|
|
|
|
$date->modify('+1 day');
|
2017-10-05 15:18:19 +08:00
|
|
|
} else if ($view === 'upcoming' || $view === 'upcoming_mini') {
|
2017-10-17 09:19:14 +08:00
|
|
|
// Number of days in the future that will be used to fetch events.
|
2019-04-05 16:40:50 +02:00
|
|
|
if (!$lookahead) {
|
|
|
|
if (isset($CFG->calendar_lookahead)) {
|
|
|
|
$defaultlookahead = intval($CFG->calendar_lookahead);
|
|
|
|
} else {
|
|
|
|
$defaultlookahead = CALENDAR_DEFAULT_UPCOMING_LOOKAHEAD;
|
|
|
|
}
|
|
|
|
$lookahead = get_user_preferences('calendar_lookahead', $defaultlookahead);
|
2017-08-08 08:11:31 +08:00
|
|
|
}
|
2017-10-17 09:19:14 +08:00
|
|
|
|
|
|
|
// Maximum number of events to be displayed on upcoming view.
|
2017-10-11 11:42:41 +08:00
|
|
|
$defaultmaxevents = CALENDAR_DEFAULT_UPCOMING_MAXEVENTS;
|
|
|
|
if (isset($CFG->calendar_maxevents)) {
|
|
|
|
$defaultmaxevents = intval($CFG->calendar_maxevents);
|
|
|
|
}
|
2017-10-24 13:31:02 +08:00
|
|
|
$eventlimit = get_user_preferences('calendar_maxevents', $defaultmaxevents);
|
2017-10-17 09:19:14 +08:00
|
|
|
|
|
|
|
$tstart = $type->convert_to_timestamp($calendardate['year'], $calendardate['mon'], $calendardate['mday'],
|
|
|
|
$calendardate['hours']);
|
|
|
|
$date->setTimestamp($tstart);
|
|
|
|
$date->modify('+' . $lookahead . ' days');
|
2017-07-19 09:34:34 +08:00
|
|
|
} else {
|
2017-10-17 09:19:14 +08:00
|
|
|
$tstart = $type->convert_to_timestamp($calendardate['year'], $calendardate['mon'], 1);
|
|
|
|
$monthdays = $type->get_num_days_in_month($calendardate['year'], $calendardate['mon']);
|
|
|
|
$date->setTimestamp($tstart);
|
|
|
|
$date->modify('+' . $monthdays . ' days');
|
|
|
|
|
2017-09-10 13:21:33 +08:00
|
|
|
if ($view === 'mini' || $view === 'minithree') {
|
2017-09-01 14:58:14 +08:00
|
|
|
$template = 'core_calendar/calendar_mini';
|
|
|
|
} else {
|
|
|
|
$template = 'core_calendar/calendar_month';
|
|
|
|
}
|
2017-07-19 09:34:34 +08:00
|
|
|
}
|
|
|
|
|
2017-10-17 09:19:14 +08:00
|
|
|
// We need to extract 1 second to ensure that we don't get into the next day.
|
|
|
|
$date->modify('-1 second');
|
|
|
|
$tend = $date->getTimestamp();
|
|
|
|
|
2017-09-19 11:39:39 +08:00
|
|
|
list($userparam, $groupparam, $courseparam, $categoryparam) = array_map(function($param) {
|
2017-07-19 09:34:34 +08:00
|
|
|
// If parameter is true, return null.
|
|
|
|
if ($param === true) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If parameter is false, return an empty array.
|
|
|
|
if ($param === false) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the parameter is a scalar value, enclose it in an array.
|
|
|
|
if (!is_array($param)) {
|
|
|
|
return [$param];
|
|
|
|
}
|
|
|
|
|
|
|
|
// No normalisation required.
|
|
|
|
return $param;
|
2017-09-19 11:39:39 +08:00
|
|
|
}, [$calendar->users, $calendar->groups, $calendar->courses, $calendar->categories]);
|
2017-07-19 09:34:34 +08:00
|
|
|
|
2017-11-10 03:41:06 +00:00
|
|
|
if ($skipevents) {
|
|
|
|
$events = [];
|
|
|
|
} else {
|
|
|
|
$events = \core_calendar\local\api::get_events(
|
|
|
|
$tstart,
|
|
|
|
$tend,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
$eventlimit,
|
|
|
|
null,
|
|
|
|
$userparam,
|
|
|
|
$groupparam,
|
|
|
|
$courseparam,
|
|
|
|
$categoryparam,
|
|
|
|
true,
|
|
|
|
true,
|
|
|
|
function ($event) {
|
|
|
|
if ($proxy = $event->get_course_module()) {
|
|
|
|
$cminfo = $proxy->get_proxied_instance();
|
|
|
|
return $cminfo->uservisible;
|
|
|
|
}
|
2017-09-19 11:15:13 +08:00
|
|
|
|
2017-11-10 03:41:06 +00:00
|
|
|
if ($proxy = $event->get_category()) {
|
|
|
|
$category = $proxy->get_proxied_instance();
|
2017-07-19 09:34:34 +08:00
|
|
|
|
2017-11-10 03:41:06 +00:00
|
|
|
return $category->is_uservisible();
|
|
|
|
}
|
2017-07-19 09:34:34 +08:00
|
|
|
|
2017-11-10 03:41:06 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2017-07-19 09:34:34 +08:00
|
|
|
|
|
|
|
$related = [
|
|
|
|
'events' => $events,
|
|
|
|
'cache' => new \core_calendar\external\events_related_objects_cache($events),
|
2017-09-18 10:53:26 +08:00
|
|
|
'type' => $type,
|
2017-07-19 09:34:34 +08:00
|
|
|
];
|
|
|
|
|
2017-09-18 10:53:26 +08:00
|
|
|
$data = [];
|
2021-07-21 23:49:23 +08:00
|
|
|
$calendar->set_viewmode($view);
|
2021-07-01 10:56:52 +08:00
|
|
|
if ($view == "month" || $view == "monthblock" || $view == "mini" || $view == "minithree" ) {
|
2017-09-18 10:53:26 +08:00
|
|
|
$month = new \core_calendar\external\month_exporter($calendar, $type, $related);
|
2017-09-20 13:25:19 +08:00
|
|
|
$month->set_includenavigation($includenavigation);
|
2017-11-10 03:41:06 +00:00
|
|
|
$month->set_initialeventsloaded(!$skipevents);
|
2021-07-01 10:56:52 +08:00
|
|
|
$month->set_showcoursefilter(($view == "month" || $view == "monthblock"));
|
2017-09-18 10:53:26 +08:00
|
|
|
$data = $month->export($renderer);
|
|
|
|
} else if ($view == "day") {
|
2017-09-29 11:16:57 +08:00
|
|
|
$day = new \core_calendar\external\calendar_day_exporter($calendar, $related);
|
2017-09-18 10:53:26 +08:00
|
|
|
$data = $day->export($renderer);
|
2019-05-27 10:38:02 +08:00
|
|
|
$data->viewingday = true;
|
2021-07-01 10:56:52 +08:00
|
|
|
$data->showviewselector = true;
|
2017-09-29 11:16:57 +08:00
|
|
|
$template = 'core_calendar/calendar_day';
|
2017-10-05 15:18:19 +08:00
|
|
|
} else if ($view == "upcoming" || $view == "upcoming_mini") {
|
2017-09-29 12:59:24 +08:00
|
|
|
$upcoming = new \core_calendar\external\calendar_upcoming_exporter($calendar, $related);
|
|
|
|
$data = $upcoming->export($renderer);
|
2017-10-05 15:18:19 +08:00
|
|
|
|
|
|
|
if ($view == "upcoming") {
|
|
|
|
$template = 'core_calendar/calendar_upcoming';
|
2019-05-27 10:38:02 +08:00
|
|
|
$data->viewingupcoming = true;
|
2021-07-01 10:56:52 +08:00
|
|
|
$data->showviewselector = true;
|
2017-10-05 15:18:19 +08:00
|
|
|
} else if ($view == "upcoming_mini") {
|
2017-10-26 10:13:57 +08:00
|
|
|
$template = 'core_calendar/calendar_upcoming_mini';
|
2017-10-05 15:18:19 +08:00
|
|
|
}
|
2017-09-18 10:53:26 +08:00
|
|
|
}
|
2017-07-19 09:34:34 +08:00
|
|
|
|
|
|
|
return [$data, $template];
|
|
|
|
}
|
|
|
|
|
2017-07-28 14:32:09 +08:00
|
|
|
/**
|
|
|
|
* Request and render event form fragment.
|
|
|
|
*
|
|
|
|
* @param array $args The fragment arguments.
|
|
|
|
* @return string The rendered mform fragment.
|
|
|
|
*/
|
2017-07-24 08:01:14 +00:00
|
|
|
function calendar_output_fragment_event_form($args) {
|
2017-08-30 01:08:22 +00:00
|
|
|
global $CFG, $OUTPUT, $USER;
|
2018-07-16 10:10:50 +08:00
|
|
|
require_once($CFG->libdir . '/grouplib.php');
|
2017-07-24 08:01:14 +00:00
|
|
|
$html = '';
|
2017-08-30 01:08:22 +00:00
|
|
|
$data = [];
|
2017-07-24 08:01:14 +00:00
|
|
|
$eventid = isset($args['eventid']) ? clean_param($args['eventid'], PARAM_INT) : null;
|
2017-08-21 06:27:47 +00:00
|
|
|
$starttime = isset($args['starttime']) ? clean_param($args['starttime'], PARAM_INT) : null;
|
2018-07-16 10:10:50 +08:00
|
|
|
$courseid = (isset($args['courseid']) && $args['courseid'] != SITEID) ? clean_param($args['courseid'], PARAM_INT) : null;
|
2017-09-22 09:56:18 +08:00
|
|
|
$categoryid = isset($args['categoryid']) ? clean_param($args['categoryid'], PARAM_INT) : null;
|
2017-07-24 08:01:14 +00:00
|
|
|
$event = null;
|
|
|
|
$hasformdata = isset($args['formdata']) && !empty($args['formdata']);
|
2017-08-30 01:08:22 +00:00
|
|
|
$context = \context_user::instance($USER->id);
|
|
|
|
$editoroptions = \core_calendar\local\event\forms\create::build_editor_options($context);
|
2018-07-16 10:10:50 +08:00
|
|
|
$formoptions = ['editoroptions' => $editoroptions, 'courseid' => $courseid];
|
2017-08-30 01:08:22 +00:00
|
|
|
$draftitemid = 0;
|
2017-07-24 08:01:14 +00:00
|
|
|
|
|
|
|
if ($hasformdata) {
|
|
|
|
parse_str(clean_param($args['formdata'], PARAM_TEXT), $data);
|
2017-08-30 01:08:22 +00:00
|
|
|
if (isset($data['description']['itemid'])) {
|
|
|
|
$draftitemid = $data['description']['itemid'];
|
|
|
|
}
|
2017-07-24 08:01:14 +00:00
|
|
|
}
|
|
|
|
|
2017-08-21 06:27:47 +00:00
|
|
|
if ($starttime) {
|
|
|
|
$formoptions['starttime'] = $starttime;
|
|
|
|
}
|
2019-08-13 09:50:48 +08:00
|
|
|
// Let's check first which event types user can add.
|
|
|
|
$eventtypes = calendar_get_allowed_event_types($courseid);
|
|
|
|
$formoptions['eventtypes'] = $eventtypes;
|
2017-08-21 06:27:47 +00:00
|
|
|
|
2017-07-24 08:01:14 +00:00
|
|
|
if (is_null($eventid)) {
|
2018-07-16 10:10:50 +08:00
|
|
|
if (!empty($courseid)) {
|
|
|
|
$groupcoursedata = groups_get_course_data($courseid);
|
|
|
|
$formoptions['groups'] = [];
|
|
|
|
foreach ($groupcoursedata->groups as $groupid => $groupdata) {
|
|
|
|
$formoptions['groups'][$groupid] = $groupdata->name;
|
|
|
|
}
|
|
|
|
}
|
2019-06-25 19:14:06 +08:00
|
|
|
|
2017-07-24 08:01:14 +00:00
|
|
|
$mform = new \core_calendar\local\event\forms\create(
|
|
|
|
null,
|
|
|
|
$formoptions,
|
|
|
|
'post',
|
|
|
|
'',
|
|
|
|
null,
|
|
|
|
true,
|
|
|
|
$data
|
|
|
|
);
|
2017-10-18 15:41:05 +08:00
|
|
|
|
|
|
|
// If the user is on course context and is allowed to add course events set the event type default to course.
|
2019-05-24 11:52:54 +08:00
|
|
|
if (!empty($courseid) && !empty($eventtypes['course'])) {
|
2017-08-30 10:45:05 +08:00
|
|
|
$data['eventtype'] = 'course';
|
|
|
|
$data['courseid'] = $courseid;
|
|
|
|
$data['groupcourseid'] = $courseid;
|
2018-07-16 10:10:50 +08:00
|
|
|
} else if (!empty($categoryid) && !empty($eventtypes['category'])) {
|
2017-09-22 09:56:18 +08:00
|
|
|
$data['eventtype'] = 'category';
|
|
|
|
$data['categoryid'] = $categoryid;
|
2018-07-16 10:10:50 +08:00
|
|
|
} else if (!empty($groupcoursedata) && !empty($eventtypes['group'])) {
|
|
|
|
$data['groupcourseid'] = $courseid;
|
|
|
|
$data['groups'] = $groupcoursedata->groups;
|
2017-08-30 10:45:05 +08:00
|
|
|
}
|
|
|
|
$mform->set_data($data);
|
2017-07-24 08:01:14 +00:00
|
|
|
} else {
|
|
|
|
$event = calendar_event::load($eventid);
|
2019-02-20 14:54:53 +11:00
|
|
|
|
|
|
|
if (!calendar_edit_event_allowed($event)) {
|
|
|
|
print_error('nopermissiontoupdatecalendar');
|
|
|
|
}
|
|
|
|
|
2017-08-30 01:08:22 +00:00
|
|
|
$mapper = new \core_calendar\local\event\mappers\create_update_form_mapper();
|
|
|
|
$eventdata = $mapper->from_legacy_event_to_data($event);
|
|
|
|
$data = array_merge((array) $eventdata, $data);
|
2017-07-24 08:01:14 +00:00
|
|
|
$event->count_repeats();
|
|
|
|
$formoptions['event'] = $event;
|
2018-07-16 10:10:50 +08:00
|
|
|
|
|
|
|
if (!empty($event->courseid)) {
|
|
|
|
$groupcoursedata = groups_get_course_data($event->courseid);
|
|
|
|
$formoptions['groups'] = [];
|
|
|
|
foreach ($groupcoursedata->groups as $groupid => $groupdata) {
|
|
|
|
$formoptions['groups'][$groupid] = $groupdata->name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-30 01:08:22 +00:00
|
|
|
$data['description']['text'] = file_prepare_draft_area(
|
|
|
|
$draftitemid,
|
|
|
|
$event->context->id,
|
|
|
|
'calendar',
|
|
|
|
'event_description',
|
|
|
|
$event->id,
|
|
|
|
null,
|
|
|
|
$data['description']['text']
|
|
|
|
);
|
|
|
|
$data['description']['itemid'] = $draftitemid;
|
|
|
|
|
2017-07-24 08:01:14 +00:00
|
|
|
$mform = new \core_calendar\local\event\forms\update(
|
|
|
|
null,
|
|
|
|
$formoptions,
|
|
|
|
'post',
|
|
|
|
'',
|
|
|
|
null,
|
|
|
|
true,
|
|
|
|
$data
|
|
|
|
);
|
|
|
|
$mform->set_data($data);
|
|
|
|
|
|
|
|
// Check to see if this event is part of a subscription or import.
|
|
|
|
// If so display a warning on edit.
|
|
|
|
if (isset($event->subscriptionid) && ($event->subscriptionid != null)) {
|
|
|
|
$renderable = new \core\output\notification(
|
|
|
|
get_string('eventsubscriptioneditwarning', 'calendar'),
|
|
|
|
\core\output\notification::NOTIFY_INFO
|
|
|
|
);
|
|
|
|
|
|
|
|
$html .= $OUTPUT->render($renderable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-30 01:08:22 +00:00
|
|
|
if ($hasformdata) {
|
|
|
|
$mform->is_validated();
|
|
|
|
}
|
|
|
|
|
2017-07-24 08:01:14 +00:00
|
|
|
$html .= $mform->render();
|
|
|
|
return $html;
|
|
|
|
}
|
2017-07-19 09:27:35 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate the timestamp from the supplied Gregorian Year, Month, and Day.
|
|
|
|
*
|
|
|
|
* @param int $d The day
|
|
|
|
* @param int $m The month
|
|
|
|
* @param int $y The year
|
|
|
|
* @param int $time The timestamp to use instead of a separate y/m/d.
|
|
|
|
* @return int The timestamp
|
|
|
|
*/
|
|
|
|
function calendar_get_timestamp($d, $m, $y, $time = 0) {
|
|
|
|
// If a day, month and year were passed then convert it to a timestamp. If these were passed
|
|
|
|
// then we can assume the day, month and year are passed as Gregorian, as no where in core
|
|
|
|
// should we be passing these values rather than the time.
|
|
|
|
if (!empty($d) && !empty($m) && !empty($y)) {
|
|
|
|
if (checkdate($m, $d, $y)) {
|
|
|
|
$time = make_timestamp($y, $m, $d);
|
|
|
|
} else {
|
|
|
|
$time = time();
|
|
|
|
}
|
|
|
|
} else if (empty($time)) {
|
|
|
|
$time = time();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $time;
|
|
|
|
}
|
2017-07-19 08:58:11 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the calendar footer options.
|
|
|
|
*
|
|
|
|
* @param calendar_information $calendar The calendar information object.
|
2021-07-13 14:14:33 +08:00
|
|
|
* @param array $options Display options for the footer. If an option is not set, a default value will be provided.
|
|
|
|
* It consists of:
|
|
|
|
* - showfullcalendarlink - bool - Whether to show the full calendar link or not. Defaults to false.
|
|
|
|
*
|
2017-07-19 08:58:11 +08:00
|
|
|
* @return array The data for template and template name.
|
|
|
|
*/
|
2021-07-13 14:14:33 +08:00
|
|
|
function calendar_get_footer_options($calendar, array $options = []) {
|
2020-09-16 18:41:47 +02:00
|
|
|
global $CFG, $USER, $PAGE;
|
2017-07-19 08:58:11 +08:00
|
|
|
|
|
|
|
// Generate hash for iCal link.
|
2020-09-16 18:41:47 +02:00
|
|
|
$authtoken = calendar_get_export_token($USER);
|
2017-07-19 08:58:11 +08:00
|
|
|
|
|
|
|
$renderer = $PAGE->get_renderer('core_calendar');
|
2021-07-13 14:14:33 +08:00
|
|
|
$footer = new \core_calendar\external\footer_options_exporter($calendar, $USER->id, $authtoken, $options);
|
2017-07-19 08:58:11 +08:00
|
|
|
$data = $footer->export($renderer);
|
|
|
|
$template = 'core_calendar/footer_options';
|
|
|
|
|
|
|
|
return [$data, $template];
|
|
|
|
}
|
2017-09-01 14:58:14 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the list of potential calendar filter types as a type => name
|
|
|
|
* combination.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function calendar_get_filter_types() {
|
|
|
|
$types = [
|
|
|
|
'site',
|
2017-09-18 12:47:37 +08:00
|
|
|
'category',
|
2017-09-01 14:58:14 +08:00
|
|
|
'course',
|
|
|
|
'group',
|
|
|
|
'user',
|
2020-04-30 18:55:36 +02:00
|
|
|
'other'
|
2017-09-01 14:58:14 +08:00
|
|
|
];
|
|
|
|
|
|
|
|
return array_map(function($type) {
|
|
|
|
return [
|
2017-09-12 09:45:59 +08:00
|
|
|
'eventtype' => $type,
|
2017-09-01 14:58:14 +08:00
|
|
|
'name' => get_string("eventtype{$type}", "calendar"),
|
2019-09-02 14:42:24 +02:00
|
|
|
'icon' => true,
|
|
|
|
'key' => 'i/' . $type . 'event',
|
|
|
|
'component' => 'core'
|
2017-09-01 14:58:14 +08:00
|
|
|
];
|
|
|
|
}, $types);
|
|
|
|
}
|
2017-09-18 12:47:37 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether the specified event type is valid.
|
|
|
|
*
|
|
|
|
* @param string $type
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
function calendar_is_valid_eventtype($type) {
|
|
|
|
$validtypes = [
|
|
|
|
'user',
|
|
|
|
'group',
|
|
|
|
'course',
|
|
|
|
'category',
|
|
|
|
'site',
|
|
|
|
];
|
|
|
|
return in_array($type, $validtypes);
|
|
|
|
}
|
2018-07-16 10:00:03 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get event types the user can create event based on categories, courses and groups
|
|
|
|
* the logged in user belongs to.
|
|
|
|
*
|
|
|
|
* @param int|null $courseid The course id.
|
|
|
|
* @return array The array of allowed types.
|
|
|
|
*/
|
|
|
|
function calendar_get_allowed_event_types(int $courseid = null) {
|
|
|
|
global $DB, $CFG, $USER;
|
|
|
|
|
|
|
|
$types = [
|
|
|
|
'user' => false,
|
|
|
|
'site' => false,
|
|
|
|
'course' => false,
|
|
|
|
'group' => false,
|
|
|
|
'category' => false
|
|
|
|
];
|
|
|
|
|
|
|
|
if (!empty($courseid) && $courseid != SITEID) {
|
|
|
|
$context = \context_course::instance($courseid);
|
|
|
|
$types['user'] = has_capability('moodle/calendar:manageownentries', $context);
|
2019-11-19 11:46:40 +00:00
|
|
|
calendar_internal_update_course_and_group_permission($courseid, $context, $types);
|
2018-07-16 10:00:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (has_capability('moodle/calendar:manageentries', \context_course::instance(SITEID))) {
|
|
|
|
$types['site'] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_capability('moodle/calendar:manageownentries', \context_system::instance())) {
|
|
|
|
$types['user'] = true;
|
|
|
|
}
|
2018-06-16 11:57:27 +08:00
|
|
|
if (core_course_category::has_manage_capability_on_any()) {
|
2018-07-16 10:00:03 +08:00
|
|
|
$types['category'] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We still don't know if the user can create group and course events, so iterate over the courses to find out
|
|
|
|
// if the user has capabilities in one of the courses.
|
2018-07-16 10:10:50 +08:00
|
|
|
if ($types['course'] == false || $types['group'] == false) {
|
2018-07-25 08:40:28 +08:00
|
|
|
if ($CFG->calendar_adminseesall && has_capability('moodle/calendar:manageentries', context_system::instance())) {
|
|
|
|
$sql = "SELECT c.id, " . context_helper::get_preload_record_columns_sql('ctx') . "
|
|
|
|
FROM {course} c
|
|
|
|
JOIN {context} ctx ON ctx.contextlevel = ? AND ctx.instanceid = c.id
|
|
|
|
WHERE c.id IN (
|
|
|
|
SELECT DISTINCT courseid FROM {groups}
|
|
|
|
)";
|
|
|
|
$courseswithgroups = $DB->get_recordset_sql($sql, [CONTEXT_COURSE]);
|
|
|
|
foreach ($courseswithgroups as $course) {
|
|
|
|
context_helper::preload_from_record($course);
|
|
|
|
$context = context_course::instance($course->id);
|
|
|
|
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $context)) {
|
|
|
|
if (has_any_capability(['moodle/site:accessallgroups', 'moodle/calendar:managegroupentries'], $context)) {
|
|
|
|
// The user can manage group entries or access any group.
|
|
|
|
$types['group'] = true;
|
|
|
|
$types['course'] = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$courseswithgroups->close();
|
|
|
|
|
|
|
|
if (false === $types['course']) {
|
|
|
|
// Course is still not confirmed. There may have been no courses with a group in them.
|
|
|
|
$ctxfields = context_helper::get_preload_record_columns_sql('ctx');
|
|
|
|
$sql = "SELECT
|
|
|
|
c.id, c.visible, {$ctxfields}
|
2018-09-29 10:37:31 +02:00
|
|
|
FROM {course} c
|
2018-07-25 08:40:28 +08:00
|
|
|
JOIN {context} ctx ON (ctx.instanceid = c.id AND ctx.contextlevel = :contextlevel)";
|
|
|
|
$params = [
|
|
|
|
'contextlevel' => CONTEXT_COURSE,
|
|
|
|
];
|
|
|
|
$courses = $DB->get_recordset_sql($sql, $params);
|
|
|
|
foreach ($courses as $course) {
|
|
|
|
context_helper::preload_from_record($course);
|
|
|
|
$context = context_course::instance($course->id);
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $context)) {
|
|
|
|
$types['course'] = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$courses->close();
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$courses = calendar_get_default_courses(null, 'id');
|
|
|
|
if (empty($courses)) {
|
|
|
|
return $types;
|
|
|
|
}
|
2018-07-16 10:00:03 +08:00
|
|
|
|
|
|
|
$courseids = array_map(function($c) {
|
|
|
|
return $c->id;
|
|
|
|
}, $courses);
|
|
|
|
|
2018-07-25 10:31:54 +08:00
|
|
|
// Check whether the user has access to create events within courses which have groups.
|
2018-07-25 08:40:28 +08:00
|
|
|
list($insql, $params) = $DB->get_in_or_equal($courseids, SQL_PARAMS_NAMED);
|
2018-07-16 10:00:03 +08:00
|
|
|
$sql = "SELECT c.id, " . context_helper::get_preload_record_columns_sql('ctx') . "
|
|
|
|
FROM {course} c
|
2018-07-25 08:40:28 +08:00
|
|
|
JOIN {context} ctx ON ctx.contextlevel = :contextlevel AND ctx.instanceid = c.id
|
2018-07-16 10:00:03 +08:00
|
|
|
WHERE c.id $insql
|
2018-07-25 08:40:28 +08:00
|
|
|
AND c.id IN (SELECT DISTINCT courseid FROM {groups})";
|
|
|
|
$params['contextlevel'] = CONTEXT_COURSE;
|
|
|
|
$courseswithgroups = $DB->get_recordset_sql($sql, $params);
|
2018-07-16 10:00:03 +08:00
|
|
|
foreach ($courseswithgroups as $coursewithgroup) {
|
|
|
|
context_helper::preload_from_record($coursewithgroup);
|
|
|
|
$context = context_course::instance($coursewithgroup->id);
|
|
|
|
|
2019-11-19 11:46:40 +00:00
|
|
|
calendar_internal_update_course_and_group_permission($coursewithgroup->id, $context, $types);
|
2018-07-16 10:00:03 +08:00
|
|
|
|
|
|
|
// Okay, course and group event types are allowed, no need to keep the loop iteration.
|
|
|
|
if ($types['course'] == true && $types['group'] == true) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-07-25 08:40:28 +08:00
|
|
|
$courseswithgroups->close();
|
|
|
|
|
|
|
|
if (false === $types['course']) {
|
|
|
|
list($insql, $params) = $DB->get_in_or_equal($courseids, SQL_PARAMS_NAMED);
|
|
|
|
$contextsql = "SELECT c.id, " . context_helper::get_preload_record_columns_sql('ctx') . "
|
|
|
|
FROM {course} c
|
|
|
|
JOIN {context} ctx ON ctx.contextlevel = :contextlevel AND ctx.instanceid = c.id
|
2018-07-25 10:31:54 +08:00
|
|
|
WHERE c.id $insql";
|
2018-07-25 08:40:28 +08:00
|
|
|
$params['contextlevel'] = CONTEXT_COURSE;
|
|
|
|
$contextrecords = $DB->get_recordset_sql($contextsql, $params);
|
|
|
|
foreach ($contextrecords as $course) {
|
|
|
|
context_helper::preload_from_record($course);
|
|
|
|
$coursecontext = context_course::instance($course->id);
|
|
|
|
if (has_capability('moodle/calendar:manageentries', $coursecontext)
|
|
|
|
&& ($courseid == $course->id || empty($courseid))) {
|
|
|
|
$types['course'] = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$contextrecords->close();
|
|
|
|
}
|
|
|
|
|
2018-07-16 10:00:03 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $types;
|
|
|
|
}
|
2019-11-19 11:46:40 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Given a course id, and context, updates the permission types array to add the 'course' or 'group'
|
|
|
|
* permission if it is relevant for that course.
|
|
|
|
*
|
|
|
|
* For efficiency, if they already have 'course' or 'group' then it skips checks.
|
|
|
|
*
|
|
|
|
* Do not call this function directly, it is only for use by calendar_get_allowed_event_types().
|
|
|
|
*
|
|
|
|
* @param int $courseid Course id
|
|
|
|
* @param context $context Context for that course
|
|
|
|
* @param array $types Current permissions
|
|
|
|
*/
|
|
|
|
function calendar_internal_update_course_and_group_permission(int $courseid, context $context, array &$types) {
|
|
|
|
if (!$types['course']) {
|
2019-11-26 22:16:38 +01:00
|
|
|
// If they have manageentries permission on the course, then they can update this course.
|
2019-11-19 11:46:40 +00:00
|
|
|
if (has_capability('moodle/calendar:manageentries', $context)) {
|
|
|
|
$types['course'] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// To update group events they must have EITHER manageentries OR managegroupentries.
|
|
|
|
if (!$types['group'] && (has_capability('moodle/calendar:manageentries', $context) ||
|
|
|
|
has_capability('moodle/calendar:managegroupentries', $context))) {
|
|
|
|
// And they also need for a group to exist on the course.
|
|
|
|
$groups = groups_get_all_groups($courseid);
|
|
|
|
if (!empty($groups)) {
|
|
|
|
// And either accessallgroups, or belong to one of the groups.
|
|
|
|
if (has_capability('moodle/site:accessallgroups', $context)) {
|
|
|
|
$types['group'] = true;
|
|
|
|
} else {
|
|
|
|
foreach ($groups as $group) {
|
|
|
|
if (groups_is_member($group->id)) {
|
|
|
|
$types['group'] = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-09-16 18:41:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the auth token for exporting the given user calendar.
|
|
|
|
* @param stdClass $user The user to export the calendar for
|
|
|
|
*
|
|
|
|
* @return string The export token.
|
|
|
|
*/
|
|
|
|
function calendar_get_export_token(stdClass $user): string {
|
|
|
|
global $CFG, $DB;
|
|
|
|
|
|
|
|
return sha1($user->id . $DB->get_field('user', 'password', ['id' => $user->id]) . $CFG->calendar_exportsalt);
|
|
|
|
}
|
2021-07-07 14:53:01 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the list of URL parameters for calendar expport and import links.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function calendar_get_export_import_link_params(): array {
|
|
|
|
global $PAGE;
|
|
|
|
|
|
|
|
$params = [];
|
|
|
|
if ($courseid = $PAGE->url->get_param('course')) {
|
|
|
|
$params['course'] = $courseid;
|
|
|
|
}
|
|
|
|
if ($categoryid = $PAGE->url->get_param('category')) {
|
|
|
|
$params['category'] = $categoryid;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $params;
|
|
|
|
}
|
2021-08-01 21:18:51 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements the inplace editable feature.
|
|
|
|
*
|
|
|
|
* @param string $itemtype Type of the inplace editable element
|
|
|
|
* @param int $itemid Id of the item to edit
|
|
|
|
* @param int $newvalue New value of the item
|
|
|
|
* @return \core\output\inplace_editable
|
|
|
|
*/
|
|
|
|
function calendar_inplace_editable(string $itemtype, int $itemid, int $newvalue): \core\output\inplace_editable {
|
|
|
|
global $OUTPUT;
|
|
|
|
|
|
|
|
if ($itemtype === 'refreshinterval') {
|
|
|
|
|
|
|
|
$subscription = calendar_get_subscription($itemid);
|
|
|
|
$context = calendar_get_calendar_context($subscription);
|
|
|
|
\external_api::validate_context($context);
|
|
|
|
|
|
|
|
$updateresult = \core_calendar\output\refreshintervalcollection::update($itemid, $newvalue);
|
|
|
|
|
|
|
|
$refreshresults = calendar_update_subscription_events($itemid);
|
|
|
|
\core\notification::add($OUTPUT->render_from_template(
|
|
|
|
'core_calendar/subscription_update_result',
|
|
|
|
array_merge($refreshresults, [
|
|
|
|
'subscriptionname' => s($subscription->name),
|
|
|
|
])
|
|
|
|
), \core\notification::INFO);
|
|
|
|
|
|
|
|
return $updateresult;
|
|
|
|
}
|
|
|
|
|
|
|
|
\external_api::validate_context(context_system::instance());
|
|
|
|
}
|