* @package infolog * @copyright (c) 2003-16 by Ralf Becker * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @version $Id$ */ use EGroupware\Api; use EGroupware\Api\Hooks; use EGroupware\Api\Link; use EGroupware\Api\Framework; use EGroupware\Api\Egw; use EGroupware\Api\Acl; /** * Class containing admin, preferences and sidebox-menus (used as hooks) */ class infolog_hooks { /** * For which groups should no group acl be used: infolog group owners * * @param array|string $location location and other parameters (not used) * @return boolean|array true, false or array with group-account_id's */ static function not_enum_group_acls($location) { unset($location); // not used, but part of hook signature $config = Api\Config::read('infolog'); return $config['group_owners']; } /** * Hook called by link-class to include infolog 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 part of hook signature return array( 'query' => 'infolog.infolog_bo.link_query', 'title' => 'infolog.infolog_bo.link_title', 'titles' => 'infolog.infolog_bo.link_titles', 'view' => array( 'menuaction' => 'infolog.infolog_ui.edit', ), 'view_id' => 'info_id', 'view_popup' => '760x570', 'list' => array( 'menuaction' => 'infolog.infolog_ui.index', 'ajax' => 'true' ), 'add' => array( 'menuaction' => 'infolog.infolog_ui.edit', 'type' => $GLOBALS['egw_info']['user']['preferences']['preferred_type'] ), 'add_app' => 'action', 'add_id' => 'action_id', 'add_popup' => '760x570', 'file_access'=> 'infolog.infolog_bo.file_access', 'file_access_user' => true, // file_access supports 4th parameter $user 'edit' => array( 'menuaction' => 'infolog.infolog_ui.edit', ), 'edit_id' => 'info_id', 'edit_popup' => '760x570', 'notify' => self::class.'::link_notify', 'merge' => true, 'push_data' => self::class.'::prepareEventPush' ); } /** * Get notified about changes to linked entries * * 'type' => $type, * 'id' => $notify_id, * 'target_app' => $target_app, * 'target_id' => $target_id, * 'link_id' => $link_id, * 'data' => $data, */ static public function link_notify($data) { switch($data['target_app']) { case 'timesheet': // Something changed with a timesheet linked to an infolog. Trigger update in place to update times. $bo = new infolog_bo(); $push_data = $bo->read($data['id']); Hooks::process([ 'location' => 'notify-all', 'type' => 'update-in-place', 'app' => 'infolog', 'id' => $data['id'], 'data' => $push_data, ], null, true); break; } } /** * Prepare entry to be pushed via Link::notify_update() * * Get linked contact ID for CRM view * * @param $entry * @return array */ static public function prepareEventPush($entry) { $info = array_intersect_key($entry, array_flip(['info_type', 'info_owner','info_responsible', 'info_modified'])); // Add in contact IDs for CRM view if($entry['info_contact'] && $entry['info_contact']['app'] == 'addressbook') { $info['contact_id'] = [$entry['info_contact']['id']]; } foreach(Api\Link::get_links('infolog',$entry['info_id'],'addressbook') as $link) { if(!is_array($info['contact_id'])) { $info['contact_id'] = []; } // Skip if already there if(in_array($link,$info['contact_id'])) { continue; } $info['contact_id'][] = $link; } return $info; } /** * hooks to build sidebox-menu plus the admin and preferences sections * * @param string/array $args hook args */ static function all_hooks($args) { $appname = 'infolog'; $location = is_array($args) ? $args['location'] : $args; //echo "

admin_prefs_sidebox_hooks::all_hooks(".print_r($args,True).") appname='$appname', location='$location'

\n"; if ($location == 'sidebox_menu') { // Magic etemplate2 favorites menu (from nextmatch widget) display_sidebox($appname, lang('Favorites'), Framework\Favorites::list_favorites($appname, 'nextmatch-nextmatch-infolog.index.rows-favorite')); $file = array( 'infolog list' => Egw::link('/index.php',array( 'menuaction' => 'infolog.infolog_ui.index', 'ajax' => 'true')), array( 'text' => lang('Add %1',lang(Link::get_registry($appname, 'entry'))), 'no_lang' => true, 'link' => "javascript:app.infolog.add_link_sidemenu();" ), ['text'=>'--'], 'Placeholders' => Egw::link('/index.php','menuaction=infolog.infolog_merge.show_replacements') ); display_sidebox($appname,$GLOBALS['egw_info']['apps']['infolog']['title'].' '.lang('Menu'),$file); } if ($GLOBALS['egw_info']['user']['apps']['admin'] && !Api\Header\UserAgent::mobile()) { $file = Array( 'Site configuration' => Egw::link('/index.php',array( 'menuaction' => 'infolog.infolog_ui.admin', 'ajax' => 'true', )), 'Global Categories' => Egw::link('/index.php',array( 'menuaction' => 'admin.admin_categories.index', 'appname' => $appname, 'global_cats'=> True, 'ajax' => 'true', )), 'Custom fields, type and status' => Egw::link('/index.php',array( 'menuaction' => 'infolog.infolog_customfields.index', 'ajax' => 'true', )), ); if ($location == 'admin') { display_section($appname,$file); } else { display_sidebox($appname,lang('Admin'),$file); } } } /** * populates $settings for the Api\Preferences * * @return array */ static function settings() { /* Setup some values to fill the array of this app's settings below */ $info = new infolog_bo(); // need some labels from $filters = $show_home = array(); $show_home[] = lang("DON'T show InfoLog"); $filters['none'] = $info->filters['']; foreach($info->filters as $key => $label) { $show_home[$key] = $filters[$key] = lang($label); } // migrage old filter-pref 1,2 to the filter one 'own-open-today' if (isset($GLOBALS['type']) && in_array($GLOBALS['egw']->preferences->{$GLOBALS['type']}['homeShowEvents'],array('1','2'))) { $GLOBALS['egw']->preferences->add('infolog','homeShowEvents','own-open-today',$GLOBALS['type']); $GLOBALS['egw']->preferences->save_repository(); } $show_links = array( 'all' => lang('all links and attachments'), 'links' => lang('only the links'), 'attach' => lang('only the attachments'), 'none' => lang('no links or attachments'), 'no_describtion' => lang('no describtion, links or attachments'), ); $show_details = array( 0 => lang('No'), 1 => lang('Yes'), 2 => lang('Only for details'), ); /* Settings array for this app */ $settings = array( array( 'type' => 'section', 'title' => lang('General settings'), 'no_lang'=> true, 'xmlrpc' => False, 'admin' => False ), 'defaultFilter' => array( 'type' => 'select', 'label' => 'Default Filter for InfoLog', 'name' => 'defaultFilter', 'values' => $filters, 'help' => 'This is the filter InfoLog uses when you enter the application. Filters limit the entries to show in the actual view. There are filters to show only finished, still open or futures entries of yourself or all users.', 'xmlrpc' => True, 'admin' => False, 'default'=> '', ), /* disabled until we have a home app again 'homeShowEvents' => array( 'type' => 'select', 'label' => 'InfoLog filter for the home screen', 'name' => 'homeShowEvents', 'values' => $show_home, 'help' => 'Should InfoLog show up on the home screen and with which filter. Works only if you dont selected an application for the home screen (in your preferences).', 'xmlrpc' => True, 'admin' => False, 'default'=> 'responsible-open-today', ),*/ 'set_start' => array( 'type' => 'select', 'label' => 'Startdate for new entries', 'name' => 'set_start', 'values' => array( 'date' => lang('todays date'), 'datetime' => lang('actual date and time'), 'empty' => lang('leave it empty'), ), 'help' => 'To what should the startdate of new entries be set.', 'xmlrpc' => True, 'admin' => False, 'default'=> 'date', ), 'cat_add_default' => array( 'type' => 'select', 'label' => 'Default category for new Infolog entries', 'name' => 'cat_add_default', 'values' => self::all_cats(), 'help' => 'You can choose a categorie to be preselected, when you create a new Infolog entry', 'xmlrpc' => True, 'admin' => False, ), 'show_id' => array( 'type' => 'select', 'label' => 'Show ticket Id', 'name' => 'show_id', 'values' => $show_details, 'help' => 'Should the Infolog list show a unique numerical Id, which can be used eg. as ticket Id.', 'xmlrpc' => True, 'admin' => False, 'default'=> '1', // Yes ), 'listNoSubs' => array( 'type' => 'select', 'label' => 'Show sub-entries', 'name' => 'listNoSubs', 'values' => array( '0' => lang('Always show them'), 'filter' => lang('Only show them if there is a filter'), '1' => lang('Only show them while searching'), ), 'help' => 'Should InfoLog show Subtasks, -calls or -notes in the normal view or not. You can always view the Subs via there parent.', 'xmlrpc' => True, 'admin' => False, 'default'=> '0', // Allways show them ), 'show_links' => array( 'type' => 'select', 'label' => 'Show in the InfoLog list', 'name' => 'show_links', 'values' => $show_links, 'help' => 'Should InfoLog show the links to other applications and/or the file-attachments in the InfoLog list (normal view when you enter InfoLog).', 'xmlrpc' => True, 'admin' => False, 'default'=> 'all', ), 'show_percent' => array( 'type' => 'select', 'label' => 'Show status and percent done separate', 'name' => 'show_percent', 'values' => $show_details, 'help' => 'Should the Infolog list show the percent done only for status ongoing or two separate icons.', 'xmlrpc' => True, 'admin' => False, 'default'=> 1, // Yes ), 'limit_des_lines' => array( 'type' => 'input', 'size' => 5, 'label' => 'Limit number of description lines (default 5, 0 for no limit)', 'name' => 'limit_des_lines', 'help' => 'How many description lines should be directly visible. Further lines are available via a scrollbar.', 'xmlrpc' => True, 'admin' => False, 'default'=> 5, ), ); $settings[] = array( 'type' => 'section', 'title' => lang('Notification settings'), 'no_lang'=> true, 'xmlrpc' => False, 'admin' => False ); // notification preferences $settings['notify_creator'] = array( 'type' => 'check', 'label' => 'Receive notifications about own items', 'name' => 'notify_creator', 'help' => 'Do you want a notification, if items you created get updated?', 'xmlrpc' => True, 'admin' => False, 'default'=> '1', // Yes ); $settings['notify_assigned'] = array( 'type' => 'select', 'label' => 'Receive notifications about items assigned to you', 'name' => 'notify_assigned', 'help' => 'Do you want a notification, if items get assigned to you or assigned items get updated?', 'values' => array( '0' => lang('No'), '1' => lang('Yes'), 'assignment' => lang('Only if I get assigned or removed'), ), 'xmlrpc' => True, 'admin' => False, 'default'=> '1', // Yes ); // to add options for more then 3 days back or in advance, you need to update soinfolog::users_with_open_entries()! $options = array( '0' => lang('No'), '-1d' => lang('one day after'), '0d' => lang('same day'), '1d' => lang('one day in advance'), '2d' => lang('%1 days in advance',2), '3d' => lang('%1 days in advance',3), ); $settings['notify_due_delegated'] = array( 'type' => 'select', 'label' => 'Receive notifications about due entries you delegated', 'name' => 'notify_due_delegated', 'help' => 'Do you want a notification, if items you delegated are due?', 'values' => $options, 'xmlrpc' => True, 'admin' => False, 'default'=> '0', // No ); $settings['notify_due_responsible'] = array( 'type' => 'select', 'label' => 'Receive notifications about due entries you are responsible for', 'name' => 'notify_due_responsible', 'help' => 'Do you want a notification, if items you are responsible for are due?', 'values' => $options, 'xmlrpc' => True, 'admin' => False, 'default'=> '0d', // Same day ); $settings['notify_start_delegated'] = array( 'type' => 'select', 'label' => 'Receive notifications about starting entries you delegated', 'name' => 'notify_start_delegated', 'help' => 'Do you want a notification, if items you delegated are about to start?', 'values' => $options, 'xmlrpc' => True, 'admin' => False, 'default'=> '0', // No ); $settings['notify_start_responsible'] = array( 'type' => 'select', 'label' => 'Receive notifications about starting entries you are responsible for', 'name' => 'notify_start_responsible', 'help' => 'Do you want a notification, if items you are responsible for are about to start?', 'values' => $options, 'xmlrpc' => True, 'admin' => False, 'default'=> '0d', // Same day ); // receive notification for items owned by groups you are part of $settings['notify_owner_group_member'] = array( 'type' => 'check', 'label' => 'Receive notifications about items of type owned by groups you are part of', 'name' => 'notify_owner_group_member', 'help' => 'Do you want a notification if items owned by groups you are part of get updated ?', 'xmlrpc' => True, 'admin' => False, 'default'=> '0', // No ); // Notify for types (default) $settings['no_notification_types'] = array( 'type' => 'multiselect', 'label' => 'Do not notify', 'name' => 'no_notification_types', 'help' => "Automatically check 'Do not notify' for these types", 'values' => $info->enums['type'], 'default'=> array(), // None ); $settings[] = 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('infolog')).' '. lang('The document can contain placeholder like {{%1}}, to be replaced with the data.','info_subject').' '. lang('The following document-types are supported:').'*.rtf, *.txt', '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('infolog')) . ' ' . lang('The document can contain placeholder like {{%1}}, to be replaced with the data.', 'info_subject') . ' ' . lang('The following document-types are supported:') . '*.rtf, *.txt', 'run_lang' => false, 'xmlrpc' => True, 'admin' => False, 'default' => '/templates/infolog', ); $settings[Api\Storage\Merge::PREF_DOCUMENT_FILENAME] = array( 'type' => 'taglist', 'label' => 'Document download filename', 'name' => 'document_download_name', 'values' => Api\Storage\Merge::DOCUMENT_FILENAME_OPTIONS, 'help' => 'Choose the default filename for downloaded documents.', 'xmlrpc' => True, 'admin' => False, 'default' => 'document', ); } if ($GLOBALS['egw_info']['user']['apps']['calendar']) { $settings['cal_show'] = array( 'type' => 'multiselect', 'label' => 'Which types should the calendar show', 'name' => 'cal_show', 'values' => array(0 => lang('None')) + $info->enums['type'], 'help' => 'Can be used to show further InfoLog types in the calendar or limit it to show eg. only tasks.', 'xmlrpc' => True, 'admin' => False, 'default'=> 'tasks,phone', ); $settings['calendar_set'] = array( 'type' => 'multiselect', 'label' => 'Participants for scheduling an appointment', 'name' => 'calendar_set', 'values' => array( 'responsible' => lang('Responsible'), 'contact' => lang('Contact'), 'owner' => lang('Owner'), 'user' => lang('Current user'), 'selected' => lang('Selected calendars'), ), 'help' => 'Which participants should be preselected when scheduling an appointment.', 'xmlrpc' => True, 'admin' => False, 'default'=> 'responsible,contact,user', ); } return $settings; } /** * Return InoLog Categories (used for setting ) * * @return array */ private static function all_cats() { $categories = new Api\Categories('','infolog'); $accountId = $GLOBALS['egw_info']['user']['account_id']; foreach((array)$categories->return_sorted_array(0,False,'','','',true) as $cat) { $s = str_repeat(' ',$cat['level']) . stripslashes($cat['name']); if ($cat['app_name'] == 'phpgw' || $cat['owner'] == '-1') { $s .= ' ♦'; } elseif ($cat['owner'] != $accountId) { $s .= '<' . $GLOBALS['egw']->accounts->id2name($cat['owner'], 'account_fullname') . '>'; } elseif ($cat['access'] == 'private') { $s .= ' ♥'; } $sel_options[$cat['id']] = $s; // 0.9.14 only } return $sel_options; } /** * Verification hook called if settings / preferences get stored * * Installs a task to send async infolog notifications at 2h everyday * * @param array $data */ static function verify_settings($data) { if ($data['prefs']['notify_due_delegated'] || $data['prefs']['notify_due_responsible'] || $data['prefs']['notify_start_delegated'] || $data['prefs']['notify_start_responsible']) { $async = new Api\Asyncservice(); if (!$async->read('infolog-async-notification')) { $async->set_timer(array('hour' => 2),'infolog-async-notification','infolog.infolog_bo.async_notification',null); } } } /** * 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) { unset($params); // not used, but default function signature for hooks return array( Acl::READ => 'read', Acl::ADD => 'add', Acl::EDIT => 'edit', Acl::DELETE => 'delete', Acl::PRIVAT => 'private', ); } /** * Hook to tell framework we use standard categories method * * @param array|string $location location and other parameters (not used) * @return boolean */ public static function categories($location) { unset($location); // not used, but part of hook signature return true; } /** * Mail integration hook to import mail message contents into an infolog entry * * @return array */ public static function mail_import($args) { unset($args); // not used, but required by function signature return array ( 'menuaction' => 'infolog.infolog_ui.mail_import', 'popup' => Link::get_registry('infolog', 'edit_popup') ); } }