1 |
efrain |
1 |
<?php
|
|
|
2 |
// This file is part of Moodle - http://moodle.org/
|
|
|
3 |
//
|
|
|
4 |
// Moodle is free software: you can redistribute it and/or modify
|
|
|
5 |
// it under the terms of the GNU General Public License as published by
|
|
|
6 |
// the Free Software Foundation, either version 3 of the License, or
|
|
|
7 |
// (at your option) any later version.
|
|
|
8 |
//
|
|
|
9 |
// Moodle is distributed in the hope that it will be useful,
|
|
|
10 |
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
11 |
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
12 |
// GNU General Public License for more details.
|
|
|
13 |
//
|
|
|
14 |
// You should have received a copy of the GNU General Public License
|
|
|
15 |
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
|
|
16 |
|
|
|
17 |
/**
|
|
|
18 |
* Contains event class for displaying a calendar event.
|
|
|
19 |
*
|
|
|
20 |
* @package core_calendar
|
|
|
21 |
* @copyright 2017 Ryan Wyllie <ryan@moodle.com>
|
|
|
22 |
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
23 |
*/
|
|
|
24 |
|
|
|
25 |
namespace core_calendar\external;
|
|
|
26 |
|
|
|
27 |
defined('MOODLE_INTERNAL') || die();
|
|
|
28 |
|
|
|
29 |
use \core_calendar\local\event\container;
|
|
|
30 |
use \renderer_base;
|
|
|
31 |
require_once($CFG->dirroot . '/course/lib.php');
|
|
|
32 |
/**
|
|
|
33 |
* Class for displaying a calendar event.
|
|
|
34 |
*
|
|
|
35 |
* @package core_calendar
|
|
|
36 |
* @copyright 2017 Ryan Wyllie <ryan@moodle.com>
|
|
|
37 |
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
38 |
*/
|
|
|
39 |
class calendar_event_exporter extends event_exporter_base {
|
|
|
40 |
|
|
|
41 |
/**
|
|
|
42 |
* Return the list of additional properties.
|
|
|
43 |
*
|
|
|
44 |
* @return array
|
|
|
45 |
*/
|
|
|
46 |
protected static function define_other_properties() {
|
|
|
47 |
|
|
|
48 |
$values = parent::define_other_properties();
|
|
|
49 |
$values['url'] = ['type' => PARAM_URL];
|
|
|
50 |
$values['islastday'] = [
|
|
|
51 |
'type' => PARAM_BOOL,
|
|
|
52 |
'default' => false,
|
|
|
53 |
];
|
|
|
54 |
$values['popupname'] = [
|
|
|
55 |
'type' => PARAM_RAW,
|
|
|
56 |
];
|
|
|
57 |
$values['mindaytimestamp'] = [
|
|
|
58 |
'type' => PARAM_INT,
|
|
|
59 |
'optional' => true
|
|
|
60 |
];
|
|
|
61 |
$values['mindayerror'] = [
|
|
|
62 |
'type' => PARAM_TEXT,
|
|
|
63 |
'optional' => true
|
|
|
64 |
];
|
|
|
65 |
$values['maxdaytimestamp'] = [
|
|
|
66 |
'type' => PARAM_INT,
|
|
|
67 |
'optional' => true
|
|
|
68 |
];
|
|
|
69 |
$values['maxdayerror'] = [
|
|
|
70 |
'type' => PARAM_TEXT,
|
|
|
71 |
'optional' => true
|
|
|
72 |
];
|
|
|
73 |
$values['draggable'] = [
|
|
|
74 |
'type' => PARAM_BOOL,
|
|
|
75 |
'default' => false
|
|
|
76 |
];
|
|
|
77 |
|
|
|
78 |
return $values;
|
|
|
79 |
}
|
|
|
80 |
|
|
|
81 |
/**
|
|
|
82 |
* Get the additional values to inject while exporting.
|
|
|
83 |
*
|
|
|
84 |
* @param renderer_base $output The renderer.
|
|
|
85 |
* @return array Keys are the property names, values are their values.
|
|
|
86 |
*/
|
|
|
87 |
protected function get_other_values(renderer_base $output) {
|
|
|
88 |
global $CFG;
|
|
|
89 |
|
|
|
90 |
$values = parent::get_other_values($output);
|
|
|
91 |
$event = $this->event;
|
|
|
92 |
$course = $this->related['course'];
|
|
|
93 |
$hascourse = !empty($course);
|
|
|
94 |
|
|
|
95 |
// By default all events that can be edited are
|
|
|
96 |
// draggable.
|
|
|
97 |
$values['draggable'] = $values['canedit'];
|
|
|
98 |
|
|
|
99 |
if ($moduleproxy = $event->get_course_module()) {
|
|
|
100 |
$modulename = $moduleproxy->get('modname');
|
|
|
101 |
$moduleid = $moduleproxy->get('id');
|
|
|
102 |
$url = new \moodle_url(sprintf('/mod/%s/view.php', $modulename), ['id' => $moduleid]);
|
|
|
103 |
|
|
|
104 |
// Build edit event url for action events.
|
|
|
105 |
$params = array('update' => $moduleid, 'return' => true, 'sesskey' => sesskey());
|
|
|
106 |
$editurl = new \moodle_url('/course/mod.php', $params);
|
|
|
107 |
$values['editurl'] = $editurl->out(false);
|
|
|
108 |
} else if ($event->get_type() == 'category') {
|
|
|
109 |
$url = $event->get_category()->get_proxied_instance()->get_view_link();
|
|
|
110 |
} else {
|
|
|
111 |
$url = course_get_url($hascourse ? $course : SITEID);
|
|
|
112 |
}
|
|
|
113 |
|
|
|
114 |
$values['url'] = $url->out(false);
|
|
|
115 |
$values['islastday'] = false;
|
|
|
116 |
$today = $this->related['type']->timestamp_to_date_array($this->related['today']);
|
|
|
117 |
|
|
|
118 |
if ($hascourse) {
|
|
|
119 |
$values['popupname'] = \core_external\util::format_string(
|
|
|
120 |
$this->event->get_name(),
|
|
|
121 |
\context_course::instance($course->id),
|
|
|
122 |
true
|
|
|
123 |
);
|
|
|
124 |
} else {
|
|
|
125 |
$values['popupname'] = \core_external\util::format_string($this->event->get_name(), \context_system::instance(), true);
|
|
|
126 |
}
|
|
|
127 |
|
|
|
128 |
$times = $this->event->get_times();
|
|
|
129 |
if ($duration = $times->get_duration()) {
|
|
|
130 |
$enddate = $this->related['type']->timestamp_to_date_array($times->get_end_time()->getTimestamp());
|
|
|
131 |
$values['islastday'] = true;
|
|
|
132 |
$values['islastday'] = $values['islastday'] && $enddate['year'] == $today['year'];
|
|
|
133 |
$values['islastday'] = $values['islastday'] && $enddate['mon'] == $today['mon'];
|
|
|
134 |
$values['islastday'] = $values['islastday'] && $enddate['mday'] == $today['mday'];
|
|
|
135 |
}
|
|
|
136 |
|
|
|
137 |
$subscription = $this->event->get_subscription();
|
|
|
138 |
if ($subscription && !empty($subscription->get('id')) && $CFG->calendar_showicalsource) {
|
|
|
139 |
$a = (object) [
|
|
|
140 |
'name' => $values['popupname'],
|
|
|
141 |
'source' => $subscription->get('name'),
|
|
|
142 |
];
|
|
|
143 |
$values['popupname'] = get_string('namewithsource', 'calendar', $a);
|
|
|
144 |
} else {
|
|
|
145 |
if ($values['islastday']) {
|
|
|
146 |
$startdate = $this->related['type']->timestamp_to_date_array($times->get_start_time()->getTimestamp());
|
|
|
147 |
$samedate = true;
|
|
|
148 |
$samedate = $samedate && $startdate['mon'] == $enddate['mon'];
|
|
|
149 |
$samedate = $samedate && $startdate['year'] == $enddate['year'];
|
|
|
150 |
$samedate = $samedate && $startdate['mday'] == $enddate['mday'];
|
|
|
151 |
|
|
|
152 |
if (!$samedate) {
|
|
|
153 |
$values['popupname'] = get_string('eventendtimewrapped', 'calendar', $values['popupname']);
|
|
|
154 |
}
|
|
|
155 |
}
|
|
|
156 |
}
|
|
|
157 |
|
|
|
158 |
// Include category name into the event name, if applicable.
|
|
|
159 |
$proxy = $this->event->get_category();
|
|
|
160 |
if ($proxy && $proxy->get('id')) {
|
|
|
161 |
$category = $proxy->get_proxied_instance();
|
|
|
162 |
$eventnameparams = (object) [
|
|
|
163 |
'name' => $values['popupname'],
|
|
|
164 |
'category' => $category->get_formatted_name(),
|
|
|
165 |
];
|
|
|
166 |
$values['popupname'] = get_string('eventnameandcategory', 'calendar', $eventnameparams);
|
|
|
167 |
}
|
|
|
168 |
|
|
|
169 |
// Include course's shortname into the event name, if applicable.
|
|
|
170 |
if ($hascourse && $course->id !== SITEID) {
|
|
|
171 |
$eventnameparams = (object) [
|
|
|
172 |
'name' => $values['popupname'],
|
|
|
173 |
'course' => $values['course']->shortname,
|
|
|
174 |
];
|
|
|
175 |
$values['popupname'] = get_string('eventnameandcourse', 'calendar', $eventnameparams);
|
|
|
176 |
}
|
|
|
177 |
|
|
|
178 |
if ($event->get_course_module()) {
|
|
|
179 |
$values = array_merge($values, $this->get_module_timestamp_limits($event));
|
|
|
180 |
} else if ($hascourse && $course->id != SITEID && empty($event->get_group())) {
|
|
|
181 |
// This is a course event.
|
|
|
182 |
$values = array_merge($values, $this->get_course_timestamp_limits($event));
|
|
|
183 |
}
|
|
|
184 |
|
|
|
185 |
return $values;
|
|
|
186 |
}
|
|
|
187 |
|
|
|
188 |
/**
|
|
|
189 |
* Returns a list of objects that are related.
|
|
|
190 |
*
|
|
|
191 |
* @return array
|
|
|
192 |
*/
|
|
|
193 |
protected static function define_related() {
|
|
|
194 |
$related = parent::define_related();
|
|
|
195 |
$related['daylink'] = \moodle_url::class;
|
|
|
196 |
$related['type'] = '\core_calendar\type_base';
|
|
|
197 |
$related['today'] = 'int';
|
|
|
198 |
$related['moduleinstance'] = 'stdClass?';
|
|
|
199 |
|
|
|
200 |
return $related;
|
|
|
201 |
}
|
|
|
202 |
|
|
|
203 |
/**
|
|
|
204 |
* Return the normalised event type.
|
|
|
205 |
* Activity events are normalised to be course events.
|
|
|
206 |
*
|
|
|
207 |
* @return string
|
|
|
208 |
*/
|
|
|
209 |
public function get_calendar_event_type() {
|
|
|
210 |
if ($this->event->get_course_module()) {
|
|
|
211 |
return 'course';
|
|
|
212 |
}
|
|
|
213 |
|
|
|
214 |
return $this->event->get_type();
|
|
|
215 |
}
|
|
|
216 |
|
|
|
217 |
/**
|
|
|
218 |
* Return the set of minimum and maximum date timestamp values
|
|
|
219 |
* for the given event.
|
|
|
220 |
*
|
|
|
221 |
* @param \core_calendar\local\event\entities\event_interface $event
|
|
|
222 |
* @return array
|
|
|
223 |
*/
|
|
|
224 |
protected function get_course_timestamp_limits($event) {
|
|
|
225 |
$values = [];
|
|
|
226 |
$mapper = container::get_event_mapper();
|
|
|
227 |
$starttime = $event->get_times()->get_start_time();
|
|
|
228 |
|
|
|
229 |
list($min, $max) = component_callback(
|
|
|
230 |
'core_course',
|
|
|
231 |
'core_calendar_get_valid_event_timestart_range',
|
|
|
232 |
[$mapper->from_event_to_legacy_event($event), $event->get_course()->get_proxied_instance()],
|
|
|
233 |
[false, false]
|
|
|
234 |
);
|
|
|
235 |
|
|
|
236 |
// The callback will return false for either of the
|
|
|
237 |
// min or max cutoffs to indicate that there are no
|
|
|
238 |
// valid timestart values. In which case the event is
|
|
|
239 |
// not draggable.
|
|
|
240 |
if ($min === false || $max === false) {
|
|
|
241 |
return ['draggable' => false];
|
|
|
242 |
}
|
|
|
243 |
|
|
|
244 |
if ($min) {
|
|
|
245 |
$values = array_merge($values, $this->get_timestamp_min_limit($starttime, $min));
|
|
|
246 |
}
|
|
|
247 |
|
|
|
248 |
if ($max) {
|
|
|
249 |
$values = array_merge($values, $this->get_timestamp_max_limit($starttime, $max));
|
|
|
250 |
}
|
|
|
251 |
|
|
|
252 |
return $values;
|
|
|
253 |
}
|
|
|
254 |
|
|
|
255 |
/**
|
|
|
256 |
* Return the set of minimum and maximum date timestamp values
|
|
|
257 |
* for the given event.
|
|
|
258 |
*
|
|
|
259 |
* @param \core_calendar\local\event\entities\event_interface $event
|
|
|
260 |
* @return array
|
|
|
261 |
*/
|
|
|
262 |
protected function get_module_timestamp_limits($event) {
|
|
|
263 |
$values = [];
|
|
|
264 |
$mapper = container::get_event_mapper();
|
|
|
265 |
$starttime = $event->get_times()->get_start_time();
|
|
|
266 |
$modname = $event->get_course_module()->get('modname');
|
|
|
267 |
$moduleinstance = $this->related['moduleinstance'];
|
|
|
268 |
|
|
|
269 |
list($min, $max) = component_callback(
|
|
|
270 |
'mod_' . $modname,
|
|
|
271 |
'core_calendar_get_valid_event_timestart_range',
|
|
|
272 |
[$mapper->from_event_to_legacy_event($event), $moduleinstance],
|
|
|
273 |
[false, false]
|
|
|
274 |
);
|
|
|
275 |
|
|
|
276 |
// The callback will return false for either of the
|
|
|
277 |
// min or max cutoffs to indicate that there are no
|
|
|
278 |
// valid timestart values. In which case the event is
|
|
|
279 |
// not draggable.
|
|
|
280 |
if ($min === false || $max === false) {
|
|
|
281 |
return ['draggable' => false];
|
|
|
282 |
}
|
|
|
283 |
|
|
|
284 |
if ($min) {
|
|
|
285 |
$values = array_merge($values, $this->get_timestamp_min_limit($starttime, $min));
|
|
|
286 |
}
|
|
|
287 |
|
|
|
288 |
if ($max) {
|
|
|
289 |
$values = array_merge($values, $this->get_timestamp_max_limit($starttime, $max));
|
|
|
290 |
}
|
|
|
291 |
|
|
|
292 |
return $values;
|
|
|
293 |
}
|
|
|
294 |
|
|
|
295 |
/**
|
|
|
296 |
* Get the correct minimum midnight day limit based on the event start time
|
|
|
297 |
* and the minimum timestamp limit of what the event belongs to.
|
|
|
298 |
*
|
|
|
299 |
* @param DateTimeInterface $starttime The event start time
|
|
|
300 |
* @param array $min The module's minimum limit for the event
|
|
|
301 |
* @return array Returns an array with mindaytimestamp and mindayerror keys.
|
|
|
302 |
*/
|
|
|
303 |
protected function get_timestamp_min_limit(\DateTimeInterface $starttime, $min) {
|
|
|
304 |
// We need to check that the minimum valid time is earlier in the
|
|
|
305 |
// day than the current event time so that if the user drags and drops
|
|
|
306 |
// the event to this day (which changes the date but not the time) it
|
|
|
307 |
// will result in a valid time start for the event.
|
|
|
308 |
//
|
|
|
309 |
// For example:
|
|
|
310 |
// An event that starts on 2017-01-10 08:00 with a minimum cutoff
|
|
|
311 |
// of 2017-01-05 09:00 means that 2017-01-05 is not a valid start day
|
|
|
312 |
// for the drag and drop because it would result in the event start time
|
|
|
313 |
// being set to 2017-01-05 08:00, which is invalid. Instead the minimum
|
|
|
314 |
// valid start day would be 2017-01-06.
|
|
|
315 |
$values = [];
|
|
|
316 |
$timestamp = $min[0];
|
|
|
317 |
$errorstring = $min[1];
|
|
|
318 |
$mindate = (new \DateTimeImmutable())->setTimestamp($timestamp);
|
|
|
319 |
$minstart = $mindate->setTime(
|
|
|
320 |
$starttime->format('H'),
|
|
|
321 |
$starttime->format('i'),
|
|
|
322 |
$starttime->format('s')
|
|
|
323 |
);
|
|
|
324 |
$midnight = usergetmidnight($timestamp);
|
|
|
325 |
|
|
|
326 |
if ($mindate <= $minstart) {
|
|
|
327 |
$values['mindaytimestamp'] = $midnight;
|
|
|
328 |
} else {
|
|
|
329 |
$tomorrow = (new \DateTime())->setTimestamp($midnight)->modify('+1 day');
|
|
|
330 |
$values['mindaytimestamp'] = $tomorrow->getTimestamp();
|
|
|
331 |
}
|
|
|
332 |
|
|
|
333 |
// Get the human readable error message to display if the min day
|
|
|
334 |
// timestamp is violated.
|
|
|
335 |
$values['mindayerror'] = $errorstring;
|
|
|
336 |
return $values;
|
|
|
337 |
}
|
|
|
338 |
|
|
|
339 |
/**
|
|
|
340 |
* Get the correct maximum midnight day limit based on the event start time
|
|
|
341 |
* and the maximum timestamp limit of what the event belongs to.
|
|
|
342 |
*
|
|
|
343 |
* @param DateTimeInterface $starttime The event start time
|
|
|
344 |
* @param array $max The module's maximum limit for the event
|
|
|
345 |
* @return array Returns an array with maxdaytimestamp and maxdayerror keys.
|
|
|
346 |
*/
|
|
|
347 |
protected function get_timestamp_max_limit(\DateTimeInterface $starttime, $max) {
|
|
|
348 |
// We're doing a similar calculation here as we are for the minimum
|
|
|
349 |
// day timestamp. See the explanation above.
|
|
|
350 |
$values = [];
|
|
|
351 |
$timestamp = $max[0];
|
|
|
352 |
$errorstring = $max[1];
|
|
|
353 |
$maxdate = (new \DateTimeImmutable())->setTimestamp($timestamp);
|
|
|
354 |
$maxstart = $maxdate->setTime(
|
|
|
355 |
$starttime->format('H'),
|
|
|
356 |
$starttime->format('i'),
|
|
|
357 |
$starttime->format('s')
|
|
|
358 |
);
|
|
|
359 |
$midnight = usergetmidnight($timestamp);
|
|
|
360 |
|
|
|
361 |
if ($maxdate >= $maxstart) {
|
|
|
362 |
$values['maxdaytimestamp'] = $midnight;
|
|
|
363 |
} else {
|
|
|
364 |
$yesterday = (new \DateTime())->setTimestamp($midnight)->modify('-1 day');
|
|
|
365 |
$values['maxdaytimestamp'] = $yesterday->getTimestamp();
|
|
|
366 |
}
|
|
|
367 |
|
|
|
368 |
// Get the human readable error message to display if the max day
|
|
|
369 |
// timestamp is violated.
|
|
|
370 |
$values['maxdayerror'] = $errorstring;
|
|
|
371 |
return $values;
|
|
|
372 |
}
|
|
|
373 |
|
|
|
374 |
/**
|
|
|
375 |
* Get the correct minimum midnight day limit based on the event start time
|
|
|
376 |
* and the module's minimum timestamp limit.
|
|
|
377 |
*
|
|
|
378 |
* @deprecated since Moodle 3.6. Please use get_timestamp_min_limit().
|
|
|
379 |
* @todo final deprecation. To be removed in Moodle 3.10
|
|
|
380 |
* @param \DateTimeInterface $starttime The event start time
|
|
|
381 |
* @param array $min The module's minimum limit for the event
|
|
|
382 |
* @return array Returns an array with mindaytimestamp and mindayerror keys.
|
|
|
383 |
*/
|
|
|
384 |
protected function get_module_timestamp_min_limit(\DateTimeInterface $starttime, $min) {
|
|
|
385 |
debugging('get_module_timestamp_min_limit() has been deprecated. Please call get_timestamp_min_limit() instead.',
|
|
|
386 |
DEBUG_DEVELOPER);
|
|
|
387 |
return $this->get_timestamp_min_limit($starttime, $min);
|
|
|
388 |
}
|
|
|
389 |
|
|
|
390 |
/**
|
|
|
391 |
* Get the correct maximum midnight day limit based on the event start time
|
|
|
392 |
* and the module's maximum timestamp limit.
|
|
|
393 |
*
|
|
|
394 |
* @deprecated since Moodle 3.6. Please use get_timestamp_max_limit().
|
|
|
395 |
* @todo final deprecation. To be removed in Moodle 3.10
|
|
|
396 |
* @param \DateTimeInterface $starttime The event start time
|
|
|
397 |
* @param array $max The module's maximum limit for the event
|
|
|
398 |
* @return array Returns an array with maxdaytimestamp and maxdayerror keys.
|
|
|
399 |
*/
|
|
|
400 |
protected function get_module_timestamp_max_limit(\DateTimeInterface $starttime, $max) {
|
|
|
401 |
debugging('get_module_timestamp_max_limit() has been deprecated. Please call get_timestamp_max_limit() instead.',
|
|
|
402 |
DEBUG_DEVELOPER);
|
|
|
403 |
return $this->get_timestamp_max_limit($starttime, $max);
|
|
|
404 |
}
|
|
|
405 |
}
|