Calendar.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. <?php
  2. /**
  3. * This file has only one real task, showing the calendar.
  4. * Original module by Aaron O'Neil - [email protected]
  5. *
  6. * Simple Machines Forum (SMF)
  7. *
  8. * @package SMF
  9. * @author Simple Machines http://www.simplemachines.org
  10. * @copyright 2011 Simple Machines
  11. * @license http://www.simplemachines.org/about/smf/license.php BSD
  12. *
  13. * @version 2.0
  14. */
  15. if (!defined('SMF'))
  16. die('Hacking attempt...');
  17. /**
  18. * Show the calendar.
  19. * It loads the specified month's events, holidays, and birthdays.
  20. * It requires the calendar_view permission.
  21. * It depends on the cal_enabled setting, and many of the other cal_ settings.
  22. * It uses the calendar_start_day theme option. (Monday/Sunday)
  23. * It uses the main sub template in the Calendar template.
  24. * It goes to the month and year passed in 'month' and 'year' by get or post.
  25. * It is accessed through ?action=calendar.
  26. */
  27. function CalendarMain()
  28. {
  29. global $txt, $context, $modSettings, $scripturl, $options, $sourcedir;
  30. // Permissions, permissions, permissions.
  31. isAllowedTo('calendar_view');
  32. // Doing something other than calendar viewing?
  33. $subActions = array(
  34. 'ical' => 'iCalDownload',
  35. 'post' => 'CalendarPost',
  36. );
  37. if (isset($_GET['sa']) && isset($subActions[$_GET['sa']]) && !WIRELESS)
  38. return $subActions[$_GET['sa']]();
  39. // This is gonna be needed...
  40. loadTemplate('Calendar');
  41. // You can't do anything if the calendar is off.
  42. if (empty($modSettings['cal_enabled']))
  43. fatal_lang_error('calendar_off', false);
  44. // Set the page title to mention the calendar ;).
  45. $context['page_title'] = $txt['calendar'];
  46. // Is this a week view?
  47. $context['view_week'] = isset($_GET['viewweek']);
  48. // Don't let search engines index weekly calendar pages.
  49. if ($context['view_week'])
  50. $context['robot_no_index'] = true;
  51. // Get the current day of month...
  52. require_once($sourcedir . '/Subs-Calendar.php');
  53. $today = getTodayInfo();
  54. // If the month and year are not passed in, use today's date as a starting point.
  55. $curPage = array(
  56. 'day' => isset($_REQUEST['day']) ? (int) $_REQUEST['day'] : $today['day'],
  57. 'month' => isset($_REQUEST['month']) ? (int) $_REQUEST['month'] : $today['month'],
  58. 'year' => isset($_REQUEST['year']) ? (int) $_REQUEST['year'] : $today['year']
  59. );
  60. // Make sure the year and month are in valid ranges.
  61. if ($curPage['month'] < 1 || $curPage['month'] > 12)
  62. fatal_lang_error('invalid_month', false);
  63. if ($curPage['year'] < $modSettings['cal_minyear'] || $curPage['year'] > $modSettings['cal_maxyear'])
  64. fatal_lang_error('invalid_year', false);
  65. // If we have a day clean that too.
  66. if ($context['view_week'])
  67. {
  68. // Note $isValid is -1 < PHP 5.1
  69. $isValid = mktime(0, 0, 0, $curPage['month'], $curPage['day'], $curPage['year']);
  70. if ($curPage['day'] > 31 || !$isValid || $isValid == -1)
  71. fatal_lang_error('invalid_day', false);
  72. }
  73. // Load all the context information needed to show the calendar grid.
  74. $calendarOptions = array(
  75. 'start_day' => !empty($options['calendar_start_day']) ? $options['calendar_start_day'] : 0,
  76. 'show_birthdays' => in_array($modSettings['cal_showbdays'], array(1, 2)),
  77. 'show_events' => in_array($modSettings['cal_showevents'], array(1, 2)),
  78. 'show_holidays' => in_array($modSettings['cal_showholidays'], array(1, 2)),
  79. 'show_week_num' => true,
  80. 'short_day_titles' => false,
  81. 'show_next_prev' => true,
  82. 'show_week_links' => true,
  83. 'size' => 'large',
  84. );
  85. // Load up the main view.
  86. if ($context['view_week'])
  87. $context['calendar_grid_main'] = getCalendarWeek($curPage['month'], $curPage['year'], $curPage['day'], $calendarOptions);
  88. else
  89. $context['calendar_grid_main'] = getCalendarGrid($curPage['month'], $curPage['year'], $calendarOptions);
  90. // Load up the previous and next months.
  91. $calendarOptions['show_birthdays'] = $calendarOptions['show_events'] = $calendarOptions['show_holidays'] = false;
  92. $calendarOptions['short_day_titles'] = true;
  93. $calendarOptions['show_next_prev'] = false;
  94. $calendarOptions['show_week_links'] = false;
  95. $calendarOptions['size'] = 'small';
  96. $context['calendar_grid_current'] = getCalendarGrid($curPage['month'], $curPage['year'], $calendarOptions);
  97. // Only show previous month if it isn't pre-January of the min-year
  98. if ($context['calendar_grid_current']['previous_calendar']['year'] > $modSettings['cal_minyear'] || $curPage['month'] != 1)
  99. $context['calendar_grid_prev'] = getCalendarGrid($context['calendar_grid_current']['previous_calendar']['month'], $context['calendar_grid_current']['previous_calendar']['year'], $calendarOptions);
  100. // Only show next month if it isn't post-December of the max-year
  101. if ($context['calendar_grid_current']['next_calendar']['year'] < $modSettings['cal_maxyear'] || $curPage['month'] != 12)
  102. $context['calendar_grid_next'] = getCalendarGrid($context['calendar_grid_current']['next_calendar']['month'], $context['calendar_grid_current']['next_calendar']['year'], $calendarOptions);
  103. // Basic template stuff.
  104. $context['can_post'] = allowedTo('calendar_post');
  105. $context['current_day'] = $curPage['day'];
  106. $context['current_month'] = $curPage['month'];
  107. $context['current_year'] = $curPage['year'];
  108. $context['show_all_birthdays'] = isset($_GET['showbd']);
  109. // Set the page title to mention the month or week, too
  110. $context['page_title'] .= ' - ' . ($context['view_week'] ? sprintf($txt['calendar_week_title'], $context['calendar_grid_main']['week_number'], ($context['calendar_grid_main']['week_number'] == 53 ? $context['current_year'] - 1 : $context['current_year'])) : $txt['months'][$context['current_month']] . ' ' . $context['current_year']);
  111. // Load up the linktree!
  112. $context['linktree'][] = array(
  113. 'url' => $scripturl . '?action=calendar',
  114. 'name' => $txt['calendar']
  115. );
  116. // Add the current month to the linktree.
  117. $context['linktree'][] = array(
  118. 'url' => $scripturl . '?action=calendar;year=' . $context['current_year'] . ';month=' . $context['current_month'],
  119. 'name' => $txt['months'][$context['current_month']] . ' ' . $context['current_year']
  120. );
  121. // If applicable, add the current week to the linktree.
  122. if ($context['view_week'])
  123. $context['linktree'][] = array(
  124. 'url' => $scripturl . '?action=calendar;viewweek;year=' . $context['current_year'] . ';month=' . $context['current_month'] . ';day=' . $context['current_day'],
  125. 'name' => $txt['calendar_week'] . ' ' . $context['calendar_grid_main']['week_number']
  126. );
  127. }
  128. /**
  129. * This function processes posting/editing/deleting a calendar event.
  130. * - calls Post() function if event is linked to a post.
  131. * - calls insertEvent() to insert the event if not linked to post.
  132. * It requires the calendar_post permission to use.
  133. * It uses the event_post sub template in the Calendar template.
  134. * It is accessed with ?action=calendar;sa=post.
  135. */
  136. function CalendarPost()
  137. {
  138. global $context, $txt, $user_info, $sourcedir, $scripturl;
  139. global $modSettings, $topic, $smcFunc;
  140. // Well - can they?
  141. isAllowedTo('calendar_post');
  142. // We need this for all kinds of useful functions.
  143. require_once($sourcedir . '/Subs-Calendar.php');
  144. // Cast this for safety...
  145. if (isset($_REQUEST['eventid']))
  146. $_REQUEST['eventid'] = (int) $_REQUEST['eventid'];
  147. // Submitting?
  148. if (isset($_POST[$context['session_var']], $_REQUEST['eventid']))
  149. {
  150. checkSession();
  151. // Validate the post...
  152. if (!isset($_POST['link_to_board']))
  153. validateEventPost();
  154. // If you're not allowed to edit any events, you have to be the poster.
  155. if ($_REQUEST['eventid'] > 0 && !allowedTo('calendar_edit_any'))
  156. isAllowedTo('calendar_edit_' . (!empty($user_info['id']) && getEventPoster($_REQUEST['eventid']) == $user_info['id'] ? 'own' : 'any'));
  157. // New - and directing?
  158. if ($_REQUEST['eventid'] == -1 && isset($_POST['link_to_board']))
  159. {
  160. $_REQUEST['calendar'] = 1;
  161. require_once($sourcedir . '/Post.php');
  162. return Post();
  163. }
  164. // New...
  165. elseif ($_REQUEST['eventid'] == -1)
  166. {
  167. $eventOptions = array(
  168. 'board' => 0,
  169. 'topic' => 0,
  170. 'title' => substr($_REQUEST['evtitle'], 0, 60),
  171. 'member' => $user_info['id'],
  172. 'start_date' => sprintf('%04d-%02d-%02d', $_POST['year'], $_POST['month'], $_POST['day']),
  173. 'span' => isset($_POST['span']) && $_POST['span'] > 0 ? min((int) $modSettings['cal_maxspan'], (int) $_POST['span'] - 1) : 0,
  174. );
  175. insertEvent($eventOptions);
  176. }
  177. // Deleting...
  178. elseif (isset($_REQUEST['deleteevent']))
  179. removeEvent($_REQUEST['eventid']);
  180. // ... or just update it?
  181. else
  182. {
  183. $eventOptions = array(
  184. 'title' => substr($_REQUEST['evtitle'], 0, 60),
  185. 'span' => empty($modSettings['cal_allowspan']) || empty($_POST['span']) || $_POST['span'] == 1 || empty($modSettings['cal_maxspan']) || $_POST['span'] > $modSettings['cal_maxspan'] ? 0 : min((int) $modSettings['cal_maxspan'], (int) $_POST['span'] - 1),
  186. 'start_date' => strftime('%Y-%m-%d', mktime(0, 0, 0, (int) $_REQUEST['month'], (int) $_REQUEST['day'], (int) $_REQUEST['year'])),
  187. );
  188. modifyEvent($_REQUEST['eventid'], $eventOptions);
  189. }
  190. updateSettings(array(
  191. 'calendar_updated' => time(),
  192. ));
  193. // No point hanging around here now...
  194. redirectexit($scripturl . '?action=calendar;month=' . $_POST['month'] . ';year=' . $_POST['year']);
  195. }
  196. // If we are not enabled... we are not enabled.
  197. if (empty($modSettings['cal_allow_unlinked']) && empty($_REQUEST['eventid']))
  198. {
  199. $_REQUEST['calendar'] = 1;
  200. require_once($sourcedir . '/Post.php');
  201. return Post();
  202. }
  203. // New?
  204. if (!isset($_REQUEST['eventid']))
  205. {
  206. $today = getdate();
  207. $context['event'] = array(
  208. 'boards' => array(),
  209. 'board' => 0,
  210. 'new' => 1,
  211. 'eventid' => -1,
  212. 'year' => isset($_REQUEST['year']) ? $_REQUEST['year'] : $today['year'],
  213. 'month' => isset($_REQUEST['month']) ? $_REQUEST['month'] : $today['mon'],
  214. 'day' => isset($_REQUEST['day']) ? $_REQUEST['day'] : $today['mday'],
  215. 'title' => '',
  216. 'span' => 1,
  217. );
  218. $context['event']['last_day'] = (int) strftime('%d', mktime(0, 0, 0, $context['event']['month'] == 12 ? 1 : $context['event']['month'] + 1, 0, $context['event']['month'] == 12 ? $context['event']['year'] + 1 : $context['event']['year']));
  219. // Get list of boards that can be posted in.
  220. $boards = boardsAllowedTo('post_new');
  221. if (empty($boards))
  222. fatal_lang_error('cannot_post_new', 'permission');
  223. // Load the list of boards and categories in the context.
  224. require_once($sourcedir . '/Subs-MessageIndex.php');
  225. $boardListOptions = array(
  226. 'included_boards' => in_array(0, $boards) ? null : $boards,
  227. 'not_redirection' => true,
  228. 'use_permissions' => true,
  229. 'selected_board' => $modSettings['cal_defaultboard'],
  230. );
  231. $context['event']['categories'] = getBoardList($boardListOptions);
  232. }
  233. else
  234. {
  235. $context['event'] = getEventProperties($_REQUEST['eventid']);
  236. if ($context['event'] === false)
  237. fatal_lang_error('no_access', false);
  238. // If it has a board, then they should be editing it within the topic.
  239. if (!empty($context['event']['topic']['id']) && !empty($context['event']['topic']['first_msg']))
  240. {
  241. // We load the board up, for a check on the board access rights...
  242. $topic = $context['event']['topic']['id'];
  243. loadBoard();
  244. }
  245. // Make sure the user is allowed to edit this event.
  246. if ($context['event']['member'] != $user_info['id'])
  247. isAllowedTo('calendar_edit_any');
  248. elseif (!allowedTo('calendar_edit_any'))
  249. isAllowedTo('calendar_edit_own');
  250. }
  251. // Template, sub template, etc.
  252. loadTemplate('Calendar');
  253. $context['sub_template'] = 'event_post';
  254. $context['page_title'] = isset($_REQUEST['eventid']) ? $txt['calendar_edit'] : $txt['calendar_post_event'];
  255. $context['linktree'][] = array(
  256. 'name' => $context['page_title'],
  257. );
  258. }
  259. /**
  260. * This function offers up a download of an event in iCal 2.0 format.
  261. * @todo lol... User interface! :P
  262. */
  263. function iCalDownload()
  264. {
  265. global $smcFunc, $sourcedir, $forum_version, $context, $modSettings;
  266. // Goes without saying that this is required.
  267. if (!isset($_REQUEST['eventid']))
  268. fatal_lang_error('no_access', false);
  269. // This is kinda wanted.
  270. require_once($sourcedir . '/Subs-Calendar.php');
  271. // Load up the event in question and check it exists.
  272. $event = getEventProperties($_REQUEST['eventid']);
  273. if ($event === false)
  274. fatal_lang_error('no_access', false);
  275. // Check the title isn't too long - iCal requires some formatting if so.
  276. $title = str_split($event['title'], 30);
  277. foreach ($title as $id => $line)
  278. {
  279. if ($id != 0)
  280. $title[$id] = ' ' . $title[$id];
  281. $title[$id] .= "\n";
  282. }
  283. // Format the date.
  284. $date = $event['year'] . '-' . ($event['month'] < 10 ? '0' . $event['month'] : $event['month']) . '-' . ($event['day'] < 10 ? '0' . $event['day'] : $event['day']) . 'T';
  285. $date .= '1200:00:00Z';
  286. // This is what we will be sending later.
  287. $filecontents = '';
  288. $filecontents .= 'BEGIN:VCALENDAR' . "\n";
  289. $filecontents .= 'VERSION:2.0' . "\n";
  290. $filecontents .= 'PRODID:-//SimpleMachines//SMF ' . (empty($forum_version) ? 1.0 : strtr($forum_version, array('SMF ' => ''))) . '//EN' . "\n";
  291. $filecontents .= 'BEGIN:VEVENT' . "\n";
  292. $filecontents .= 'DTSTART:' . $date . "\n";
  293. $filecontents .= 'DTEND:' . $date . "\n";
  294. $filecontents .= 'SUMMARY:' . implode('', $title);
  295. $filecontents .= 'END:VEVENT' . "\n";
  296. $filecontents .= 'END:VCALENDAR';
  297. // Send some standard headers.
  298. ob_end_clean();
  299. if (!empty($modSettings['enableCompressedOutput']))
  300. @ob_start('ob_gzhandler');
  301. else
  302. ob_start();
  303. // Send the file headers
  304. header('Pragma: ');
  305. header('Cache-Control: no-cache');
  306. if (!$context['browser']['is_gecko'])
  307. header('Content-Transfer-Encoding: binary');
  308. header('Expires: ' . gmdate('D, d M Y H:i:s', time() + 525600 * 60) . ' GMT');
  309. header('Last-Modified: ' . gmdate('D, d M Y H:i:s', time()) . 'GMT');
  310. header('Accept-Ranges: bytes');
  311. header('Connection: close');
  312. header('Content-Disposition: attachment; filename=' . $event['title'] . '.ics');
  313. // How big is it?
  314. if (empty($modSettings['enableCompressedOutput']))
  315. header('Content-Length: ' . $smcFunc['strlen']($filecontents));
  316. // This is a calendar item!
  317. header('Content-Type: text/calendar');
  318. // Chuck out the card.
  319. echo $filecontents;
  320. // Off we pop - lovely!
  321. obExit(false);
  322. }
  323. /**
  324. * Nothing to see here. Move along.
  325. */
  326. function clock()
  327. {
  328. global $settings, $context;
  329. $context['onimg'] = $settings['images_url'] . '/bbc/bbc_bg.gif';
  330. $context['offimg'] = $settings['images_url'] . '/bbc/bbc_hoverbg.gif';
  331. $context['page_title'] = 'Anyone know what time it is?';
  332. $context['robot_no_index'] = true;
  333. $omfg = isset($_REQUEST['omfg']);
  334. $bcd = !isset($_REQUEST['rb']) && !isset($_REQUEST['omfg']) && !isset($_REQUEST['time']);
  335. loadTemplate('Calendar');
  336. if ($bcd && !$omfg)
  337. {
  338. $context['sub_template'] = 'bcd';
  339. $context['clockicons'] = unserialize(base64_decode('YTo2OntzOjI6ImgxIjthOjI6e2k6MDtpOjI7aToxO2k6MTt9czoyOiJoMiI7YTo0OntpOjA7aTo4O2k6MTtpOjQ7aToyO2k6MjtpOjM7aToxO31zOjI6Im0xIjthOjM6e2k6MDtpOjQ7aToxO2k6MjtpOjI7aToxO31zOjI6Im0yIjthOjQ6e2k6MDtpOjg7aToxO2k6NDtpOjI7aToyO2k6MztpOjE7fXM6MjoiczEiO2E6Mzp7aTowO2k6NDtpOjE7aToyO2k6MjtpOjE7fXM6MjoiczIiO2E6NDp7aTowO2k6ODtpOjE7aTo0O2k6MjtpOjI7aTozO2k6MTt9fQ=='));
  340. }
  341. elseif (!$omfg && !isset($_REQUEST['time']))
  342. {
  343. $context['sub_template'] = 'hms';
  344. $context['clockicons'] = unserialize(base64_decode('YTozOntzOjE6ImgiO2E6NTp7aTowO2k6MTY7aToxO2k6ODtpOjI7aTo0O2k6MztpOjI7aTo0O2k6MTt9czoxOiJtIjthOjY6e2k6MDtpOjMyO2k6MTtpOjE2O2k6MjtpOjg7aTozO2k6NDtpOjQ7aToyO2k6NTtpOjE7fXM6MToicyI7YTo2OntpOjA7aTozMjtpOjE7aToxNjtpOjI7aTo4O2k6MztpOjQ7aTo0O2k6MjtpOjU7aToxO319'));
  345. }
  346. elseif ($omfg)
  347. {
  348. $context['sub_template'] = 'omfg';
  349. $context['clockicons'] = unserialize(base64_decode('YTo2OntzOjQ6InllYXIiO2E6Nzp7aTowO2k6NjQ7aToxO2k6MzI7aToyO2k6MTY7aTozO2k6ODtpOjQ7aTo0O2k6NTtpOjI7aTo2O2k6MTt9czo1OiJtb250aCI7YTo0OntpOjA7aTo4O2k6MTtpOjQ7aToyO2k6MjtpOjM7aToxO31zOjM6ImRheSI7YTo1OntpOjA7aToxNjtpOjE7aTo4O2k6MjtpOjQ7aTozO2k6MjtpOjQ7aToxO31zOjQ6ImhvdXIiO2E6NTp7aTowO2k6MTY7aToxO2k6ODtpOjI7aTo0O2k6MztpOjI7aTo0O2k6MTt9czozOiJtaW4iO2E6Njp7aTowO2k6MzI7aToxO2k6MTY7aToyO2k6ODtpOjM7aTo0O2k6NDtpOjI7aTo1O2k6MTt9czozOiJzZWMiO2E6Njp7aTowO2k6MzI7aToxO2k6MTY7aToyO2k6ODtpOjM7aTo0O2k6NDtpOjI7aTo1O2k6MTt9fQ=='));
  350. }
  351. elseif (isset($_REQUEST['time']))
  352. {
  353. $context['sub_template'] = 'thetime';
  354. $time = getdate($_REQUEST['time'] == 'now' ? time() : (int) $_REQUEST['time']);
  355. $context['clockicons'] = array(
  356. 'year' => array(
  357. 64 => false,
  358. 32 => false,
  359. 16 => false,
  360. 8 => false,
  361. 4 => false,
  362. 2 => false,
  363. 1 => false
  364. ),
  365. 'month' => array(
  366. 8 => false,
  367. 4 => false,
  368. 2 => false,
  369. 1 => false
  370. ),
  371. 'day' => array(
  372. 16 => false,
  373. 4 => false,
  374. 8 => false,
  375. 2 => false,
  376. 1 => false
  377. ),
  378. 'hour' => array(
  379. 32 => false,
  380. 16 => false,
  381. 8 => false,
  382. 4 => false,
  383. 2 => false,
  384. 1 => false
  385. ),
  386. 'min' => array(
  387. 32 => false,
  388. 16 => false,
  389. 8 => false,
  390. 4 => false,
  391. 2 => false,
  392. 1 => false
  393. ),
  394. 'sec' => array(
  395. 32 => false,
  396. 16 => false,
  397. 8 => false,
  398. 4 => false,
  399. 2 => false,
  400. 1 => false
  401. ),
  402. );
  403. $year = $time['year'] % 100;
  404. $month = $time['mon'];
  405. $day = $time['mday'];
  406. $hour = $time['hours'];
  407. $min = $time['minutes'];
  408. $sec = $time['seconds'];
  409. foreach ($context['clockicons'] as $t => $vs)
  410. foreach ($vs as $v => $dumb)
  411. {
  412. if ($$t >= $v)
  413. {
  414. $$t -= $v;
  415. $context['clockicons'][$t][$v] = true;
  416. }
  417. }
  418. }
  419. }
  420. ?>