* @copyright (c) 2004-16 by RalfBecker-At-outdoor-training.de * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @version $Id$ */ /** * diverse static calendar hooks */ class calendar_hooks { /** * Hook called by link-class to include calendar in the appregistry of the linkage * * @param array|string $location location and other parameters (not used) * @return array with method-names */ static function search_link($location) { unset($location); // not used, but in function signature for hooks return array( 'query' => 'calendar.calendar_bo.link_query', 'title' => 'calendar.calendar_bo.link_title', 'view' => array( 'menuaction' => 'calendar.calendar_uiforms.edit', ), 'view_id' => 'cal_id', 'view_popup' => '790x400', 'edit_popup' => '790x400', 'list' => array( 'menuaction' => 'calendar.calendar_uiviews.index', 'view' => 'listview', 'ajax'=>'true' ), 'add' => array( 'menuaction' => 'calendar.calendar_uiforms.edit', ), 'add_app' => 'link_app', 'add_id' => 'link_id', 'add_popup' => '790x400', 'file_access' => 'calendar.calendar_bo.file_access', 'file_access_user' => true, // file_access supports 4th parameter $user 'mime' => array( 'text/calendar' => array( 'menuaction' => 'calendar.calendar_uiforms.edit', 'mime_data' => 'ical_data', 'mime_url' => 'ical_url', 'mime_popup' => '790x400', 'mime_target' => '_blank' ), ), 'merge' => true, 'entry' => 'Event', 'entries' => 'Events', ); } /** * Hook called to retrieve a app specific exportLimit * * @param array|string $location location and other parameters (not used) * @return the export_limit to be applied for the app, may be empty, int or string */ static function getAppExportLimit($location) { unset($location); // not used, but in function signature for hooks return $GLOBALS['egw_info']['server']['calendar_export_limit']; } /** * Entries for calendar's admin menu */ static function admin() { $file = Array( 'Site Configuration' => egw::link('/index.php','menuaction=admin.admin_config.index&appname=calendar&ajax=true'), 'Custom fields' => egw::link('/index.php','menuaction=admin.customfields.index&appname=calendar'), 'Calendar Holiday Management' => egw::link('/index.php','menuaction=calendar.uiholiday.admin'), 'Global Categories' => egw::link('/index.php','menuaction=admin.admin_categories.index&appname=calendar'), 'Category ACL' => egw::link('/index.php','menuaction=calendar.calendar_uiforms.cat_acl'), 'Update timezones' => egw::link('/index.php','menuaction=calendar.calendar_timezones.update'), ); display_section('calendar','calendar',$file); } /** * Calendar preferences * * @param array $hook_data */ static function settings($hook_data) { if (!$hook_data['setup']) // does not work on setup time { $bo = new calendar_bo(); $bo->check_set_default_prefs(); } $mainscreen = array( '0' => lang('None'), 'day' => lang('Dayview'), 'day4' => lang('Four days view'), '1' => lang('Weekview'), 'weekN' => lang('Multiple week view'), 'month' => lang('Monthview'), 'year' => lang('yearview'), 'planner_cat' => lang('Planner by category'), 'planner_user' => lang('Planner by user'), 'listview' => lang('Listview'), ); $yesno = array( '1' => lang('Yes'), '0' => lang('No'), ); $list_views = array( 0 => lang('None'), 'weekN' => lang('Multiple week view'), 'week' => lang('Weekview'), 'day4' => lang('Four days view'), 'day' => lang('Dayview'), ); $updates = array( 'no' => lang('Never'), 'add_cancel' => lang('on invitation / cancelation only'), 'time_change_4h' => lang('on time change of more than 4 hours too'), 'time_change' => lang('on any time change too'), 'modifications' => lang('on all modification, but responses'), 'responses' => lang('on participant responses too') ); $update_formats = array( 'none' => lang('None'), 'extended' => lang('Extended'), 'ical' => lang('iCal / rfc2445'), ); $event_details = array( 'to-fullname' => lang('Fullname of person to notify'), 'to-firstname'=> lang('Firstname of person to notify'), 'to-lastname' => lang('Lastname of person to notify'), 'title' => lang('Title of the event'), 'description' => lang('Description'), 'startdate' => lang('Start Date/Time'), 'enddate' => lang('End Date/Time'), 'olddate' => lang('Old Startdate'), 'category' => lang('Category'), 'location' => lang('Location'), 'priority' => lang('Priority'), 'participants'=> lang('Participants'), 'owner' => lang('Owner'), 'repetition' => lang('Repetitiondetails (or empty)'), 'action' => lang('Action that caused the notify: Added, Canceled, Accepted, Rejected, ...'), 'link' => lang('Link to view the event'), 'disinvited' => lang('Participants disinvited from an event'), ); $weekdaystarts = array( 'Monday' => lang('Monday'), 'Sunday' => lang('Sunday'), 'Saturday' => lang('Saturday') ); for ($i=0; $i < 24; ++$i) { $times[$i] = $GLOBALS['egw']->common->formattime($i,'00'); } for ($i = 2; $i <= 4; ++$i) { $muliple_weeks[$i] = lang('%1 weeks',$i); } for ($i = 2; $i <= 9; $i++) { $consolidated[$i] = $i; } $intervals = array( 5 => '5', 10 => '10', 15 => '15', 20 => '20', 30 => '30', 45 => '45', 60 => '60' ); $default_participants = array( 0 => lang('Just me'), 'selected' => lang('Selected users/groups') ); $defaultresource_sel = array( 'resources_conflict' => lang('resources with conflict detection'), 'resources_without_conflict' => lang('resources except conflicting ones'), 'resources' => lang('resources') ); $reset_stati_on_shifts = array( 'no' => lang('Never'), 'all' => lang('Always'), 'startday' => lang('If start day differs'), ); $freebusy_values = array( 0 => lang('No'), 1 => lang('Yes'), 2 => lang('With credentials included'), ); if (!$hook_data['setup']) // does not work at setup time { $options = array('0' => lang('none')); foreach($GLOBALS['egw']->accounts->search(array('type' => 'owngroups','app' => 'calendar')) as $group) { $options[$group['account_id']] = common::grab_owner_name($group['account_id']); } $freebusy_url = calendar_bo::freebusy_url($GLOBALS['egw_info']['user']['account_lid'],$GLOBALS['egw_info']['user']['preferences']['calendar']['freebusy_pw']); $freebusy_url = ''.$freebusy_url.''; $freebusy_help = lang('Should not loged in persons be able to see your freebusy information? You can set an extra password, different from your normal password, to protect this informations. The freebusy information is in iCal format and only include the times when you are busy. It does not include the event-name, description or locations. The URL to your freebusy information is'); $freebusy_help .= ' ' . $freebusy_url; // Timezone for file exports $export_tzs = array('0' => 'Use Event TZ'); $export_tzs += egw_time::getTimezones(); } $link_title_options = ExecMethod('calendar.calendar_bo.get_link_options'); $settings = array( '1.section' => array( 'type' => 'section', 'title' => lang('General settings'), 'no_lang'=> true, 'xmlrpc' => False, 'admin' => False ), /* disabled until we have a home app again 'mainscreen_showevents' => array( 'type' => 'select', 'label' => 'Which view to show on home page', 'name' => 'mainscreen_showevents', 'values' => $mainscreen, 'help' => 'Displays this calendar view on the home page (page you get when you enter EGroupware or click on the home page icon)?', 'xmlrpc' => True, 'admin' => False, 'default'=> '1', // 1 = week ),*/ 'multiple_weeks' => array( 'type' => 'select', 'label' => 'Weeks in multiple week view', 'name' => 'multiple_weeks', 'values' => $muliple_weeks, 'help' => 'How many weeks should the multiple week view show?', 'xmlrpc' => True, 'admin' => False, 'default'=> 2, ), 'weekdaystarts' => array( 'type' => 'select', 'label' => 'weekday starts on', 'name' => 'weekdaystarts', 'values' => $weekdaystarts, 'help' => 'This day is shown as first day in the week or month view.', 'xmlrpc' => True, 'admin' => False, 'forced' => 'Monday', ), 'workdaystarts' => array( 'type' => 'select', 'label' => 'work day starts on', 'name' => 'workdaystarts', 'values' => $times, 'help' => 'This defines the start of your dayview. Events before this time, are shown above the dayview.
This time is also used as a default starttime for new events.', 'xmlrpc' => True, 'admin' => False, 'default'=> 9, ), 'workdayends' => array( 'type' => 'select', 'label' => 'work day ends on', 'name' => 'workdayends', 'values' => $times, 'help' => 'This defines the end of your dayview. Events after this time, are shown below the dayview.', 'xmlrpc' => True, 'admin' => False, 'default'=> 18, ), 'interval' => array( 'type' => 'select', 'label' => 'Length of the time interval', 'name' => 'interval', 'values' => $intervals, 'help' => 'How many minutes should each interval last?', 'xmlrpc' => True, 'admin' => False, 'default'=> 30, ), 'day_consolidate' => array( 'type' => 'select', 'label' => 'Minimum number of users for showing day view as consolidated.', 'name' => 'day_consolidate', 'values'=> $consolidated, 'help' => 'How many seperate calendars to show before merging them together', 'default'=> 6 ), 'week_consolidate' => array( 'type' => 'select', 'label' => 'Minimum number of users for showing week view as consolidated.', 'name' => 'week_consolidate', 'values'=> $consolidated, 'help' => 'How many seperate calendars to show before merging them together', 'default'=> 4 ), 'limit_des_lines' => array( 'type' => 'input', 'size' => 5, 'label' => 'Limit number of description lines in list view (default 5, 0 for no limit)', 'name' => 'limit_des_lines', 'help' => 'How many describtion lines should be directly visible. Further lines are available via a scrollbar.', 'xmlrpc' => True, 'admin' => False ), 'planner_show_empty_rows' => array( 'type' => 'select', 'label' => 'Show empty rows in Planner', 'name' => 'planner_show_empty_rows', 'values' => array( 0 => lang('no'), 'user' => lang('Planner by user'), 'cat' => lang('Planner by category'), 'both' => lang('All'), ), 'help' => 'Should the planner display an empty row for users or categories without any appointment.', 'xmlrpc' => True, 'admin' => False, 'forced' => 'user', ), 'use_time_grid' => array( 'type' => 'multiselect', 'label' => 'Views showing a list of events', 'name' => 'use_time_grid', 'values' => $list_views, 'help' => 'For which views should calendar just a list of events instead of distinct lines with a fixed time interval.', 'xmlrpc' => True, 'admin' => False, 'default' => ['weekN'], ), 'link_title' => array( 'type' => 'multiselect', 'label' => 'Link title for events to show', 'name' => 'link_title', 'values' => $link_title_options, 'help' => 'What should links to the calendar events display in other applications.', 'xmlrpc' => True, 'admin' => false, 'default'=> '', ), '2.section' => array( 'type' => 'section', 'title' => lang('appointment settings'), 'no_lang'=> true, 'xmlrpc' => False, 'admin' => False ), 'defaultlength' => array( 'type' => 'input', 'label' => 'default appointment length (in minutes)', 'name' => 'defaultlength', 'help' => 'Default length of newly created events. The length is in minutes, eg. 60 for 1 hour.', 'default' => '', 'size' => 3, 'xmlrpc' => True, 'admin' => False, 'default'=> 60, ), 'default_participant' => array( 'type' => 'select', 'label' => 'New event participants', 'name' => 'default_participant', 'values'=> $default_participants, 'help' => 'Participants automatically added to new events', 'default' => 'selected', 'xmlrpc' => False, 'admin' => False ), 'default-alarm' => array( 'type' => 'date-duration',//'select', 'label' => lang('Default alarm for regular events').' ('.lang('empty = no alarm').')', 'name' => 'default-alarm', 'help' => 'Alarm added automatic to new events before event start-time', 'xmlrpc' => True, 'admin' => False, 'default' => '', ), 'default-alarm-wholeday' => array( 'type' => 'date-duration',//'select', 'label' => lang('Default alarm for whole-day events').' ('.lang('empty = no alarm').')', 'name' => 'default-alarm-wholeday', 'help' => lang('Alarm added automatic to new events before event start-time').' ('.lang('Midnight').')', 'xmlrpc' => True, 'admin' => False, 'default' => '', ), ); if (isset($bo)) // add custom time-spans set by CalDAV clients, not in our prefs { $prefs = $GLOBALS['egw_info']['user']['preferences']['calendar']; $data = array( 'prefs' => &$prefs, // use reference to get preference value back 'preprocess' => true, 'type' => 'user', ); self::verify_settings_reference($data); } $settings += array( 'defaultresource_sel' => array( 'type' => 'select', 'label' => 'default type of resources selection', 'name' => 'defaultresource_sel', 'values' => $defaultresource_sel, 'help' => 'Default type of resources application selected in the calendar particpants research form.', 'xmlrpc' => True, 'admin' => False, 'forced' => 'resources', 'default' => 'resources' ), 'default_private' => array( 'type' => 'check', 'label' => 'Set new events to private', 'name' => 'default_private', 'help' => 'Should new events created as private by default ?', 'xmlrpc' => True, 'admin' => False, 'forced' => '0', ), 'reset_stati' => array( 'type' => 'select', 'label' => 'Reset participant stati on event shifts', 'name' => 'reset_stati', 'help' => 'Select whether you want the pariticpant stati reset to unkown, if an event is shifted later on.', 'values' => $reset_stati_on_shifts, 'default' => 'no', 'xmlrpc' => True, 'admin' => False, ), '3.section' => array( 'type' => 'section', 'title' => lang('notification settings'), 'no_lang'=> true, 'xmlrpc' => False, 'admin' => False ), 'receive_updates' => array( 'type' => 'select', 'label' => 'Receive email updates', 'name' => 'receive_updates', 'values' => $updates, 'help' => "Do you want to be notified about new or changed appointments? You be notified about changes you make yourself.
You can limit the notifications to certain changes only. Each item includes all the notification listed above it. All modifications include changes of title, description, participants, but no participant responses. If the owner of an event requested any notifcations, he will always get the participant responses like acceptions and rejections too.", 'xmlrpc' => True, 'admin' => False, 'default'=> 'time_change', ), 'receive_own_updates' => array( 'type' => 'select', 'label' => 'Receive notifications about events you created/modified/deleted', 'name' => 'receive_own_updates', 'values' => $yesno, 'help' => "Do you want to be notified about changes of appointments you modified?", 'xmlrpc' => True, 'admin' => False, 'default'=> 'false', ), 'notify_externals' => array( 'type' => 'select', 'label' => 'Notify non-EGroupware users about event updates', 'name' => 'notify_externals', 'values' => $updates, 'help' => 'Do you want non-EGroupware participants of events you created to be automatically notified about new or changed appointments?', 'xmlrpc' => True, 'admin' => False, 'default'=> 'no', ), 'update_format' => array( 'type' => 'select', 'label' => 'Format of event updates', 'name' => 'update_format', 'values' => $update_formats, 'help' => 'Extended updates always include the complete event-details. iCal\'s can be imported by certain other calendar-applications.', 'xmlrpc' => True, 'admin' => False, 'forced' => 'ical', ), 'notifyAdded' => array( 'type' => 'notify', 'label' => 'Notification messages for added events', 'name' => 'notifyAdded', 'rows' => 5, 'cols' => 50, 'help' => 'This message is sent to every participant of events you own, who has requested notifcations about new events.
You can use certain variables which get substituted with the data of the event. The first line is the subject of the email.', 'default' => '', 'values' => $event_details, 'xmlrpc' => True, 'admin' => False, ), 'notifyCanceled' => array( 'type' => 'notify', 'label' => 'Notification messages for canceled events', 'name' => 'notifyCanceled', 'rows' => 5, 'cols' => 50, 'help' => 'This message is sent for canceled or deleted events.', 'default' => '', 'values' => $event_details, 'subst_help' => False, 'xmlrpc' => True, 'admin' => False, ), 'notifyModified' => array( 'type' => 'notify', 'label' => 'Notification messages for modified events', 'name' => 'notifyModified', 'rows' => 5, 'cols' => 50, 'help' => 'This message is sent for modified or moved events.', 'default' => '', 'values' => $event_details, 'subst_help' => False, 'xmlrpc' => True, 'admin' => False, ), 'notifyDisinvited' => array( 'type' => 'notify', 'label' => 'Notification messages for disinvited participants', 'name' => 'notifyDisinvited', 'rows' => 5, 'cols' => 50, 'help' => 'This message is sent to disinvited participants.', 'values' => $event_details, 'subst_help' => False, 'xmlrpc' => True, 'admin' => False, ), 'notifyResponse' => array( 'type' => 'notify', 'label' => 'Notification messages for your responses', 'name' => 'notifyResponse', 'rows' => 5, 'cols' => 50, 'help' => 'This message is sent when you accept, tentative accept or reject an event.', 'values' => $event_details, 'subst_help' => False, 'xmlrpc' => True, 'admin' => False, ), 'notifyAlarm' => array( 'type' => 'notify', 'label' => 'Notification messages for your alarms', 'name' => 'notifyAlarm', 'rows' => 5, 'cols' => 50, 'help' => 'This message is sent when you set an Alarm for a certain event. Include all information you might need.', 'values' => $event_details, 'subst_help' => False, 'xmlrpc' => True, 'admin' => False, ), '4.section' => array( 'type' => 'section', 'title' => lang('Data exchange settings'), 'no_lang'=> true, 'xmlrpc' => False, 'admin' => False ), ); // Merge print if ($GLOBALS['egw_info']['user']['apps']['filemanager']) { $settings['default_document'] = array( 'type' => 'vfs_file', 'size' => 60, 'label' => 'Default document to insert entries', 'name' => 'default_document', 'help' => lang('If you specify a document (full vfs path) here, %1 displays an extra document icon for each entry. That icon allows to download the specified document with the data inserted.',lang('calendar')).' '. lang('The document can contain placeholder like {{%1}}, to be replaced with the data.','calendar_title').' '. lang('The following document-types are supported:'). implode(',',bo_merge::get_file_extensions()), 'run_lang' => false, 'xmlrpc' => True, 'admin' => False, ); $settings['document_dir'] = array( 'type' => 'vfs_dirs', 'size' => 60, 'label' => 'Directory with documents to insert entries', 'name' => 'document_dir', 'help' => lang('If you specify a directory (full vfs path) here, %1 displays an action for each document. That action allows to download the specified document with the data inserted.',lang('calendar')).' '. lang('The document can contain placeholder like {{%1}}, to be replaced with the data.','calendar_title').' '. lang('The following document-types are supported:'). implode(',',bo_merge::get_file_extensions()), 'run_lang' => false, 'xmlrpc' => True, 'admin' => False, 'default' => '/templates/calendar', ); } // Import / Export for nextmatch if ($GLOBALS['egw_info']['user']['apps']['importexport']) { $definitions = new importexport_definitions_bo(array( 'type' => 'export', 'application' => 'calendar' )); $options = array( '~nextmatch~' => lang('Old fixed definition') ); foreach ((array)$definitions->get_definitions() as $identifier) { try { $definition = new importexport_definition($identifier); } catch (Exception $e) { unset($e); // not used // permission error continue; } if (($title = $definition->get_title())) { $options[$title] = $title; } unset($definition); } $default_def = 'export-calendar-csv'; $settings['nextmatch-export-definition'] = array( 'type' => 'select', 'values' => $options, 'label' => 'Export definition to use for nextmatch export', 'name' => 'nextmatch-export-definition', 'help' => lang('If you specify an export definition, it will be used when you export'), 'run_lang' => false, 'xmlrpc' => True, 'admin' => False, 'default'=> isset($options[$default_def]) ? $default_def : false, ); } $settings += array( 'export_timezone' => array( 'type' => 'select', 'label' => 'Timezone of event iCal file import/export', 'name' => 'export_timezone', 'values' => $export_tzs, 'help' => 'Use this timezone to import/export calendar data.', 'xmlrpc' => True, 'admin' => False, 'default' => '0', // Use event's TZ ), 'freebusy' => array( 'type' => 'select', 'label' => 'Make freebusy information available to not loged in persons?', 'name' => 'freebusy', 'help' => $freebusy_help, 'values' => $freebusy_values, 'run_lang' => false, 'subst_help' => False, 'xmlrpc' => True, 'admin' => False, 'forced' => 0, ), 'freebusy_pw' => array( 'type' => 'input', 'label' => 'Password for not loged in users to your freebusy information?', 'name' => 'freebusy_pw', 'help' => 'If you dont set a password here, the information is available to everyone, who knows the URL!!!', 'xmlrpc' => True, 'admin' => False, 'forced' => '' ), ); return $settings; } /** * Verify settings hook called to generate errors about settings used here to store default alarms in CalDAV prefs * * @param array $data * array $data['prefs'] * string $data['type'] 'user', 'default' or 'forced' * boolean $data['preprocess'] true data just shown to user, false: data stored by user */ public static function verify_settings(array $data) { self::verify_settings_reference($data); } /** * Verify settings hook called to generate errors about settings used here to store default alarms in CalDAV prefs * * @param array& $data * array $data['prefs'] * string $data['type'] 'user', 'default' or 'forced' * boolean $data['preprocess'] true data just shown to user, false: data stored by user */ public static function verify_settings_reference(array &$data) { //error_log(__METHOD__."(".array2string($data).")"); // caldav perfs are always user specific and cant by switched off if ($data['type'] != 'user') return; $account_lid = $GLOBALS['egw_info']['user']['account_lid']; foreach(array( 'default-alarm' => 'default-alarm-vevent-datetime:/'.$account_lid.'/:urn:ietf:params:xml:ns:caldav', 'default-alarm-wholeday' => 'default-alarm-vevent-date:/'.$account_lid.'/:urn:ietf:params:xml:ns:caldav', ) as $name => $dav) { $pref =& $GLOBALS['egw_info']['user']['preferences']['groupdav'][$dav]; if (true) $pref = str_replace("\r", '', $pref); // remove CR messing up multiline preg_match $val =& $data['prefs'][$name]; //error_log(__METHOD__."() groupdav[$dav]=$pref, calendar[$name]=$val"); if ($data['preprocess']) // showing preferences { if (!isset($val)) // no calendar pref --> read value from caldav { $matches = null; if (preg_match('/^ACTION:NONE$/mi', $pref)) { $val = ''; } elseif (preg_match('/^TRIGGER:-PT(\d+(M|H|D))$/mi', $pref, $matches)) { static $factors = array( 'M' => 1, 'H' => 60, 'D' => 1440, ); $factor = $factors[strtoupper($matches[2])]; $val = $factor*(int)$matches[1]; } else { $val = ''; } $GLOBALS['egw']->preferences->add('calendar', $name, $val, 'user'); //error_log(__METHOD__."() setting $name={$val} from $dav='$pref'"); } } else // storing preferences { if (empty($pref) || !preg_match('/^TRIGGER:/m', $pref)) { $pref = 'BEGIN:VALARM TRIGGER:-PT1H ATTACH;VALUE=URI:Basso ACTION:AUDIO END:VALARM'; } $trigger = $val < 0 ? 'TRIGGER:PT' : 'TRIGGER:-PT'; if ((string)$val === '') { $pref = preg_replace('/^ACTION:.*$/m', 'ACTION:NONE', $pref); } elseif (abs($val) < 60) { $pref = preg_replace('/^TRIGGER:.*$/m', $trigger.number_format(abs($val), 0).'M', $pref); } else { $pref = preg_replace('/^TRIGGER:.*$/m', $trigger.number_format(abs($val)/60, 0).'H', $pref); } $GLOBALS['egw']->preferences->add('groupdav', $dav, $pref, 'user'); //error_log(__METHOD__."() storing $name=$val --> $dav='$pref'"); } } } /** * Sync default alarms from CalDAV to Calendar * * Gets called by groupdav::PROPPATCH() for 'default-alarm-vevent-date(time)' changes */ public static function sync_default_alarms() { self::verify_settings(array( 'prefs' => array(), 'preprocess' => true, 'type' => 'user', )); } public static function config_validate() { $GLOBALS['egw_info']['server']['found_validation_hook'] = array('calendar_purge_old'); } /** * ACL rights and labels used * * @param string|array string with location or array with parameters incl. "location", specially "owner" for selected acl owner * @return array acl::(READ|ADD|EDIT|DELETE|PRIVAT|CUSTOM(1|2|3)) => $label pairs */ public static function acl_rights($params) { $rights = array( acl::CUSTOM2 => 'freebusy', acl::CUSTOM3 => 'invite', acl::READ => 'read', acl::ADD => 'add', acl::EDIT => 'edit', acl::DELETE => 'delete', acl::PRIVAT => 'private', ); $require_acl_invite = $GLOBALS['egw_info']['server']['require_acl_invite']; if (!$require_acl_invite || $require_acl_invite == 'groups' && !($params['owner'] < 0)) { unset($rights[acl::CUSTOM3]); } return $rights; } /** * Hook to tell framework we use standard categories method * * @param string|array $data hook-data or location * @return boolean */ public static function categories($data) { unset($data); // not used, but in function signature for hooks return true; } /** * Mail integration hook to import mail message contents into a calendar entry * * @return string method to be executed for calendar mail integration */ public static function mail_import($args) { return array ( 'menuaction' => 'calendar.calendar_uiforms.mail_import', 'popup' => egw_link::get_registry('calendar', 'edit_popup') ); } } // Not part of the class, since config hooks are still using the old style function calendar_purge_old($config) { $id = 'calendar_purge'; // Cancel old purge ExecMethod('phpgwapi.asyncservice.cancel_timer', $id); if((float)$config > 0) { $result = ExecMethod2('phpgwapi.asyncservice.set_timer', array('month' => '*', 'day' => 1), $id, 'calendar.calendar_boupdate.purge', (float)$config ); if(!$result) { $GLOBALS['config_error'] = 'Unable to schedule purge'; } } }