3 +--------------------------------------------------------------------+
4 | Copyright CiviCRM LLC. All rights reserved. |
6 | This work is published under the GNU AGPLv3 license with some |
7 | permitted exceptions and without any warranty. For full license |
8 | and copyright information, see https://civicrm.org/licensing |
9 +--------------------------------------------------------------------+
13 * Class to generate various "icalendar" type event feeds
15 class CRM_Event_ICalendar
{
18 * Heart of the iCalendar data assignment process. The runner gets all the meta
19 * data for the event and calls the method to output the iCalendar
20 * to the user. If gData param is passed on the URL, outputs gData XML format.
21 * Else outputs iCalendar format per IETF RFC2445. Page param true means send
22 * to browser as inline content. Else, we send .ics file as attachment.
24 public static function run() {
25 $id = CRM_Utils_Request
::retrieveValue('id', 'Positive', NULL, FALSE, 'GET');
26 $type = CRM_Utils_Request
::retrieveValue('type', 'Positive', 0);
27 $start = CRM_Utils_Request
::retrieveValue('start', 'Positive', 0);
28 $end = CRM_Utils_Request
::retrieveValue('end', 'Positive', 0);
30 // We used to handle the event list as a html page at civicrm/event/ical - redirect to the new URL if that was what we requested.
31 if (CRM_Utils_Request
::retrieveValue('html', 'Positive', 0)) {
35 $id ?
$urlParams['id'] = $id : NULL;
36 $type ?
$urlParams['type'] = $type : NULL;
37 $start ?
$urlParams['start'] = $start : NULL;
38 $end ?
$urlParams['end'] = $end : NULL;
39 CRM_Utils_System
::redirect(CRM_Utils_System
::url('civicrm/event/list', $urlParams, FALSE, NULL, FALSE, TRUE));
42 $iCalPage = CRM_Utils_Request
::retrieveValue('list', 'Positive', 0);
43 $gData = CRM_Utils_Request
::retrieveValue('gData', 'Positive', 0);
44 $rss = CRM_Utils_Request
::retrieveValue('rss', 'Positive', 0);
45 $gCalendar = CRM_Utils_Request
::retrieveValue('gCalendar', 'Positive', 0);
47 $info = CRM_Event_BAO_Event
::getCompleteInfo($start, $type, $id, $end);
50 return self
::gCalRedirect($info);
53 $template = CRM_Core_Smarty
::singleton();
54 $config = CRM_Core_Config
::singleton();
56 $template->assign('events', $info);
58 $timezones = [@date_default_timezone_get
()];
60 $template->assign('timezone', $timezones[0]);
62 // Send data to the correct template for formatting (iCal vs. gData)
64 // rss 2.0 requires lower case dash delimited locale
65 $template->assign('rssLang', str_replace('_', '-', strtolower($config->lcMessages
)));
66 $calendar = $template->fetch('CRM/Core/Calendar/Rss.tpl');
69 $calendar = $template->fetch('CRM/Core/Calendar/GData.tpl');
73 array_map(function ($event) {
74 return strtotime($event['start_date']);
78 array_map(function ($event) {
79 return strtotime($event['end_date'] ??
$event['start_date']);
82 $template->assign('timezones', CRM_Utils_ICalendar
::generate_timezones($timezones, $date_min, $date_max));
83 $calendar = $template->fetch('CRM/Core/Calendar/ICal.tpl');
84 $calendar = preg_replace('/(?<!\r)\n/', "\r\n", $calendar);
87 // Push output for feed or download
90 CRM_Utils_ICalendar
::send($calendar, 'text/xml', 'utf-8');
93 CRM_Utils_ICalendar
::send($calendar, 'text/calendar', 'utf-8');
97 CRM_Utils_ICalendar
::send($calendar, 'text/calendar', 'utf-8', 'civicrm_ical.ics', 'attachment');
99 CRM_Utils_System
::civiExit();
102 protected static function gCalRedirect(array $events) {
103 if (count($events) != 1) {
104 throw new CRM_Core_Exception(ts('Expected one %1, found %2', [1 => ts('Event'), 2 => count($events)]));
107 $event = reset($events);
109 // Fetch the required Date TimeStamps
110 $start_date = date_create($event['start_date']);
112 // Google Requires that a Full Day event end day happens on the next Day
113 $end_date = ($event['end_date']
114 ?
date_create($event['end_date'])
115 : date_create($event['start_date'])
116 ->add(DateInterval
::createFromDateString('1 day'))
120 $dates = $start_date->format('Ymd\THis') . '/' . $end_date->format('Ymd\THis');
122 $event_details = $event['description'];
124 // Add space after paragraph
125 $event_details = str_replace('</p>', '</p> ', $event_details);
126 $event_details = strip_tags($event_details);
128 // Truncate Event Description and add permalink if greater than 996 characters
129 if (strlen($event_details) > 996) {
130 if (preg_match('/^.{0,996}(?=\s|$_)/', $event_details, $m)) {
131 $event_details = $m[0] . '...';
135 $event_details .= "\n\n<a href=\"{$event['url']}\">" . ts('View %1 Details', [1 => $event['event_type']]) . '</a>';
138 'action' => 'TEMPLATE',
139 'text' => strip_tags($event['title']),
141 'details' => $event_details,
142 'location' => str_replace("\n", "\t", $event['location']),
144 'sprop' => 'website:' . CRM_Utils_System
::baseCMSURL(),
145 'ctz' => @date_default_timezone_get
(),
148 $url = 'https://www.google.com/calendar/event?' . CRM_Utils_System
::makeQueryString($params);
150 CRM_Utils_System
::redirect($url);