+++ /dev/null
-<?php
-/**
- * @copyright Copyright (C) 2010-2022, the Friendica project
- *
- * @license GNU AGPL version 3 or any later version
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Affero General Public License as
- * published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * This program 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 Affero General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- *
- * The calendar module
- *
- * This calendar is for profile visitors and contains only the events
- * of the profile owner
- */
-
-use Friendica\App;
-use Friendica\Content\Nav;
-use Friendica\Content\Widget;
-use Friendica\Core\Renderer;
-use Friendica\Core\System;
-use Friendica\Database\DBA;
-use Friendica\DI;
-use Friendica\Model\Event;
-use Friendica\Model\Item;
-use Friendica\Model\User;
-use Friendica\Module\BaseProfile;
-use Friendica\Network\HTTPException;
-use Friendica\Util\DateTimeFormat;
-use Friendica\Util\Temporal;
-
-function cal_init(App $a)
-{
- if (DI::config()->get('system', 'block_public') && !DI::userSession()->isAuthenticated()) {
- throw new HTTPException\ForbiddenException(DI::l10n()->t('Access denied.'));
- }
-
- if (DI::args()->getArgc() < 2) {
- throw new HTTPException\ForbiddenException(DI::l10n()->t('Access denied.'));
- }
-
- Nav::setSelected('events');
-
- // if it's a json request abort here becaus we don't
- // need the widget data
- if (!empty(DI::args()->getArgv()[2]) && (DI::args()->getArgv()[2] === 'json')) {
- return;
- }
-
- $owner = User::getOwnerDataByNick(DI::args()->getArgv()[1]);
- if (empty($owner)) {
- throw new HTTPException\NotFoundException(DI::l10n()->t('User not found.'));
- }
-
- if (empty(DI::page()['aside'])) {
- DI::page()['aside'] = '';
- }
-
- DI::page()['aside'] .= Widget\VCard::getHTML($owner);
- DI::page()['aside'] .= Widget\CalendarExport::getHTML($owner['uid']);
-
- return;
-}
-
-function cal_content(App $a)
-{
- $owner = User::getOwnerDataByNick(DI::args()->getArgv()[1]);
- if (empty($owner)) {
- throw new HTTPException\NotFoundException(DI::l10n()->t('User not found.'));
- }
-
- Nav::setSelected('events');
-
- // get the translation strings for the callendar
- $i18n = Event::getStrings();
-
- DI::page()->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.min.css');
- DI::page()->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.print.min.css', 'print');
- DI::page()->registerFooterScript('view/asset/moment/min/moment-with-locales.min.js');
- DI::page()->registerFooterScript('view/asset/fullcalendar/dist/fullcalendar.min.js');
-
- $htpl = Renderer::getMarkupTemplate('event_head.tpl');
- DI::page()['htmlhead'] .= Renderer::replaceMacros($htpl, [
- '$module_url' => '/cal/' . $owner['nickname'],
- '$modparams' => 2,
- '$i18n' => $i18n,
- ]);
-
- $mode = 'view';
- $y = 0;
- $m = 0;
- $ignored = (!empty($_REQUEST['ignored']) ? intval($_REQUEST['ignored']) : 0);
-
- // Setup permissions structures
- $owner_uid = intval($owner['uid']);
-
- $contact_id = DI::userSession()->getRemoteContactID($owner['uid']);
-
- $remote_contact = $contact_id && DBA::exists('contact', ['id' => $contact_id, 'uid' => $owner['uid']]);
-
- $is_owner = DI::userSession()->getLocalUserId() == $owner['uid'];
-
- if ($owner['hidewall'] && !$is_owner && !$remote_contact) {
- DI::sysmsg()->addNotice(DI::l10n()->t('Access to this profile has been restricted.'));
- return;
- }
-
- // get the permissions
- $sql_perms = Item::getPermissionsSQLByUserId($owner_uid);
- // we only want to have the events of the profile owner
- $sql_extra = " AND `event`.`cid` = 0 " . $sql_perms;
-
- // get the tab navigation bar
- $tabs = BaseProfile::getTabsHTML($a, 'cal', false, $owner['nickname'], $owner['hide-friends']);
-
- // The view mode part is similiar to /mod/events.php
- if ($mode == 'view') {
- $thisyear = DateTimeFormat::localNow('Y');
- $thismonth = DateTimeFormat::localNow('m');
- if (!$y) {
- $y = intval($thisyear);
- }
-
- if (!$m) {
- $m = intval($thismonth);
- }
-
- // Put some limits on dates. The PHP date functions don't seem to do so well before 1900.
- // An upper limit was chosen to keep search engines from exploring links millions of years in the future.
-
- if ($y < 1901) {
- $y = 1900;
- }
-
- if ($y > 2099) {
- $y = 2100;
- }
-
- $nextyear = $y;
- $nextmonth = $m + 1;
- if ($nextmonth > 12) {
- $nextmonth = 1;
- $nextyear ++;
- }
-
- $prevyear = $y;
- if ($m > 1) {
- $prevmonth = $m - 1;
- } else {
- $prevmonth = 12;
- $prevyear --;
- }
-
- $dim = Temporal::getDaysInMonth($y, $m);
- $start = sprintf('%d-%d-%d %d:%d:%d', $y, $m, 1, 0, 0, 0);
- $finish = sprintf('%d-%d-%d %d:%d:%d', $y, $m, $dim, 23, 59, 59);
-
-
- if (!empty(DI::args()->getArgv()[2]) && (DI::args()->getArgv()[2] === 'json')) {
- if (!empty($_GET['start'])) {
- $start = $_GET['start'];
- }
-
- if (!empty($_GET['end'])) {
- $finish = $_GET['end'];
- }
- }
-
- $start = DateTimeFormat::utc($start);
- $finish = DateTimeFormat::utc($finish);
-
- // put the event parametes in an array so we can better transmit them
- $event_params = [
- 'event_id' => intval($_GET['id'] ?? 0),
- 'start' => $start,
- 'finish' => $finish,
- 'ignore' => $ignored,
- ];
-
- // get events by id or by date
- if ($event_params['event_id']) {
- $r = Event::getListById($owner_uid, $event_params['event_id'], $sql_extra);
- } else {
- $r = Event::getListByDate($owner_uid, $event_params, $sql_extra);
- }
-
- $links = [];
-
- if (DBA::isResult($r)) {
- $r = Event::sortByDate($r);
- foreach ($r as $rr) {
- $j = DateTimeFormat::local($rr['start'], 'j');
- if (empty($links[$j])) {
- $links[$j] = DI::baseUrl() . '/' . DI::args()->getCommand() . '#link-' . $j;
- }
- }
- }
-
- // transform the event in a usable array
- $events = Event::prepareListForTemplate($r);
-
- if (!empty(DI::args()->getArgv()[2]) && (DI::args()->getArgv()[2] === 'json')) {
- System::jsonExit($events);
- }
-
- // links: array('href', 'text', 'extra css classes', 'title')
- if (!empty($_GET['id'])) {
- $tpl = Renderer::getMarkupTemplate("event.tpl");
- } else {
- $tpl = Renderer::getMarkupTemplate("events_js.tpl");
- }
-
- // Get rid of dashes in key names, Smarty3 can't handle them
- foreach ($events as $key => $event) {
- $event_item = [];
- foreach ($event['item'] as $k => $v) {
- $k = str_replace('-', '_', $k);
- $event_item[$k] = $v;
- }
- $events[$key]['item'] = $event_item;
- }
-
- $o = Renderer::replaceMacros($tpl, [
- '$tabs' => $tabs,
- '$title' => DI::l10n()->t('Events'),
- '$view' => DI::l10n()->t('View'),
- '$previous' => [DI::baseUrl() . "/events/$prevyear/$prevmonth", DI::l10n()->t('Previous'), '', ''],
- '$next' => [DI::baseUrl() . "/events/$nextyear/$nextmonth", DI::l10n()->t('Next'), '', ''],
- '$calendar' => Temporal::getCalendarTable($y, $m, $links, ' eventcal'),
- '$events' => $events,
- "today" => DI::l10n()->t("today"),
- "month" => DI::l10n()->t("month"),
- "week" => DI::l10n()->t("week"),
- "day" => DI::l10n()->t("day"),
- "list" => DI::l10n()->t("list"),
- ]);
-
- if (!empty($_GET['id'])) {
- System::httpExit($o);
- }
-
- return $o;
- }
-}
+++ /dev/null
-<?php
-/**
- * @copyright Copyright (C) 2010-2022, the Friendica project
- *
- * @license GNU AGPL version 3 or any later version
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Affero General Public License as
- * published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * This program 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 Affero General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- *
- * The events module
- */
-
-use Friendica\App;
-use Friendica\Content\Nav;
-use Friendica\Content\Widget\CalendarExport;
-use Friendica\Core\ACL;
-use Friendica\Core\Logger;
-use Friendica\Core\Protocol;
-use Friendica\Core\Renderer;
-use Friendica\Core\System;
-use Friendica\Core\Theme;
-use Friendica\Core\Worker;
-use Friendica\Database\DBA;
-use Friendica\DI;
-use Friendica\Model\Conversation;
-use Friendica\Model\Event;
-use Friendica\Model\Item;
-use Friendica\Model\Post;
-use Friendica\Model\User;
-use Friendica\Module\BaseProfile;
-use Friendica\Module\Security\Login;
-use Friendica\Util\DateTimeFormat;
-use Friendica\Util\Strings;
-use Friendica\Util\Temporal;
-use Friendica\Worker\Delivery;
-
-function events_init(App $a)
-{
- if (!DI::userSession()->getLocalUserId()) {
- return;
- }
-
- if (empty(DI::page()['aside'])) {
- DI::page()['aside'] = '';
- }
-
- $cal_widget = CalendarExport::getHTML(DI::userSession()->getLocalUserId());
-
- DI::page()['aside'] .= $cal_widget;
-
- return;
-}
-
-function events_post(App $a)
-{
- Logger::debug('post', ['request' => $_REQUEST]);
- if (!DI::userSession()->getLocalUserId()) {
- return;
- }
-
- $event_id = !empty($_POST['event_id']) ? intval($_POST['event_id']) : 0;
- $cid = !empty($_POST['cid']) ? intval($_POST['cid']) : 0;
- $uid = DI::userSession()->getLocalUserId();
-
- $start_text = Strings::escapeHtml($_REQUEST['start_text'] ?? '');
- $finish_text = Strings::escapeHtml($_REQUEST['finish_text'] ?? '');
-
- $nofinish = intval($_POST['nofinish'] ?? 0);
-
- $share = intval($_POST['share'] ?? 0);
-
- // The default setting for the `private` field in event_store() is false, so mirror that
- $private_event = false;
-
- $start = DBA::NULL_DATETIME;
- $finish = DBA::NULL_DATETIME;
-
- if ($start_text) {
- $start = $start_text;
- }
-
- if ($finish_text) {
- $finish = $finish_text;
- }
-
- $start = DateTimeFormat::convert($start, 'UTC', $a->getTimeZone());
- if (!$nofinish) {
- $finish = DateTimeFormat::convert($finish, 'UTC', $a->getTimeZone());
- }
-
- // Don't allow the event to finish before it begins.
- // It won't hurt anything, but somebody will file a bug report
- // and we'll waste a bunch of time responding to it. Time that
- // could've been spent doing something else.
-
- $summary = trim($_POST['summary'] ?? '');
- $desc = trim($_POST['desc'] ?? '');
- $location = trim($_POST['location'] ?? '');
- $type = 'event';
-
- $params = [
- 'summary' => $summary,
- 'description' => $desc,
- 'location' => $location,
- 'start' => $start_text,
- 'finish' => $finish_text,
- 'nofinish' => $nofinish,
- ];
-
- $action = ($event_id == '') ? 'new' : 'event/' . $event_id;
- $onerror_path = 'events/' . $action . '?' . http_build_query($params, '', '&', PHP_QUERY_RFC3986);
-
- if (strcmp($finish, $start) < 0 && !$nofinish) {
- DI::sysmsg()->addNotice(DI::l10n()->t('Event can not end before it has started.'));
- if (intval($_REQUEST['preview'])) {
- System::httpExit(DI::l10n()->t('Event can not end before it has started.'));
- }
- DI::baseUrl()->redirect($onerror_path);
- }
-
- if (!$summary || ($start === DBA::NULL_DATETIME)) {
- DI::sysmsg()->addNotice(DI::l10n()->t('Event title and start time are required.'));
- if (intval($_REQUEST['preview'])) {
- System::httpExit(DI::l10n()->t('Event title and start time are required.'));
- }
- DI::baseUrl()->redirect($onerror_path);
- }
-
- $self = \Friendica\Model\Contact::getPublicIdByUserId($uid);
-
- $aclFormatter = DI::aclFormatter();
-
- if ($share) {
- $user = User::getById($uid, ['allow_cid', 'allow_gid', 'deny_cid', 'deny_gid']);
- if (!DBA::isResult($user)) {
- return;
- }
-
- $str_contact_allow = isset($_REQUEST['contact_allow']) ? $aclFormatter->toString($_REQUEST['contact_allow']) : $user['allow_cid'] ?? '';
- $str_group_allow = isset($_REQUEST['group_allow']) ? $aclFormatter->toString($_REQUEST['group_allow']) : $user['allow_gid'] ?? '';
- $str_contact_deny = isset($_REQUEST['contact_deny']) ? $aclFormatter->toString($_REQUEST['contact_deny']) : $user['deny_cid'] ?? '';
- $str_group_deny = isset($_REQUEST['group_deny']) ? $aclFormatter->toString($_REQUEST['group_deny']) : $user['deny_gid'] ?? '';
-
- $visibility = $_REQUEST['visibility'] ?? '';
- if ($visibility === 'public') {
- // The ACL selector introduced in version 2019.12 sends ACL input data even when the Public visibility is selected
- $str_contact_allow = $str_group_allow = $str_contact_deny = $str_group_deny = '';
- } else if ($visibility === 'custom') {
- // Since we know from the visibility parameter the item should be private, we have to prevent the empty ACL
- // case that would make it public. So we always append the author's contact id to the allowed contacts.
- // See https://github.com/friendica/friendica/issues/9672
- $str_contact_allow .= $aclFormatter->toString($self);
- }
- } else {
- $str_contact_allow = $aclFormatter->toString($self);
- $str_group_allow = $str_contact_deny = $str_group_deny = '';
- }
-
- // Make sure to set the `private` field as true. This is necessary to
- // have the posts show up correctly in Diaspora if an event is created
- // as visible only to self at first, but then edited to display to others.
- if (strlen($str_group_allow) || strlen($str_contact_allow) || strlen($str_group_deny) || strlen($str_contact_deny)) {
- $private_event = true;
- }
-
- $datarray = [
- 'start' => $start,
- 'finish' => $finish,
- 'summary' => $summary,
- 'desc' => $desc,
- 'location' => $location,
- 'type' => $type,
- 'nofinish' => $nofinish,
- 'uid' => $uid,
- 'cid' => $cid,
- 'allow_cid' => $str_contact_allow,
- 'allow_gid' => $str_group_allow,
- 'deny_cid' => $str_contact_deny,
- 'deny_gid' => $str_group_deny,
- 'private' => $private_event,
- 'id' => $event_id,
- ];
-
- if (intval($_REQUEST['preview'])) {
- System::httpExit(Event::getHTML($datarray));
- }
-
- $event_id = Event::store($datarray);
-
- $item = ['network' => Protocol::DFRN, 'protocol' => Conversation::PARCEL_DIRECT, 'direction' => Conversation::PUSH];
- $item = Event::getItemArrayForId($event_id, $item);
- if (Item::insert($item)) {
- $uri_id = $item['uri-id'];
- } else {
- $uri_id = 0;
- }
-
- if (!$cid && $uri_id) {
- Worker::add(Worker::PRIORITY_HIGH, "Notifier", Delivery::POST, (int)$uri_id, (int)$uid);
- }
-
- DI::baseUrl()->redirect('events');
-}
-
-function events_content(App $a)
-{
- if (!DI::userSession()->getLocalUserId()) {
- DI::sysmsg()->addNotice(DI::l10n()->t('Permission denied.'));
- return Login::form();
- }
-
- if (DI::args()->getArgc() == 1) {
- $_SESSION['return_path'] = DI::args()->getCommand();
- }
-
- if ((DI::args()->getArgc() > 2) && (DI::args()->getArgv()[1] === 'ignore') && intval(DI::args()->getArgv()[2])) {
- DBA::update('event', ['ignore' => true], ['id' => DI::args()->getArgv()[2], 'uid' => DI::userSession()->getLocalUserId()]);
- }
-
- if ((DI::args()->getArgc() > 2) && (DI::args()->getArgv()[1] === 'unignore') && intval(DI::args()->getArgv()[2])) {
- DBA::update('event', ['ignore' => false], ['id' => DI::args()->getArgv()[2], 'uid' => DI::userSession()->getLocalUserId()]);
- }
-
- if ($a->getThemeInfoValue('events_in_profile')) {
- Nav::setSelected('home');
- } else {
- Nav::setSelected('events');
- }
-
- // get the translation strings for the callendar
- $i18n = Event::getStrings();
-
- DI::page()->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.min.css');
- DI::page()->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.print.min.css', 'print');
- DI::page()->registerFooterScript('view/asset/moment/min/moment-with-locales.min.js');
- DI::page()->registerFooterScript('view/asset/fullcalendar/dist/fullcalendar.min.js');
-
- $htpl = Renderer::getMarkupTemplate('event_head.tpl');
- DI::page()['htmlhead'] .= Renderer::replaceMacros($htpl, [
- '$module_url' => '/events',
- '$modparams' => 1,
- '$i18n' => $i18n,
- ]);
-
- $o = '';
- $tabs = '';
- // tabs
- if ($a->getThemeInfoValue('events_in_profile')) {
- $tabs = BaseProfile::getTabsHTML($a, 'events', true, $a->getLoggedInUserNickname(), false);
- }
-
- $mode = 'view';
- $y = 0;
- $m = 0;
- $ignored = !empty($_REQUEST['ignored']) ? intval($_REQUEST['ignored']) : 0;
-
- if (DI::args()->getArgc() > 1) {
- if (DI::args()->getArgc() > 2 && DI::args()->getArgv()[1] == 'event') {
- $mode = 'edit';
- $event_id = intval(DI::args()->getArgv()[2]);
- }
- if (DI::args()->getArgc() > 2 && DI::args()->getArgv()[1] == 'drop') {
- $mode = 'drop';
- $event_id = intval(DI::args()->getArgv()[2]);
- }
- if (DI::args()->getArgc() > 2 && DI::args()->getArgv()[1] == 'copy') {
- $mode = 'copy';
- $event_id = intval(DI::args()->getArgv()[2]);
- }
- if (DI::args()->getArgv()[1] === 'new') {
- $mode = 'new';
- $event_id = 0;
- }
- if (DI::args()->getArgc() > 2 && intval(DI::args()->getArgv()[1]) && intval(DI::args()->getArgv()[2])) {
- $mode = 'view';
- $y = intval(DI::args()->getArgv()[1]);
- $m = intval(DI::args()->getArgv()[2]);
- }
- }
-
- // The view mode part is similiar to /mod/cal.php
- if ($mode == 'view') {
- $thisyear = DateTimeFormat::localNow('Y');
- $thismonth = DateTimeFormat::localNow('m');
- if (!$y) {
- $y = intval($thisyear);
- }
- if (!$m) {
- $m = intval($thismonth);
- }
-
- // Put some limits on dates. The PHP date functions don't seem to do so well before 1900.
- // An upper limit was chosen to keep search engines from exploring links millions of years in the future.
-
- if ($y < 1901) {
- $y = 1900;
- }
- if ($y > 2099) {
- $y = 2100;
- }
-
- $dim = Temporal::getDaysInMonth($y, $m);
- $start = sprintf('%d-%d-%d %d:%d:%d', $y, $m, 1, 0, 0, 0);
- $finish = sprintf('%d-%d-%d %d:%d:%d', $y, $m, $dim, 23, 59, 59);
-
- // put the event parametes in an array so we can better transmit them
- $event_params = [
- 'event_id' => intval($_GET['id'] ?? 0),
- 'start' => $start,
- 'finish' => $finish,
- 'ignore' => $ignored,
- ];
-
- // get events by id or by date
- if ($event_params['event_id']) {
- $r = Event::getListById(DI::userSession()->getLocalUserId(), $event_params['event_id']);
- } else {
- $r = Event::getListByDate(DI::userSession()->getLocalUserId(), $event_params);
- }
-
- $links = [];
-
- if (DBA::isResult($r)) {
- $r = Event::sortByDate($r);
- foreach ($r as $rr) {
- $j = DateTimeFormat::local($rr['start'], 'j');
- if (empty($links[$j])) {
- $links[$j] = DI::baseUrl() . '/' . DI::args()->getCommand() . '#link-' . $j;
- }
- }
- }
-
- $events = [];
-
- // transform the event in a usable array
- if (DBA::isResult($r)) {
- $r = Event::sortByDate($r);
- $events = Event::prepareListForTemplate($r);
- }
-
- if (!empty($_GET['id'])) {
- $tpl = Renderer::getMarkupTemplate("event.tpl");
- } else {
- $tpl = Renderer::getMarkupTemplate("events_js.tpl");
- }
-
- // Get rid of dashes in key names, Smarty3 can't handle them
- foreach ($events as $key => $event) {
- $event_item = [];
- foreach ($event['item'] as $k => $v) {
- $k = str_replace('-', '_', $k);
- $event_item[$k] = $v;
- }
- $events[$key]['item'] = $event_item;
- }
-
- // ACL blocks are loaded in modals in frio
- DI::page()->registerFooterScript(Theme::getPathForFile('asset/typeahead.js/dist/typeahead.bundle.js'));
- DI::page()->registerFooterScript(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.js'));
- DI::page()->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.css'));
- DI::page()->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput-typeahead.css'));
-
- $o = Renderer::replaceMacros($tpl, [
- '$tabs' => $tabs,
- '$title' => DI::l10n()->t('Events'),
- '$view' => DI::l10n()->t('View'),
- '$new_event' => [DI::baseUrl() . '/events/new', DI::l10n()->t('Create New Event'), '', ''],
- '$previous' => [DI::baseUrl() . '/events/$prevyear/$prevmonth', DI::l10n()->t('Previous'), '', ''],
- '$next' => [DI::baseUrl() . '/events/$nextyear/$nextmonth', DI::l10n()->t('Next'), '', ''],
- '$calendar' => Temporal::getCalendarTable($y, $m, $links, ' eventcal'),
-
- '$events' => $events,
-
- '$today' => DI::l10n()->t('today'),
- '$month' => DI::l10n()->t('month'),
- '$week' => DI::l10n()->t('week'),
- '$day' => DI::l10n()->t('day'),
- '$list' => DI::l10n()->t('list'),
- ]);
-
- if (!empty($_GET['id'])) {
- System::httpExit($o);
- }
-
- return $o;
- }
-
- if (($mode === 'edit' || $mode === 'copy') && $event_id) {
- $orig_event = DBA::selectFirst('event', [], ['id' => $event_id, 'uid' => DI::userSession()->getLocalUserId()]);
- }
-
- // Passed parameters overrides anything found in the DB
- if (in_array($mode, ['edit', 'new', 'copy'])) {
- $share_checked = '';
- $share_disabled = '';
-
- if (empty($orig_event)) {
- $orig_event = User::getById(DI::userSession()->getLocalUserId(), ['allow_cid', 'allow_gid', 'deny_cid', 'deny_gid']);;
- } elseif ($orig_event['allow_cid'] !== '<' . DI::userSession()->getLocalUserId() . '>'
- || $orig_event['allow_gid']
- || $orig_event['deny_cid']
- || $orig_event['deny_gid']) {
- $share_checked = ' checked="checked" ';
- }
-
- // In case of an error the browser is redirected back here, with these parameters filled in with the previous values
- if (!empty($_REQUEST['nofinish'])) {$orig_event['nofinish'] = $_REQUEST['nofinish'];}
- if (!empty($_REQUEST['summary'])) {$orig_event['summary'] = $_REQUEST['summary'];}
- if (!empty($_REQUEST['desc'])) {$orig_event['desc'] = $_REQUEST['desc'];}
- if (!empty($_REQUEST['location'])) {$orig_event['location'] = $_REQUEST['location'];}
- if (!empty($_REQUEST['start'])) {$orig_event['start'] = $_REQUEST['start'];}
- if (!empty($_REQUEST['finish'])) {$orig_event['finish'] = $_REQUEST['finish'];}
-
- $n_checked = (!empty($orig_event['nofinish']) ? ' checked="checked" ' : '');
-
- $t_orig = $orig_event['summary'] ?? '';
- $d_orig = $orig_event['desc'] ?? '';
- $l_orig = $orig_event['location'] ?? '';
- $eid = $orig_event['id'] ?? 0;
- $cid = $orig_event['cid'] ?? 0;
- $uri = $orig_event['uri'] ?? '';
-
- if ($cid || $mode === 'edit') {
- $share_disabled = 'disabled="disabled"';
- }
-
- $sdt = $orig_event['start'] ?? 'now';
- $fdt = $orig_event['finish'] ?? 'now';
-
- $syear = DateTimeFormat::local($sdt, 'Y');
- $smonth = DateTimeFormat::local($sdt, 'm');
- $sday = DateTimeFormat::local($sdt, 'd');
-
- $shour = !empty($orig_event) ? DateTimeFormat::local($sdt, 'H') : '00';
- $sminute = !empty($orig_event) ? DateTimeFormat::local($sdt, 'i') : '00';
-
- $fyear = DateTimeFormat::local($fdt, 'Y');
- $fmonth = DateTimeFormat::local($fdt, 'm');
- $fday = DateTimeFormat::local($fdt, 'd');
-
- $fhour = !empty($orig_event) ? DateTimeFormat::local($fdt, 'H') : '00';
- $fminute = !empty($orig_event) ? DateTimeFormat::local($fdt, 'i') : '00';
-
- if (!$cid && in_array($mode, ['new', 'copy'])) {
- $acl = ACL::getFullSelectorHTML(DI::page(), $a->getLoggedInUserId(), false, ACL::getDefaultUserPermissions($orig_event));
- } else {
- $acl = '';
- }
-
- // If we copy an old event, we need to remove the ID and URI
- // from the original event.
- if ($mode === 'copy') {
- $eid = 0;
- $uri = '';
- }
-
- $tpl = Renderer::getMarkupTemplate('event_form.tpl');
-
- $o .= Renderer::replaceMacros($tpl, [
- '$post' => DI::baseUrl() . '/events',
- '$eid' => $eid,
- '$cid' => $cid,
- '$uri' => $uri,
-
- '$title' => DI::l10n()->t('Event details'),
- '$desc' => DI::l10n()->t('Starting date and Title are required.'),
- '$s_text' => DI::l10n()->t('Event Starts:') . ' <span class="required" title="' . DI::l10n()->t('Required') . '">*</span>',
- '$s_dsel' => Temporal::getDateTimeField(
- new DateTime(),
- DateTime::createFromFormat('Y', intval($syear) + 5),
- DateTime::createFromFormat('Y-m-d H:i', "$syear-$smonth-$sday $shour:$sminute"),
- DI::l10n()->t('Event Starts:'),
- 'start_text',
- true,
- true,
- '',
- '',
- true
- ),
- '$n_text' => DI::l10n()->t('Finish date/time is not known or not relevant'),
- '$n_checked' => $n_checked,
- '$f_text' => DI::l10n()->t('Event Finishes:'),
- '$f_dsel' => Temporal::getDateTimeField(
- new DateTime(),
- DateTime::createFromFormat('Y', intval($fyear) + 5),
- DateTime::createFromFormat('Y-m-d H:i', "$fyear-$fmonth-$fday $fhour:$fminute"),
- DI::l10n()->t('Event Finishes:'),
- 'finish_text',
- true,
- true,
- 'start_text'
- ),
- '$d_text' => DI::l10n()->t('Description:'),
- '$d_orig' => $d_orig,
- '$l_text' => DI::l10n()->t('Location:'),
- '$l_orig' => $l_orig,
- '$t_text' => DI::l10n()->t('Title:') . ' <span class="required" title="' . DI::l10n()->t('Required') . '">*</span>',
- '$t_orig' => $t_orig,
- '$summary' => ['summary', DI::l10n()->t('Title:'), $t_orig, '', '*'],
- '$sh_text' => DI::l10n()->t('Share this event'),
- '$share' => ['share', DI::l10n()->t('Share this event'), $share_checked, '', $share_disabled],
- '$sh_checked' => $share_checked,
- '$nofinish' => ['nofinish', DI::l10n()->t('Finish date/time is not known or not relevant'), $n_checked],
- '$preview' => DI::l10n()->t('Preview'),
- '$acl' => $acl,
- '$submit' => DI::l10n()->t('Submit'),
- '$basic' => DI::l10n()->t('Basic'),
- '$advanced' => DI::l10n()->t('Advanced'),
- '$permissions' => DI::l10n()->t('Permissions'),
- ]);
-
- return $o;
- }
-
- // Remove an event from the calendar and its related items
- if ($mode === 'drop' && $event_id) {
- $ev = Event::getListById(DI::userSession()->getLocalUserId(), $event_id);
-
- // Delete only real events (no birthdays)
- if (DBA::isResult($ev) && $ev[0]['type'] == 'event') {
- Item::deleteForUser(['id' => $ev[0]['itemid']], DI::userSession()->getLocalUserId());
- }
-
- if (Post::exists(['id' => $ev[0]['itemid']])) {
- DI::sysmsg()->addNotice(DI::l10n()->t('Failed to remove event'));
- }
-
- DI::baseUrl()->redirect('events');
- }
-}
'settings' => null,
'contacts' => null,
'delegation'=> null,
- 'events' => null,
+ 'calendar' => null,
'register' => null
];
'apps' => null,
'community' => null,
'home' => null,
- 'events' => null,
+ 'calendar' => null,
'login' => null,
'logout' => null,
'langselector' => null,
$nav['usermenu'][] = ['profile/' . $a->getLoggedInUserNickname() . '/profile', DI::l10n()->t('Profile'), '', DI::l10n()->t('Your profile page')];
$nav['usermenu'][] = ['photos/' . $a->getLoggedInUserNickname(), DI::l10n()->t('Photos'), '', DI::l10n()->t('Your photos')];
$nav['usermenu'][] = ['profile/' . $a->getLoggedInUserNickname() . '/media', DI::l10n()->t('Media'), '', DI::l10n()->t('Your postings with media')];
- $nav['usermenu'][] = ['events/', DI::l10n()->t('Events'), '', DI::l10n()->t('Your events')];
+ $nav['usermenu'][] = ['calendar/', DI::l10n()->t('Calendar'), '', DI::l10n()->t('Your calendar')];
$nav['usermenu'][] = ['notes/', DI::l10n()->t('Personal notes'), '', DI::l10n()->t('Your personal notes')];
// user info
}
if (DI::userSession()->getLocalUserId()) {
- $nav['events'] = ['events', DI::l10n()->t('Events'), '', DI::l10n()->t('Events and Calendar')];
+ $nav['calendar'] = ['calendar', DI::l10n()->t('Calendar'), '', DI::l10n()->t('Calendar')];
}
$nav['directory'] = [$gdirpath, DI::l10n()->t('Directory'), '', DI::l10n()->t('People directory')];
use Friendica\Content\Text\BBCode;
use Friendica\Core\Hook;
use Friendica\Core\Logger;
-use Friendica\Core\Protocol;
use Friendica\Core\Renderer;
use Friendica\Core\System;
use Friendica\Database\DBA;
use Friendica\DI;
+use Friendica\Network\HTTPException\NotFoundException;
+use Friendica\Network\HTTPException\UnauthorizedException;
use Friendica\Protocol\Activity;
use Friendica\Util\DateTimeFormat;
use Friendica\Util\Map;
use Friendica\Util\Strings;
+use Friendica\Util\Temporal;
use Friendica\Util\XML;
/**
*
* @param int $owner_uid The User ID of the owner of the event
* @param int $event_id The ID of the event in the event table
- * @param string $sql_extra
+ * @param string $nickname a possible nickname to search for instead of the own uid
* @return array Query result
* @throws \Exception
*/
- public static function getListById(int $owner_uid, int $event_id, string $sql_extra = ''): array
+ public static function getByIdAndUid(int $owner_uid, int $event_id, string $nickname = null): array
{
- $return = [];
+ if (!empty($nickname)) {
+ $owner = static::getOwnerForNickname($nickname, true);
+ $owner_uid = $owner['uid'];
+
+ // get the permissions
+ $sql_perms = Item::getPermissionsSQLByUserId($owner_uid);
+ // we only want to have the events of the profile owner
+ $sql_extra = " AND `event`.`cid` = 0 " . $sql_perms;
+ } else {
+ $sql_extra = "";
+ }
- // Ownly allow events if there is a valid owner_id.
+ // Only allow events if there is a valid owner_id.
if ($owner_uid == 0) {
- return $return;
+ return [];
}
// Query for the event by event id
WHERE `event`.`uid` = ? AND `event`.`id` = ? $sql_extra",
$owner_uid, $event_id));
- if (DBA::isResult($events)) {
- $return = self::removeDuplicates($events);
+ if (empty($events)) {
+ throw new NotFoundException(DI::l10n()->t('Event not found.'));
+ } else {
+ $events = self::removeDuplicates($events);
+ return $events[0];
+ }
+ }
+
+ /**
+ * Returns the owner array of a given nickname
+ * Additionally, it can check if the owner array is selectable
+ *
+ * @param string $nickname
+ * @param bool $check
+ *
+ * @return array the owner array
+ * @throws NotFoundException The given nickname does not exist
+ * @throws UnauthorizedException The access for the given nickname is restricted
+ */
+ public static function getOwnerForNickname(string $nickname, bool $check = true): array
+ {
+ $owner = User::getOwnerDataByNick($nickname);
+ if (empty($owner)) {
+ throw new NotFoundException(DI::l10n()->t('User not found.'));
}
- return $return;
+ if ($check) {
+ $contact_id = DI::userSession()->getRemoteContactID($owner['uid']);
+
+ $remote_contact = $contact_id && DBA::exists('contact', ['id' => $contact_id, 'uid' => $owner['uid']]);
+
+ $is_owner = DI::userSession()->getLocalUserId() == $owner['uid'];
+
+ if ($owner['hidewall'] && !$is_owner && !$remote_contact) {
+ throw new UnauthorizedException(DI::l10n()->t('Access to this profile has been restricted.'));
+ }
+ }
+
+ return $owner;
}
/**
* Get all events in a specific time frame.
*
- * @param int $owner_uid The User ID of the owner of the events.
- * @param array $event_params An associative array with
- * int 'ignore' =>
- * string 'start' => Start time of the timeframe.
- * string 'finish' => Finish time of the timeframe.
- *
- * @param string $sql_extra Additional sql conditions (e.g. permission request).
+ * @param int $owner_uid The User ID of the owner of the events.
+ * @param string|null $start Start time of the timeframe.
+ * @param string|null $finish Finish time of the timeframe.
+ * @param bool $ignore
+ * @param string|null $nickname
*
* @return array Query results.
- * @throws \Exception
+ * @throws NotFoundException
+ * @throws UnauthorizedException
*/
- public static function getListByDate(int $owner_uid, array $event_params, string $sql_extra = ''): array
+ public static function getListByDate(int $owner_uid, string $start = null, string $finish = null, bool $ignore = false, string $nickname = null): array
{
- $return = [];
+ if (!empty($nickname)) {
+ $owner = static::getOwnerForNickname($nickname, true);
+ $owner_uid = $owner['uid'];
+
+ // get the permissions
+ $sql_perms = Item::getPermissionsSQLByUserId($owner_uid);
+ // we only want to have the events of the profile owner
+ $sql_extra = " AND `event`.`cid` = 0 " . $sql_perms;
+ } else {
+ $sql_extra = "";
+ }
// Only allow events if there is a valid owner_id.
if ($owner_uid == 0) {
- return $return;
+ return [];
+ }
+
+ if (empty($start) || empty($finish)) {
+
+ $y = intval(DateTimeFormat::localNow('Y'));
+ $m = intval(DateTimeFormat::localNow('m'));
+
+ // Put some limit on dates. The PHP date functions don't seem to do so well before 1900.
+ if ($y < 1901) {
+ $y = 1900;
+ }
+
+ if ($y > 2099) {
+ $y = 2100;
+ }
+
+ if (empty($start)) {
+ $start = sprintf('%d-%d-%d %d:%d:%d', $y, $m, 1, 0, 0, 0);
+ } else {
+ $dim = Temporal::getDaysInMonth($y, $m);
+ $finish = sprintf('%d-%d-%d %d:%d:%d', $y, $m, $dim, 23, 59, 59);
+ }
}
// Query for the event by date.
WHERE `event`.`uid` = ? AND `event`.`ignore` = ?
AND (`finish` >= ? OR (`nofinish` AND `start` >= ?)) AND `start` <= ?
" . $sql_extra,
- $owner_uid, $event_params['ignore'],
- $event_params['start'], $event_params['start'], $event_params['finish']
+ $owner_uid, $ignore,
+ $start, $start, $finish
));
- if (DBA::isResult($events)) {
- $return = self::removeDuplicates($events);
- }
-
- return $return;
+ $events = self::removeDuplicates($events ?? []);
+ return self::sortByDate($events);
}
/**
{
$event_list = [];
- $last_date = '';
- $fmt = DI::l10n()->t('l, F j');
foreach ($event_result as $event) {
- $item = Post::selectFirst(['plink', 'author-name', 'author-network', 'author-id', 'author-avatar', 'author-link', 'private', 'uri-id'], ['id' => $event['itemid']]);
- if (!DBA::isResult($item)) {
- // Using default values when no item had been found
- $item = ['plink' => '', 'author-name' => '', 'author-avatar' => '', 'author-link' => '', 'private' => Item::PUBLIC, 'uri-id' => ($event['uri-id'] ?? 0)];
- }
+ $event_list[] = static::prepareForItem($event);
+ }
- $event = array_merge($event, $item);
+ return $event_list;
+ }
- $start = DateTimeFormat::local($event['start'], 'c');
- $j = DateTimeFormat::local($event['start'], 'j');
- $day = DateTimeFormat::local($event['start'], $fmt);
- $day = DI::l10n()->getDay($day);
+ /**
+ * Convert an one event in an array which could be used by the events template.
+ *
+ * @param array $event Event query array.
+ * @return array Event array for the template.
+ * @throws \Friendica\Network\HTTPException\InternalServerErrorException
+ * @throws \ImagickException
+ */
+ public static function prepareForItem(array $event): array
+ {
+ $fmt = DI::l10n()->t('l, F j');
- if ($event['nofinish']) {
- $end = null;
- } else {
- $end = DateTimeFormat::local($event['finish'], 'c');
- }
+ $item = Post::selectFirst(['plink', 'author-name', 'author-network', 'author-id', 'author-avatar', 'author-link', 'private', 'uri-id'], ['id' => $event['itemid']]);
+ if (!DBA::isResult($item)) {
+ // Using default values when no item had been found
+ $item = ['plink' => '', 'author-name' => '', 'author-avatar' => '', 'author-link' => '', 'private' => Item::PUBLIC, 'uri-id' => ($event['uri-id'] ?? 0)];
+ }
- $is_first = ($day !== $last_date);
+ $event = array_merge($event, $item);
- $last_date = $day;
+ $start = DateTimeFormat::local($event['start'], 'c');
+ $j = DateTimeFormat::local($event['start'], 'j');
+ $day = DateTimeFormat::local($event['start'], $fmt);
+ $day = DI::l10n()->getDay($day);
- // Show edit and drop actions only if the user is the owner of the event and the event
- // is a real event (no bithdays).
- $edit = null;
- $copy = null;
- $drop = null;
- if (DI::userSession()->getLocalUserId() && DI::userSession()->getLocalUserId() == $event['uid'] && $event['type'] == 'event') {
- $edit = !$event['cid'] ? [DI::baseUrl() . '/events/event/' . $event['id'], DI::l10n()->t('Edit event') , '', ''] : null;
- $copy = !$event['cid'] ? [DI::baseUrl() . '/events/copy/' . $event['id'] , DI::l10n()->t('Duplicate event'), '', ''] : null;
- $drop = [DI::baseUrl() . '/events/drop/' . $event['id'] , DI::l10n()->t('Delete event') , '', ''];
- }
+ if ($event['nofinish']) {
+ $end = null;
+ } else {
+ $end = DateTimeFormat::local($event['finish'], 'c');
+ }
- $title = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['summary']));
- if (!$title) {
- list($title, $_trash) = explode("<br", BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['desc'])), BBCode::TWITTER_API);
- }
+ // Show edit and drop actions only if the user is the owner of the event and the event
+ // is a real event (no bithdays).
+ $edit = null;
+ $copy = null;
+ $drop = null;
+ if (DI::userSession()->getLocalUserId() && DI::userSession()->getLocalUserId() == $event['uid'] && $event['type'] == 'event') {
+ $edit = !$event['cid'] ? [DI::baseUrl() . '/calendar/event/edit/' . $event['id'], DI::l10n()->t('Edit event') , '', ''] : null;
+ $copy = !$event['cid'] ? [DI::baseUrl() . '/calendar/event/copy/' . $event['id'] , DI::l10n()->t('Duplicate event'), '', ''] : null;
+ $drop = [DI::baseUrl() . '/calendar/api/delete/' . $event['id'] , DI::l10n()->t('Delete event') , '', ''];
+ }
- $author_link = $event['author-link'];
-
- $event['author-link'] = Contact::magicLink($author_link);
-
- $html = self::getHTML($event);
- $event['summary'] = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['summary']));
- $event['desc'] = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['desc']));
- $event['location'] = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['location']));
- $event_list[] = [
- 'id' => $event['id'],
- 'start' => $start,
- 'end' => $end,
- 'allDay' => false,
- 'title' => $title,
- 'j' => $j,
- 'd' => $day,
- 'edit' => $edit,
- 'drop' => $drop,
- 'copy' => $copy,
- 'is_first' => $is_first,
- 'item' => $event,
- 'html' => $html,
- 'plink' => Item::getPlink($event),
- ];
+ $title = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['summary']));
+ if (!$title) {
+ [$title, $_trash] = explode("<br", BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['desc'])), BBCode::TWITTER_API);
}
- return $event_list;
+ $author_link = $event['author-link'];
+
+ $event['author-link'] = Contact::magicLink($author_link);
+
+ $html = self::getHTML($event);
+ $event['summary'] = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['summary']));
+ $event['desc'] = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['desc']));
+ $event['location'] = BBCode::convertForUriId($event['uri-id'], Strings::escapeHtml($event['location']));
+
+ return [
+ 'id' => $event['id'],
+ 'start' => $start,
+ 'end' => $end,
+ 'allDay' => false,
+ 'title' => $title,
+ 'j' => $j,
+ 'd' => $day,
+ 'edit' => $edit,
+ 'drop' => $drop,
+ 'copy' => $copy,
+ 'item' => $event,
+ 'html' => $html,
+ 'plink' => Item::getPlink($event),
+ ];
}
/**
// Check if self::store() was success
return (self::store($values) > 0);
}
+
+ public static function setIgnore(int $uid, int $eventId, bool $ignore = true)
+ {
+ DBA::update('event', ['ignore' => $ignore], ['id' => $eventId, 'uid' => $uid]);
+ }
}
*/
abstract public function getNotifications();
- public function __construct(L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, IHandleUserSessions $userSession, array $server, array $parameters = [])
+ public function __construct(L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, IHandleUserSessions $session, array $server, array $parameters = [])
{
parent::__construct($l10n, $baseUrl, $args, $logger, $profiler, $response, $server, $parameters);
- if (!$userSession->getLocalUserId()) {
+ if (!$session->getLocalUserId()) {
throw new ForbiddenException($this->t('Permission denied.'));
}
// the calendar link for the full featured events calendar
if ($is_owner && $a->getThemeInfoValue('events_in_profile')) {
$tabs[] = [
- 'label' => DI::l10n()->t('Events'),
- 'url' => DI::baseUrl() . '/events',
- 'sel' => $current == 'events' ? 'active' : '',
- 'title' => DI::l10n()->t('Events and Calendar'),
- 'id' => 'events-tab',
- 'accesskey' => 'e',
+ 'label' => DI::l10n()->t('Calendar'),
+ 'url' => DI::baseUrl() . '/calendar',
+ 'sel' => $current == 'calendar' ? 'active' : '',
+ 'title' => DI::l10n()->t('Calendar'),
+ 'id' => 'calendar-tab',
+ 'accesskey' => 'c',
];
// if the user is not the owner of the calendar we only show a calendar
// with the public events of the calendar owner
} elseif (!$is_owner) {
$tabs[] = [
- 'label' => DI::l10n()->t('Events'),
- 'url' => DI::baseUrl() . '/cal/' . $nickname,
- 'sel' => $current == 'cal' ? 'active' : '',
- 'title' => DI::l10n()->t('Events and Calendar'),
- 'id' => 'events-tab',
- 'accesskey' => 'e',
+ 'label' => DI::l10n()->t('Calendar'),
+ 'url' => DI::baseUrl() . '/calendar/show/' . $nickname,
+ 'sel' => $current == 'calendar' ? 'active' : '',
+ 'title' => DI::l10n()->t('Calendar'),
+ 'id' => 'calendar-tab',
+ 'accesskey' => 'c',
];
}
--- /dev/null
+<?php
+/**
+ * @copyright Copyright (C) 2010-2022, the Friendica project
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program 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 Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+
+namespace Friendica\Module\Calendar\Event;
+
+use Friendica\App;
+use Friendica\BaseModule;
+use Friendica\Core\L10n;
+use Friendica\Core\Protocol;
+use Friendica\Core\Session\Capability\IHandleUserSessions;
+use Friendica\Core\System;
+use Friendica\Core\Worker;
+use Friendica\Database\DBA;
+use Friendica\Model\Contact;
+use Friendica\Model\Conversation;
+use Friendica\Model\Event;
+use Friendica\Model\Item;
+use Friendica\Model\Post;
+use Friendica\Model\User;
+use Friendica\Module\Response;
+use Friendica\Navigation\SystemMessages;
+use Friendica\Network\HTTPException\BadRequestException;
+use Friendica\Network\HTTPException\UnauthorizedException;
+use Friendica\Util\ACLFormatter;
+use Friendica\Util\DateTimeFormat;
+use Friendica\Util\Profiler;
+use Friendica\Util\Strings;
+use Friendica\Worker\Delivery;
+use Psr\Log\LoggerInterface;
+
+/**
+ * Basic API class for events
+ * currently supports create, delete, ignore, unignore
+ *
+ * @todo: make create/update as REST-call instead of POST
+ */
+class API extends BaseModule
+{
+ const ACTION_CREATE = 'create';
+ const ACTION_DELETE = 'delete';
+ const ACTION_IGNORE = 'ignore';
+ const ACTION_UNIGNORE = 'unignore';
+
+ const ALLOWED_ACTIONS = [
+ self::ACTION_CREATE,
+ self::ACTION_DELETE,
+ self::ACTION_IGNORE,
+ self::ACTION_UNIGNORE,
+ ];
+
+ /** @var IHandleUserSessions */
+ protected $session;
+ /** @var SystemMessages */
+ protected $sysMessages;
+ /** @var ACLFormatter */
+ protected $aclFormatter;
+ /** @var string */
+ protected $timezone;
+
+ public function __construct(L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, IHandleUserSessions $session, SystemMessages $sysMessages, ACLFormatter $aclFormatter, App $app, array $server, array $parameters = [])
+ {
+ parent::__construct($l10n, $baseUrl, $args, $logger, $profiler, $response, $server, $parameters);
+
+ $this->session = $session;
+ $this->sysMessages = $sysMessages;
+ $this->aclFormatter = $aclFormatter;
+ $this->timezone = $app->getTimeZone();
+
+ if (!$this->session->getLocalUserId()) {
+ throw new UnauthorizedException($this->t('Permission denied.'));
+ }
+ }
+
+ protected function post(array $request = [])
+ {
+ $this->createEvent($request);
+ }
+
+ protected function rawContent(array $request = [])
+ {
+ if (empty($this->parameters['action']) || !in_array($this->parameters['action'], self::ALLOWED_ACTIONS)) {
+ throw new BadRequestException($this->t('Invalid Request'));
+ }
+
+ // CREATE is done per POSt, so nothing to do left
+ if ($this->parameters['action'] === static::ACTION_CREATE) {
+ return;
+ }
+
+ if (empty($this->parameters['id'])) {
+ throw new BadRequestException($this->t('Event id is missing.'));
+ }
+
+ $returnPath = $request['return_path'] ?? 'calendar';
+
+ switch ($this->parameters['action']) {
+ case self::ACTION_IGNORE:
+ Event::setIgnore($this->session->getLocalUserId(), $this->parameters['id']);
+ break;
+ case self::ACTION_UNIGNORE:
+ Event::setIgnore($this->session->getLocalUserId(), $this->parameters['id'], false);
+ break;
+ case self::ACTION_DELETE:
+ // Remove an event from the calendar and its related items
+ $event = Event::getByIdAndUid($this->session->getLocalUserId(), $this->parameters['id']);
+
+ // Delete only real events (no birthdays)
+ if (!empty($event) && $event['type'] == 'event') {
+ Item::deleteForUser(['id' => $event['itemid']], $this->session->getLocalUserId());
+ }
+
+ if (Post::exists(['id' => $event['itemid']])) {
+ $this->sysMessages->addNotice($this->t('Failed to remove event'));
+ }
+ break;
+ default:
+ throw new BadRequestException($this->t('Invalid Request'));
+ }
+
+ $this->baseUrl->redirect($returnPath);
+ }
+
+ protected function createEvent(array $request)
+ {
+ $eventId = !empty($request['event_id']) ? intval($request['event_id']) : 0;
+ $uid = (int)$this->session->getLocalUserId();
+ $cid = !empty($request['cid']) ? intval($request['cid']) : 0;
+
+ $strStartDateTime = Strings::escapeHtml($request['start_text'] ?? '');
+ $strFinishDateTime = Strings::escapeHtml($request['finish_text'] ?? '');
+
+ $noFinish = intval($request['nofinish'] ?? 0);
+
+ $share = intval($request['share'] ?? 0);
+ $isPreview = intval($request['preview'] ?? 0);
+
+ $start = DateTimeFormat::convert($strStartDateTime ?? DBA::NULL_DATETIME, $this->timezone);
+ if (!$noFinish) {
+ $finish = DateTimeFormat::convert($strFinishDateTime ?? DBA::NULL_DATETIME, 'UTC', $this->timezone);
+ } else {
+ $finish = DBA::NULL_DATETIME;
+ }
+
+ // Don't allow the event to finish before it begins.
+ // It won't hurt anything, but somebody will file a bug report,
+ // and we'll waste a bunch of time responding to it. Time that
+ // could've been spent doing something else.
+
+ $summary = trim($request['summary'] ?? '');
+ $desc = trim($request['desc'] ?? '');
+ $location = trim($request['location'] ?? '');
+ $type = 'event';
+
+ $params = [
+ 'summary' => $summary,
+ 'description' => $desc,
+ 'location' => $location,
+ 'start' => $strStartDateTime,
+ 'finish' => $strFinishDateTime,
+ 'nofinish' => $noFinish,
+ ];
+
+ $action = empty($eventId) ? 'new' : 'edit/' . $eventId;
+ $redirectOnError = 'calendar/event/' . $action . '?' . http_build_query($params, '', '&', PHP_QUERY_RFC3986);
+
+ if (strcmp($finish, $start) < 0 && !$noFinish) {
+ if ($isPreview) {
+ System::httpExit($this->t('Event can not end before it has started.'));
+ return;
+ } else {
+ $this->sysMessages->addNotice($this->t('Event can not end before it has started.'));
+ $this->baseUrl->redirect($redirectOnError);
+ }
+ }
+
+ if (empty($summary) || ($start === DBA::NULL_DATETIME)) {
+ if ($isPreview) {
+ System::httpExit($this->t('Event title and start time are required.'));
+ return;
+ } else {
+ $this->sysMessages->addNotice($this->t('Event title and start time are required.'));
+ $this->baseUrl->redirect($redirectOnError);
+ }
+ }
+
+ $self = Contact::getPublicIdByUserId($uid);
+
+ $aclFormatter = $this->aclFormatter;
+
+ if ($share) {
+ $user = User::getById($uid, ['allow_cid', 'allow_gid', 'deny_cid', 'deny_gid']);
+ if (empty($user)) {
+ $this->logger->warning('Cannot find user for an event.', ['uid' => $uid, 'event' => $eventId]);
+ $this->response->setStatus(500);
+ return;
+ }
+
+ $strAclContactAllow = isset($request['contact_allow']) ? $aclFormatter->toString($request['contact_allow']) : $user['allow_cid'] ?? '';
+ $strAclGroupAllow = isset($request['group_allow']) ? $aclFormatter->toString($request['group_allow']) : $user['allow_gid'] ?? '';
+ $strContactDeny = isset($request['contact_deny']) ? $aclFormatter->toString($request['contact_deny']) : $user['deny_cid'] ?? '';
+ $strGroupDeny = isset($request['group_deny']) ? $aclFormatter->toString($request['group_deny']) : $user['deny_gid'] ?? '';
+
+ $visibility = $request['visibility'] ?? '';
+ if ($visibility === 'public') {
+ // The ACL selector introduced in version 2019.12 sends ACL input data even when the Public visibility is selected
+ $strAclContactAllow = $strAclGroupAllow = $strContactDeny = $strGroupDeny = '';
+ } else if ($visibility === 'custom') {
+ // Since we know from the visibility parameter the item should be private, we have to prevent the empty ACL
+ // case that would make it public. So we always append the author's contact id to the allowed contacts.
+ // See https://github.com/friendica/friendica/issues/9672
+ $strAclContactAllow .= $aclFormatter->toString($self);
+ }
+ } else {
+ $strAclContactAllow = $aclFormatter->toString($self);
+ $strAclGroupAllow = $strContactDeny = $strGroupDeny = '';
+ }
+
+ $datarray = [
+ 'start' => $start,
+ 'finish' => $finish,
+ 'summary' => $summary,
+ 'desc' => $desc,
+ 'location' => $location,
+ 'type' => $type,
+ 'nofinish' => $noFinish,
+ 'uid' => $uid,
+ 'cid' => $cid,
+ 'allow_cid' => $strAclContactAllow,
+ 'allow_gid' => $strAclGroupAllow,
+ 'deny_cid' => $strContactDeny,
+ 'deny_gid' => $strGroupDeny,
+ 'id' => $eventId,
+ ];
+
+ if (intval($request['preview'])) {
+ System::httpExit(Event::getHTML($datarray));
+ return;
+ }
+
+ $eventId = Event::store($datarray);
+
+ $newItem = Event::getItemArrayForId($eventId, [
+ 'network' => Protocol::DFRN,
+ 'protocol' => Conversation::PARCEL_DIRECT,
+ 'direction' => Conversation::PUSH
+ ]);
+ if (Item::insert($newItem)) {
+ $uriId = (int)$newItem['uri-id'];
+ } else {
+ $uriId = 0;
+ }
+
+ if (!$cid && $uriId) {
+ Worker::add(Worker::PRIORITY_HIGH, "Notifier", Delivery::POST, $uriId, $uid);
+ }
+
+ $this->baseUrl->redirect('calendar');
+ }
+}
--- /dev/null
+<?php
+/**
+ * @copyright Copyright (C) 2010-2022, the Friendica project
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program 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 Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+
+namespace Friendica\Module\Calendar\Event;
+
+use Friendica\App;
+use Friendica\BaseModule;
+use Friendica\Content\Widget\CalendarExport;
+use Friendica\Core\ACL;
+use Friendica\Core\L10n;
+use Friendica\Core\Renderer;
+use Friendica\Core\Session\Capability\IHandleUserSessions;
+use Friendica\Model\Event as EventModel;
+use Friendica\Model\User;
+use Friendica\Module\Response;
+use Friendica\Module\Security\Login;
+use Friendica\Navigation\SystemMessages;
+use Friendica\Network\HTTPException\BadRequestException;
+use Friendica\Util\ACLFormatter;
+use Friendica\Util\DateTimeFormat;
+use Friendica\Util\Profiler;
+use Friendica\Util\Temporal;
+use Psr\Log\LoggerInterface;
+
+/**
+ * The editor-view of an event
+ */
+class Form extends BaseModule
+{
+ const MODE_NEW = 'new';
+ const MODE_EDIT = 'edit';
+ const MODE_COPY = 'copy';
+
+ const ALLOWED_MODES = [
+ self::MODE_NEW,
+ self::MODE_EDIT,
+ self::MODE_COPY,
+ ];
+
+ /** @var IHandleUserSessions */
+ protected $session;
+ /** @var SystemMessages */
+ protected $sysMessages;
+ /** @var ACLFormatter */
+ protected $aclFormatter;
+ /** @var App\Page */
+ protected $page;
+
+ public function __construct(L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, IHandleUserSessions $session, SystemMessages $sysMessages, ACLFormatter $aclFormatter, App\Page $page, array $server, array $parameters = [])
+ {
+ parent::__construct($l10n, $baseUrl, $args, $logger, $profiler, $response, $server, $parameters);
+
+ $this->session = $session;
+ $this->sysMessages = $sysMessages;
+ $this->aclFormatter = $aclFormatter;
+ $this->page = $page;
+ }
+
+ protected function content(array $request = []): string
+ {
+ if (empty($this->parameters['mode']) || !in_array($this->parameters['mode'], self::ALLOWED_MODES)) {
+ throw new BadRequestException($this->t('Invalid Request'));
+ }
+
+ if (!$this->session->getLocalUserId()) {
+ $this->sysMessages->addNotice($this->t('Permission denied.'));
+ return Login::form();
+ }
+
+ $mode = $this->parameters['mode'];
+
+ if (($mode === self::MODE_EDIT || $mode === self::MODE_COPY)) {
+ if (empty($this->parameters['id'])) {
+ throw new BadRequestException('Invalid Request');
+ }
+ $orig_event = EventModel::getByIdAndUid($this->session->getLocalUserId(), $this->parameters['id']);
+ if (empty($orig_event)) {
+ throw new BadRequestException('Invalid Request');
+ }
+ }
+
+ if ($mode === self::MODE_NEW) {
+ $this->session->set('return_path', $this->args->getCommand());
+ }
+
+ // get the translation strings for the calendar
+ $i18n = EventModel::getStrings();
+
+ $this->page->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.min.css');
+ $this->page->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.print.min.css', 'print');
+ $this->page->registerFooterScript('view/asset/moment/min/moment-with-locales.min.js');
+ $this->page->registerFooterScript('view/asset/fullcalendar/dist/fullcalendar.min.js');
+
+ $htpl = Renderer::getMarkupTemplate('calendar/calendar_head.tpl');
+ $this->page['htmlhead'] .= Renderer::replaceMacros($htpl, [
+ '$calendar_api' => $this->baseUrl . '/calendar/api/get',
+ '$event_api' => $this->baseUrl . '/calendar/event/show',
+ '$modparams' => 2,
+ '$i18n' => $i18n,
+ ]);
+
+ $share_checked = '';
+ $share_disabled = '';
+
+ if (empty($orig_event)) {
+ $orig_event = User::getById($this->session->getLocalUserId(), ['allow_cid', 'allow_gid', 'deny_cid',
+ 'deny_gid']);;
+ } else if ($orig_event['allow_cid'] !== '<' . $this->session->getLocalUserId() . '>'
+ || $orig_event['allow_gid']
+ || $orig_event['deny_cid']
+ || $orig_event['deny_gid']) {
+ $share_checked = ' checked="checked" ';
+ }
+
+ // In case of an error the browser is redirected back here, with these parameters filled in with the previous values
+ if (!empty($request['nofinish'])) {
+ $orig_event['nofinish'] = $request['nofinish'];
+ }
+ if (!empty($request['summary'])) {
+ $orig_event['summary'] = $request['summary'];
+ }
+ if (!empty($request['desc'])) {
+ $orig_event['desc'] = $request['desc'];
+ }
+ if (!empty($request['location'])) {
+ $orig_event['location'] = $request['location'];
+ }
+ if (!empty($request['start'])) {
+ $orig_event['start'] = $request['start'];
+ }
+ if (!empty($request['finish'])) {
+ $orig_event['finish'] = $request['finish'];
+ }
+
+ $n_checked = (!empty($orig_event['nofinish']) ? ' checked="checked" ' : '');
+
+ $t_orig = $orig_event['summary'] ?? '';
+ $d_orig = $orig_event['desc'] ?? '';
+ $l_orig = $orig_event['location'] ?? '';
+ $eid = $orig_event['id'] ?? 0;
+ $cid = $orig_event['cid'] ?? 0;
+ $uri = $orig_event['uri'] ?? '';
+
+ if ($cid || $mode === 'edit') {
+ $share_disabled = 'disabled="disabled"';
+ }
+
+ $sdt = $orig_event['start'] ?? 'now';
+ $fdt = $orig_event['finish'] ?? 'now';
+
+ $syear = DateTimeFormat::local($sdt, 'Y');
+ $smonth = DateTimeFormat::local($sdt, 'm');
+ $sday = DateTimeFormat::local($sdt, 'd');
+
+ $shour = !empty($orig_event) ? DateTimeFormat::local($sdt, 'H') : '00';
+ $sminute = !empty($orig_event) ? DateTimeFormat::local($sdt, 'i') : '00';
+
+ $fyear = DateTimeFormat::local($fdt, 'Y');
+ $fmonth = DateTimeFormat::local($fdt, 'm');
+ $fday = DateTimeFormat::local($fdt, 'd');
+
+ $fhour = !empty($orig_event) ? DateTimeFormat::local($fdt, 'H') : '00';
+ $fminute = !empty($orig_event) ? DateTimeFormat::local($fdt, 'i') : '00';
+
+ if (!$cid && in_array($mode, [self::MODE_NEW, self::MODE_COPY])) {
+ $acl = ACL::getFullSelectorHTML($this->page, $this->session->getLocalUserId(), false, ACL::getDefaultUserPermissions($orig_event));
+ } else {
+ $acl = '';
+ }
+
+ // If we copy an old event, we need to remove the ID and URI
+ // from the original event.
+ if ($mode === self::MODE_COPY) {
+ $eid = 0;
+ $uri = '';
+ }
+
+ $this->page['aside'] .= CalendarExport::getHTML($this->session->getLocalUserId());
+
+ $tpl = Renderer::getMarkupTemplate('calendar/event_form.tpl');
+
+ return Renderer::replaceMacros($tpl, [
+ '$post' => $this->baseUrl . '/calendar/api/create',
+ '$eid' => $eid,
+ '$cid' => $cid,
+ '$uri' => $uri,
+
+ '$title' => $this->t('Event details'),
+ '$desc' => $this->t('Starting date and Title are required.'),
+ '$s_text' => $this->t('Event Starts:') . ' <span class="required" title="' . $this->t('Required') . '">*</span>',
+ '$s_dsel' => Temporal::getDateTimeField(
+ new \DateTime(),
+ \DateTime::createFromFormat('Y', intval($syear) + 5),
+ \DateTime::createFromFormat('Y-m-d H:i', "$syear-$smonth-$sday $shour:$sminute"),
+ $this->t('Event Starts:'),
+ 'start_text',
+ true,
+ true,
+ '',
+ '',
+ true
+ ),
+ '$n_text' => $this->t('Finish date/time is not known or not relevant'),
+ '$n_checked' => $n_checked,
+ '$f_text' => $this->t('Event Finishes:'),
+ '$f_dsel' => Temporal::getDateTimeField(
+ new \DateTime(),
+ \DateTime::createFromFormat('Y', intval($fyear) + 5),
+ \DateTime::createFromFormat('Y-m-d H:i', "$fyear-$fmonth-$fday $fhour:$fminute"),
+ $this->t('Event Finishes:'),
+ 'finish_text',
+ true,
+ true,
+ 'start_text'
+ ),
+ '$d_text' => $this->t('Description:'),
+ '$d_orig' => $d_orig,
+ '$l_text' => $this->t('Location:'),
+ '$l_orig' => $l_orig,
+ '$t_text' => $this->t('Title:') . ' <span class="required" title="' . $this->t('Required') . '">*</span>',
+ '$t_orig' => $t_orig,
+ '$summary' => ['summary', $this->t('Title:'), $t_orig, '', '*'],
+ '$sh_text' => $this->t('Share this event'),
+ '$share' => ['share', $this->t('Share this event'), $share_checked, '', $share_disabled],
+ '$sh_checked' => $share_checked,
+ '$nofinish' => ['nofinish', $this->t('Finish date/time is not known or not relevant'), $n_checked],
+ '$preview' => $this->t('Preview'),
+ '$acl' => $acl,
+ '$submit' => $this->t('Submit'),
+ '$basic' => $this->t('Basic'),
+ '$advanced' => $this->t('Advanced'),
+ '$permissions' => $this->t('Permissions'),
+ ]);
+ }
+}
--- /dev/null
+<?php
+/**
+ * @copyright Copyright (C) 2010-2022, the Friendica project
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program 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 Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+
+namespace Friendica\Module\Calendar\Event;
+
+use Friendica\App;
+use Friendica\Core\L10n;
+use Friendica\Core\Session\Capability\IHandleUserSessions;
+use Friendica\Core\System;
+use Friendica\Model\Event;
+use Friendica\Model\Item;
+use Friendica\Model\Post;
+use Friendica\Module\Response;
+use Friendica\Network\HTTPException;
+use Friendica\Util\DateTimeFormat;
+use Friendica\Util\Profiler;
+use Psr\Log\LoggerInterface;
+
+/**
+ * GET-Controller for event
+ * returns the result as JSON
+ */
+class Get extends \Friendica\BaseModule
+{
+ /** @var IHandleUserSessions */
+ protected $session;
+
+ public function __construct(L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, IHandleUserSessions $session, array $server, array $parameters = [])
+ {
+ parent::__construct($l10n, $baseUrl, $args, $logger, $profiler, $response, $server, $parameters);
+
+ $this->session = $session;
+ }
+
+ protected function rawContent(array $request = [])
+ {
+ if (!$this->session->getLocalUserId()) {
+ throw new HTTPException\UnauthorizedException();
+ }
+
+ // get events by id or by date
+ if (!empty($request['id'])) {
+ $events = [Event::getByIdAndUid($this->session->getLocalUserId(), $request['id'], $this->parameters['nickname'] ?? null)];
+ } else {
+ $events = Event::getListByDate($this->session->getLocalUserId(), $request['start'] ?? '', $request['end'] ?? '', false, $this->parameters['nickname'] ?? null);
+ }
+
+ System::jsonExit($events ? self::map($events) : []);
+ }
+
+ private static function map(array $events): array
+ {
+ return array_map(function ($event) {
+ $item = Post::selectFirst(['plink', 'author-name', 'author-avatar', 'author-link', 'private', 'uri-id'], ['id' => $event['itemid']]);
+ if (empty($item)) {
+ // Using default values when no item had been found
+ $item = ['plink' => '', 'author-name' => '', 'author-avatar' => '', 'author-link' => '', 'private' => Item::PUBLIC, 'uri-id' => ($event['uri-id'] ?? 0)];
+ }
+
+ return [
+ 'id' => $event['id'],
+ 'title' => $event['summary'],
+ 'start' => DateTimeFormat::local($event['start']),
+ 'end' => DateTimeFormat::local($event['finish']),
+ 'nofinish' => $event['nofinish'],
+ 'desc' => $event['desc'],
+ 'location' => $event['location'],
+ 'item' => $item,
+ ];
+ }, $events);
+ }
+}
--- /dev/null
+<?php
+/**
+ * @copyright Copyright (C) 2010-2022, the Friendica project
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program 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 Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+
+namespace Friendica\Module\Calendar\Event;
+
+use Friendica\App;
+use Friendica\BaseModule;
+use Friendica\Core\L10n;
+use Friendica\Core\Renderer;
+use Friendica\Core\Session\Capability\IHandleUserSessions;
+use Friendica\Core\System;
+use Friendica\Model\Event;
+use Friendica\Module\Response;
+use Friendica\Network\HTTPException;
+use Friendica\Util\Profiler;
+use Psr\Log\LoggerInterface;
+
+/**
+ * Displays one specific event in a <div> container
+ */
+class Show extends BaseModule
+{
+ /** @var IHandleUserSessions */
+ protected $session;
+
+ public function __construct(L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, IHandleUserSessions $session, array $server, array $parameters = [])
+ {
+ parent::__construct($l10n, $baseUrl, $args, $logger, $profiler, $response, $server, $parameters);
+
+ $this->session = $session;
+ }
+
+ protected function rawContent(array $request = [])
+ {
+ if (!$this->session->getLocalUserId()) {
+ throw new HTTPException\UnauthorizedException($this->t('Permission denied.'));
+ }
+
+ if (empty($this->parameters['id'])) {
+ throw new HTTPException\BadRequestException($this->t('Invalid Request'));
+ }
+
+ $event = Event::getByIdAndUid($this->session->getLocalUserId(), (int)$this->parameters['id'], $this->parameters['nickname'] ?? '');
+
+ if (empty($event)) {
+ throw new HTTPException\NotFoundException($this->t('Event not found.'));
+ }
+
+ $tplEvent = Event::prepareForItem($event);
+
+ $event_item = [];
+ foreach ($tplEvent['item'] as $k => $v) {
+ $k = str_replace('-', '_', $k);
+ $event_item[$k] = $v;
+ }
+ $tplEvent['item'] = $event_item;
+
+ $tpl = Renderer::getMarkupTemplate('calendar/event.tpl');
+
+ $o = Renderer::replaceMacros($tpl, [
+ '$event' => $tplEvent,
+ ]);
+
+ System::httpExit($o);
+ }
+}
// If it is the own calendar return to the events page
// otherwise to the profile calendar page
if ($this->session->getLocalUserId() === $ownerUid) {
- $returnPath = 'events';
+ $returnPath = 'calendar';
} else {
- $returnPath = 'events/' . $this->parameters['nickname'];
+ $returnPath = 'calendar/show/' . $this->parameters['nickname'];
}
$this->baseUrl->redirect($returnPath);
+++ /dev/null
-<?php
-/**
- * @copyright Copyright (C) 2010-2022, the Friendica project
- *
- * @license GNU AGPL version 3 or any later version
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Affero General Public License as
- * published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * This program 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 Affero General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- *
- */
-
-namespace Friendica\Module\Calendar;
-
-use Friendica\Core\System;
-use Friendica\Database\DBA;
-use Friendica\DI;
-use Friendica\Model\Event;
-use Friendica\Model\Item;
-use Friendica\Model\Post;
-use Friendica\Network\HTTPException;
-use Friendica\Util\DateTimeFormat;
-use Friendica\Util\Temporal;
-
-class Json extends \Friendica\BaseModule
-{
- protected function rawContent(array $request = [])
- {
- if (!DI::userSession()->getLocalUserId()) {
- throw new HTTPException\UnauthorizedException();
- }
-
- $y = intval(DateTimeFormat::localNow('Y'));
- $m = intval(DateTimeFormat::localNow('m'));
-
- // Put some limit on dates. The PHP date functions don't seem to do so well before 1900.
- if ($y < 1901) {
- $y = 1900;
- }
-
- $dim = Temporal::getDaysInMonth($y, $m);
- $start = sprintf('%d-%d-%d %d:%d:%d', $y, $m, 1, 0, 0, 0);
- $finish = sprintf('%d-%d-%d %d:%d:%d', $y, $m, $dim, 23, 59, 59);
-
- if (!empty($request['start'])) {
- $start = $request['start'];
- }
-
- if (!empty($request['end'])) {
- $finish = $request['end'];
- }
-
- // put the event parametes in an array so we can better transmit them
- $event_params = [
- 'event_id' => intval($request['id'] ?? 0),
- 'start' => $start,
- 'finish' => $finish,
- 'ignore' => 0,
- ];
-
- // get events by id or by date
- if ($event_params['event_id']) {
- $r = Event::getListById(DI::userSession()->getLocalUserId(), $event_params['event_id']);
- } else {
- $r = Event::getListByDate(DI::userSession()->getLocalUserId(), $event_params);
- }
-
- $links = [];
-
- if (DBA::isResult($r)) {
- $r = Event::sortByDate($r);
- foreach ($r as $rr) {
- $j = DateTimeFormat::utc($rr['start'], 'j');
- if (empty($links[$j])) {
- $links[$j] = DI::baseUrl() . '/' . DI::args()->getCommand() . '#link-' . $j;
- }
- }
- }
-
- $events = [];
-
- // transform the event in a usable array
- if (DBA::isResult($r)) {
- $events = Event::sortByDate($r);
-
- $events = self::map($events);
- }
-
- System::jsonExit($events);
- }
-
- private static function map(array $events): array
- {
- return array_map(function ($event) {
- $item = Post::selectFirst(['plink', 'author-name', 'author-avatar', 'author-link', 'private', 'uri-id'], ['id' => $event['itemid']]);
- if (!DBA::isResult($item)) {
- // Using default values when no item had been found
- $item = ['plink' => '', 'author-name' => '', 'author-avatar' => '', 'author-link' => '', 'private' => Item::PUBLIC, 'uri-id' => ($event['uri-id'] ?? 0)];
- }
-
- return [
- 'id' => $event['id'],
- 'title' => $event['summary'],
- 'start' => DateTimeFormat::local($event['start']),
- 'end' => DateTimeFormat::local($event['finish']),
- 'nofinish' => $event['nofinish'],
- 'desc' => $event['desc'],
- 'location' => $event['location'],
- 'item' => $item,
- ];
- }, $events);
- }
-}
--- /dev/null
+<?php
+/**
+ * @copyright Copyright (C) 2010-2022, the Friendica project
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program 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 Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+
+namespace Friendica\Module\Calendar;
+
+use Friendica\App;
+use Friendica\BaseModule;
+use Friendica\Content\Nav;
+use Friendica\Content\Widget;
+use Friendica\Core\L10n;
+use Friendica\Core\Renderer;
+use Friendica\Core\Session\Capability\IHandleUserSessions;
+use Friendica\Core\Theme;
+use Friendica\Model\Event;
+use Friendica\Module\BaseProfile;
+use Friendica\Module\Response;
+use Friendica\Module\Security\Login;
+use Friendica\Navigation\SystemMessages;
+use Friendica\Util\Profiler;
+use Psr\Log\LoggerInterface;
+
+class Show extends BaseModule
+{
+ /** @var IHandleUserSessions */
+ protected $session;
+ /** @var SystemMessages */
+ protected $sysMessages;
+ /** @var App\Page */
+ protected $page;
+ /** @var App */
+ protected $app;
+
+ public function __construct(L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, IHandleUserSessions $session, SystemMessages $sysMessages, App\Page $page, App $app, array $server, array $parameters = [])
+ {
+ parent::__construct($l10n, $baseUrl, $args, $logger, $profiler, $response, $server, $parameters);
+
+ $this->session = $session;
+ $this->sysMessages = $sysMessages;
+ $this->page = $page;
+ $this->app = $app;
+ }
+
+ protected function content(array $request = []): string
+ {
+ if (!$this->session->getLocalUserId()) {
+ $this->sysMessages->addNotice($this->t('Permission denied.'));
+ return Login::form();
+ }
+
+ // get the translation strings for the calendar
+ $i18n = Event::getStrings();
+
+ $this->page->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.min.css');
+ $this->page->registerStylesheet('view/asset/fullcalendar/dist/fullcalendar.print.min.css', 'print');
+ $this->page->registerFooterScript('view/asset/moment/min/moment-with-locales.min.js');
+ $this->page->registerFooterScript('view/asset/fullcalendar/dist/fullcalendar.min.js');
+
+ $htpl = Renderer::getMarkupTemplate('calendar/calendar_head.tpl');
+ $this->page['htmlhead'] .= Renderer::replaceMacros($htpl, [
+ '$calendar_api' => $this->baseUrl . '/calendar/api/get' . (!empty($this->parameters['nickname']) ? '/' . $this->parameters['nickname'] : ''),
+ '$event_api' => $this->baseUrl . '/calendar/event/show' . (!empty($this->parameters['nickname']) ? '/' . $this->parameters['nickname'] : ''),
+ '$modparams' => 2,
+ '$i18n' => $i18n,
+ ]);
+
+ $tabs = '';
+
+ if (empty($this->parameters['nickname'])) {
+ if ($this->app->getThemeInfoValue('events_in_profile')) {
+ Nav::setSelected('home');
+ } else {
+ Nav::setSelected('calendar');
+ }
+
+ // tabs
+ if ($this->app->getThemeInfoValue('events_in_profile')) {
+ $tabs = BaseProfile::getTabsHTML($this->app, 'calendar', true, $this->app->getLoggedInUserNickname(), false);
+ }
+
+ $this->page['aside'] .= Widget\CalendarExport::getHTML($this->session->getLocalUserId());
+ } else {
+ $owner = Event::getOwnerForNickname($this->parameters['nickname'], true);
+
+ Nav::setSelected('calendar');
+
+ // get the tab navigation bar
+ $tabs = BaseProfile::getTabsHTML($this->app, 'calendar', false, $owner['nickname'], $owner['hide-friends']);
+
+ $this->page['aside'] .= Widget\VCard::getHTML($owner);
+ $this->page['aside'] .= Widget\CalendarExport::getHTML($owner['uid']);
+ }
+
+ // ACL blocks are loaded in modals in frio
+ $this->page->registerFooterScript(Theme::getPathForFile('asset/typeahead.js/dist/typeahead.bundle.js'));
+ $this->page->registerFooterScript(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.js'));
+ $this->page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.css'));
+ $this->page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput-typeahead.css'));
+
+ $tpl = Renderer::getMarkupTemplate("calendar/calendar.tpl");
+ $o = Renderer::replaceMacros($tpl, [
+ '$tabs' => $tabs,
+ '$title' => $this->t('Events'),
+ '$view' => $this->t('View'),
+ '$new_event' => [$this->baseUrl . '/calendar/event/new', $this->t('Create New Event'), '', ''],
+
+ '$today' => $this->t('today'),
+ '$month' => $this->t('month'),
+ '$week' => $this->t('week'),
+ '$day' => $this->t('day'),
+ '$list' => $this->t('list'),
+ ]);
+
+ return $o;
+ }
+}
'url' => '/contact'
],
[
- 'name' => 'Events',
- 'url' => '/events'
+ 'name' => 'Calendar',
+ 'url' => '/calendar'
]
]
];
if (DI::userSession()->getLocalUserId()) {
if (Strings::compareLink(DI::session()->get('my_url'), $item['author-link'])) {
if ($item['event-id'] != 0) {
- $edpost = ['events/event/' . $item['event-id'], DI::l10n()->t('Edit')];
+ $edpost = ['calendar/event/edit/' . $item['event-id'], DI::l10n()->t('Edit')];
} else {
$edpost = ['editpost/' . $item['id'], DI::l10n()->t('Edit')];
}
'/bookmarklet' => [Module\Bookmarklet::class, [R::GET]],
- '/calendar/{nickname}/export[/{format}]' => [Module\Calendar\Export::class, [R::GET]],
- '/calendar/json' => [Module\Calendar\Json::class, [R::GET]],
+ '/calendar' => [
+ '[/]' => [Module\Calendar\Show::class, [R::GET ]],
+ '/show/{nickname}' => [Module\Calendar\Show::class, [R::GET ]],
+ '/export/{nickname}[/{format:csv|ical}]' => [Module\Calendar\Export::class, [R::GET ]],
+ '/api/{action:ignore|unignore|delete}/{id:\d+}' => [Module\Calendar\Event\API::class, [R::GET ]],
+ '/api/{action:create}' => [Module\Calendar\Event\API::class, [ R::POST]],
+ '/api/get[/{nickname}]' => [Module\Calendar\Event\Get::class, [R::GET ]],
+ '/event/show/{id:\d+}' => [Module\Calendar\Event\Show::class, [R::GET ]],
+ '/event/show/{nickname}/{id:\d+}' => [Module\Calendar\Event\Show::class, [R::GET ]],
+ '/event/{mode:new}' => [Module\Calendar\Event\Form::class, [R::GET ]],
+ '/event/{mode:edit|copy}/{id:\d+}' => [Module\Calendar\Event\Form::class, [R::GET ]],
+ ],
'/community[/{content}]' => [Module\Conversation\Community::class, [R::GET]],
--- /dev/null
+
+{{$tabs nofilter}}
+<h2>{{$title}}</h2>
+
+<div id="new-event-link"><a href="{{$new_event.0}}">{{$new_event.1}}</a></div>
+
+<div id="events-calendar"></div>
--- /dev/null
+<script>
+ function showEvent(eventid) {
+ $.get(
+ '{{$event_api}}/'+eventid,
+ function(data){
+ $.colorbox({html:data});
+ }
+ );
+ }
+
+ function doEventPreview() {
+ $('#event-edit-preview').val(1);
+ $.post('calendar',$('#event-edit-form').serialize(), function(data) {
+ $.colorbox({ html: data });
+ });
+ $('#event-edit-preview').val(0);
+ }
+
+ // disable the input for the finish date if it is not available
+ function enableDisableFinishDate() {
+ if( $('#id_nofinish').is(':checked'))
+ $('#id_finish_text').prop("disabled", true);
+ else
+ $('#id_finish_text').prop("disabled", false);
+ }
+
+ $(document).ready(function() {
+ $('#events-calendar').fullCalendar({
+ firstDay: '{{$i18n.firstDay|escape:'quotes'}}',
+ monthNames: [
+ '{{$i18n.January|escape:'quotes'}}',
+ '{{$i18n.February|escape:'quotes'}}',
+ '{{$i18n.March|escape:'quotes'}}',
+ '{{$i18n.April|escape:'quotes'}}',
+ '{{$i18n.May|escape:'quotes'}}',
+ '{{$i18n.June|escape:'quotes'}}',
+ '{{$i18n.July|escape:'quotes'}}',
+ '{{$i18n.August|escape:'quotes'}}',
+ '{{$i18n.September|escape:'quotes'}}',
+ '{{$i18n.October|escape:'quotes'}}',
+ '{{$i18n.November|escape:'quotes'}}',
+ '{{$i18n.December|escape:'quotes'}}'
+ ],
+ monthNamesShort: [
+ '{{$i18n.Jan|escape:'quotes'}}',
+ '{{$i18n.Feb|escape:'quotes'}}',
+ '{{$i18n.Mar|escape:'quotes'}}',
+ '{{$i18n.Apr|escape:'quotes'}}',
+ '{{$i18n.May|escape:'quotes'}}',
+ '{{$i18n.Jun|escape:'quotes'}}',
+ '{{$i18n.Jul|escape:'quotes'}}',
+ '{{$i18n.Aug|escape:'quotes'}}',
+ '{{$i18n.Sep|escape:'quotes'}}',
+ '{{$i18n.Oct|escape:'quotes'}}',
+ '{{$i18n.Nov|escape:'quotes'}}',
+ '{{$i18n.Dec|escape:'quotes'}}'
+ ],
+ dayNames: [
+ '{{$i18n.Sunday|escape:'quotes'}}',
+ '{{$i18n.Monday|escape:'quotes'}}',
+ '{{$i18n.Tuesday|escape:'quotes'}}',
+ '{{$i18n.Wednesday|escape:'quotes'}}',
+ '{{$i18n.Thursday|escape:'quotes'}}',
+ '{{$i18n.Friday|escape:'quotes'}}',
+ '{{$i18n.Saturday|escape:'quotes'}}'
+ ],
+ dayNamesShort: [
+ '{{$i18n.Sun|escape:'quotes'}}',
+ '{{$i18n.Mon|escape:'quotes'}}',
+ '{{$i18n.Tue|escape:'quotes'}}',
+ '{{$i18n.Wed|escape:'quotes'}}',
+ '{{$i18n.Thu|escape:'quotes'}}',
+ '{{$i18n.Fri|escape:'quotes'}}',
+ '{{$i18n.Sat|escape:'quotes'}}'
+ ],
+ allDayText: '{{$i18n.allday|escape:'quotes'}}',
+ noEventsMessage: '{{$i18n.noevent|escape:'quotes'}}',
+ buttonText: {
+ today: '{{$i18n.today|escape:'quotes'}}',
+ month: '{{$i18n.month|escape:'quotes'}}',
+ week: '{{$i18n.week|escape:'quotes'}}',
+ day: '{{$i18n.day|escape:'quotes'}}'
+ },
+ events: '{{$calendar_api}}',
+ header: {
+ left: 'prev,next today',
+ center: 'title',
+ right: 'month,agendaWeek,agendaDay'
+ },
+ timeFormat: 'H(:mm)',
+ eventClick: function(calEvent, jsEvent, view) {
+ showEvent(calEvent.id);
+ },
+ loading: function(isLoading, view) {
+ if(!isLoading) {
+ $('td.fc-day').dblclick(function() { window.location.href='/calendar/event/new?start='+$(this).data('date'); });
+ }
+ },
+
+ eventRender: function(event, element, view) {
+ if (event.item['author-name']==null) return;
+ switch(view.name){
+ case "month":
+ element.find(".fc-title").html(
+ "<img src='{0}' style='height:10px;width:10px'>{1} : {2}".format(
+ event.item['author-avatar'],
+ event.item['author-name'],
+ event.title
+ ));
+ break;
+ case "agendaWeek":
+ element.find(".fc-title").html(
+ "<img src='{0}' style='height:12px; width:12px'>{1}<p>{2}</p><p>{3}</p>".format(
+ event.item['author-avatar'],
+ event.item['author-name'],
+ event.item.desc,
+ event.item.location
+ ));
+ break;
+ case "agendaDay":
+ element.find(".fc-title").html(
+ "<img src='{0}' style='height:24px;width:24px'>{1}<p>{2}</p><p>{3}</p>".format(
+ event.item['author-avatar'],
+ event.item['author-name'],
+ event.item.desc,
+ event.item.location
+ ));
+ break;
+ }
+ }
+
+ })
+
+ // center on date
+ var args=location.href.replace(baseurl,"").split("/");
+{{if $modparams == 2}}
+ if (args.length>=5) {
+ $("#events-calendar").fullCalendar('gotoDate',args[3] , args[4]-1);
+ }
+{{else}}
+ if (args.length>=4) {
+ $("#events-calendar").fullCalendar('gotoDate',args[2] , args[3]-1);
+ }
+{{/if}}
+
+ // show event popup
+ var hash = location.hash.split("-")
+ if (hash.length==2 && hash[0]=="#link") showEvent(hash[1]);
+
+ });
+</script>
+
+<script language="javascript" type="text/javascript">
+
+ $(document).ready(function() {
+ $("#comment-edit-text-desc").bbco_autocomplete('bbcode');
+
+ $('#id_share').change(function() {
+
+ if ($('#id_share').is(':checked')) {
+ $('#acl-wrapper').show();
+ }
+ else {
+ $('#acl-wrapper').hide();
+ }
+ }).trigger('change');
+
+ $('#contact_allow, #contact_deny, #group_allow, #group_deny').change(function() {
+ var selstr;
+ $('#contact_allow option:selected, #contact_deny option:selected, #group_allow option:selected, #group_deny option:selected').each( function() {
+ selstr = $(this).html();
+ $('#jot-public').hide();
+ });
+ if(selstr == null) {
+ $('#jot-public').show();
+ }
+
+ }).trigger('change');
+
+ // disable the finish time input if the user disable it
+ $('#id_nofinish').change(function() {
+ enableDisableFinishDate()
+ }).trigger('change');
+
+ });
+
+</script>
+
--- /dev/null
+<div class="event">
+
+{{if $event.item.author_name}}<a href="{{$event.item.author_link}}"><img src="{{$event.item.author_avatar}}" height="32" width="32" />{{$event.item.author_name}}</a>{{/if}}
+{{$event.html nofilter}}
+{{if $event.plink.orig}}<a href="{{$event.plink.orig}}" title="{{$event.plink.orig_title}}" target="_blank" rel="noopener noreferrer" class="plink-event-link icon s22 remote-link"></a>{{/if}}
+{{if $event.edit}}<a href="{{$event.edit.0}}" title="{{$event.edit.1}}" class="edit-event-link icon s22 pencil"></a>{{/if}}
+{{if $event.copy}}<a href="{{$event.copy.0}}" title="{{$event.copy.1}}" class="copy-event-link icon s22 copy"></a>{{/if}}
+{{if $event.drop}}<a href="{{$event.drop.0}}" onclick="return confirmDelete();" title="{{$event.drop.1}}" class="drop-event-link icon s22 delete"></a>{{/if}}
+</div>
+<div class="clear"></div>
\ No newline at end of file
--- /dev/null
+
+<h3>{{$title}}</h3>
+
+<p>
+{{$desc nofilter}}
+</p>
+
+<form id="event-edit-form" action="{{$post}}" method="post">
+
+<input type="hidden" name="event_id" value="{{$eid}}" />
+<input type="hidden" name="cid" value="{{$cid}}" />
+<input type="hidden" name="uri" value="{{$uri}}" />
+<input type="hidden" name="preview" id="event-edit-preview" value="0" />
+
+{{$s_dsel nofilter}}
+
+{{$f_dsel nofilter}}
+
+{{include file="field_checkbox.tpl" field=$nofinish}}
+
+{{include file="field_input.tpl" field=$summary}}
+
+
+<div id="event-desc-text">{{$d_text}}</div>
+<textarea id="event-desc-textarea" name="desc">{{$d_orig}}</textarea>
+
+
+<div id="event-location-text">{{$l_text}}</div>
+<textarea id="event-location-textarea" name="location">{{$l_orig}}</textarea>
+
+<div id="event-location-break"></div>
+
+{{if ! $eid}}
+{{include file="field_checkbox.tpl" field=$share}}
+{{/if}}
+
+{{$acl nofilter}}
+
+<div class="clear"></div>
+<input id="event-edit-preview" type="submit" name="preview" value="{{$preview}}" onclick="doEventPreview(); return false;" />
+<input id="event-submit" type="submit" name="submit" value="{{$submit}}" />
+</form>
+++ /dev/null
-
-{{foreach $events as $event}}
- <div class="event">
-
- {{if $event.item.author_name}}<a href="{{$event.item.author_link}}"><img src="{{$event.item.author_avatar}}" height="32" width="32" />{{$event.item.author_name}}</a>{{/if}}
- {{$event.html nofilter}}
- {{if $event.plink.orig}}<a href="{{$event.plink.orig}}" title="{{$event.plink.orig_title}}" target="_blank" rel="noopener noreferrer" class="plink-event-link icon s22 remote-link"></a>{{/if}}
- {{if $event.edit}}<a href="{{$event.edit.0}}" title="{{$event.edit.1}}" class="edit-event-link icon s22 pencil"></a>{{/if}}
- {{if $event.copy}}<a href="{{$event.copy.0}}" title="{{$event.copy.1}}" class="copy-event-link icon s22 copy"></a>{{/if}}
- {{if $event.drop}}<a href="{{$event.drop.0}}" onclick="return confirmDelete();" title="{{$event.drop.1}}" class="drop-event-link icon s22 delete"></a>{{/if}}
- </div>
- <div class="clear"></div>
-{{/foreach}}
+++ /dev/null
-
-<h3>{{$title}}</h3>
-
-<p>
-{{$desc nofilter}}
-</p>
-
-<form id="event-edit-form" action="{{$post}}" method="post">
-
-<input type="hidden" name="event_id" value="{{$eid}}" />
-<input type="hidden" name="cid" value="{{$cid}}" />
-<input type="hidden" name="uri" value="{{$uri}}" />
-<input type="hidden" name="preview" id="event-edit-preview" value="0" />
-
-{{$s_dsel nofilter}}
-
-{{$f_dsel nofilter}}
-
-{{include file="field_checkbox.tpl" field=$nofinish}}
-
-{{include file="field_input.tpl" field=$summary}}
-
-
-<div id="event-desc-text">{{$d_text}}</div>
-<textarea id="event-desc-textarea" name="desc">{{$d_orig}}</textarea>
-
-
-<div id="event-location-text">{{$l_text}}</div>
-<textarea id="event-location-textarea" name="location">{{$l_orig}}</textarea>
-
-<div id="event-location-break"></div>
-
-{{if ! $eid}}
-{{include file="field_checkbox.tpl" field=$share}}
-{{/if}}
-
-{{$acl nofilter}}
-
-<div class="clear"></div>
-<input id="event-edit-preview" type="submit" name="preview" value="{{$preview}}" onclick="doEventPreview(); return false;" />
-<input id="event-submit" type="submit" name="submit" value="{{$submit}}" />
-</form>
+++ /dev/null
-<script>
- function showEvent(eventid) {
- $.get(
- '{{$baseurl}}{{$module_url}}/?id='+eventid,
- function(data){
- $.colorbox({html:data});
- }
- );
- }
-
- function doEventPreview() {
- $('#event-edit-preview').val(1);
- $.post('events',$('#event-edit-form').serialize(), function(data) {
- $.colorbox({ html: data });
- });
- $('#event-edit-preview').val(0);
- }
-
- // disable the input for the finish date if it is not available
- function enableDisableFinishDate() {
- if( $('#id_nofinish').is(':checked'))
- $('#id_finish_text').prop("disabled", true);
- else
- $('#id_finish_text').prop("disabled", false);
- }
-
- $(document).ready(function() {
- $('#events-calendar').fullCalendar({
- firstDay: '{{$i18n.firstDay|escape:'quotes'}}',
- monthNames: [
- '{{$i18n.January|escape:'quotes'}}',
- '{{$i18n.February|escape:'quotes'}}',
- '{{$i18n.March|escape:'quotes'}}',
- '{{$i18n.April|escape:'quotes'}}',
- '{{$i18n.May|escape:'quotes'}}',
- '{{$i18n.June|escape:'quotes'}}',
- '{{$i18n.July|escape:'quotes'}}',
- '{{$i18n.August|escape:'quotes'}}',
- '{{$i18n.September|escape:'quotes'}}',
- '{{$i18n.October|escape:'quotes'}}',
- '{{$i18n.November|escape:'quotes'}}',
- '{{$i18n.December|escape:'quotes'}}'
- ],
- monthNamesShort: [
- '{{$i18n.Jan|escape:'quotes'}}',
- '{{$i18n.Feb|escape:'quotes'}}',
- '{{$i18n.Mar|escape:'quotes'}}',
- '{{$i18n.Apr|escape:'quotes'}}',
- '{{$i18n.May|escape:'quotes'}}',
- '{{$i18n.Jun|escape:'quotes'}}',
- '{{$i18n.Jul|escape:'quotes'}}',
- '{{$i18n.Aug|escape:'quotes'}}',
- '{{$i18n.Sep|escape:'quotes'}}',
- '{{$i18n.Oct|escape:'quotes'}}',
- '{{$i18n.Nov|escape:'quotes'}}',
- '{{$i18n.Dec|escape:'quotes'}}'
- ],
- dayNames: [
- '{{$i18n.Sunday|escape:'quotes'}}',
- '{{$i18n.Monday|escape:'quotes'}}',
- '{{$i18n.Tuesday|escape:'quotes'}}',
- '{{$i18n.Wednesday|escape:'quotes'}}',
- '{{$i18n.Thursday|escape:'quotes'}}',
- '{{$i18n.Friday|escape:'quotes'}}',
- '{{$i18n.Saturday|escape:'quotes'}}'
- ],
- dayNamesShort: [
- '{{$i18n.Sun|escape:'quotes'}}',
- '{{$i18n.Mon|escape:'quotes'}}',
- '{{$i18n.Tue|escape:'quotes'}}',
- '{{$i18n.Wed|escape:'quotes'}}',
- '{{$i18n.Thu|escape:'quotes'}}',
- '{{$i18n.Fri|escape:'quotes'}}',
- '{{$i18n.Sat|escape:'quotes'}}'
- ],
- allDayText: '{{$i18n.allday|escape:'quotes'}}',
- noEventsMessage: '{{$i18n.noevent|escape:'quotes'}}',
- buttonText: {
- today: '{{$i18n.today|escape:'quotes'}}',
- month: '{{$i18n.month|escape:'quotes'}}',
- week: '{{$i18n.week|escape:'quotes'}}',
- day: '{{$i18n.day|escape:'quotes'}}'
- },
- events: '{{$baseurl}}/calendar/json',
- header: {
- left: 'prev,next today',
- center: 'title',
- right: 'month,agendaWeek,agendaDay'
- },
- timeFormat: 'H(:mm)',
- eventClick: function(calEvent, jsEvent, view) {
- showEvent(calEvent.id);
- },
- loading: function(isLoading, view) {
- if(!isLoading) {
- $('td.fc-day').dblclick(function() { window.location.href='/events/new?start='+$(this).data('date'); });
- }
- },
-
- eventRender: function(event, element, view) {
- if (event.item['author-name']==null) return;
- switch(view.name){
- case "month":
- element.find(".fc-title").html(
- "<img src='{0}' style='height:10px;width:10px'>{1} : {2}".format(
- event.item['author-avatar'],
- event.item['author-name'],
- event.title
- ));
- break;
- case "agendaWeek":
- element.find(".fc-title").html(
- "<img src='{0}' style='height:12px; width:12px'>{1}<p>{2}</p><p>{3}</p>".format(
- event.item['author-avatar'],
- event.item['author-name'],
- event.item.desc,
- event.item.location
- ));
- break;
- case "agendaDay":
- element.find(".fc-title").html(
- "<img src='{0}' style='height:24px;width:24px'>{1}<p>{2}</p><p>{3}</p>".format(
- event.item['author-avatar'],
- event.item['author-name'],
- event.item.desc,
- event.item.location
- ));
- break;
- }
- }
-
- })
-
- // center on date
- var args=location.href.replace(baseurl,"").split("/");
-{{if $modparams == 2}}
- if (args.length>=5) {
- $("#events-calendar").fullCalendar('gotoDate',args[3] , args[4]-1);
- }
-{{else}}
- if (args.length>=4) {
- $("#events-calendar").fullCalendar('gotoDate',args[2] , args[3]-1);
- }
-{{/if}}
-
- // show event popup
- var hash = location.hash.split("-")
- if (hash.length==2 && hash[0]=="#link") showEvent(hash[1]);
-
- });
-</script>
-
-<script language="javascript" type="text/javascript">
-
- $(document).ready(function() {
- $("#comment-edit-text-desc").bbco_autocomplete('bbcode');
-
- $('#id_share').change(function() {
-
- if ($('#id_share').is(':checked')) {
- $('#acl-wrapper').show();
- }
- else {
- $('#acl-wrapper').hide();
- }
- }).trigger('change');
-
- $('#contact_allow, #contact_deny, #group_allow, #group_deny').change(function() {
- var selstr;
- $('#contact_allow option:selected, #contact_deny option:selected, #group_allow option:selected, #group_deny option:selected').each( function() {
- selstr = $(this).html();
- $('#jot-public').hide();
- });
- if(selstr == null) {
- $('#jot-public').show();
- }
-
- }).trigger('change');
-
- // disable the finish time input if the user disable it
- $('#id_nofinish').change(function() {
- enableDisableFinishDate()
- }).trigger('change');
-
- });
-
-</script>
-
+++ /dev/null
-
-{{$tabs nofilter}}
-{{include file="section_title.tpl"}}
-
-<div id="new-event-link"><a href="{{$new_event.0}}">{{$new_event.1}}</a></div>
-
-<div id="event-calendar-wrapper">
- <a href="{{$previus.0}}" class="prevcal {{$previus.2}}"><div id="event-calendar-prev" class="icon s22 prev" title="{{$previus.1}}"></div></a>
- {{$calendar}}
- <a href="{{$next.0}}" class="nextcal {{$next.2}}"><div id="event-calendar-prev" class="icon s22 next" title="{{$next.1}}"></div></a>
-</div>
-<div class="event-calendar-end"></div>
-
-
-{{foreach $events as $event}}
- <div class="event">
- {{if $event.is_first}}<hr /><a name="link-{{$event.j}}"><div class="event-list-date">{{$event.d}}</div></a>{{/if}}
- {{if $event.item.author_name}}<a href="{{$event.item.author_link}}"><img src="{{$event.item.author_avatar}}" height="32" width="32" />{{$event.item.author_name}}</a>{{/if}}
- {{$event.html nofilter}}
- {{if $event.item.plink}}<a href="{{$event.plink.0}}" title="{{$event.plink.1}}" target="_blank" rel="noopener noreferrer" class="plink-event-link icon s22 remote-link"></a>{{/if}}
- {{if $event.edit}}<a href="{{$event.edit.0}}" title="{{$event.edit.1}}" class="edit-event-link icon s22 pencil"></a>{{/if}}
- </div>
- <div class="clear"></div>
-
-{{/foreach}}
+++ /dev/null
-
-{{$tabs nofilter}}
-<h2>{{$title}}</h2>
-
-<div id="new-event-link"><a href="{{$new_event.0}}">{{$new_event.1}}</a></div>
-
-<div id="events-calendar"></div>
<div id="event-wrapper" style="display: none;"><div id="event-title">{{$event_title}}</div>
<div id="event-title-end"></div>
{{foreach $events as $event}}
-<div class="event-list" id="event-{{$event.id}}"> <a class="ajax-popupbox" href="events/?id={{$event.id}}">{{$event.title}}</a> - {{$event.date}} </div>
+<div class="event-list" id="event-{{$event.id}}"> <a class="ajax-popupbox" href="calendar/event/show/{{$event.id}}">{{$event.title}}</a> - {{$event.date}} </div>
{{/foreach}}
</div>
{{/if}}
-<div id="sidebar-events" class="widget">
+<div id="sidebar-calendar" class="widget">
<h3>{{$etitle}}</h3>
<ul class="sidebar-calendar-export-ul">
- <li role="menuitem" class="sidebar-calendar-export-li"><a href="{{$baseurl}}/calendar/{{$user}}/export/ical">{{$export_ical}}</a></li>
- <li role="menuitem" class="sidebar-calendar-export-li"><a href="{{$baseurl}}/calendar/{{$user}}/export/csv">{{$export_csv}}</a></li>
+ <li role="menuitem" class="sidebar-calendar-export-li"><a href="{{$baseurl}}/calendar/export/{{$user}}/ical">{{$export_ical}}</a></li>
+ <li role="menuitem" class="sidebar-calendar-export-li"><a href="{{$baseurl}}/calendar/export/{{$user}}/csv">{{$export_csv}}</a></li>
</ul>
</div>
week: aStr.week,
day: aStr.day,
},
- events: baseurl + moduleUrl + "/json/",
+ events: calendar_api,
header: {
left: "",
// center: 'title',
loading: function (isLoading, view) {
if (!isLoading) {
$("td.fc-day").dblclick(function () {
- addToModal("/events/new?start=" + $(this).data("date"));
+ addToModal("/calendar/event/new?start=" + $(this).data("date"));
});
}
},
},
});
- // center on date
- var args = location.href.replace(baseurl, "").split("/");
- if (modparams == 2) {
- if (args.length >= 5) {
- $("#events-calendar").fullCalendar("gotoDate", args[3], args[4] - 1);
- }
- } else {
- if (args.length >= 4) {
- $("#events-calendar").fullCalendar("gotoDate", args[2], args[3] - 1);
- }
- }
-
// echo the title
var view = $("#events-calendar").fullCalendar("getView");
$("#fc-title").text(view.title);
// loads the event into a modal
function showEvent(eventid) {
- addToModal(baseurl + moduleUrl + "/?id=" + eventid);
+ addToModal(event_api + '/' + eventid);
}
function changeView(action, viewName) {
// event-edit section.
function doEventPreview() {
$("#event-edit-preview").val(1);
- $.post("events", $("#event-edit-form").serialize(), function (data) {
+ $.post("calendar/api/create", $("#event-edit-form").serialize(), function (data) {
$("#event-preview").append(data);
});
$("#event-edit-preview").val(0);
// But first we have to test if the url links to an event. So we will split up
// the url in its parts.
var splitURL = parseUrl(url);
- // Test if in the url path containing "events/event". If the path containing this
+ // Test if in the url path containing "calendar/event/show". If the path containing this
// expression then we will call the addToModal function and exit this function at
// this point.
- if (splitURL.path.indexOf("events/event") > -1) {
+ if (splitURL.path.indexOf("calendar/event/show") > -1) {
addToModal(splitURL.path);
return;
}
$modalpages = [
'message/new',
'settings/oauth/add',
- 'events/new',
+ 'calendar/event/new',
// 'fbrowser/image/'
];
--- /dev/null
+<div class="generic-page-wrapper">
+ {{$tabs nofilter}}
+ {{include file="section_title.tpl" title=$title pullright=1}}
+
+ {{* The link to create a new event *}}
+ {{if $new_event.0}}
+ <div class="pull-right" id="new-event-link">
+ <button type="button" class="btn-link page-action faded-icon" onclick="addToModal('{{$new_event.0}}')" title="{{$new_event.1}}" data-toggle="tooltip">
+ <i class="fa fa-plus"></i>
+ </button>
+ </div>
+ {{/if}}
+
+ {{* We create our own fullcallendar header (with title & calendar view *}}
+ <div id="fc-header" class="clear">
+ <div id="fc-header-right" class="pull-right">
+ {{* The dropdown to change the callendar view *}}
+ <ul class="nav nav-pills">
+ <li class="dropdown pull-right">
+ <button class="btn btn-link btn-sm dropdown-toggle" type="button" id="event-calendar-views" data-toggle="dropdown" aria-expanded="false">
+ <i class="fa fa-angle-down" aria-hidden="true"></i> {{$view}}
+ </button>
+ <ul class="dropdown-menu pull-right" role="menu" aria-labelledby="event-calendar-views">
+ <li role="presentation">
+ <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'month');$('#events-calendar').fullCalendar('option', {contentHeight: '', aspectRatio: 1});">{{$month}}</button>
+ </li>
+ <li role="presentation">
+ <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'agendaWeek');$('#events-calendar').fullCalendar('option', 'contentHeight', 'auto');">{{$week}}</button>
+ </li>
+ <li role="presentation">
+ <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'agendaDay');$('#events-calendar').fullCalendar('option', 'contentHeight', 'auto');">{{$day}}</button>
+ </li>
+ <li role="presentation">
+ <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'listMonth');$('#events-calendar').fullCalendar('option', 'contentHeight', 'auto');">{{$list}}</button>
+ </li>
+ </ul>
+ </li>
+ </ul>
+ </div>
+
+ {{* The buttons to change the month/weeks/days *}}
+ <div id="fc-fc-header-left" class="btn-group">
+ <button class="btn btn-eventnav" onclick="changeView('prev', false);" title="{{$previous.1}}"><i class="fa fa-angle-up" aria-hidden="true"></i></button>
+ <button class="btn btn-eventnav btn-separator" onclick="changeView('next', false);" title="{{$next.1}}"><i class="fa fa-angle-down" aria-hidden="true"></i></button>
+ <button class="btn btn-eventnav btn-separator" onclick="changeView('today', false);" title="{{$today}}"><i class="fa fa-bullseye" aria-hidden="true"></i></button>
+ </div>
+
+ {{* The title (e.g. name of the mont/week/day) *}}
+ <div id="event-calendar-title"><h4 id="fc-title"></h4></div>
+
+ </div>
+
+ {{* This is the container where the fullCalendar is inserted through js *}}
+ <div id="events-calendar"></div>
+</div>
--- /dev/null
+
+<script type="text/javascript" src="view/theme/frio/js/mod_events.js?v={{$smarty.const.FRIENDICA_VERSION}}"></script>
+
+<script type="text/javascript">
+ // pass php translation strings to js variables/arrays so we can make use of it in js files
+ aStr.monthNames = [
+ '{{$i18n.January|escape:'quotes'}}',
+ '{{$i18n.February|escape:'quotes'}}',
+ '{{$i18n.March|escape:'quotes'}}',
+ '{{$i18n.April|escape:'quotes'}}',
+ '{{$i18n.May|escape:'quotes'}}',
+ '{{$i18n.June|escape:'quotes'}}',
+ '{{$i18n.July|escape:'quotes'}}',
+ '{{$i18n.August|escape:'quotes'}}',
+ '{{$i18n.September|escape:'quotes'}}',
+ '{{$i18n.October|escape:'quotes'}}',
+ '{{$i18n.November|escape:'quotes'}}',
+ '{{$i18n.December|escape:'quotes'}}'
+ ];
+ aStr.monthNamesShort = [
+ '{{$i18n.Jan|escape:'quotes'}}',
+ '{{$i18n.Feb|escape:'quotes'}}',
+ '{{$i18n.Mar|escape:'quotes'}}',
+ '{{$i18n.Apr|escape:'quotes'}}',
+ '{{$i18n.May|escape:'quotes'}}',
+ '{{$i18n.Jun|escape:'quotes'}}',
+ '{{$i18n.Jul|escape:'quotes'}}',
+ '{{$i18n.Aug|escape:'quotes'}}',
+ '{{$i18n.Sep|escape:'quotes'}}',
+ '{{$i18n.Oct|escape:'quotes'}}',
+ '{{$i18n.Nov|escape:'quotes'}}',
+ '{{$i18n.Dec|escape:'quotes'}}'
+ ];
+ aStr.dayNames = [
+ '{{$i18n.Sunday|escape:'quotes'}}',
+ '{{$i18n.Monday|escape:'quotes'}}',
+ '{{$i18n.Tuesday|escape:'quotes'}}',
+ '{{$i18n.Wednesday|escape:'quotes'}}',
+ '{{$i18n.Thursday|escape:'quotes'}}',
+ '{{$i18n.Friday|escape:'quotes'}}',
+ '{{$i18n.Saturday|escape:'quotes'}}'
+ ];
+ aStr.dayNamesShort = [
+ '{{$i18n.Sun|escape:'quotes'}}',
+ '{{$i18n.Mon|escape:'quotes'}}',
+ '{{$i18n.Tue|escape:'quotes'}}',
+ '{{$i18n.Wed|escape:'quotes'}}',
+ '{{$i18n.Thu|escape:'quotes'}}',
+ '{{$i18n.Fri|escape:'quotes'}}',
+ '{{$i18n.Sat|escape:'quotes'}}'
+ ];
+
+ aStr.firstDay = '{{$i18n.firstDay|escape:'quotes'}}';
+ aStr.today = '{{$i18n.today|escape:'quotes'}}';
+ aStr.month = '{{$i18n.month|escape:'quotes'}}';
+ aStr.week = '{{$i18n.week|escape:'quotes'}}';
+ aStr.day = '{{$i18n.day|escape:'quotes'}}';
+
+ aStr.allday = '{{$i18n.allday|escape:'quotes'}}';
+ aStr.noevent = '{{$i18n.noevent|escape:'quotes'}}';
+
+ aStr.dtstartLabel = '{{$i18n.dtstart_label|escape:'quotes'}}';
+ aStr.dtendLabel = '{{$i18n.dtend_label|escape:'quotes'}}';
+ aStr.locationLabel = '{{$i18n.location_label|escape:'quotes'}}';
+
+ var calendar_api = '{{$calendar_api}}';
+ var event_api = '{{$event_api}}';
+
+ $(document).ready(function() {
+ $("#comment-edit-text-desc").bbco_autocomplete('bbcode');
+ });
+</script>
--- /dev/null
+
+<div class="event-wrapper">
+ <div class="event">
+ <div class="media">
+ <div class="event-owner media-left">
+ {{if $event.item.author_name}}
+ <a href="{{$event.item.author_link}}"><img src="{{$event.item.author_avatar}}" /></a>
+ <a href="{{$event.item.author_link}}">{{$event.item.author_name}}</a>
+ {{/if}}
+ </div>
+ <div class="media-body">
+ {{$event.html nofilter}}
+ </div>
+ </div>
+
+ <div class="event-buttons pull-right">
+ {{if $event.edit}}<button type="button" class="btn" onclick="eventEdit('{{$event.edit.0}}')" title="{{$event.edit.1}}"><i class="fa fa-pencil" aria-hidden="true"></i></button>{{/if}}
+ {{if $event.copy}}<button type="button" class="btn" onclick="eventEdit('{{$event.copy.0}}')" title="{{$event.copy.1}}"><i class="fa fa-files-o" aria-hidden="true"></i></button>{{/if}}
+ {{if $event.drop}}<a href="{{$event.drop.0}}" onclick="return confirmDelete();" title="{{$event.drop.1}}" class="drop-event-link btn"><i class="fa fa-trash-o" aria-hidden="true"></i></a>{{/if}}
+ {{if $event.plink.orig}}<a href="{{$event.plink.orig}}" title="{{$event.plink.orig_title}}" class="plink-event-link btn" aria-label="{{$event.plink.1}}"><i class="fa fa-external-link" aria-hidden="true"></i></a>{{/if}}
+ </div>
+ <div class="clear"></div>
+ </div>
+</div>
--- /dev/null
+<div id="event-form-wrapper">
+ <h3 class="heading">{{$title}}</h3>
+
+ {{* The event edit navigation menu (text input, permissions, preview, filebrowser) *}}
+ <ul id="event-nav" class="nav nav-tabs event-nav" role="menubar" data-tabs="tabs">
+ {{* Mark the first list entry as active because it is the first which is active after opening
+ the modal. Changing of the activity status is done by js in calendar_head.tpl *}}
+ <li class="active" role="menuitem">
+ <a id="event-edit-lnk" onclick="eventEditActive(); return false;">{{$basic}}</a>
+ </li>
+ <li role="menuitem">
+ <a id="event-desc-lnk" onclick="eventDescActive(); return false;">{{$advanced}}</a>
+ </li>
+ {{if $acl}}
+ <li role="menuitem" {{if !$sh_checked}} style="display: none"{{/if}}>
+ <a id="event-perms-lnk" onclick="eventAclActive(); return false;">{{$permissions}}</a>
+ </li>
+ {{/if}}
+ {{if $preview}}
+ <li role="menuitem">
+ <a id="event-preview-lnk" onclick="eventPreviewActive(); return false;">{{$preview}}</a>
+ </li>
+ {{/if}}
+ {{* commented out because it isn't implemented yet
+ <li role="menuitem"><a id="event-preview-link" onclick="fbrowserActive(); return false;"> Browser </a></li>
+ *}}
+ </ul>
+
+ <div id="event-edit-form-wrapper">
+ <form id="event-edit-form" action="{{$post}}" method="post">
+
+ <input type="hidden" name="event_id" value="{{$eid}}" />
+ <input type="hidden" name="cid" value="{{$cid}}" />
+ <input type="hidden" name="uri" value="{{$uri}}" />
+ <input type="hidden" name="preview" id="event-edit-preview" value="0" />
+
+ {{* The tab conten with the necessary basic settings *}}
+ <div id="event-edit-wrapper">
+
+ {{* The event title *}}
+ {{include file="field_input.tpl" field=$summary}}
+
+ <div id="event-edit-time">
+ {{* The field for event starting time *}}
+ {{$s_dsel nofilter}}
+
+ {{* The field for event finish time *}}
+ {{$f_dsel nofilter}}
+
+ {{* checkbox if the the event doesn't have a finish time *}}
+ {{include file="field_checkbox.tpl" field=$nofinish}}
+ </div>
+
+ {{* checkbox to enable event sharing and the permissions tab *}}
+ {{if ! $eid}}
+ {{include file="field_checkbox.tpl" field=$share}}
+ {{/if}}
+
+ {{* The submit button - saves the event *}}
+ <div class="pull-right">
+ <button id="event-submit" type="submit" name="submit" class="btn btn-primary" value="{{$submit}}">{{$submit}}</button>
+ </div>
+ <div class="clear"></div>
+ </div>
+
+ {{* The advanced tab *}}
+ <div id="event-desc-wrapper" style="display: none">
+
+ {{* The textarea for the event description *}}
+ <div class="form-group">
+ <div id="event-desc-text"><b>{{$d_text}}</b></div>
+ <textarea id="comment-edit-text-desc" class="form-control text-autosize" name="desc" dir="auto">{{$d_orig}}</textarea>
+ <ul id="event-desc-text-edit-bb" class="comment-edit-bb comment-icon-list nav nav-pills hidden-xs pull-left">
+ {{* commented out because it isn't implemented yet
+ <li>
+ <button type="button" class="btn-link icon bb-img" style="cursor: pointer;" title="{{$edimg}}" data-role="insert-formatting" data-comment=" " data-bbcode="img" data-id="desc">
+ <i class="fa fa-picture-o"></i>
+ </button>
+ </li>
+ *}}
+ <li>
+ <button type="button" class="btn-link icon bb-url" style="cursor: pointer;" title="{{$edurl}}" data-role="insert-formatting" data-comment=" " data-bbcode="url" data-id="desc">
+ <i class="fa fa-link"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon bb-video" style="cursor: pointer;" title="{{$edvideo}}" data-role="insert-formatting" data-comment=" " data-bbcode="video" data-id="desc">
+ <i class="fa fa-video-camera"></i>
+ </button>
+ </li>
+
+ <li>
+ <button type="button" class="btn-link icon underline" style="cursor: pointer;" title="{{$eduline}}" data-role="insert-formatting" data-comment=" " data-bbcode="u" data-id="desc">
+ <i class="fa fa-underline"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon italic" style="cursor: pointer;" title="{{$editalic}}" data-role="insert-formatting" data-comment=" " data-bbcode="i" data-id="desc">
+ <i class="fa fa-italic"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon bold" style="cursor: pointer;" title="{{$edbold}}" data-role="insert-formatting" data-comment=" " data-bbcode="b" data-id="desc">
+ <i class="fa fa-bold"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon quote" style="cursor: pointer;" title="{{$edquote}}" data-role="insert-formatting" data-comment=" " data-bbcode="quote" data-id="desc">
+ <i class="fa fa-quote-left"></i>
+ </button>
+ </li>
+ </ul>
+ <div class="clear"></div>
+ </div>
+
+ {{* The textarea for the event location *}}
+ <div class="form-group">
+ <div id="event-location-text"><b>{{$l_text}}</b></div>
+ <textarea id="comment-edit-text-loc" class="form-control text-autosize" name="location" dir="auto">{{$l_orig}}</textarea>
+ <ul id="comment-tools-loc" class="comment-edit-bb comment-icon-list nav nav-pills hidden-xs pull-left">
+ {{* commented out because it isn't implemented yet
+ <li>
+ <button type="button" class="btn-link icon bb-img" style="cursor: pointer;" title="{{$edimg}}" data-role="insert-formatting" data-comment=" " data-bbcode="img" data-id="loc">
+ <i class="fa fa-picture-o"></i>
+ </button>
+ </li>
+ *}}
+ <li>
+ <button type="button" class="btn-link icon bb-url" style="cursor: pointer;" title="{{$edurl}}" data-role="insert-formatting" data-comment=" " data-bbcode="url" data-id="loc">
+ <i class="fa fa-link"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon bb-video" style="cursor: pointer;" title="{{$edvideo}}" data-role="insert-formatting" data-comment=" " data-bbcode="video" data-id="loc">
+ <i class="fa fa-video-camera"></i>
+ </button>
+ </li>
+
+ <li>
+ <button type="button" class="btn-link icon underline" style="cursor: pointer;" title="{{$eduline}}" data-role="insert-formatting" data-comment=" " data-bbcode="u" data-id="loc">
+ <i class="fa fa-underline"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon italic" style="cursor: pointer;" title="{{$editalic}}" data-role="insert-formatting" data-comment=" " data-bbcode="i" data-id="loc">
+ <i class="fa fa-italic"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon bold" style="cursor: pointer;" title="{{$edbold}}" data-role="insert-formatting" data-comment=" " data-bbcode="b" data-id="loc">
+ <i class="fa fa-bold"></i>
+ </button>
+ </li>
+ <li>
+ <button type="button" class="btn-link icon quote" style="cursor: pointer;" title="{{$edquote}}" data-role="insert-formatting" data-comment=" " data-bbcode="quote" data-id="loc">
+ <i class="fa fa-quote-left"></i>
+ </button>
+ </li>
+ </ul>
+ <div class="clear"></div>
+ </div>
+ </div>
+
+ {{* The tab for the permissions (if event sharing is enabled) *}}
+ <div id="event-acl-wrapper" style="display: none">
+ {{$acl nofilter}}
+ </div>
+
+ {{* The tab for the event preview (content is inserted by js) *}}
+ <div id="event-preview" style="display: none"></div>
+
+ <div class="clear"></div>
+
+ </form>
+ </div>
+</div>
+
+<script type="text/javascript">
+ $(document).ready( function() {
+ // disable finish date input if it isn't available
+ enableDisableFinishDate();
+ // load bbcode autocomplete for the description textarea
+ $('#comment-edit-text-desc, #comment-edit-text-loc').bbco_autocomplete('bbcode');
+
+ // initiale autosize for the textareas
+ autosize($("textarea.text-autosize"));
+ });
+</script>
+++ /dev/null
-
-{{foreach $events as $event}}
- <div class="event-wrapper">
- <div class="event">
- <div class="media">
- <div class="event-owner media-left">
- {{if $event.item.author_name}}
- <a href="{{$event.item.author_link}}"><img src="{{$event.item.author_avatar}}" /></a>
- <a href="{{$event.item.author_link}}">{{$event.item.author_name}}</a>
- {{/if}}
- </div>
- <div class="media-body">
- {{$event.html nofilter}}
- </div>
- </div>
-
- <div class="event-buttons pull-right">
- {{if $event.edit}}<button type="button" class="btn" onclick="eventEdit('{{$event.edit.0}}')" title="{{$event.edit.1}}"><i class="fa fa-pencil" aria-hidden="true"></i></button>{{/if}}
- {{if $event.copy}}<button type="button" class="btn" onclick="eventEdit('{{$event.copy.0}}')" title="{{$event.copy.1}}"><i class="fa fa-files-o" aria-hidden="true"></i></button>{{/if}}
- {{if $event.drop}}<a href="{{$event.drop.0}}" onclick="return confirmDelete();" title="{{$event.drop.1}}" class="drop-event-link btn"><i class="fa fa-trash-o" aria-hidden="true"></i></a>{{/if}}
- {{if $event.plink.orig}}<a href="{{$event.plink.orig}}" title="{{$event.plink.orig_title}}" class="plink-event-link btn" aria-label="{{$event.plink.1}}"><i class="fa fa-external-link" aria-hidden="true"></i></a>{{/if}}
- </div>
- <div class="clear"></div>
- </div>
- </div>
-{{/foreach}}
+++ /dev/null
-<div id="event-form-wrapper">
- <h3 class="heading">{{$title}}</h3>
-
- {{* The event edit navigation menu (text input, permissions, preview, filebrowser) *}}
- <ul id="event-nav" class="nav nav-tabs event-nav" role="menubar" data-tabs="tabs">
- {{* Mark the first list entry as active because it is the first which is active after opening
- the modal. Changing of the activity status is done by js in event_head.tpl *}}
- <li class="active" role="menuitem">
- <a id="event-edit-lnk" onclick="eventEditActive(); return false;">{{$basic}}</a>
- </li>
- <li role="menuitem">
- <a id="event-desc-lnk" onclick="eventDescActive(); return false;">{{$advanced}}</a>
- </li>
- {{if $acl}}
- <li role="menuitem" {{if !$sh_checked}} style="display: none"{{/if}}>
- <a id="event-perms-lnk" onclick="eventAclActive(); return false;">{{$permissions}}</a>
- </li>
- {{/if}}
- {{if $preview}}
- <li role="menuitem">
- <a id="event-preview-lnk" onclick="eventPreviewActive(); return false;">{{$preview}}</a>
- </li>
- {{/if}}
- {{* commented out because it isn't implemented yet
- <li role="menuitem"><a id="event-preview-link" onclick="fbrowserActive(); return false;"> Browser </a></li>
- *}}
- </ul>
-
- <div id="event-edit-form-wrapper">
- <form id="event-edit-form" action="{{$post}}" method="post">
-
- <input type="hidden" name="event_id" value="{{$eid}}" />
- <input type="hidden" name="cid" value="{{$cid}}" />
- <input type="hidden" name="uri" value="{{$uri}}" />
- <input type="hidden" name="preview" id="event-edit-preview" value="0" />
-
- {{* The tab conten with the necessary basic settings *}}
- <div id="event-edit-wrapper">
-
- {{* The event title *}}
- {{include file="field_input.tpl" field=$summary}}
-
- <div id="event-edit-time">
- {{* The field for event starting time *}}
- {{$s_dsel nofilter}}
-
- {{* The field for event finish time *}}
- {{$f_dsel nofilter}}
-
- {{* checkbox if the the event doesn't have a finish time *}}
- {{include file="field_checkbox.tpl" field=$nofinish}}
- </div>
-
- {{* checkbox to enable event sharing and the permissions tab *}}
- {{if ! $eid}}
- {{include file="field_checkbox.tpl" field=$share}}
- {{/if}}
-
- {{* The submit button - saves the event *}}
- <div class="pull-right">
- <button id="event-submit" type="submit" name="submit" class="btn btn-primary" value="{{$submit}}">{{$submit}}</button>
- </div>
- <div class="clear"></div>
- </div>
-
- {{* The advanced tab *}}
- <div id="event-desc-wrapper" style="display: none">
-
- {{* The textarea for the event description *}}
- <div class="form-group">
- <div id="event-desc-text"><b>{{$d_text}}</b></div>
- <textarea id="comment-edit-text-desc" class="form-control text-autosize" name="desc" dir="auto">{{$d_orig}}</textarea>
- <ul id="event-desc-text-edit-bb" class="comment-edit-bb comment-icon-list nav nav-pills hidden-xs pull-left">
- {{* commented out because it isn't implemented yet
- <li>
- <button type="button" class="btn-link icon bb-img" style="cursor: pointer;" title="{{$edimg}}" data-role="insert-formatting" data-comment=" " data-bbcode="img" data-id="desc">
- <i class="fa fa-picture-o"></i>
- </button>
- </li>
- *}}
- <li>
- <button type="button" class="btn-link icon bb-url" style="cursor: pointer;" title="{{$edurl}}" data-role="insert-formatting" data-comment=" " data-bbcode="url" data-id="desc">
- <i class="fa fa-link"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon bb-video" style="cursor: pointer;" title="{{$edvideo}}" data-role="insert-formatting" data-comment=" " data-bbcode="video" data-id="desc">
- <i class="fa fa-video-camera"></i>
- </button>
- </li>
-
- <li>
- <button type="button" class="btn-link icon underline" style="cursor: pointer;" title="{{$eduline}}" data-role="insert-formatting" data-comment=" " data-bbcode="u" data-id="desc">
- <i class="fa fa-underline"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon italic" style="cursor: pointer;" title="{{$editalic}}" data-role="insert-formatting" data-comment=" " data-bbcode="i" data-id="desc">
- <i class="fa fa-italic"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon bold" style="cursor: pointer;" title="{{$edbold}}" data-role="insert-formatting" data-comment=" " data-bbcode="b" data-id="desc">
- <i class="fa fa-bold"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon quote" style="cursor: pointer;" title="{{$edquote}}" data-role="insert-formatting" data-comment=" " data-bbcode="quote" data-id="desc">
- <i class="fa fa-quote-left"></i>
- </button>
- </li>
- </ul>
- <div class="clear"></div>
- </div>
-
- {{* The textarea for the event location *}}
- <div class="form-group">
- <div id="event-location-text"><b>{{$l_text}}</b></div>
- <textarea id="comment-edit-text-loc" class="form-control text-autosize" name="location" dir="auto">{{$l_orig}}</textarea>
- <ul id="comment-tools-loc" class="comment-edit-bb comment-icon-list nav nav-pills hidden-xs pull-left">
- {{* commented out because it isn't implemented yet
- <li>
- <button type="button" class="btn-link icon bb-img" style="cursor: pointer;" title="{{$edimg}}" data-role="insert-formatting" data-comment=" " data-bbcode="img" data-id="loc">
- <i class="fa fa-picture-o"></i>
- </button>
- </li>
- *}}
- <li>
- <button type="button" class="btn-link icon bb-url" style="cursor: pointer;" title="{{$edurl}}" data-role="insert-formatting" data-comment=" " data-bbcode="url" data-id="loc">
- <i class="fa fa-link"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon bb-video" style="cursor: pointer;" title="{{$edvideo}}" data-role="insert-formatting" data-comment=" " data-bbcode="video" data-id="loc">
- <i class="fa fa-video-camera"></i>
- </button>
- </li>
-
- <li>
- <button type="button" class="btn-link icon underline" style="cursor: pointer;" title="{{$eduline}}" data-role="insert-formatting" data-comment=" " data-bbcode="u" data-id="loc">
- <i class="fa fa-underline"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon italic" style="cursor: pointer;" title="{{$editalic}}" data-role="insert-formatting" data-comment=" " data-bbcode="i" data-id="loc">
- <i class="fa fa-italic"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon bold" style="cursor: pointer;" title="{{$edbold}}" data-role="insert-formatting" data-comment=" " data-bbcode="b" data-id="loc">
- <i class="fa fa-bold"></i>
- </button>
- </li>
- <li>
- <button type="button" class="btn-link icon quote" style="cursor: pointer;" title="{{$edquote}}" data-role="insert-formatting" data-comment=" " data-bbcode="quote" data-id="loc">
- <i class="fa fa-quote-left"></i>
- </button>
- </li>
- </ul>
- <div class="clear"></div>
- </div>
- </div>
-
- {{* The tab for the permissions (if event sharing is enabled) *}}
- <div id="event-acl-wrapper" style="display: none">
- {{$acl nofilter}}
- </div>
-
- {{* The tab for the event preview (content is inserted by js) *}}
- <div id="event-preview" style="display: none"></div>
-
- <div class="clear"></div>
-
- </form>
- </div>
-</div>
-
-<script type="text/javascript">
- $(document).ready( function() {
- // disable finish date input if it isn't available
- enableDisableFinishDate();
- // load bbcode autocomplete for the description textarea
- $('#comment-edit-text-desc, #comment-edit-text-loc').bbco_autocomplete('bbcode');
-
- // initiale autosize for the textareas
- autosize($("textarea.text-autosize"));
- });
-</script>
+++ /dev/null
-
-<script type="text/javascript" src="view/theme/frio/js/mod_events.js?v={{$smarty.const.FRIENDICA_VERSION}}"></script>
-
-<script type="text/javascript">
- // pass php translation strings to js variables/arrays so we can make use of it in js files
- aStr.monthNames = [
- '{{$i18n.January|escape:'quotes'}}',
- '{{$i18n.February|escape:'quotes'}}',
- '{{$i18n.March|escape:'quotes'}}',
- '{{$i18n.April|escape:'quotes'}}',
- '{{$i18n.May|escape:'quotes'}}',
- '{{$i18n.June|escape:'quotes'}}',
- '{{$i18n.July|escape:'quotes'}}',
- '{{$i18n.August|escape:'quotes'}}',
- '{{$i18n.September|escape:'quotes'}}',
- '{{$i18n.October|escape:'quotes'}}',
- '{{$i18n.November|escape:'quotes'}}',
- '{{$i18n.December|escape:'quotes'}}'
- ];
- aStr.monthNamesShort = [
- '{{$i18n.Jan|escape:'quotes'}}',
- '{{$i18n.Feb|escape:'quotes'}}',
- '{{$i18n.Mar|escape:'quotes'}}',
- '{{$i18n.Apr|escape:'quotes'}}',
- '{{$i18n.May|escape:'quotes'}}',
- '{{$i18n.Jun|escape:'quotes'}}',
- '{{$i18n.Jul|escape:'quotes'}}',
- '{{$i18n.Aug|escape:'quotes'}}',
- '{{$i18n.Sep|escape:'quotes'}}',
- '{{$i18n.Oct|escape:'quotes'}}',
- '{{$i18n.Nov|escape:'quotes'}}',
- '{{$i18n.Dec|escape:'quotes'}}'
- ];
- aStr.dayNames = [
- '{{$i18n.Sunday|escape:'quotes'}}',
- '{{$i18n.Monday|escape:'quotes'}}',
- '{{$i18n.Tuesday|escape:'quotes'}}',
- '{{$i18n.Wednesday|escape:'quotes'}}',
- '{{$i18n.Thursday|escape:'quotes'}}',
- '{{$i18n.Friday|escape:'quotes'}}',
- '{{$i18n.Saturday|escape:'quotes'}}'
- ];
- aStr.dayNamesShort = [
- '{{$i18n.Sun|escape:'quotes'}}',
- '{{$i18n.Mon|escape:'quotes'}}',
- '{{$i18n.Tue|escape:'quotes'}}',
- '{{$i18n.Wed|escape:'quotes'}}',
- '{{$i18n.Thu|escape:'quotes'}}',
- '{{$i18n.Fri|escape:'quotes'}}',
- '{{$i18n.Sat|escape:'quotes'}}'
- ];
-
- aStr.firstDay = '{{$i18n.firstDay|escape:'quotes'}}';
- aStr.today = '{{$i18n.today|escape:'quotes'}}';
- aStr.month = '{{$i18n.month|escape:'quotes'}}';
- aStr.week = '{{$i18n.week|escape:'quotes'}}';
- aStr.day = '{{$i18n.day|escape:'quotes'}}';
-
- aStr.allday = '{{$i18n.allday|escape:'quotes'}}';
- aStr.noevent = '{{$i18n.noevent|escape:'quotes'}}';
-
- aStr.dtstartLabel = '{{$i18n.dtstart_label|escape:'quotes'}}';
- aStr.dtendLabel = '{{$i18n.dtend_label|escape:'quotes'}}';
- aStr.locationLabel = '{{$i18n.location_label|escape:'quotes'}}';
-
- var moduleUrl = '{{$module_url}}';
- var modparams = {{$modparams}}
-
-
- $(document).ready(function() {
- $("#comment-edit-text-desc").bbco_autocomplete('bbcode');
- });
-</script>
+++ /dev/null
-<div class="generic-page-wrapper">
- {{$tabs nofilter}}
- {{include file="section_title.tpl" title=$title pullright=1}}
-
- {{* The link to create a new event *}}
- {{if $new_event.0}}
- <div class="pull-right" id="new-event-link">
- <button type="button" class="btn-link page-action faded-icon" onclick="addToModal('{{$new_event.0}}')" title="{{$new_event.1}}" data-toggle="tooltip">
- <i class="fa fa-plus"></i>
- </button>
- </div>
- {{/if}}
-
- {{* We create our own fullcallendar header (with title & calendar view *}}
- <div id="fc-header" class="clear">
- <div id="fc-header-right" class="pull-right">
- {{* The dropdown to change the callendar view *}}
- <ul class="nav nav-pills">
- <li class="dropdown pull-right">
- <button class="btn btn-link btn-sm dropdown-toggle" type="button" id="event-calendar-views" data-toggle="dropdown" aria-expanded="false">
- <i class="fa fa-angle-down" aria-hidden="true"></i> {{$view}}
- </button>
- <ul class="dropdown-menu pull-right" role="menu" aria-labelledby="event-calendar-views">
- <li role="presentation">
- <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'month');$('#events-calendar').fullCalendar('option', {contentHeight: '', aspectRatio: 1});">{{$month}}</button>
- </li>
- <li role="presentation">
- <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'agendaWeek');$('#events-calendar').fullCalendar('option', 'contentHeight', 'auto');">{{$week}}</button>
- </li>
- <li role="presentation">
- <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'agendaDay');$('#events-calendar').fullCalendar('option', 'contentHeight', 'auto');">{{$day}}</button>
- </li>
- <li role="presentation">
- <button role="menuitem" type="button" class="btn-link" onclick="changeView('changeView', 'listMonth');$('#events-calendar').fullCalendar('option', 'contentHeight', 'auto');">{{$list}}</button>
- </li>
- </ul>
- </li>
- </ul>
- </div>
-
- {{* The buttons to change the month/weeks/days *}}
- <div id="fc-fc-header-left" class="btn-group">
- <button class="btn btn-eventnav" onclick="changeView('prev', false);" title="{{$previous.1}}"><i class="fa fa-angle-up" aria-hidden="true"></i></button>
- <button class="btn btn-eventnav btn-separator" onclick="changeView('next', false);" title="{{$next.1}}"><i class="fa fa-angle-down" aria-hidden="true"></i></button>
- <button class="btn btn-eventnav btn-separator" onclick="changeView('today', false);" title="{{$today}}"><i class="fa fa-bullseye" aria-hidden="true"></i></button>
- </div>
-
- {{* The title (e.g. name of the mont/week/day) *}}
- <div id="event-calendar-title"><h4 id="fc-title"></h4></div>
-
- </div>
-
- {{* This is the container where the fullCalendar is inserted through js *}}
- <div id="events-calendar"></div>
-</div>
</li>
{{/if}}
- {{if $nav.events}}
+ {{if $nav.calendar}}
<li class="nav-segment hidden-xs">
- <a accesskey="e" id="nav-events-link" href="{{$nav.events.0}}" data-toggle="tooltip"
- aria-label="{{$nav.events.1}}" title="{{$nav.events.1}}" class="nav-menu"><i
+ <a accesskey="e" id="nav-calendar-link" href="{{$nav.calendar.0}}" data-toggle="tooltip"
+ aria-label="{{$nav.calendar.1}}" title="{{$nav.calendar.1}}" class="nav-menu"><i
class="fa fa-lg fa-calendar fa-fw"></i></a>
</li>
{{/if}}
$nav_info['nav']['usermenu'][] = [$server_url . '/profile/' . $remoteUser['nick'] . '/profile', DI::l10n()->t('Profile'), '', DI::l10n()->t('Your profile page')];
$nav_info['nav']['usermenu'][] = [$server_url . '/photos/' . $remoteUser['nick'], DI::l10n()->t('Photos'), '', DI::l10n()->t('Your photos')];
$nav_info['nav']['usermenu'][] = [$server_url . '/profile/' . $remoteUser['nick'] . '/media', DI::l10n()->t('Media'), '', DI::l10n()->t('Your postings with media')];
- $nav_info['nav']['usermenu'][] = [$server_url . '/events/', DI::l10n()->t('Events'), '', DI::l10n()->t('Your events')];
+ $nav_info['nav']['usermenu'][] = [$server_url . '/calendar/', DI::l10n()->t('Calendar'), '', DI::l10n()->t('Your calendar')];
// navbar links
$nav_info['nav']['network'] = [$server_url . '/network', DI::l10n()->t('Network'), '', DI::l10n()->t('Conversations from your friends')];
- $nav_info['nav']['events'] = [$server_url . '/events', DI::l10n()->t('Events'), '', DI::l10n()->t('Events and Calendar')];
+ $nav_info['nav']['calendar'] = [$server_url . '/calendar', DI::l10n()->t('Calendar'), '', DI::l10n()->t('Calendar')];
$nav_info['nav']['messages'] = [$server_url . '/message', DI::l10n()->t('Messages'), '', DI::l10n()->t('Private mail')];
$nav_info['nav']['settings'] = [$server_url . '/settings', DI::l10n()->t('Settings'), '', DI::l10n()->t('Account settings')];
$nav_info['nav']['contacts'] = [$server_url . '/contact', DI::l10n()->t('Contacts'), '', DI::l10n()->t('Manage/edit friends and contacts')];
--- /dev/null
+
+{{$tabs nofilter}}
+<h2>{{$title}} <a class="actionbutton" href="{{$new_event.0}}"><i class="icon add s10"></i> {{$new_event.1}}</a></h2>
+
+<div id="new-event-link"></div>
+
+<div id="events-calendar"></div>
--- /dev/null
+
+<h3>{{$title}}</h3>
+
+<p>{{$desc nofilter}}</p>
+
+<form id="event-edit-form" action="{{$post}}" method="post">
+
+ <input type="hidden" name="event_id" value="{{$eid}}" />
+ <input type="hidden" name="cid" value="{{$cid}}" />
+ <input type="hidden" name="uri" value="{{$uri}}" />
+ <input type="hidden" name="preview" id="event-edit-preview" value="0" />
+
+ {{$s_dsel nofilter}}
+
+ {{$f_dsel nofilter}}
+
+ {{include file="field_checkbox.tpl" field=$nofinish}}
+ <hr>
+ {{include file="field_input.tpl" field=$summary}}
+ {{include file="field_textarea.tpl" field=array('desc', $d_text, $d_orig, "")}}
+
+ {{include file="field_textarea.tpl" field=array('location', $l_text, $l_orig, "")}}
+ <hr>
+
+ {{if ! $eid}}
+ {{include file="field_checkbox.tpl" field=$share}}
+ {{/if}}
+
+ {{$acl nofilter}}
+
+ <div class="settings-submit-wrapper">
+ <input id="event-edit-preview" type="submit" name="preview" value="{{$preview}}" onclick="doEventPreview(); return false;" />
+ <input id="event-submit" type="submit" name="submit" value="{{$submit}}" />
+ </div>
+</form>
+
+<script language="javascript" type="text/javascript">
+ $(document).ready(function() {
+ $('#id_share').change(function() {
+
+ if ($('#id_share').is(':checked')) {
+ $('#acl-wrapper').show();
+ }
+ else {
+ $('#acl-wrapper').hide();
+ }
+ }).trigger('change');
+ });
+</script>
+++ /dev/null
-
-<h3>{{$title}}</h3>
-
-<p>{{$desc nofilter}}</p>
-
-<form id="event-edit-form" action="{{$post}}" method="post">
-
- <input type="hidden" name="event_id" value="{{$eid}}" />
- <input type="hidden" name="cid" value="{{$cid}}" />
- <input type="hidden" name="uri" value="{{$uri}}" />
- <input type="hidden" name="preview" id="event-edit-preview" value="0" />
-
- {{$s_dsel nofilter}}
-
- {{$f_dsel nofilter}}
-
- {{include file="field_checkbox.tpl" field=$nofinish}}
- <hr>
- {{include file="field_input.tpl" field=$summary}}
- {{include file="field_textarea.tpl" field=array('desc', $d_text, $d_orig, "")}}
-
- {{include file="field_textarea.tpl" field=array('location', $l_text, $l_orig, "")}}
- <hr>
-
- {{if ! $eid}}
- {{include file="field_checkbox.tpl" field=$share}}
- {{/if}}
-
- {{$acl nofilter}}
-
- <div class="settings-submit-wrapper">
- <input id="event-edit-preview" type="submit" name="preview" value="{{$preview}}" onclick="doEventPreview(); return false;" />
- <input id="event-submit" type="submit" name="submit" value="{{$submit}}" />
- </div>
-</form>
-
-<script language="javascript" type="text/javascript">
- $(document).ready(function() {
- $('#id_share').change(function() {
-
- if ($('#id_share').is(':checked')) {
- $('#acl-wrapper').show();
- }
- else {
- $('#acl-wrapper').hide();
- }
- }).trigger('change');
- });
-</script>
+++ /dev/null
-
-{{$tabs nofilter}}
-<h2>{{$title}} <a class="actionbutton" href="{{$new_event.0}}"><i class="icon add s10"></i> {{$new_event.1}}</a></h2>
-
-<div id="new-event-link"></div>
-
-<div id="events-calendar"></div>
+++ /dev/null
-{{$tabs nofilter}}
-<h2>{{$title}} <a class="actionbutton" href="{{$new_event.0}}"><i class="icon add s10"></i> {{$new_event.1}}</a></h2>
-
-<div id="event-calendar-wrapper">
- <a href="{{$previus.0}}" class="prevcal {{$previus.2}}"><div id="event-calendar-prev" class="icon s22 prev" title="{{$previus.1}}"></div></a>
- {{$calendar}}
- <a href="{{$next.0}}" class="nextcal {{$next.2}}"><div id="event-calendar-prev" class="icon s22 next" title="{{$next.1}}"></div></a>
-</div>
-<div class="event-calendar-end"></div>
-
-{{foreach $events as $event}}
- <div class="event">
- {{if $event.is_first}}<hr /><a name="link-{{$event.j}}"><div class="event-list-date">{{$event.d}}</div></a>{{/if}}
- {{if $event.item.author_name}}<a href="{{$event.item.author_link}}"><img src="{{$event.item.author_avatar}}" height="32" width="32" />{{$event.item.author_name}}</a>{{/if}}
- {{$event.html nofilter}}
- {{if $event.item.plink}}<a href="{{$event.plink.0}}" title="{{$event.plink.1}}" target="_blank" rel="noopener noreferrer" class="plink-event-link icon s22 remote-link"></a>{{/if}}
- {{if $event.edit}}<a href="{{$event.edit.0}}" title="{{$event.edit.1}}" class="edit-event-link icon s22 pencil"></a>{{/if}}
- </div>
- <div class="clear"></div>
-{{/foreach}}
function showEvent(eventid) {
$.get(
- '{{$baseurl}}/events/?id='+eventid,
+ '{{$event_api}}/'+eventid,
function(data){
$.colorbox({html:data});
}
year: yesterday.getFullYear(),
month: yesterday.getMonth(),
date: yesterday.getDate(),
- events: '{{$baseurl}}/calendar/json',
+ events: '{{$baseurl}}/calendar/api/get',
header: false,
timeFormat: 'H(:mm)',
defaultView: 'basicWeek',
--- /dev/null
+
+<link rel="stylesheet" type="text/css" href="{{$baseurl}}/view/asset/fullcalendar/dist/fullcalendar.min.css?v={{$smarty.const.FRIENDICA_VERSION}}" />
+<link rel="stylesheet" type="text/css" href="{{$baseurl}}/view/asset/fullcalendar/dist/fullcalendar.print.min.css?v={{$smarty.const.FRIENDICA_VERSION}}" media="print" />
+<script type="text/javascript" src="{{$baseurl}}/view/asset/moment/min/moment-with-locales.min.js?v={{$smarty.const.FRIENDICA_VERSION}}"></script>
+<script type="text/javascript" src="{{$baseurl}}/view/asset/fullcalendar/dist/fullcalendar.min.js?v={{$smarty.const.FRIENDICA_VERSION}}"></script>
+
+<script>
+ function showEvent(eventid) {
+ $.get(
+ '{{$event_api}}/'+eventid,
+ function(data){
+ $.colorbox({html:data});
+ }
+ );
+ }
+
+ function doEventPreview() {
+ $('#event-edit-preview').val(1);
+ $.post('events',$('#event-edit-form').serialize(), function(data) {
+ $.colorbox({ html: data });
+ });
+ $('#event-edit-preview').val(0);
+ }
+
+ // disable the input for the finish date if it is not available
+ function enableDisableFinishDate() {
+ if( $('#id_nofinish').is(':checked'))
+ $('#id_finish_text').prop("disabled", true);
+ else
+ $('#id_finish_text').prop("disabled", false);
+ }
+
+ $(document).ready(function() {
+ $('#events-calendar').fullCalendar({
+ firstDay: '{{$i18n.firstDay|escape:'quotes'}}',
+ monthNames: [
+ '{{$i18n.January|escape:'quotes'}}',
+ '{{$i18n.February|escape:'quotes'}}',
+ '{{$i18n.March|escape:'quotes'}}',
+ '{{$i18n.April|escape:'quotes'}}',
+ '{{$i18n.May|escape:'quotes'}}',
+ '{{$i18n.June|escape:'quotes'}}',
+ '{{$i18n.July|escape:'quotes'}}',
+ '{{$i18n.August|escape:'quotes'}}',
+ '{{$i18n.September|escape:'quotes'}}',
+ '{{$i18n.October|escape:'quotes'}}',
+ '{{$i18n.November|escape:'quotes'}}',
+ '{{$i18n.December|escape:'quotes'}}'
+ ],
+ monthNamesShort: [
+ '{{$i18n.Jan|escape:'quotes'}}',
+ '{{$i18n.Feb|escape:'quotes'}}',
+ '{{$i18n.Mar|escape:'quotes'}}',
+ '{{$i18n.Apr|escape:'quotes'}}',
+ '{{$i18n.May|escape:'quotes'}}',
+ '{{$i18n.Jun|escape:'quotes'}}',
+ '{{$i18n.Jul|escape:'quotes'}}',
+ '{{$i18n.Aug|escape:'quotes'}}',
+ '{{$i18n.Sep|escape:'quotes'}}',
+ '{{$i18n.Oct|escape:'quotes'}}',
+ '{{$i18n.Nov|escape:'quotes'}}',
+ '{{$i18n.Dec|escape:'quotes'}}'
+ ],
+ dayNames: [
+ '{{$i18n.Sunday|escape:'quotes'}}',
+ '{{$i18n.Monday|escape:'quotes'}}',
+ '{{$i18n.Tuesday|escape:'quotes'}}',
+ '{{$i18n.Wednesday|escape:'quotes'}}',
+ '{{$i18n.Thursday|escape:'quotes'}}',
+ '{{$i18n.Friday|escape:'quotes'}}',
+ '{{$i18n.Saturday|escape:'quotes'}}'
+ ],
+ dayNamesShort: [
+ '{{$i18n.Sun|escape:'quotes'}}',
+ '{{$i18n.Mon|escape:'quotes'}}',
+ '{{$i18n.Tue|escape:'quotes'}}',
+ '{{$i18n.Wed|escape:'quotes'}}',
+ '{{$i18n.Thu|escape:'quotes'}}',
+ '{{$i18n.Fri|escape:'quotes'}}',
+ '{{$i18n.Sat|escape:'quotes'}}'
+ ],
+ allDayText: '{{$i18n.allday|escape:'quotes'}}',
+ noEventsMessage: '{{$i18n.noevent|escape:'quotes'}}',
+ buttonText: {
+ today: '{{$i18n.today|escape:'quotes'}}',
+ month: '{{$i18n.month|escape:'quotes'}}',
+ week: '{{$i18n.week|escape:'quotes'}}',
+ day: '{{$i18n.day|escape:'quotes'}}'
+ },
+ events: '{{$calendar_api}}',
+ header: {
+ left: 'prev,next today',
+ center: 'title',
+ right: 'month,agendaWeek,agendaDay'
+ },
+ timeFormat: 'H:mm',
+ eventClick: function(calEvent, jsEvent, view) {
+ showEvent(calEvent.id);
+ },
+ loading: function(isLoading, view) {
+ if(!isLoading) {
+ $('td.fc-day').dblclick(function() { window.location.href='/calendar/event/new?start='+$(this).data('date'); });
+ }
+ },
+
+ eventRender: function(event, element, view) {
+ if (event.item['author-name']==null) return;
+ switch(view.name){
+ case "month":
+ element.find(".fc-title").html(
+ "{0}".format(
+ event.title
+ ));
+ break;
+ case "agendaWeek":
+ element.find(".fc-title").html(
+ "{0}<p>{1}</p><p>{2}</p>".format(
+ event.item['author-name'],
+ event.item.desc,
+ event.item.location
+ ));
+ break;
+ case "agendaDay":
+ element.find(".fc-title").html(
+ "{0}<p>{1}</p><p>{2}</p>".format(
+ event.item['author-name'],
+ event.item.desc,
+ event.item.location
+ ));
+ break;
+ }
+ }
+
+ })
+
+ // show event popup
+ var hash = location.hash.split("-")
+ if (hash.length==2 && hash[0]=="#link") showEvent(hash[1]);
+
+ });
+</script>
+
+<script language="javascript" type="text/javascript">
+
+ $(document).ready(function() {
+ $("#comment-edit-text-desc").bbco_autocomplete('bbcode');
+
+ $('#id_share').change(function() {
+
+ if ($('#id_share').is(':checked')) {
+ $('#acl-wrapper').show();
+ }
+ else {
+ $('#acl-wrapper').hide();
+ }
+ }).trigger('change');
+
+ $('#contact_allow, #contact_deny, #group_allow, #group_deny').change(function() {
+ var selstr;
+ $('#contact_allow option:selected, #contact_deny option:selected, #group_allow option:selected, #group_deny option:selected').each( function() {
+ selstr = $(this).html();
+ $('#jot-public').hide();
+ });
+ if(selstr == null) {
+ $('#jot-public').show();
+ }
+
+ }).trigger('change');
+
+ // disable the finish time input if the user disable it
+ $('#id_nofinish').change(function() {
+ enableDisableFinishDate()
+ }).trigger('change');
+
+ });
+
+</script>
+
--- /dev/null
+
+
+<h3>{{$title}}</h3>
+
+<p>
+{{$desc nofilter}}
+</p>
+
+<form id="event-edit-form" action="{{$post}}" method="post">
+
+<input type="hidden" name="event_id" value="{{$eid}}" />
+<input type="hidden" name="cid" value="{{$cid}}" />
+<input type="hidden" name="uri" value="{{$uri}}" />
+<input type="hidden" name="preview" id="event-edit-preview" value="0" />
+
+{{$s_dsel nofilter}}
+
+{{$f_dsel nofilter}}
+
+{{include file="field_checkbox.tpl" field=$nofinish}}
+
+{{include file="field_input.tpl" field=$summary}}
+
+
+<div id="event-desc-text">{{$d_text}}</div>
+<textarea id="comment-edit-text-desc" rows="8" cols="64" name="desc" autocomplete="off" dir="auto">{{$d_orig}}</textarea>
+<div id="event-desc-text-edit-bb" class="comment-edit-bb">
+ <a title="{{$edimg}}" data-role="insert-formatting" data-bbcode="img" data-id="desc"><i class="icon-picture"></i></a>
+ <a title="{{$edurl}}" data-role="insert-formatting" data-bbcode="url" data-id="desc"><i class="icon-link"></i></a>
+ <a title="{{$edvideo}}" data-role="insert-formatting" data-bbcode="video" data-id="desc"><i class="icon-film"></i></a>
+
+ <a title="{{$eduline}}" data-role="insert-formatting" data-bbcode="u" data-id="desc"><i class="icon-underline"></i></a>
+ <a title="{{$editalic}}" data-role="insert-formatting" data-bbcode="i" data-id="desc"><i class="icon-italic"></i></a>
+ <a title="{{$edbold}}" data-role="insert-formatting" data-bbcode="b" data-id="desc"><i class="icon-bold"></i></a>
+ <a title="{{$edquote}}" data-role="insert-formatting" data-bbcode="quote" data-id="desc"><i class="icon-quote-left"></i></a>
+</div>
+
+<div id="event-location-text">{{$l_text}}</div>
+<textarea id="comment-edit-text-location" rows="4" cols="64" name="location" dir="auto">{{$l_orig}}</textarea>
+<div id="event-location-text-edit-bb" class="comment-edit-bb">
+ <a title="{{$edimg}}" data-role="insert-formatting" data-bbcode="img" data-id="location"><i class="icon-picture"></i></a>
+ <a title="{{$edurl}}" data-role="insert-formatting" data-bbcode="url" data-id="location"><i class="icon-link"></i></a>
+ <a title="{{$edvideo}}" data-role="insert-formatting" data-bbcode="video" data-id="location"><i class="icon-film"></i></a>
+
+ <a title="{{$eduline}}" data-role="insert-formatting" data-bbcode="u" data-id="location"><i class="icon-underline"></i></a>
+ <a title="{{$editalic}}" data-role="insert-formatting" data-bbcode="i" data-id="location"><i class="icon-italic"></i></a>
+ <a title="{{$edbold}}" data-role="insert-formatting" data-bbcode="b" data-id="location"><i class="icon-bold"></i></a>
+ <a title="{{$edquote}}" data-role="insert-formatting" data-bbcode="quote" data-id="location"><i class="icon-quote-left"></i></a>
+</div>
+
+<div id="event-location-break"></div>
+
+{{if ! $eid}}
+{{include file="field_checkbox.tpl" field=$share}}
+{{/if}}
+
+{{$acl nofilter}}
+
+<div class="clear"></div>
+<input id="event-edit-preview" type="submit" name="preview" value="{{$preview}}" onclick="doEventPreview(); return false;" />
+<input id="event-submit" type="submit" name="submit" value="{{$submit}}" />
+</form>
+++ /dev/null
-
-
-<h3>{{$title}}</h3>
-
-<p>
-{{$desc nofilter}}
-</p>
-
-<form id="event-edit-form" action="{{$post}}" method="post">
-
-<input type="hidden" name="event_id" value="{{$eid}}" />
-<input type="hidden" name="cid" value="{{$cid}}" />
-<input type="hidden" name="uri" value="{{$uri}}" />
-<input type="hidden" name="preview" id="event-edit-preview" value="0" />
-
-{{$s_dsel nofilter}}
-
-{{$f_dsel nofilter}}
-
-{{include file="field_checkbox.tpl" field=$nofinish}}
-
-{{include file="field_input.tpl" field=$summary}}
-
-
-<div id="event-desc-text">{{$d_text}}</div>
-<textarea id="comment-edit-text-desc" rows="8" cols="64" name="desc" autocomplete="off" dir="auto">{{$d_orig}}</textarea>
-<div id="event-desc-text-edit-bb" class="comment-edit-bb">
- <a title="{{$edimg}}" data-role="insert-formatting" data-bbcode="img" data-id="desc"><i class="icon-picture"></i></a>
- <a title="{{$edurl}}" data-role="insert-formatting" data-bbcode="url" data-id="desc"><i class="icon-link"></i></a>
- <a title="{{$edvideo}}" data-role="insert-formatting" data-bbcode="video" data-id="desc"><i class="icon-film"></i></a>
-
- <a title="{{$eduline}}" data-role="insert-formatting" data-bbcode="u" data-id="desc"><i class="icon-underline"></i></a>
- <a title="{{$editalic}}" data-role="insert-formatting" data-bbcode="i" data-id="desc"><i class="icon-italic"></i></a>
- <a title="{{$edbold}}" data-role="insert-formatting" data-bbcode="b" data-id="desc"><i class="icon-bold"></i></a>
- <a title="{{$edquote}}" data-role="insert-formatting" data-bbcode="quote" data-id="desc"><i class="icon-quote-left"></i></a>
-</div>
-
-<div id="event-location-text">{{$l_text}}</div>
-<textarea id="comment-edit-text-location" rows="4" cols="64" name="location" dir="auto">{{$l_orig}}</textarea>
-<div id="event-location-text-edit-bb" class="comment-edit-bb">
- <a title="{{$edimg}}" data-role="insert-formatting" data-bbcode="img" data-id="location"><i class="icon-picture"></i></a>
- <a title="{{$edurl}}" data-role="insert-formatting" data-bbcode="url" data-id="location"><i class="icon-link"></i></a>
- <a title="{{$edvideo}}" data-role="insert-formatting" data-bbcode="video" data-id="location"><i class="icon-film"></i></a>
-
- <a title="{{$eduline}}" data-role="insert-formatting" data-bbcode="u" data-id="location"><i class="icon-underline"></i></a>
- <a title="{{$editalic}}" data-role="insert-formatting" data-bbcode="i" data-id="location"><i class="icon-italic"></i></a>
- <a title="{{$edbold}}" data-role="insert-formatting" data-bbcode="b" data-id="location"><i class="icon-bold"></i></a>
- <a title="{{$edquote}}" data-role="insert-formatting" data-bbcode="quote" data-id="location"><i class="icon-quote-left"></i></a>
-</div>
-
-<div id="event-location-break"></div>
-
-{{if ! $eid}}
-{{include file="field_checkbox.tpl" field=$share}}
-{{/if}}
-
-{{$acl nofilter}}
-
-<div class="clear"></div>
-<input id="event-edit-preview" type="submit" name="preview" value="{{$preview}}" onclick="doEventPreview(); return false;" />
-<input id="event-submit" type="submit" name="submit" value="{{$submit}}" />
-</form>
+++ /dev/null
-
-<link rel="stylesheet" type="text/css" href="{{$baseurl}}/view/asset/fullcalendar/dist/fullcalendar.min.css?v={{$smarty.const.FRIENDICA_VERSION}}" />
-<link rel="stylesheet" type="text/css" href="{{$baseurl}}/view/asset/fullcalendar/dist/fullcalendar.print.min.css?v={{$smarty.const.FRIENDICA_VERSION}}" media="print" />
-<script type="text/javascript" src="{{$baseurl}}/view/asset/moment/min/moment-with-locales.min.js?v={{$smarty.const.FRIENDICA_VERSION}}"></script>
-<script type="text/javascript" src="{{$baseurl}}/view/asset/fullcalendar/dist/fullcalendar.min.js?v={{$smarty.const.FRIENDICA_VERSION}}"></script>
-
-<script>
- function showEvent(eventid) {
- $.get(
- '{{$baseurl}}{{$module_url}}/?id='+eventid,
- function(data){
- $.colorbox({html:data});
- }
- );
- }
-
- function doEventPreview() {
- $('#event-edit-preview').val(1);
- $.post('events',$('#event-edit-form').serialize(), function(data) {
- $.colorbox({ html: data });
- });
- $('#event-edit-preview').val(0);
- }
-
- // disable the input for the finish date if it is not available
- function enableDisableFinishDate() {
- if( $('#id_nofinish').is(':checked'))
- $('#id_finish_text').prop("disabled", true);
- else
- $('#id_finish_text').prop("disabled", false);
- }
-
- $(document).ready(function() {
- $('#events-calendar').fullCalendar({
- firstDay: '{{$i18n.firstDay|escape:'quotes'}}',
- monthNames: [
- '{{$i18n.January|escape:'quotes'}}',
- '{{$i18n.February|escape:'quotes'}}',
- '{{$i18n.March|escape:'quotes'}}',
- '{{$i18n.April|escape:'quotes'}}',
- '{{$i18n.May|escape:'quotes'}}',
- '{{$i18n.June|escape:'quotes'}}',
- '{{$i18n.July|escape:'quotes'}}',
- '{{$i18n.August|escape:'quotes'}}',
- '{{$i18n.September|escape:'quotes'}}',
- '{{$i18n.October|escape:'quotes'}}',
- '{{$i18n.November|escape:'quotes'}}',
- '{{$i18n.December|escape:'quotes'}}'
- ],
- monthNamesShort: [
- '{{$i18n.Jan|escape:'quotes'}}',
- '{{$i18n.Feb|escape:'quotes'}}',
- '{{$i18n.Mar|escape:'quotes'}}',
- '{{$i18n.Apr|escape:'quotes'}}',
- '{{$i18n.May|escape:'quotes'}}',
- '{{$i18n.Jun|escape:'quotes'}}',
- '{{$i18n.Jul|escape:'quotes'}}',
- '{{$i18n.Aug|escape:'quotes'}}',
- '{{$i18n.Sep|escape:'quotes'}}',
- '{{$i18n.Oct|escape:'quotes'}}',
- '{{$i18n.Nov|escape:'quotes'}}',
- '{{$i18n.Dec|escape:'quotes'}}'
- ],
- dayNames: [
- '{{$i18n.Sunday|escape:'quotes'}}',
- '{{$i18n.Monday|escape:'quotes'}}',
- '{{$i18n.Tuesday|escape:'quotes'}}',
- '{{$i18n.Wednesday|escape:'quotes'}}',
- '{{$i18n.Thursday|escape:'quotes'}}',
- '{{$i18n.Friday|escape:'quotes'}}',
- '{{$i18n.Saturday|escape:'quotes'}}'
- ],
- dayNamesShort: [
- '{{$i18n.Sun|escape:'quotes'}}',
- '{{$i18n.Mon|escape:'quotes'}}',
- '{{$i18n.Tue|escape:'quotes'}}',
- '{{$i18n.Wed|escape:'quotes'}}',
- '{{$i18n.Thu|escape:'quotes'}}',
- '{{$i18n.Fri|escape:'quotes'}}',
- '{{$i18n.Sat|escape:'quotes'}}'
- ],
- allDayText: '{{$i18n.allday|escape:'quotes'}}',
- noEventsMessage: '{{$i18n.noevent|escape:'quotes'}}',
- buttonText: {
- today: '{{$i18n.today|escape:'quotes'}}',
- month: '{{$i18n.month|escape:'quotes'}}',
- week: '{{$i18n.week|escape:'quotes'}}',
- day: '{{$i18n.day|escape:'quotes'}}'
- },
- events: '{{$baseurl}}/calendar/json',
- header: {
- left: 'prev,next today',
- center: 'title',
- right: 'month,agendaWeek,agendaDay'
- },
- timeFormat: 'H:mm',
- eventClick: function(calEvent, jsEvent, view) {
- showEvent(calEvent.id);
- },
- loading: function(isLoading, view) {
- if(!isLoading) {
- $('td.fc-day').dblclick(function() { window.location.href='/events/new?start='+$(this).data('date'); });
- }
- },
-
- eventRender: function(event, element, view) {
- if (event.item['author-name']==null) return;
- switch(view.name){
- case "month":
- element.find(".fc-title").html(
- "{0}".format(
- event.title
- ));
- break;
- case "agendaWeek":
- element.find(".fc-title").html(
- "{0}<p>{1}</p><p>{2}</p>".format(
- event.item['author-name'],
- event.item.desc,
- event.item.location
- ));
- break;
- case "agendaDay":
- element.find(".fc-title").html(
- "{0}<p>{1}</p><p>{2}</p>".format(
- event.item['author-name'],
- event.item.desc,
- event.item.location
- ));
- break;
- }
- }
-
- })
-
- // center on date
- var args=location.href.replace(baseurl,"").split("/");
-{{if $modparams == 2}}
- if (args.length>=5) {
- $("#events-calendar").fullCalendar('gotoDate',args[3] , args[4]-1);
- }
-{{else}}
- if (args.length>=4) {
- $("#events-calendar").fullCalendar('gotoDate',args[2] , args[3]-1);
- }
-{{/if}}
-
- // show event popup
- var hash = location.hash.split("-")
- if (hash.length==2 && hash[0]=="#link") showEvent(hash[1]);
-
- });
-</script>
-
-<script language="javascript" type="text/javascript">
-
- $(document).ready(function() {
- $("#comment-edit-text-desc").bbco_autocomplete('bbcode');
-
- $('#id_share').change(function() {
-
- if ($('#id_share').is(':checked')) {
- $('#acl-wrapper').show();
- }
- else {
- $('#acl-wrapper').hide();
- }
- }).trigger('change');
-
- $('#contact_allow, #contact_deny, #group_allow, #group_deny').change(function() {
- var selstr;
- $('#contact_allow option:selected, #contact_deny option:selected, #group_allow option:selected, #group_deny option:selected').each( function() {
- selstr = $(this).html();
- $('#jot-public').hide();
- });
- if(selstr == null) {
- $('#jot-public').show();
- }
-
- }).trigger('change');
-
- // disable the finish time input if the user disable it
- $('#id_nofinish').change(function() {
- enableDisableFinishDate()
- }).trigger('change');
-
- });
-
-</script>
-
</li>
{{/if}}
{{if $nav.events}}
- <li role="menuitem" id="nav-events-link" class="nav-menu {{$sel.events}}">
+ <li role="menuitem" id="nav-calendar-link" class="nav-menu {{$sel.events}}">
<a accesskey="e" class="{{$nav.events.2}} desktop-view" href="{{$nav.events.0}}" title="{{$nav.events.3}}">{{$nav.events.1}}</a>
<a class="{{$nav.events.2}} mobile-view" href="{{$nav.events.0}}" title="{{$nav.events.3}}"><i class="icon s22 icon-calendar"></i></a>
</li>
// Hide the left menu bar
/// @TODO maybe move this static array out where it should belong?
- if (empty(DI::page()['aside']) && in_array($args->get(0), ["community", "events", "help", "delegation", "notifications",
+ if (empty(DI::page()['aside']) && in_array($args->get(0), ["community", "calendar", "help", "delegation", "notifications",
"probe", "webfinger", "login", "invite", "credits"])) {
DI::page()['htmlhead'] .= "<link rel='stylesheet' href='view/theme/vier/hide.css' />";
}