* @author Ralf Becker * @copyright (c) 2005-19 by Ralf Becker * @copyright (c) 2005/6 by Cornelius Weiss * @package addressbook * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License */ use EGroupware\Api; use EGroupware\Api\Link; use EGroupware\Api\Framework; use EGroupware\Api\Egw; use EGroupware\Api\Acl; use EGroupware\Api\Vfs; use EGroupware\Api\Etemplate; use EGroupware\Kanban\Hooks; /** * General user interface object of the adressbook */ class addressbook_ui extends addressbook_bo { public $public_functions = array( 'extSearch' => True, 'edit' => True, 'view' => True, 'index' => True, 'photo' => True, 'emailpopup'=> True, 'migrate2ldap' => True, 'admin_set_fileas' => True, 'admin_set_all_cleanup' => True, 'cat_add' => True, ); protected $org_views; /** * Addressbook configuration (stored as phpgwapi = general server config) * * @var array */ protected $config; /** * Fields to copy, default if nothing specified in config * * @var array */ static public $copy_fields = array( 'org_name', 'org_unit', 'adr_one_street', 'adr_one_street2', 'adr_one_locality', 'adr_one_region', 'adr_one_postalcode', 'adr_one_countryname', 'adr_one_countrycode', 'email', 'url', 'tel_work', 'cat_id' ); /** * Instance of eTemplate class * * @var Etemplate */ protected $tmpl; /** * @var array */ public $grouped_views; /** * Constructor * * @param string $contact_app */ function __construct($contact_app='addressbook') { parent::__construct($contact_app); $this->tmpl = new Etemplate(); $this->grouped_views = array( 'org_name' => lang('Organisations'), 'org_name,adr_one_locality' => lang('Organisations by location'), 'org_name,org_unit' => lang('Organisations by departments'), 'duplicates' => lang('Duplicates'), 'shared_by_me' => lang('Shared by me'), ); // make sure the hook for export_limit is registered if (!Api\Hooks::exists('export_limit','addressbook')) Api\Hooks::read(true); $this->config =& $GLOBALS['egw_info']['server']; // check if a contact specific export limit is set, if yes use it also for etemplate's csv export $this->config['export_limit'] = $this->config['contact_export_limit'] = Api\Storage\Merge::getExportLimit($app='addressbook'); if ($this->config['copy_fields'] && ($fields = is_array($this->config['copy_fields']) ? $this->config['copy_fields'] : unserialize($this->config['copy_fields']))) { // Set country code if country name is selected $supported_fields = $this->get_fields('supported',null,0); if(in_array('adr_one_countrycode', $supported_fields) && in_array('adr_one_countryname',$fields)) { $fields[] = 'adr_one_countrycode'; } if(in_array('adr_two_countrycode', $supported_fields) && in_array('adr_two_countryname',$fields)) { $fields[] = 'adr_two_countrycode'; } self::$copy_fields = $fields; } } /** * List contacts of an addressbook * * @param array $_content =null submitted content * @param string $msg =null message to show */ function index($_content=null,$msg=null) { //echo "

uicontacts::index(".print_r($_content,true).",'$msg')

\n"; if (($re_submit = is_array($_content))) { if (isset($_content['nm']['rows']['delete'])) // handle a single delete like delete with the checkboxes { $id = @key($_content['nm']['rows']['delete']); $_content['nm']['action'] = 'delete'; $_content['nm']['selected'] = array($id); } if (isset($_content['nm']['rows']['document'])) // handle insert in default document button like an action { $id = @key($_content['nm']['rows']['document']); $_content['nm']['action'] = 'document'; $_content['nm']['selected'] = array($id); } if ($_content['nm']['action'] !== '' && $_content['nm']['action'] !== null) { if (!count($_content['nm']['selected']) && !$_content['nm']['select_all'] && $_content['nm']['action'] != 'delete_list') { $msg = lang('You need to select some contacts first'); } elseif ($_content['nm']['action'] == 'view_org' || $_content['nm']['action'] == 'view_duplicates') { // grouped view via context menu $_content['nm']['grouped_view'] = array_shift($_content['nm']['selected']); } else { $success = $failed = $action_msg = null; if ($this->action($_content['nm']['action'],$_content['nm']['selected'],$_content['nm']['select_all'], $success,$failed,$action_msg,'index',$msg,$_content['nm']['checkboxes'], $error_msg)) { $msg .= lang('%1 contact(s) %2',$success,$action_msg); Framework::message($msg); } elseif(is_null($msg)) { if (empty($error_msg)) { $msg .= lang('%1 contact(s) %2, %3 failed because of insufficent rights !!!', $success, $action_msg, $failed); } else { $msg .= lang('%1 contact(s) %2, %3 failed because of %4 !!!', $success, $action_msg, $failed, $error_msg); } Framework::message($msg,'error'); } $msg = ''; } } if (!empty($_content['nm']['rows']['infolog'])) { $org = key($_content['nm']['rows']['infolog']); return $this->infolog_org_view($org); } if (!empty($_content['nm']['rows']['view'])) // show all contacts of an organisation { $grouped_view = key($_content['nm']['rows']['view']); } else { $grouped_view = $_content['nm']['grouped_view']; } $typeselection = $_content['nm']['col_filter']['tid']; } elseif($_GET['add_list']) { $list = $this->add_list($_GET['add_list'],$_GET['owner']?$_GET['owner']:$this->user); if ($list === true) { $msg = lang('List already exists!'); } elseif ($list) { $msg = lang('List created'); } else { $msg = lang('List creation failed, no rights!'); } } $preserv = array(); $content = array(); if($msg || $_GET['msg']) { Framework::message($msg ? $msg : $_GET['msg']); } $content['nm'] = Api\Cache::getSession('addressbook', 'index'); if (!is_array($content['nm'])) { $content['nm'] = array( 'get_rows' => 'addressbook.addressbook_ui.get_rows', // I method/callback to request the data for the rows eg. 'notes.bo.get_rows' 'bottom_too' => false, // I show the nextmatch-line (arrows, filters, search, ...) again after the rows 'never_hide' => True, // I never hide the nextmatch-line if less then maxmatch entrie 'start' => 0, // IO position in list 'cat_id' => '', // IO category, if not 'no_cat' => True 'search' => '', // IO search pattern 'order' => 'n_family', // IO name of the column to sort after (optional for the sortheaders) 'sort' => 'ASC', // IO direction of the sort: 'ASC' or 'DESC' 'col_filter' => array(), // IO array of column-name value pairs (optional for the filterheaders) //'cat_id_label' => lang('Categories'), //'filter_label' => lang('Addressbook'), // I label for filter (optional) 'filter' => '', // =All // IO filter, if not 'no_filter' => True 'filter_no_lang' => True, // I set no_lang for filter (=dont translate the options) 'no_filter2' => True, // I disable the 2. filter (params are the same as for filter) //'filter2_label'=> lang('Distribution lists'), // IO filter2, if not 'no_filter2' => True 'filter2' => '', // IO filter2, if not 'no_filter2' => True 'filter2_no_lang'=> True, // I set no_lang for filter2 (=dont translate the options) 'lettersearch' => true, // using a positiv list now, as we constantly adding new columns in addressbook, but not removing them from default 'default_cols' => 'type,n_fileas_n_given_n_family_n_family_n_given_org_name_n_family_n_given_n_fileas,'. 'number,org_name,org_unit,'. 'business_adr_one_countrycode_adr_one_postalcode,tel_work_tel_cell_tel_home,url_email_email_home', /* old negative list 'default_cols' => '!cat_id,contact_created_contact_modified,distribution_list,contact_id,owner,room',*/ 'filter2_onchange' => "return app.addressbook.filter2_onchange();", 'filter2_tags' => true, //'actions' => $this->get_actions(), // set on each request, as it depends on some filters 'row_id' => 'id', 'row_modified' => 'modified', 'add_on_top_sort_field' => 'modified', 'is_parent' => 'group_count', 'parent_id' => 'parent_id', 'favorites' => true, ); // use the state of the last session stored in the user prefs if (($state = @unserialize($this->prefs['index_state']))) { $content['nm'] = array_merge($content['nm'],$state); } } $sel_options['cat_id'] = array('' => lang('All categories'), '0' => lang('None')); $content['nm']['placeholder_actions'] = array('add'); // Edit and delete list actions depends on permissions if($this->get_lists(Acl::EDIT)) { $content['nm']['placeholder_actions'][] = 'rename_list'; $content['nm']['placeholder_actions'][] = 'delete_list'; } // Search parameter passed in if ($_GET['search']) { $content['nm']['search'] = $_GET['search']; } if(isset($typeselection)) { $content['nm']['col_filter']['tid'] = $typeselection; } // disable kanban column if we have no kanban if(empty($GLOBALS['egw_info']['user']['apps']['kanban'])) { $content['nm']['no_kanban'] = true; } // save the tid for use in creating new addressbook entrys via UI. Current tid is to be used as type of new entrys //error_log(__METHOD__.__LINE__.' '.$content['nm']['col_filter']['tid']); Api\Cache::setSession('addressbook','active_tid',$content['nm']['col_filter']['tid']); if ($this->lists_available()) { $sel_options['filter2'] = $this->get_lists(Acl::READ,array('' => lang('No distribution list'))); $sel_options['filter2']['add'] = lang('Add a new list').'...'; // put it at the end } // Organisation stuff is not (yet) availible with ldap if($GLOBALS['egw_info']['server']['contact_repository'] != 'ldap' && Api\Header\UserAgent::mobile() == '') { $content['nm']['header_left'] = 'addressbook.index.left'; } $sel_options['filter'] = $sel_options['owner'] = $this->get_addressbooks(Acl::READ, lang('All addressbooks')); $sel_options['to'] = array( 'to' => 'To', 'cc' => 'Cc', 'bcc' => 'Bcc', ); // if there is any export limit set, pass it on to the nextmatch, to be evaluated by the export if (isset($this->config['contact_export_limit']) && (int)$this->config['contact_export_limit']) $content['nm']['export_limit']=$this->config['contact_export_limit']; // Merge to email dialog needs the infolog types $infolog = new infolog_bo(); $sel_options['info_type'] = $infolog->enums['type']; // dont show tid-selection if we have only one content_type // be a bit more sophisticated about it $availabletypes = array_keys($this->content_types); if ($content['nm']['col_filter']['tid'] && !in_array($content['nm']['col_filter']['tid'],$availabletypes)) { //_debug_array(array('Typefilter:'=> $content['nm']['col_filter']['tid'],'Available Types:'=>$availabletypes,'action:'=>'remove invalid filter')); unset($content['nm']['col_filter']['tid']); } if (!isset($content['nm']['col_filter']['tid'])) $content['nm']['col_filter']['tid'] = $availabletypes[0]; if (count($this->content_types) > 1) { foreach($this->content_types as $tid => $data) { $sel_options['tid'][$tid] = $data['name']; } } else { $this->tmpl->disableElement('nm[col_filter][tid]'); } // get the availible grouped-views plus the label of the contacts view of one group $sel_options['grouped_view'] = $this->grouped_views; if (isset($grouped_view)) { $content['nm']['grouped_view'] = $grouped_view; } $content['nm']['actions'] = $this->get_actions($content['nm']['col_filter']['tid']); if (!isset($sel_options['grouped_view'][(string) $content['nm']['grouped_view']])) { $sel_options['grouped_view'] += $this->_get_grouped_name((string)$content['nm']['grouped_view']); } // unset the filters regarding grouped views, when there is no group selected if (empty($sel_options['grouped_view'][(string) $content['nm']['grouped_view']]) || stripos($grouped_view,":") === false ) { $this->unset_grouped_filters($content['nm']); } $content['nm']['grouped_view_label'] = $sel_options['grouped_view'][(string) $content['nm']['grouped_view']]; // allow to also filter by (not) shared contacts $sel_options['shared_with'] = [ 'not' => lang('not shared'), 'shared' => lang('shared'), ]; $this->tmpl->read('addressbook.index'); return $this->tmpl->exec('addressbook.addressbook_ui.index', $content,$sel_options,array(),$preserv); } /** * Get actions / context menu items * * @param ?string $tid_filter =null * @return array see Etemplate\Widget\Nextmatch::get_actions() */ public function get_actions($tid_filter=null) { // Contact view $actions = array( 'view' => array( 'caption' => 'CRM-View', 'default' => $GLOBALS['egw_info']['user']['preferences']['addressbook']['crm_list'] != '~edit~', 'allowOnMultiple' => false, 'group' => $group=1, 'onExecute' => 'javaScript:app.addressbook.view', 'enableClass' => 'contact_contact', 'hideOnDisabled' => true, // Children added below 'children' => array(), 'hideOnMobile' => true ), 'open' => array( 'caption' => 'Open', 'default' => $GLOBALS['egw_info']['user']['preferences']['addressbook']['crm_list'] == '~edit~', 'allowOnMultiple' => false, 'enableClass' => 'contact_contact', 'hideOnDisabled' => true, 'url' => 'menuaction=addressbook.addressbook_ui.edit&contact_id=$id', 'popup' => Link::get_registry('addressbook', 'edit_popup'), 'group' => $group, ), 'add' => array( 'caption' => 'Add', 'group' => $group, 'hideOnDisabled' => true, 'children' => array( 'new' => array( 'caption' => 'New', 'url' => 'menuaction=addressbook.addressbook_ui.edit', 'popup' => Link::get_registry('addressbook', 'add_popup'), 'icon' => 'new', ), 'copy' => array( 'caption' => 'Copy', 'url' => 'menuaction=addressbook.addressbook_ui.edit&makecp=1&contact_id=$id', 'popup' => Link::get_registry('addressbook', 'add_popup'), 'enableClass' => 'contact_contact', 'allowOnMultiple' => false, 'icon' => 'copy', ), ), 'hideOnMobile' => true ), ); // CRM view options $crm_apps = array('infolog','tracker'); foreach($crm_apps as $crm_index => $app) { if (!$GLOBALS['egw_info']['user']['apps'][$app]) { unset($crm_apps[$crm_index]); } } if($GLOBALS['egw_info']['user']['apps']['infolog']) { array_splice($crm_apps, 1, 0, 'infolog-organisation'); } if(count($crm_apps) > 1) { foreach($crm_apps as $app) { $actions['view']['children']["view-$app"] = array( 'caption' => $app, 'icon' => $app == 'infolog-organisation' ? "infolog/navbar" : "$app/navbar" ); } } // org view $actions += array( 'view_org' => array( 'caption' => 'View', 'default' => true, 'allowOnMultiple' => false, 'group' => $group=1, 'enableClass' => 'contact_organisation', 'hideOnDisabled' => true ), 'add_org' => array( 'caption' => 'Add', 'group' => $group, 'allowOnMultiple' => false, 'enableClass' => 'contact_organisation', 'hideOnDisabled' => true, 'url' => 'menuaction=addressbook.addressbook_ui.edit&org=$id', 'popup' => Link::get_registry('addressbook', 'add_popup'), ), ); // Duplicates view $actions += array( 'view_duplicates' => array( 'caption' => 'View', 'default' => true, 'allowOnMultiple' => false, 'group' => $group=1, 'enableClass' => 'contact_duplicate', 'hideOnDisabled' => true ) ); ++$group; // other AB related stuff group: lists, AB's, categories // categories submenu $actions['cat'] = array( 'caption' => 'Categories', 'group' => $group, 'children' => array( 'cat_add' => Etemplate\Widget\Nextmatch::category_action( 'addressbook',$group,'Add category', 'cat_add_', true, 0,Etemplate\Widget\Nextmatch::DEFAULT_MAX_MENU_LENGTH,false )+array( 'icon' => 'foldertree_nolines_plus', 'disableClass' => 'rowNoEdit', ), 'cat_del' => Etemplate\Widget\Nextmatch::category_action( 'addressbook',$group,'Delete category', 'cat_del_', true, 0,Etemplate\Widget\Nextmatch::DEFAULT_MAX_MENU_LENGTH,false )+array( 'icon' => 'foldertree_nolines_minus', 'disableClass' => 'rowNoEdit', ), ), ); if (!$GLOBALS['egw_info']['user']['apps']['preferences']) unset($actions['cats']['children']['cat_edit']); // Submenu for all distributionlist stuff $actions['lists'] = array( 'caption' => 'Distribution lists', 'children' => array( 'list_add' => array( 'caption' => 'Add a new list', 'icon' => 'new', 'onExecute' => 'javaScript:app.addressbook.add_new_list', ), ), 'group' => $group, ); if (($add_lists = $this->get_lists(Acl::EDIT))) // do we have distribution lists?, and are we allowed to edit them { $actions['lists']['children'] += array( 'to_list' => array( 'caption' => 'Add to distribution list', 'children' => $add_lists, 'prefix' => 'to_list_', 'icon' => 'foldertree_nolines_plus', 'enabled' => ($add_lists?true:false), // if there are editable lists, allow to add a contact to one of them, //'disableClass' => 'rowNoEdit', // wether you are allowed to edit the contact or not, as you alter a list, not the contact ), 'remove_from_list' => array( 'caption' => 'Remove from distribution list', 'confirm' => 'Remove selected contacts from distribution list', 'icon' => 'foldertree_nolines_minus', 'enabled' => 'javaScript:app.addressbook.nm_compare_field', 'fieldId' => 'exec[nm][filter2]', 'fieldValue' => '!', // enable if list != '' ), 'rename_list' => array( 'caption' => 'Rename selected distribution list', 'icon' => 'edit', 'enabled' => 'javaScript:app.addressbook.nm_compare_field', 'fieldId' => 'exec[nm][filter2]', 'fieldValue' => '!', // enable if list != '' 'onExecute' => 'javaScript:app.addressbook.rename_list' ), 'delete_list' => array( 'caption' => 'Delete selected distribution list!', 'confirm' => 'Delete selected distribution list!', 'icon' => 'delete', 'enabled' => 'javaScript:app.addressbook.nm_compare_field', 'fieldId' => 'exec[nm][filter2]', 'fieldValue' => '!', // enable if list != '' ), ); if(is_subclass_of('etemplate', 'etemplate_new')) { $actions['lists']['children']['remove_from_list']['fieldId'] = 'filter2'; $actions['lists']['children']['rename_list']['fieldId'] = 'filter2'; $actions['lists']['children']['delete_list']['fieldId'] = 'filter2'; } } // move to AB if (($move2addressbooks = $this->get_addressbooks(Acl::ADD))) // do we have addressbooks, we should { unset($move2addressbooks[0]); // do not offer action to move contact to an account, as we dont support that currrently foreach($move2addressbooks as $owner => $label) { $icon = $type_label = null; $this->type_icon((int)$owner, substr($owner,-1) == 'p', 'n', $icon, $type_label); $move2addressbooks[$owner] = array( 'icon' => $icon, 'caption' => $label, ); } // copy checkbox $move2addressbooks= array( 'copy' =>array( 'id' => 'move_to_copy', 'caption' => 'Copy instead of move', 'checkbox' => true, )) + $move2addressbooks; $actions['move_to'] = array( 'caption' => 'Move to addressbook', 'children' => $move2addressbooks, 'prefix' => 'move_to_', 'group' => $group, 'disableClass' => 'rowNoDelete', 'hideOnMobile' => true ); } if (($share2addressbooks = $this->get_addressbooks(Acl::EDIT|self::ACL_SHARED, null, null, false))) { unset($share2addressbooks[0]); // do not offer action to share contact into accounts AB foreach ($share2addressbooks as $owner => $label) { if (substr($owner, -1) === 'p') // share with private AB makes no sense { unset($share2addressbooks[$owner]); continue; } $icon = $type_label = null; $this->type_icon((int)$owner, substr($owner, -1) == 'p', 'n', $icon, $type_label); $share2addressbooks[$owner] = array( 'icon' => $icon, 'caption' => $label, ); } $actions['shared_with'] = [ 'caption' => 'Share into addressbook', 'children' => [ 'writable' => [ 'id' => 'writable', 'caption' => 'Share writable', 'checkbox' => true, ]] + $share2addressbooks + [ 'unshare' => [ 'icon' => 'delete', 'caption' => 'Unshare', 'group' => $group, 'enableClass' => 'unshare_contact', 'hideOnDisabled' => true, 'hideOnMobile' => true ] ], 'prefix' => 'shared_with_', 'group' => $group, 'hideOnMobile' => true ]; } $actions['change_type'] = $this->change_type_actions($group); $actions['merge'] = array( 'caption' => 'Merge contacts', 'confirm' => 'Merge into first or account, deletes all other!', 'hint' => 'Merge into first or account, deletes all other!', 'allowOnMultiple' => 'only', 'group' => $group, 'hideOnMobile' => true, 'enabled' => 'javaScript:app.addressbook.can_merge' ); // Duplicates view $actions['merge_duplicates'] = array( 'caption' => 'Merge duplicates', 'group' => $group, 'allowOnMultiple' => true, 'enableClass' => 'contact_duplicate', 'hideOnDisabled' => true ); ++$group; // integration with other apps: infolog, calendar, filemanager, messenger // Integrate Status Videoconference actions if ($GLOBALS['egw_info']['user']['apps']['status']) { $actions['videoconference'] = [ 'caption' => 'Video Conference', 'icon' => 'status/videoconference', 'group' => $group, 'children' => [ 'call' => [ 'caption' => lang('Video Call'), 'icon' => 'status/videoconference_call', 'allowOnMultiple' => true, 'onExecute' => 'javaScript:app.addressbook.videoconference_actionCall', 'enabled' => 'javaScript:app.addressbook.videoconference_isUserOnline' ], 'audiocall' => [ 'caption' => lang('Audio Call'), 'icon' => 'accept_call', 'allowOnMultiple' => true, 'onExecute' => 'javaScript:app.addressbook.videoconference_actionCall', 'enabled' => 'javaScript:app.addressbook.videoconference_isUserOnline' ], 'invite' => [ 'caption' => lang('Invite to current call'), 'icon' => 'status/videoconference_join', 'allowOnMultiple' => true, 'onExecute' => 'javaScript:app.addressbook.videoconference_actionCall', 'enabled' => 'javaScript:app.addressbook.videoconference_isThereAnyCall' ], 'schedule_call' => [ 'caption' => lang('Schedule a video conference'), 'icon' => 'calendar/navbar', 'allowOnMultiple' => true, 'onExecute' => 'javaScript:app.addressbook.add_cal', ] ] ]; } if ($GLOBALS['egw_info']['user']['apps']['infolog']) { $actions['infolog_app'] = array( 'caption' => 'InfoLog', 'icon' => 'infolog/navbar', 'group' => $group, 'children' => array( 'infolog' => array( 'caption' => lang('View linked InfoLog entries'), 'icon' => 'infolog/navbar', 'onExecute' => 'javaScript:app.addressbook.view_infolog', 'disableClass' => 'contact_duplicate', 'allowOnMultiple' => true, 'hideOnDisabled' => true, ), 'infolog_add' => array( 'caption' => 'Add a new Infolog', 'icon' => 'new', 'url' => 'menuaction=infolog.infolog_ui.edit&type=task&action=addressbook&action_id=$id', 'popup' => Link::get_registry('infolog', 'add_popup'), 'onExecute' => 'javaScript:app.addressbook.add_task', // call server for org-view only ), ), 'hideOnMobile' => true ); } if ($GLOBALS['egw_info']['user']['apps']['calendar']) { $actions['calendar'] = array( 'icon' => 'calendar/navbar', 'caption' => 'Calendar', 'group' => $group, 'enableClass' => 'contact_contact', 'children' => array( 'calendar_view' => array( 'caption' => 'Show', 'icon' => 'view', 'onExecute' => 'javaScript:app.addressbook.view_calendar', 'targetapp' => 'calendar', // open in calendar tab, 'hideOnDisabled' => true, ), 'calendar_add' => array( 'caption' => 'Add appointment', 'icon' => 'new', 'popup' => Link::get_registry('calendar', 'add_popup'), 'onExecute' => 'javaScript:app.addressbook.add_cal', ), ), 'hideOnMobile' => true ); } //Send to email $actions['email'] = array( 'caption' => 'Email', 'icon' => 'mail/navbar', 'enableClass' => 'contact_contact', 'hideOnDisabled' => true, 'group' => $group, 'children' => array( 'add_to_to' => array( 'caption' => lang('Add to To'), 'no_lang' => true, 'onExecute' => 'javaScript:app.addressbook.addEmail', ), 'add_to_cc' => array( 'caption' => lang('Add to Cc'), 'no_lang' => true, 'onExecute' => 'javaScript:app.addressbook.addEmail', ), 'add_to_bcc' => array( 'caption' => lang('Add to BCc'), 'no_lang' => true, 'onExecute' => 'javaScript:app.addressbook.addEmail', ), 'email_business' => array( 'caption' => lang('Business email'), 'no_lang' => true, 'checkbox' => true, 'group' => $group, 'onExecute' => 'javaScript:app.addressbook.mailCheckbox', 'checked' => $this->prefs['preferredMail']['business'], ), 'email_home' => array( 'caption' => lang('Home email'), 'no_lang' => true, 'checkbox' => true, 'group' => $group, 'onExecute' => 'javaScript:app.addressbook.mailCheckbox', 'checked' => $this->prefs['preferredMail']['private'], ), ), ); if (!$this->prefs['preferredMail']) $actions['email']['children']['email_business']['checked'] = true; if ($GLOBALS['egw_info']['user']['apps']['filemanager']) { $actions['filemanager'] = array( 'icon' => 'filemanager/navbar', 'caption' => 'Filemanager', 'url' => 'menuaction=filemanager.filemanager_ui.index&path=/apps/addressbook/$id&ajax=true', 'allowOnMultiple' => false, 'group' => $group, // disable for for group-views, as it needs contact-ids 'enableClass' => 'contact_contact', 'hideOnMobile' => true ); } if ($GLOBALS['egw_info']['user']['apps']['kanban']) { $actions['kanban'] = EGroupware\Kanban\Hooks::get_actions(['addressbook'], $group); } $actions['geolocation'] = array( 'caption' => 'GeoLocation', 'icon' => 'map', 'group' => ++$group, 'enableClass' => 'contact_contact', 'children' => array ( 'private' => array( 'caption' => 'Private Address', 'enabled' => 'javaScript:app.addressbook.geoLocation_enabled', 'onExecute' => 'javaScript:app.addressbook.geoLocationExec', ), 'business' => array( 'caption' => 'Business Address', 'enabled' => 'javaScript:app.addressbook.geoLocation_enabled', 'onExecute' => 'javaScript:app.addressbook.geoLocationExec', ) ) ); $actions += EGroupware\Api\Link\Sharing::get_actions('addressbook', $group); // check if user is an admin or the export is not generally turned off (contact_export_limit is non-numerical, eg. no) $exception = Api\Storage\Merge::is_export_limit_excepted(); if ((isset($GLOBALS['egw_info']['user']['apps']['admin']) || $exception) || !$this->config['contact_export_limit'] || (int)$this->config['contact_export_limit']) { $actions['export'] = array( 'caption' => 'Export', 'icon' => 'filesave', 'enableClass' => 'contact_contact', 'group' => ++$group, 'children' => array( 'csv' => array( 'caption' => 'Export as CSV', 'allowOnMultiple' => true, 'url' => 'menuaction=importexport.importexport_export_ui.export_dialog&appname=addressbook&plugin=addressbook_export_contacts_csv&selection=$id&select_all=$select_all', 'popup' => '850x440' ), 'vcard' => array( 'caption' => 'Export as VCard', 'postSubmit' => true, // download needs post submit (not Ajax) to work 'icon' => Vfs::mime_icon('text/vcard'), ), ), 'hideOnMobile' => true ); } $actions['documents'] = Api\Contacts\Merge::document_action( $this->prefs['document_dir'], $group, 'Insert in document', 'document_', $this->prefs['default_document'], $this->config['contact_export_limit'] ); if ($GLOBALS['egw_info']['user']['apps']['felamimail']||$GLOBALS['egw_info']['user']['apps']['mail']) { $actions['mail'] = array( 'caption' => lang('Mail VCard'), 'icon' => 'filemanager/mail_post_to', 'group' => $group, 'onExecute' => 'javaScript:app.addressbook.adb_mail_vcard', 'enableClass' => 'contact_contact', 'hideOnDisabled' => true, 'hideOnMobile' => true, 'disableIfNoEPL' => true ); } ++$group; if (!($tid_filter == 'D' && !$GLOBALS['egw_info']['user']['apps']['admin'] && $this->config['history'] != 'userpurge')) { $actions['delete'] = array( 'caption' => 'Delete', 'confirm' => 'Delete this contact', 'confirm_multiple' => 'Delete these entries', 'group' => $group, 'disableClass' => 'rowNoDelete', 'onExecute' => 'javaScript:app.addressbook.action', ); } if ($this->grants[0] & Acl::DELETE) { $actions['delete_account'] = array( 'caption' => 'Delete', 'icon' => 'delete', 'group' => $group, 'enableClass' => 'rowAccount', 'hideOnDisabled' => true, 'popup' => '400x200', 'url' => 'menuaction=admin.admin_account.delete&contact_id=$id', ); $actions['delete']['hideOnDisabled'] = true; } if($tid_filter == 'D') { $actions['undelete'] = array( 'caption' => 'Un-delete', 'icon' => 'revert', 'group' => $group, 'disableClass' => 'rowNoEdit', ); } if (isset($actions['export']['children']['csv']) && (!isset($GLOBALS['egw_info']['user']['apps']['importexport']) || !importexport_helper_functions::has_definitions('addressbook','export'))) { unset($actions['export']['children']['csv']); } // Intercept open action in order to open entry into view mode instead of edit if (Api\Header\UserAgent::mobile()) { $actions['open']['onExecute'] = 'javaScript:app.addressbook.viewEntry'; $actions['open']['mobileViewTemplate'] = 'view?'.filemtime(Api\Etemplate\Widget\Template::rel2path('/addressbook/templates/mobile/view.xet')); $actions['view']['default'] = false; $actions['open']['default'] = true; } // Allow contacts to be dragged /* $actions['drag'] = array( 'type' => 'drag', 'dragType' => 'addressbook' ); */ return $actions; } protected function change_type_actions($group) { $types = array(); foreach($this->content_types as $key => $type) { // Skip deleted if($key == self::DELETED_TYPE) continue; $types[$key] = array( 'caption' => $type['name'], ); } return array( 'caption' => 'Type', 'children' => $types, 'prefix' => 'to_type_', 'group' => $group, 'disableClass' => 'rowNoEdit', 'hideOnDisabled' => true, 'disabled' => (count($types) <= 1), 'hideOnMobile' => true ); } /** * Get a nice name for the given grouped view ID * * @param String $view_id Some kind of indicator for a specific group, either * organisation or duplicate. It looks like key:value pairs seperated by |||. * * @return Array(ID => name), where ID is the $view_id passed in */ protected function _get_grouped_name($view_id) { $group_name = array(); if (strpos($view_id,'*AND*')!== false) $view_id = str_replace('*AND*','&',$view_id); foreach(explode('|||',$view_id) as $part) { list(,$name) = explode(':',$part,2); if ($name) $group_name[] = $name; } $name = implode(': ',$group_name); return $name ? array($view_id => $name) : array(); } /** * Unset the relevant column filters to clear a grouped view * * @param Array $query */ protected function unset_grouped_filters(&$query) { unset($query['col_filter']['org_name']); unset($query['col_filter']['org_unit']); unset($query['col_filter']['adr_one_locality']); foreach(array_keys(static::$duplicate_fields) as $field) { unset($query['col_filter'][$field]); } } /** * Adjust the query as needed and get the rows for the grouped views (organisation * or duplicate contacts) * * @param Array $query Nextmatch query * @return array rows found */ protected function get_grouped_rows(&$query) { // Query doesn't like empties unset($query['col_filter']['parent_id']); if($query['actions'] && $query['actions']['open']) { // Just switched from contact view, update actions $query['actions'] = $this->get_actions($query['col_filter']['tid']); } $template = $query['grouped_view'] == 'duplicates' ? 'addressbook.index.duplicate_rows' : 'addressbook.index.org_rows'; if ($query['advanced_search']) { $query['op'] = $query['advanced_search']['operator']; unset($query['advanced_search']['operator']); $query['wildcard'] = $query['advanced_search']['meth_select']; unset($query['advanced_search']['meth_select']); $original_search = $query['search']; $query['search'] = $query['advanced_search']; } switch ($template) { case 'addressbook.index.org_rows': if ($query['order'] != 'org_name') { $query['sort'] = 'ASC'; $query['order'] = 'org_name'; } $query['org_view'] = $query['grouped_view']; // switch the distribution list selection off for ldap if($this->contact_repository != 'sql') { $query['no_filter2'] = true; unset($query['col_filter']['list']); // does not work here } else { $rows = parent::organisations($query); } break; case 'addressbook.index.duplicate_rows': $query['no_filter2'] = true; // switch the distribution list selection off unset($query['col_filter']['list']); // does not work for duplicates $rows = parent::duplicates($query); break; } if ($query['advanced_search']) { $query['search'] = $original_search; unset($query['wildcard']); unset($query['op']); } $GLOBALS['egw_info']['flags']['params']['manual'] = array('page' => 'ManualAddressbookIndexOrga'); return $rows; } /** * Return the contacts in an organisation via AJAX * * @param string|string[] $org Organisation ID * @param mixed $_query Query filters (category, etc) to use, or null to use session * @return array */ public function ajax_organisation_contacts($org, $_query = null) { $org_contacts = array(); $query = !$_query ? Api\Cache::getSession('addressbook', 'index') : $_query; $query['num_rows'] = -1; // all if(!is_array($query['col_filter'])) $query['col_filter'] = array(); if(!is_array($org)) $org = array($org); foreach($org as $org_name) { $query['grouped_view'] = $org_name; $checked = array(); $readonlys = null; $this->get_rows($query,$checked,$readonlys,true); // true = only return the id's if($checked[0]) { $org_contacts = array_merge($org_contacts,$checked); } } Api\Json\Response::get()->data(array_unique($org_contacts)); } /** * Show the infologs of an whole organisation * * @param string $org */ function infolog_org_view($org) { $query = Api\Cache::getSession('addressbook', 'index'); $query['num_rows'] = -1; // all $query['grouped_view'] = $org; $query['searchletter'] = ''; $checked = $readonlys = null; $this->get_rows($query,$checked,$readonlys,true); // true = only return the id's if (count($checked) > 1) // use a nicely formatted org-name as title in infolog { $parts = array(); if (strpos($org,'*AND*')!== false) $org = str_replace('*AND*','&',$org); foreach(explode('|||',$org) as $part) { list(,$part) = explode(':',$part,2); if ($part) $parts[] = $part; } $org = implode(', ',$parts); } else { $org = ''; // use infolog default of link-title } Egw::redirect_link('/index.php',array( 'menuaction' => 'infolog.infolog_ui.index', 'action' => 'addressbook', 'action_id' => implode(',',$checked), 'action_title' => $org, ),'infolog'); } /** * Create or rename an existing email list * * @param int $list_id ID of existing list, or 0 for a new one * @param string $new_name List name * @param int $_owner List owner, or empty for current user * @param string[] [$contacts] List of contacts to add to the array * @return boolean|string */ function ajax_set_list($list_id, $new_name, $_owner = false, $contacts = array()) { // Set owner to current user, if not set $owner = $_owner ? $_owner : $GLOBALS['egw_info']['user']['account_id']; // if admin forced or set default for add_default pref // consider default_addressbook as owner which already // covered all cases in contacts class. if ($owner == (int)$GLOBALS['egw']->preferences->default['addressbook']['add_default'] || $owner == (int)$GLOBALS['egw']->preferences->forced['addressbook']['add_default']) { $owner = $this->default_addressbook; } // Check for valid list & permissions if(!(int)$list_id && !$this->check_list(null,EGW_ACL_ADD|EGW_ACL_EDIT,$owner)) { Api\Json\Response::get()->apply('egw.message', array( lang('List creation failed, no rights!'),'error')); return; } if ((int)$list_id && !$this->check_list((int)$list_id, Acl::EDIT, $owner)) { Api\Json\Response::get()->apply('egw.message', array( lang('Insufficent rights to edit this list!'),'error')); return; } $list = array('list_owner' => $owner); // Rename if($list_id) { $list = $this->read_list((int)$list_id); } $list['list_name'] = $new_name; $new_id = $this->add_list(array('list_id' => (int)$list_id), $list['list_owner'],array(),$list); if($contacts) { $this->add2list($contacts,$new_id); } Api\Json\Response::get()->apply('egw.message', array( $new_id == $list_id ? lang('Distribution list renamed') : lang('List created'), 'success' )); // Success, just update selectbox to new value Api\Json\Response::get()->data($new_id == $list_id ? "true" : $new_id); } /** * Ajax function to get contact data out of provided account_id * * @param string $account_id */ function ajax_get_contact ($account_id) { $bo = new Api\Contacts(); $contact = $bo->read('account:'.$account_id); Api\Json\Response::get()->data($contact); } /** * Disable / clear advanced search * * Advanced search is stored server side in session no matter what the nextmatch * sends, so we have to clear it here. */ public static function ajax_clear_advanced_search() { $query = Api\Cache::getSession('addressbook', 'index'); unset($query['advanced_search']); Api\Cache::setSession('addressbook','index',$query); Api\Cache::setSession('addressbook', 'advanced_search', false); } /** * Apply an action to multiple events, but called via AJAX instead of submit * * @param string $action * @param string[] $selected * @param bool $all_selected All entries are selected, not just what's in $selected * @param bool $skip_notification */ public function ajax_action($action, $selected, $all_selected, $skip_notification = false) { $success = 0; $failed = 0; $action_msg = ''; $session_name = 'index'; if($this->action($action, $selected, $all_selected, $success, $failed, $action_msg, $session_name, $msg, $skip_notification)) { $msg = lang('%1 event(s) %2',$success,$action_msg); } elseif(is_null($msg)) { $msg .= lang('%1 event(s) %2, %3 failed because of insufficient rights !!!',$success,$action_msg,$failed); } Api\Json\Response::get()->message($msg); } /** * apply an action to multiple contacts * * @param string/int $action 'delete', 'vcard', 'csv' or nummerical account_id to move contacts to that addessbook * @param array $checked contact id's to use if !$use_all * @param boolean $use_all if true use all contacts of the current selection (in the session) * @param int &$success number of succeded actions * @param int &$failed number of failed actions (not enought permissions) * @param string &$action_msg translated verb for the actions, to be used in a message like %1 contacts 'deleted' * @param string/array $session_name 'index' or array with session-data depending if we are in the main list or the popup * @param ?string& $error_msg on return optional error-message * @return boolean true if all actions succeded, false otherwise */ function action($action, $checked, $use_all, &$success, &$failed, &$action_msg, $session_name, &$msg, $checkboxes = NULL, &$error_msg=null) { //echo "

uicontacts::action('$action',".print_r($checked,true).','.(int)$use_all.",...)

\n"; $success = $failed = 0; $error_msg = null; if ($use_all || in_array($action,array('remove_from_list','delete_list','unshare'))) { // get the whole selection $query = is_array($session_name) ? $session_name : Api\Cache::getSession('addressbook', $session_name); if ($use_all) { @set_time_limit(0); // switch off the execution time limit, as it's for big selections to small $query['num_rows'] = -1; // all $readonlys = null; $this->get_rows($query,$checked,$readonlys,true); // true = only return the id's } } // replace org_name:* id's with all id's of that org $grouped_contacts = $this->find_grouped_ids($action, $checked, $use_all, $success,$failed,$action_msg,$session_name, $msg); if ($grouped_contacts) $checked = array_unique($checked ? array_merge($checked,$grouped_contacts) : $grouped_contacts); //_debug_array($checked); exit; if (substr($action,0,8) == 'move_to_') { $action = (int)substr($action,8).(substr($action,-1) == 'p' ? 'p' : ''); } elseif (substr($action,0,8) == 'to_list_') { $to_list = (int)substr($action,8); $action = 'to_list'; } elseif (substr($action, 0, 8) == 'to_type_') { $to_type = substr($action,8); $action = 'to_type'; } elseif (substr($action,0,9) == 'document_') { $document = substr($action,9); $action = 'document'; } elseif(substr($action,0,4) == 'cat_') // cat_add_123 or cat_del_456 { $cat_id = (int)substr($action, 8); $action = substr($action,0,7); } elseif(substr($action, 0, 12) === 'shared_with_') { $shared_with = substr($action, 12); $action = 'shared_with'; } // Security: stop non-admins to export more then the configured number of contacts if (in_array($action,array('csv','vcard')) && $this->config['contact_export_limit'] && !Api\Storage\Merge::is_export_limit_excepted() && (!is_numeric($this->config['contact_export_limit']) || count($checked) > $this->config['contact_export_limit'])) { $action_msg = lang('exported'); $failed = count($checked); return false; } switch($action) { case 'vcard': $action_msg = lang('exported'); $vcard = new addressbook_vcal('addressbook','text/vcard'); $vcard->export($checked); // does not return! $Ok = false; break; case 'merge': $error_msg = null; $success = $this->merge($checked,$error_msg); $failed = count($checked) - (int)$success; $action_msg = lang('merged'); $checked = array(); // to not start the single actions break; case 'delete_list': if (!$query['filter2']) { $msg = lang('You need to select a distribution list'); } elseif($this->delete_list($query['filter2']) === false) { $msg = lang('Insufficent rights to delete this list!'); } else { $msg = lang('Distribution list deleted'); unset($query['filter2']); Api\Cache::setSession('addressbook', $session_name, $query); } return false; case 'document': if (!$document) $document = $this->prefs['default_document']; $document_merge = new Api\Contacts\Merge(); $msg = $document_merge->download($document, $checked, '', $this->prefs['document_dir']); $failed = count($checked); return false; case 'infolog_add': Framework::popup(Egw::link('/index.php',array( 'menuaction' => 'infolog.infolog_ui.edit', 'type' => 'task', 'action' => 'addressbook', 'action_id' => implode(',',$checked), )),'_blank',Link::get_registry('infolog', 'add_popup')); $msg = ''; // no message, as we send none in javascript too and users sees opening popup return false; case 'calendar_add': // add appointment for org-views, other views are handled directly in javascript Framework::popup(Egw::link('/index.php',array( 'menuaction' => 'calendar.calendar_uiforms.edit', 'participants' => 'c'.implode(',c',$checked), )),'_blank',Link::get_registry('calendar', 'add_popup')); $msg = ''; // no message, as we send none in javascript too and users sees opening popup return false; case 'calendar_view': // show calendar for org-views, although all views are handled directly in javascript Egw::redirect_link('/index.php',array( 'menuaction' => 'calendar.calendar_uiviews.index', 'owner' => 'c'.implode(',c',$checked), )); } foreach($checked as $id) { switch($action) { case 'cat_add': case 'cat_del': if (($Ok = !!($contact = $this->read($id)) && $this->check_perms(Acl::EDIT,$contact))) { $action_msg = $action == 'cat_add' ? lang('categorie added') : lang('categorie delete'); $cat_ids = $contact['cat_id'] ? explode(',', $contact['cat_id']) : array(); //existing Api\Categories if ($action == 'cat_add') { $cat_ids[] = $cat_id; $cat_ids = array_unique($cat_ids); } elseif ((($key = array_search($cat_id,$cat_ids))) !== false) { unset($cat_ids[$key]); } $ids = $cat_ids ? implode(',',$cat_ids) : null; if ($ids !== $contact['cat_id']) { $contact['cat_id'] = $ids; $Ok = $this->save($contact); } } break; case 'delete': $action_msg = lang('deleted'); if (($Ok = !!($contact = $this->read($id)) && $this->check_perms(Acl::DELETE,$contact))) { if ($contact['owner'] || // regular contact or empty($contact['account_id']) || // accounts without account_id // already deleted account (should no longer happen, but needed to allow for cleanup) $contact['tid'] == self::DELETED_TYPE) { $Ok = $this->delete($id, $contact['tid'] != self::DELETED_TYPE && $contact['account_id']); } // delete single account --> redirect to admin elseif (count($checked) == 1 && $contact['account_id']) { Egw::redirect_link('/index.php',array( 'menuaction' => 'admin.admin_account.delete', 'account_id' => $contact['account_id'], )); // this does NOT return! } else // no mass delete of accounts { $Ok = false; } } break; case 'undelete': $action_msg = lang('recovered'); if (($contact = $this->read($id))) { $contact['tid'] = 'n'; $Ok = $this->save($contact); } break; case 'email': case 'email_home': /* this cant work anymore, as Framework::set_onload does not longer exist $action_fallback = $action == 'email' ? 'email_home' : 'email'; $action_msg = lang('added'); if(($contact = $this->read($id))) { if(strpos($contact[$action],'@') !== false) { $email = $contact[$action]; } elseif(strpos($contact[$action_fallback],'@') !== false) { $email = $contact[$action_fallback]; } else { $Ok = $email = false; } if($email) { $contact['n_fn'] = str_replace(array(',','@'),' ',$contact['n_fn']); Framework::set_onload("addEmail('".addslashes( $contact['n_fn'] ? $contact['n_fn'].' <'.trim($email).'>' : trim($email))."');"); //error_log(__METHOD__.__LINE__."addEmail('".addslashes( // $contact['n_fn'] ? $contact['n_fn'].' <'.trim($email).'>' : trim($email))."');"); $Ok = true; } }*/ break; case 'remove_from_list': $action_msg = lang('removed from distribution list'); if (!$query['filter2']) { $msg = lang('You need to select a distribution list'); return false; } else { $Ok = $this->remove_from_list($id,$query['filter2']) !== false; } break; case 'to_list': $action_msg = lang('added to distribution list'); if (!$to_list) { $msg = lang('You need to select a distribution list'); return false; } else { $Ok = $this->add2list($id,$to_list) !== false; } break; case 'to_type': $action_msg = lang('changed type to %1', lang($this->content_types[$to_type]['name'])); if (($Ok = !!($contact = $this->read($id)) && $this->check_perms(Acl::EDIT,$contact))) { if (!$contact['owner']) // no change of accounts { $Ok = false; } else { $contact['tid'] = $to_type; $Ok = $this->save($contact); } } break; case 'shared_with': // as "unshare" is in "shared_with" submenu/children it uses "shared_with_unshare" if ($shared_with === 'unshare') { $action_msg = lang('unshared'); if (($Ok = !!($contact = $this->read($id)))) { $need_save = false; foreach($contact['shared'] as $key => $shared) { // only unshare contacts shared by current user if (($shared['shared_by'] == $this->user || $this->check_perms(ACL::EDIT, $contact)) && // only unshare from given addressbook, or all (empty($query['filter']) || $shared['shared_with'] == (int)$query['filter'])) { $need_save = true; unset($contact['shared'][$key]); } } // we might need to ignore acl, as we are allowed to share with just read-rights // setting user and update-time is explicitly desired for sync(-collection)! $Ok = !$need_save || $this->save($contact, true); } break; } $action_msg = lang('shared into addressbook %1', Api\Accounts::username($shared_with)); if (($Ok = !!($contact = $this->read($id)))) { $new_shared_with = [[ 'shared_with' => $shared_with, 'shared_by' => $this->user, 'shared_at' => new Api\DateTime('now'), // only allow to share writable, if user has edit-rights! 'shared_writable' => (int)($checkboxes['writable'] && $this->check_perms(Acl::EDIT, $contact)), 'contact_id' => $id, 'contact' => $contact, ]]; if ($this->check_shared_with($new_shared_with, $error_msg)) // returns [] if OK { $Ok = false; } else { $contact['shared'][] = $new_shared_with[0]; // we might need to ignore acl, as we are allowed to share with just read-rights // setting user and update-time is explicitly desired for sync(-collection)! $Ok = $this->save($contact, true); } } break; default: // move to an other addressbook if (!(int)$action || !($this->grants[(string) (int) $action] & Acl::EDIT)) // might be ADD in the future { return false; } if (!$checkboxes['move_to_copy']) { $action_msg = lang('moved'); if (($Ok = !!($contact = $this->read($id)) && $this->check_perms(Acl::DELETE,$contact))) { if (!$contact['owner']) // no (mass-)move of Api\Accounts { $Ok = false; } elseif ($contact['owner'] != (int)$action || $contact['private'] != (int)(substr($action,-1) == 'p')) { $contact['owner'] = (int) $action; $contact['private'] = (int)(substr($action,-1) == 'p'); $Ok = $this->save($contact); } } } else { $action_msg = lang('copied'); if (($Ok = !!($contact = $this->read($id)) && $this->check_perms(Acl::READ,$contact))) { if ($contact['owner'] != (int)$action || $contact['private'] != (int)(substr($action,-1) == 'p')) { $this->copy_contact($contact, false); // do NOT use self::$copy_fields, copy everything but uid etc. $links = $contact['link_to']['to_id']; $contact['owner'] = (int) $action; $contact['private'] = (int)(substr($action,-1) == 'p'); $Ok = $this->save($contact); if ($Ok && is_array($links)) { Link::link('addressbook', $contact['id'], $links); } } } } break; } if ($Ok) { ++$success; } elseif ($action != 'email' && $action != 'email_home') { ++$failed; } } return !$failed; } /** * Find the individual contact IDs for a list of grouped contacts * * Successful lookups are removed from the checked array. * * Used for action on organisation and duplicate views * @param string/int $action 'delete', 'vcard', 'csv' or nummerical account_id to move contacts to that addessbook * @param array $checked contact id's to use if !$use_all * @param boolean $use_all if true use all contacts of the current selection in the session (NOT used!) * @param int &$success number of succeded actions * @param int &$failed number of failed actions (not enought permissions) * @param string &$action_msg translated verb for the actions, to be used in a message like %1 contacts 'deleted' * @param string/array $session_name 'index' or 'email', or array with session-data depending if we are in the main list or the popup * * @return array List of contact IDs in the provided groups */ protected function find_grouped_ids($action,&$checked,$use_all,&$success,&$failed,&$action_msg,$session_name,&$msg) { unset($use_all); $grouped_contacts = array(); foreach((array)$checked as $n => $id) { if (substr($id,0,9) == 'org_name:' || substr($id, 0,10) == 'duplicate:') { if (count($checked) == 1 && !count($grouped_contacts) && $action == 'infolog') { return $this->infolog_org_view($id); // uses the org-name, instead of 'selected contacts' } unset($checked[$n]); $query = Api\Cache::getSession('addressbook', $session_name); $query['num_rows'] = -1; // all $query['grouped_view'] = $id; unset($query['filter2']); $extra = $readonlys = null; $this->get_rows($query,$extra,$readonlys,true); // true = only return the id's // Merge them here, so we only merge the ones that are duplicates, // not merge all selected together if($action == 'merge_duplicates') { $loop_success = $loop_fail = 0; $this->action('merge', $extra, false, $loop_success, $loop_fail, $action_msg,$session_name,$msg); $success += $loop_success; $failed += $loop_fail; } if ($extra[0]) { $grouped_contacts = array_merge($grouped_contacts,$extra); } } } return $grouped_contacts; } /** * Copy a given contact (not storing it!) * * Taken care only configured fields get copied and certain fields never to copy (uid etc.). * * @param array& $content * @param boolean $only_copy_fields =true true: only copy fields configured for copying (eg. no name), * false: copy everything, but never to copy fields */ function copy_contact(array &$content, $only_copy_fields=true) { $content['link_to']['to_id'] = 0; Link::link('addressbook',$content['link_to']['to_id'],'addressbook',$content['id'], lang('Copied by %1, from record #%2.',Api\Accounts::format_username('', $GLOBALS['egw_info']['user']['account_firstname'],$GLOBALS['egw_info']['user']['account_lastname']), $content['id'])); // create a new contact with the content of the old foreach(array_keys($content) as $key) { if($only_copy_fields && !in_array($key, self::$copy_fields) || in_array($key, array('id','etag','carddav_name','uid'))) { unset($content[$key]); } } if(!isset($content['owner'])) { $content['owner'] = $this->default_private ? $this->user.'p' : $this->default_addressbook; } $content['creator'] = $this->user; $content['created'] = $this->now_su; } /** * rows callback for index nextmatch * * @internal * @param array &$query * @param array &$rows returned rows/cups * @param array &$readonlys eg. to disable buttons based on Acl * @param boolean $id_only =false if true only return (via $rows) an array of contact-ids, dont save state to session * @return int total number of contacts matching the selection */ function get_rows(&$query,&$rows,&$readonlys,$id_only=false) { $what = $query['sitemgr_display'] ? $query['sitemgr_display'] : 'index'; if (!$id_only && !$query['csv_export']) // do NOT store state for csv_export or querying id's (no regular view) { $store_query = $query; // Do not store these foreach(array('options-cat_id','actions','action_links','placeholder_actions') as $key) { unset($store_query[$key]); } $old_state = $store_query; Api\Cache::setSession('addressbook', $what, $store_query); } else { $old_state = Api\Cache::getSession('addressbook', $what); } $GLOBALS['egw']->session->commit_session(); if ($query['grouped_view'] === 'shared_by_me') { $query['col_filter']['shared_by'] = $this->user; $query['grouped_view'] = ''; } if (!isset($this->grouped_views[(string) $query['grouped_view']]) || strpos($query['grouped_view'],':') === false) { // we don't have a grouped view, unset the according col_filters $this->unset_grouped_filters($query); } if (isset($this->grouped_views[(string) $query['grouped_view']])) { // we have a grouped view, reset the advanced search if (empty($query['search']) && !empty($old_state['advanced_search'])) { $query['advanced_search'] = $old_state['advanced_search']; } } // eg. paging in an advanced search elseif(empty($query['search']) && is_array($old_state) && array_key_exists('advanced_search', $old_state)) { $query['advanced_search'] = $old_state['advanced_search']; } // Make sure old lettersearch filter doesn't stay - current letter filter will be added later foreach($query['col_filter'] as $key => $col_filter) { if(!is_numeric($key)) continue; if(preg_match('/'.$GLOBALS['egw']->db->capabilities['case_insensitive_like']. ' '.$GLOBALS['egw']->db->quote('[a-z]%').'$/i',$col_filter) == 1 ) { unset($query['col_filter'][$key]); } } //echo "

uicontacts::get_rows(".print_r($query,true).")

\n"; if (!$id_only) { // check if accounts are stored in ldap, which does NOT yet support the org-views if ($this->so_accounts && $query['filter'] === '0' && $query['grouped_view']) { if ($old_state['filter'] === '0') // user changed to org_view { $query['filter'] = ''; // --> change filter to all contacts } else // user changed to accounts { $query['grouped_view'] = ''; // --> change to regular contacts view } } if ($query['grouped_view'] && isset($this->grouped_views[$old_state['grouped_view']]) && !isset($this->grouped_views[$query['grouped_view']])) { $query['searchletter'] = ''; // reset lettersearch if viewing the contacts of one group (org or duplicates) } // save the state of the index in the user prefs $state = serialize(array( 'filter' => $query['filter'], 'cat_id' => $query['cat_id'], 'order' => $query['order'], 'sort' => $query['sort'], 'col_filter' => array('tid' => $query['col_filter']['tid']), 'grouped_view' => $query['grouped_view'], )); if ($state != $this->prefs[$what.'_state'] && !$query['csv_export']) { $GLOBALS['egw']->preferences->add('addressbook',$what.'_state',$state); // save prefs, but do NOT invalid the cache (unnecessary) $GLOBALS['egw']->preferences->save_repository(false,'user',false); } } unset($old_state); if ((string)$query['cat_id'] != '') { $query['col_filter']['cat_id'] = $query['cat_id'] ? $query['cat_id'] : null; } else { unset($query['col_filter']['cat_id']); } if ($query['filter'] !== '') // not all addressbooks { $query['col_filter']['owner'] = (string) (int) $query['filter']; if ($this->private_addressbook) { $query['col_filter']['private'] = substr($query['filter'],-1) == 'p' ? 1 : 0; } } else { unset($query['col_filter']['owner']); unset($query['col_filter']['private']); } if ((int)$query['filter2']) // not no distribution list { $query['col_filter']['list'] = (string) (int) $query['filter2']; } else { unset($query['col_filter']['list']); } if ($GLOBALS['egw_info']['user']['preferences']['addressbook']['hide_accounts'] === '1') { $query['col_filter']['account_id'] = null; } else { unset($query['col_filter']['account_id']); } // all backends allow now at least to use groups as distribution lists $query['no_filter2'] = false; // Grouped view if (isset($this->grouped_views[(string) $query['grouped_view']]) && !$query['col_filter']['parent_id']) { $query['grouped_view_label'] = ''; $rows = $this->get_grouped_rows($query); } else // contacts view { if($query['col_filter']['parent_id']) { $query['grouped_view'] = $query['col_filter']['parent_id']; } // Query doesn't like parent_id unset($query['col_filter']['parent_id']); if ($query['grouped_view']) // view the contacts of one organisation only { if (strpos($query['grouped_view'],'*AND*') !== false) $query['grouped_view'] = str_replace('*AND*','&',$query['grouped_view']); if (!is_array($fields = $GLOBALS['egw_info']['user']['preferences']['addressbook']['duplicate_fields'] ?? [])) { $fields = explode(',', $fields); } foreach(explode('|||',$query['grouped_view']) as $part) { list($name,$value) = explode(':',$part,2); // do NOT set invalid column, as this gives an SQL error ("AND AND" in sql) if (static::$duplicate_fields[$name] && $value && ( strpos($query['grouped_view'], 'duplicate:') === 0 && in_array($name, $fields) || strpos($query['grouped_view'], 'duplicate:') !== 0 )) { $query['col_filter'][$name] = $value; } } } else if($query['actions'] && !$query['actions']['edit']) { // Just switched from grouped view, update actions $query['actions'] = $this->get_actions($query['col_filter']['tid']); } // translate the select order to the really used over all 3 columns $sort = $query['sort']; switch($query['order']) // "xxx<>'' DESC" sorts contacts with empty order-criteria always at the end { // we don't exclude them, as the total would otherwise depend on the order-criteria case 'org_name': $order = "egw_addressbook.org_name<>''DESC,egw_addressbook.org_name $sort,n_family $sort,n_given $sort"; break; default: if ($query['order'][0] == '#') // we order by a custom field { $order = "{$query['order']} $sort,org_name $sort,n_family $sort,n_given $sort"; break; } $query['order'] = 'n_family'; case 'n_family': $order = "n_family<>'' DESC,n_family $sort,n_given $sort,org_name $sort"; break; case 'n_given': $order = "n_given<>'' DESC,n_given $sort,n_family $sort,org_name $sort"; break; case 'n_fileas': $order = "n_fileas<>'' DESC,n_fileas $sort"; break; case 'adr_one_postalcode': case 'adr_two_postalcode': $order = $query['order']."<>'' DESC,".$query['order']." $sort,org_name $sort,n_family $sort,n_given $sort"; break; case 'contact_modified': case 'contact_created': $order = "$query[order] IS NULL,$query[order] $sort,org_name $sort,n_family $sort,n_given $sort"; break; case 'contact_id': $order = "egw_addressbook.$query[order] $sort"; } if ($query['searchletter']) // only show contacts if the order-criteria starts with the given letter { $no_letter_search = array('adr_one_postalcode', 'adr_two_postalcode', 'contact_id', 'contact_created','contact_modified'); $query['col_filter'][] = (in_array($query['order'],$no_letter_search) ? 'org_name' : (substr($query['order'],0,1)=='#'?'':'egw_addressbook.').$query['order']).' '. $GLOBALS['egw']->db->capabilities['case_insensitive_like'].' '.$GLOBALS['egw']->db->quote($query['searchletter'].'%'); } $wildcard = '%'; $op = 'OR'; if ($query['advanced_search']) { // Make sure op & wildcard are only valid options $op = $query['advanced_search']['operator'] == $op ? $op : 'AND'; unset($query['advanced_search']['operator']); $wildcard = $query['advanced_search']['meth_select'] == $wildcard ? $wildcard : ''; unset($query['advanced_search']['meth_select']); } $columsel = $this->prefs['nextmatch-addressbook.index.rows']; $columselection = $columsel ? explode(',',$columsel) : array(); $extracols = []; if (in_array('owner_shared_with', $columselection)) { $extracols[] = 'shared_with'; } $rows = parent::search($query['advanced_search'] ? $query['advanced_search'] : $query['search'],$id_only, $order, $extracols, $wildcard,false, $op,[(int)$query['start'], (int)$query['num_rows']], $query['col_filter']); // do we need to read the custom fields, depends on the column is enabled and customfields $available_distib_lists=$this->get_lists(Acl::READ); $ids = $calendar_participants = array(); if (!$id_only && $rows) { $show_custom_fields = (in_array('customfields',$columselection) || $this->config['index_load_cfs']) && $this->customfields; $show_calendar = $this->config['disable_event_column'] != 'True' && in_array('calendar_calendar',$columselection); $show_distributionlist = in_array('distribution_list', $columselection) || is_array($available_distib_lists) && count($available_distib_lists); if ($show_calendar || $show_custom_fields || $show_distributionlist) { foreach($rows as $val) { $ids[] = $val['id']; $calendar_participants[$val['id']] = $val['account_id'] ? $val['account_id'] : 'c'.$val['id']; } if ($show_custom_fields) { $selected_cfs = array(); if(in_array('customfields',$columselection)) { foreach($columselection as $col) { if ($col[0] == '#') $selected_cfs[] = substr($col,1); } } $selected_cfs = array_unique(array_merge($selected_cfs, (array)$this->config['index_load_cfs'])); $customfields = $this->read_customfields($ids,$selected_cfs); } if ($show_calendar && !empty($ids)) $calendar = $this->read_calendar($calendar_participants); // distributionlist memership for the entrys //_debug_array($this->get_lists(Acl::EDIT)); if ($show_distributionlist && $available_distib_lists) { $distributionlist = $this->read_distributionlist($ids,array_keys($available_distib_lists)); } } } } if (!$rows) $rows = array(); if ($id_only) { foreach($rows as $n => $row) { $rows[$n] = $row['id']; } return $this->total; // no need to set other fields or $readonlys } $order = $query['order']; $unshare_grants = []; foreach($this->grants as $grantee => $rights) { if ($rights & (ACL::EDIT|self::ACL_SHARED)) $unshare_grants[] = $grantee; } $readonlys = array(); foreach($rows as $n => &$row) { $given = $row['n_given'] ? $row['n_given'] : ($row['n_prefix'] ? $row['n_prefix'] : ''); switch($order) { default: // postalcode, created, modified, ... case 'org_name': $row['line1'] = $row['org_name']; $row['line2'] = $row['n_family'].($given ? ', '.$given : ''); break; case 'n_family': $row['line1'] = $row['n_family'].($given ? ', '.$given : ''); $row['line2'] = $row['org_name']; break; case 'n_given': $row['line1'] = $given.' '.$row['n_family']; $row['line2'] = $row['org_name']; break; case 'n_fileas': if (!$row['n_fileas']) $row['n_fileas'] = $this->fileas($row); list($row['line1'],$row['line2']) = explode(': ',$row['n_fileas']); break; } if (isset($this->grouped_views[(string) $query['grouped_view']])) { $row['type'] = 'home'; $row['type_label'] = $query['grouped_view'] == 'duplicate' ? lang('Duplicates') : lang('Organisation'); if ($query['filter'] && !($this->grants[(int)$query['filter']] & Acl::DELETE)) { $row['class'] .= 'rowNoDelete '; } $row['class'] .= 'rowNoEdit '; // no edit in OrgView $row['class'] .= $query['grouped_view'] == 'duplicates' ? 'contact_duplicate' : 'contact_organisation '; } else { $this->type_icon($row['owner'],$row['private'],$row['tid'],$row['type'],$row['type_label']); static $tel2show = array('tel_work','tel_cell','tel_home','tel_fax'); static $prefer_marker = null; if (is_null($prefer_marker)) { // as et2 adds options with .text(), it can't be entities, but php knows no string literals with utf-8 $prefer_marker = html_entity_decode(' ☆', ENT_NOQUOTES, 'utf-8'); } foreach($tel2show as $name) { $row[$name] .= ' '.($row['tel_prefer'] == $name ? $prefer_marker : ''); // .' ' to NOT remove the field } // always show the prefered phone, if not already shown if (!in_array($row['tel_prefer'],$tel2show) && $row[$row['tel_prefer']]) { $row['tel_prefered'] = $row[$row['tel_prefer']].$prefer_marker; } // Show nice name as status text if($row['tel_prefer']) { $row['tel_prefer_label'] = $this->contact_fields[$row['tel_prefer']]; } if (!$row['owner'] && $row['account_id'] > 0) { $row['class'] .= 'rowAccount rowNoDelete '; } elseif (!$this->check_perms(Acl::DELETE,$row) || (!$GLOBALS['egw_info']['user']['apps']['admin'] && $this->config['history'] != 'userpurge' && $query['col_filter']['tid'] == self::DELETED_TYPE)) { $row['class'] .= 'rowNoDelete '; } if (!$this->check_perms(Acl::EDIT,$row)) { $row['class'] .= 'rowNoEdit '; } $row['class'] .= 'contact_contact '; if (!self::hasPhoto($row)) { $row['lname'] = $row['n_family']; $row['fname'] = $row['n_given']; unset($row['photo']); // no need to send, as there is no photo } unset($row['jpegphoto']); // unused and messes up json encoding (not utf-8) if (isset($customfields[$row['id']])) { foreach($this->customfields as $name => $data) { $row['#'.$name] = $customfields[$row['id']]['#'.$name]; } } if (isset($distributionlist[$row['id']])) { $row['distrib_lists'] = implode("\n",array_values($distributionlist[$row['id']])); //if ($show_distributionlist) $readonlys['distrib_lists'] =true; } if (isset($calendar[$calendar_participants[$row['id']]])) { foreach($calendar[$calendar_participants[$row['id']]] as $name => $data) { $row[$name] = $data; } } } // hide region for address format 'postcode_city' if (($row['addr_format'] = $this->addr_format_by_country($row['adr_one_countryname']))=='postcode_city') unset($row['adr_one_region']); if (($row['addr_format2'] = $this->addr_format_by_country($row['adr_two_countryname']))=='postcode_city') unset($row['adr_two_region']); // respect category permissions if(!empty($row['cat_id'])) { $row['cat_id'] = $this->categories->check_list(Acl::READ,$row['cat_id']); } if ($query['col_filter']['shared_by'] == $this->user || !empty($row['shared_with']) && array_intersect($unshare_grants, explode(',', $row['shared_with']))) { $row['class'] .= 'unshare_contact '; } } $rows['no_distribution_list'] = (bool)$query['filter2']; // disable customfields column, if we have no customefield(s) if (!$this->customfields) { $rows['no_customfields'] = true; } // Disable next/last date if so configured if($this->config['disable_event_column'] == 'True') { $rows['no_event_column'] = true; } // If we've changed the sort order on them, update the display if($order !== $query['order'] ) { $rows['order'] = $order; } $rows['call_popup'] = $this->config['call_popup']; $rows['customfields'] = array_values($this->customfields); // full app-header with all search criteria specially for the print $header = array(); if ($query['filter'] !== '' && !isset($this->grouped_views[$query['grouped_view']])) { $header[] = ($query['filter'] == '0' ? lang('accounts') : ($GLOBALS['egw']->accounts->get_type($query['filter']) == 'g' ? lang('Group %1',$GLOBALS['egw']->accounts->id2name($query['filter'])) : Api\Accounts::username((int)$query['filter']). (substr($query['filter'],-1) == 'p' ? ' ('.lang('private').')' : ''))); } if ($query['grouped_view']) { $header[] = $query['grouped_view_label']; // Make sure option is there if(!array_key_exists($query['grouped_view'], $this->grouped_views)) { $this->grouped_views += $this->_get_grouped_name($query['grouped_view']); $rows['sel_options']['grouped_view'] = $this->grouped_views; } } if($query['advanced_search']) { $header[] = lang('Advanced search'); } if ($query['cat_id']) { $header[] = lang('Category').' '.$GLOBALS['egw']->categories->id2name($query['cat_id']); } if ($query['searchletter']) { $order = $order == 'n_given' ? lang('first name') : ($order == 'n_family' ? lang('last name') : lang('Organisation')); $header[] = lang("%1 starts with '%2'",$order,$query['searchletter']); } if ($query['search'] && !$query['advanced_search']) // do not add that, if we have advanced search active { $header[] = lang("Search for '%1'",$query['search']); } $GLOBALS['egw_info']['flags']['app_header'] = implode(': ', $header); if ($query['grouped_view'] === '' && $query['col_filter']['shared_by'] == $this->user) { $query['grouped_view'] = 'shared_by_me'; unset($query['col_filter']['shared_by']); } return $this->total; } /** * Get addressbook type icon from owner, private and tid * * @param int $owner user- or group-id or 0 for Api\Accounts * @param boolean $private * @param string $tid 'n' for regular addressbook * @param string &$icon icon-name * @param string &$label translated label */ function type_icon($owner,$private,$tid,&$icon,&$label) { if (!$owner) { $icon = 'accounts'; $label = lang('accounts'); } elseif ($private) { $icon = 'private'; $label = lang('private'); } elseif ($GLOBALS['egw']->accounts->get_type($owner) == 'g') { $icon = 'group'; $label = lang('group %1',$GLOBALS['egw']->accounts->id2name($owner)); } else { $icon = 'personal'; $label = $owner == $this->user ? lang('personal') : Api\Accounts::username($owner); } // show tid icon for tid!='n' AND only if one is defined if ($tid != 'n' && Api\Image::find('addressbook',$this->content_types[$tid]['name'])) { $icon = Api\Image::find('addressbook',$this->content_types[$tid]['name']); } // Legacy - from when icons could be anywhere if ($tid != 'n' && $this->content_types[$tid]['options']['icon']) { $icon = $this->content_types[$tid]['options']['icon']; $label = $this->content_types[$tid]['name'].' ('.$label.')'; } } /** * Edit a contact * * @param array $content=null submitted content * @param int $_GET['contact_id'] contact_id mainly for popup use * @param bool $_GET['makecp'] true if you want to copy the contact given by $_GET['contact_id'] */ function edit($content=null) { if (is_array($content)) { // sync $content['shared'] with $content['shared_values'] foreach((array)$content['shared'] as $key => $shared) { $shared_value = $shared['shared_id'].':'.$shared['shared_with'].':'.$shared['shared_by'].':'.$shared['shared_writable']; if (($k = array_search($shared_value, (array)$content['shared_values'])) === false) { unset($content['shared'][$key]); } else { unset($content['shared_values'][$k]); } } foreach((array)$content['shared_values'] as $account_id) { $content['shared'][] = [ 'contact_id' => $content['id'], 'contact' => $content, 'shared_with' => $account_id, 'shared_by' => $this->user, 'shared_at' => new Api\DateTime(), 'shared_writable' => (int)(bool)$content['shared_writable'], ]; } unset($content['shared_values']); // remove invalid shared-with entries (should not happen, as we validate already on client-side) $this->check_shared_with($content['shared']); $button = @key($content['button'] ?? []); unset($content['button']); $content['private'] = (int) ($content['owner'] && substr($content['owner'],-1) == 'p'); $content['owner'] = (string) (int) $content['owner']; $content['cat_id'] = $this->config['cat_tab'] === 'Tree' ? $content['cat_id_tree'] : $content['cat_id']; switch($button) { case 'save': case 'apply': if ($content['presets_fields']) { // unset the duplicate_filed after submit because we don't need to warn user for second time about contact duplication unset($content['presets_fields']); } // photo might be changed by ajax_upload_photo if (!array_key_exists('jpegphoto', $content)) { $content['photo_unchanged'] = true; // hint no need to store photo } $links = false; if (!$content['id'] && is_array($content['link_to']['to_id'])) { $links = $content['link_to']['to_id']; } $fullname = $old_fullname = parent::fullname($content); if ($content['id'] && $content['org_name'] && $content['change_org']) { $old_org_entry = $this->read($content['id']); $old_fullname = ($old_org_entry['n_fn'] ? $old_org_entry['n_fn'] : parent::fullname($old_org_entry)); } if ( $content['n_fn'] != $fullname || $fullname != $old_fullname) { unset($content['n_fn']); } // Country codes foreach(array('adr_one', 'adr_two') as $c_prefix) { // we store region-name not code if (!empty($content[$c_prefix.'_region'])) { $states = Api\Country::get_states($content[$c_prefix.'_countrycode']); if ($states && isset($states[$content[$c_prefix.'_region']])) { $content[$c_prefix.'_region'] = $states[$content[$c_prefix.'_region']]; } } // handling custom country-name if (!Api\Country::get_full_name($content[$c_prefix.'_countrycode'])) { $content[$c_prefix.'_countryname'] = $content[$c_prefix.'_countrycode']; unset($content[$c_prefix.'_countrycode']); } } $content['msg'] = ''; $this->error = false; foreach((array)$content['pre_save_callbacks'] as $callback) { try { if (($success_msg = call_user_func_array($callback, array(&$content)))) { $content['msg'] .= ($content['msg'] ? ', ' : '').$success_msg; } } catch (Exception $ex) { $content['msg'] .= ($content['msg'] ? ', ' : '').$ex->getMessage(); $button = 'apply'; // do not close dialog $this->error = true; break; } } if ($this->error) { // error in pre_save_callbacks } elseif ($this->save($content)) { $content['msg'] .= ($content['msg'] ? ', ' : '').lang('Contact saved'); unset($content['jpegphoto'], $content['photo_unchanged']); foreach((array)$content['post_save_callbacks'] as $callback) { try { if (($success_msg = call_user_func_array($callback, array(&$content)))) { $content['msg'] .= ', '.$success_msg; } } catch(Api\Exception\Redirect $r) { // catch it to continue execution and rethrow it later } catch (Exception $ex) { $content['msg'] .= ', '.$ex->getMessage(); $button = 'apply'; // do not close dialog $this->error = true; break; } } if ($content['change_org'] && $old_org_entry && ($changed = $this->changed_fields($old_org_entry,$content,true)) && ($members = $this->org_similar($old_org_entry['org_name'],$changed))) { //foreach($changed as $name => $old_value) echo "

$name: '$old_value' --> '{$content[$name]}'

\n"; list($changed_members,$changed_fields,$failed_members) = $this->change_org($old_org_entry['org_name'],$changed,$content,$members); if ($changed_members) { $content['msg'] .= ', '.lang('%1 fields in %2 other organisation member(s) changed',$changed_fields,$changed_members); } if ($failed_members) { $content['msg'] .= ', '.lang('failed to change %1 organisation member(s) (insufficent rights) !!!',$failed_members); } } } elseif($this->error === true) { $content['msg'] = lang('Error: the entry has been updated since you opened it for editing!').'
'. lang('Copy your changes to the clipboard, %1reload the entry%2 and merge them.','',''); break; // dont refresh the list } else { $content['msg'] = lang('Error saving the contact !!!'). ($this->error ? ' '.$this->error : ''); $button = 'apply'; // to not leave the dialog } // writing links for new entry, existing ones are handled by the widget itself if ($links && $content['id']) { Link::link('addressbook',$content['id'],$links); } // Update client side global datastore $response = Api\Json\Response::get(); $response->generic('data', array('uid' => 'addressbook::'.$content['id'], 'data' => $content)); Framework::refresh_opener($content['msg'], 'addressbook', $content['id'], $content['id'] ? 'edit' : 'add', null, null, null, $this->error ? 'error' : 'success'); // re-throw redirect exception, if there's no error if (!$this->error && isset($r)) { throw $r; } if ($button == 'save') { Framework::window_close(); } else { Framework::message($content['msg'], $this->error ? 'error' : 'success'); unset($content['msg']); } $content['link_to']['to_id'] = $content['id']; break; case 'delete': $success = $failed = $action_msg = null; if($this->action('delete',array($content['id']),false,$success,$failed,$action_msg,'',$content['msg'])) { if ($GLOBALS['egw']->currentapp == 'addressbook') { Framework::refresh_opener(lang('Contact deleted'), 'addressbook', $content['id'], 'delete' ); Framework::window_close(); } else { Framework::refresh_opener(lang('Contact deleted'), 'addressbook', $content['id'], null, 'addressbook'); Framework::window_close(); } } else { $content['msg'] = lang('Error deleting the contact !!!'); } break; } $view = !$this->check_perms(Acl::EDIT, $content); } else { $content = array(); $contact_id = $_GET['contact_id'] ? $_GET['contact_id'] : ((int)$_GET['account_id'] ? 'account:'.(int)$_GET['account_id'] : 0); $view = (boolean)$_GET['view']; // new contact --> set some defaults if ($contact_id && is_array($content = $this->read($contact_id))) { $contact_id = $content['id']; // it could have been: "account:$account_id" if (!$this->check_perms(Acl::EDIT, $content)) { $view = true; } } else // not found { $state = Api\Cache::getSession('addressbook', 'index'); // check if we create the new contact in an existing org if (($org = $_GET['org'])) { // arguments containing a comma get quoted by etemplate/js/nextmatch_action.js // leading to error in Api\Db::column_data_implode, if not unquoted if ($org[0] == '"') $org = substr($org, 1, -1); $content = $this->read_org($org); } elseif ($state['grouped_view'] && !isset($this->grouped_views[$state['grouped_view']])) { $content = $this->read_org($state['grouped_view']); } else { if ($GLOBALS['egw_info']['user']['preferences']['common']['country']) { $content['adr_one_countrycode'] = $GLOBALS['egw_info']['user']['preferences']['common']['country']; $content['adr_one_countryname'] = $GLOBALS['egw']->country->get_full_name($GLOBALS['egw_info']['user']['preferences']['common']['country']); $content['adr_two_countrycode'] = $GLOBALS['egw_info']['user']['preferences']['common']['country']; $content['adr_two_countryname'] = $GLOBALS['egw']->country->get_full_name($GLOBALS['egw_info']['user']['preferences']['common']['country']); } if ($this->prefs['fileas_default']) $content['fileas_type'] = $this->prefs['fileas_default']; } if (isset($_GET['owner']) && $_GET['owner'] !== '') { $content['owner'] = $_GET['owner']; } else { $content['owner'] = (string)($state['filter'] == 0 ? '' : $state['filter']); } $content['private'] = (int) ($content['owner'] && substr($content['owner'],-1) == 'p'); if ($content['owner'] === '' || !($this->grants[$content['owner'] = (string) (int) $content['owner']] & Acl::ADD)) { $content['owner'] = $this->default_addressbook; $content['private'] = (int)$this->default_private; if (!($this->grants[$content['owner'] = (string) (int) $content['owner']] & Acl::ADD)) { $content['owner'] = (string) $this->user; $content['private'] = 0; } } $new_type = array_keys($this->content_types); // fetch active type to preset the type, if param typeid is not passed $active_tid = Api\Cache::getSession('addressbook','active_tid'); if ($active_tid && strtoupper($active_tid) === 'D') unset($active_tid); $content['tid'] = $_GET['typeid'] ? $_GET['typeid'] : ($active_tid?$active_tid:$new_type[0]); foreach($this->get_contact_columns() as $field) { if ($_GET['presets'][$field]) { if ($field=='email'||$field=='email_home') { $singleAddress = imap_rfc822_parse_adrlist($_GET['presets'][$field],''); //error_log(__METHOD__.__LINE__.' Address:'.$singleAddress[0]->mailbox."@".$singleAddress[0]->host.", ".$singleAddress[0]->personal); if (!(!is_array($singleAddress) || count($singleAddress)<1)) { $content[$field] = $singleAddress[0]->mailbox."@".$singleAddress[0]->host; if (!empty($singleAddress[0]->personal)) { if (strpos($singleAddress[0]->personal,',')===false) { list($P_n_given,$P_n_family,$P_org_name)=explode(' ',$singleAddress[0]->personal,3); if (strlen(trim($P_n_given))>0) $content['n_given'] = trim($P_n_given); if (strlen(trim($P_n_family))>0) $content['n_family'] = trim($P_n_family); if (strlen(trim($P_org_name))>0) $content['org_name'] = trim($P_org_name); } else { list($P_n_family,$P_other)=explode(',',$singleAddress[0]->personal,2); if (strlen(trim($P_n_family))>0) $content['n_family'] = trim($P_n_family); if (strlen(trim($P_other))>0) { list($P_n_given,$P_org_name)=explode(',',$P_other,2); if (strlen(trim($P_n_given))>0) $content['n_given'] = trim($P_n_given); if (strlen(trim($P_org_name))>0) $content['org_name'] = trim($P_org_name); } } } } else { $content[$field] = $_GET['presets'][$field]; } } else { $content[$field] = $_GET['presets'][$field]; } } } if (isset($_GET['presets'])) { foreach(array('email','email_home','n_family','n_given','org_name') as $field) { if (!empty($content[$field])) { //Set the presets fields in content in order to be able to use them later in client side for checking duplication only on first time load // after save/apply we unset them $content['presets_fields'][]= $field; break; } } if (empty($content['n_fn'])) $content['n_fn'] = $this->fullname($content); } $content['creator'] = $this->user; $content['created'] = $this->now_su; unset($state); //_debug_array($content); } if ($_GET['msg']) $content['msg'] = strip_tags($_GET['msg']); // dont allow HTML! if($content && $_GET['makecp']) // copy the contact { $this->copy_contact($content); $content['msg'] = lang('%1 copied - the copy can now be edited', lang(Link::get_registry('addressbook','entry'))); $view = false; } else { if ($contact_id && is_numeric($contact_id)) $content['link_to']['to_id'] = $contact_id; } // automatic link new entries to entries specified in the url if (!$contact_id && isset($_REQUEST['link_app']) && isset($_REQUEST['link_id']) && !is_array($content['link_to']['to_id'])) { $link_ids = is_array($_REQUEST['link_id']) ? $_REQUEST['link_id'] : array($_REQUEST['link_id']); foreach(is_array($_REQUEST['link_app']) ? $_REQUEST['link_app'] : array($_REQUEST['link_app']) as $n => $link_app) { $link_id = $link_ids[$n]; if (preg_match('/^[a-z_0-9-]+:[:a-z_0-9-]+$/i',$link_app.':'.$link_id)) // gard against XSS { Link::link('addressbook',$content['link_to']['to_id'],$link_app,$link_id); } } } } // set $content[shared_options/_values] from $content[shared] $content['shared_options'] = []; $content['shared_values'] = []; foreach((array)$content['shared'] as $shared) { $shared_value = $shared['shared_id'] . ':' . $shared['shared_with'] . ':' . $shared['shared_by'] . ':' . $shared['shared_writable']; $content['shared_values'][] = $shared_value; $sel_options['shared_values'][] = [ 'value' => $shared_value, 'label' => Api\Accounts::username($shared['shared_with']), 'title' => lang('%1 shared this contact on %2 with %3 %4', Api\Accounts::username($shared['shared_by']), Api\DateTime::to($shared['shared_at']), Api\Accounts::username($shared['shared_with']), $shared['shared_writable'] ? lang('writable') : lang('readonly') ), 'icon' => $shared['shared_writable'] ? 'edit' : 'view', ]; } // disable shared with UI for non-SQL backends $content['shared_disabled'] = !is_a($this->get_backend($content['id'], $content['owner']), Api\Contacts\Sql::class); if ($content['id']) { // last and next calendar date $dates = current($this->read_calendar(array($content['account_id'] ? $content['account_id'] : 'c'.$content['id']),false)); if(is_array($dates)) $content += $dates; } // Registry has view_id as contact_id, so set it (custom fields uses it) $content['contact_id'] = $content['id']; // Avoid ID conflict with tree & selectboxes $content['cat_id_tree'] = $content['cat_id']; // how to display addresses $content['addr_format'] = $this->addr_format_by_country($content['adr_one_countryname']); $content['addr_format2'] = $this->addr_format_by_country($content['adr_two_countryname']); // Country codes foreach(array('adr_one', 'adr_two') as $c_prefix) { // handling custom country-name if (empty($content[$c_prefix.'_countrycode']) && !empty($content[$c_prefix.'_countryname'])) { $content[$c_prefix.'_countrycode'] = $content[$c_prefix.'_countryname']; } // translate from our stored state-/region-name to the code if (!empty($content[$c_prefix.'_region']) && !empty($content[$c_prefix.'_countrycode'])) { $states = Api\Country::get_states($content[$c_prefix.'_countrycode']); if ($states && ($key = array_search($content[$c_prefix.'_region'], $states))) { $content[$c_prefix.'_region'] = $key; } } } //_debug_array($content); $readonlys['button[delete]'] = !$content['owner'] || !$this->check_perms(Acl::DELETE,$content); $readonlys['button[copy]'] = $readonlys['button[edit]'] = $readonlys['button[vcard]'] = true; $readonlys['button[save]'] = $readonlys['button[apply]'] = $view; if ($view) { $readonlys['__ALL__'] = true; $readonlys['button[cancel]'] = false; } $sel_options['fileas_type'] = $this->fileas_options($content); $sel_options['owner'] = $this->get_addressbooks(Acl::ADD); if ($content['owner']) unset($sel_options['owner'][0]); // do not offer to switch to accounts, as we do not support moving contacts to accounts if ((string) $content['owner'] !== '') { if (!isset($sel_options['owner'][(int)$content['owner']])) { $sel_options['owner'][(int)$content['owner']] = !$content['owner'] ? lang('Accounts') : Api\Accounts::username($content['owner']); } $readonlys['owner'] = !$content['owner'] || // dont allow to move accounts, as this mean deleting the user incl. all content he owns $content['id'] && !$this->check_perms(Acl::DELETE,$content); // you need delete rights to move an existing contact into an other addressbook } // set the unsupported fields from the backend to readonly foreach($this->get_fields('unsupported',$content['id'],$content['owner']) as $field) { $readonlys[$field] = true; } // for editing own account, make all fields not allowed by own_account_acl readonly if (!$this->is_admin() && !$content['owner'] && $content['account_id'] == $this->user && $this->own_account_acl && !$view) { $readonlys['__ALL__'] = true; $readonlys['button[cancel]'] = false; foreach($this->own_account_acl as $field) { $readonlys[$field] = false; } if (!$readonlys['jpegphoto']) { $readonlys = array_merge($readonlys, array( 'upload_photo' => false, 'delete_photo' => false, 'addressbook.edit.upload' => false )); } if (!$readonlys['pubkey']) { $readonlys['addressbook:'.$content['id'].':.files/pgp-pubkey.asc'] = $readonlys['addressbook:'.$content['id'].':.files/smime-pubkey.crt'] = false; } } if (isset($readonlys['n_fileas'])) $readonlys['fileas_type'] = $readonlys['n_fileas']; // disable not needed tabs $readonlys['tabs']['cats'] = !($content['cat_tab'] = $this->config['cat_tab']); $readonlys['tabs']['custom'] = !$this->customfields || // only show custom fields tab for LDAP, if we have LDAP CF's defined and an existing contact (as no schema defined) $this->get_backend($content['id'],$content['owner']) == $this->so_accounts && (empty($content['id']) || !array_filter($this->customfields, static function($cf) { return substr($cf['name'], 0, 5) === 'ldap_'; })); $readonlys['tabs']['custom_private'] = $readonlys['tabs']['custom'] || !$this->config['private_cf_tab']; $readonlys['tabs']['distribution_list'] = !$content['distrib_lists'];#false; $readonlys['tabs']['history'] = $this->contact_repository != 'sql' || !$content['id'] || $this->account_repository != 'sql' && $content['account_id']; if (!$content['id']) $readonlys['button[delete]'] = !$content['id']; if ($this->config['private_cf_tab']) $content['no_private_cfs'] = 0; $content['hide_change_org'] = $readonlys['change_org'] = empty($content['org_name']) || $view; // for editing the own account (by a non-admin), enable only the fields allowed via the "own_account_acl" if (!$content['owner'] && !$this->check_perms(Acl::EDIT, $content)) { $this->_set_readonlys_for_own_account_acl($readonlys, $content['id']); } for($i = -23; $i<=23; $i++) { $tz[$i] = ($i > 0 ? '+' : '').$i; } $sel_options['tz'] = $tz; $content['tz'] = $content['tz'] ? $content['tz'] : '0'; if (count($this->content_types) > 1) { foreach($this->content_types as $type => $data) { $sel_options['tid'][$type] = $data['name']; } $content['typegfx'] = Api\Html::image('addressbook',$this->content_types[$content['tid']]['options']['icon'],'',' width="16px" height="16px"'); } else { $content['no_tid'] = true; } $content['view'] = false; $content['link_to'] = array( 'to_app' => 'addressbook', 'to_id' => $content['link_to']['to_id'], ); // Links for deleted entries if($content['tid'] == self::DELETED_TYPE) { $content['link_to']['show_deleted'] = true; if(!$GLOBALS['egw_info']['user']['apps']['admin'] && $this->config['history'] != 'userpurge') { $readonlys['button[delete]'] = true; } } // Enable history $this->setup_history($content, $sel_options); $content['photo'] = $this->photo_src($content['id'],$content['jpegphoto'],'',$content['etag']); if ($content['private']) $content['owner'] .= 'p'; // for custom types, check if we have a custom edit template named "addressbook.edit.$type", $type is the name if (in_array($content['tid'], array('n',self::DELETED_TYPE)) || !$this->tmpl->read('addressbook.edit.'.$this->content_types[$content['tid']]['name'])) { $this->tmpl->read('addressbook.edit'); } // allow other apps to add tabs to addressbook edit $preserve = $content; $preserve['old_owner'] = $content['owner']; unset($preserve['jpegphoto'], $content['jpegphoto']); // unused and messes up json encoding (not utf-8) $this->tmpl->setElementAttribute('tabs', 'add_tabs', true); $tabs =& $this->tmpl->getElementAttribute('tabs', 'extraTabs'); if (($first_call = !isset($tabs))) { $tabs = array(); } //error_log(__LINE__.': '.__METHOD__."() first_call=$first_call"); $hook_data = Api\Hooks::process(array('location' => 'addressbook_edit')+$content); //error_log(__METHOD__."() hook_data=".array2string($hook_data)); foreach($hook_data as $extra_tabs) { if (!$extra_tabs) continue; foreach(count(array_filter(array_keys($extra_tabs), 'is_int')) ? $extra_tabs : array($extra_tabs) as $extra_tab) { if ($extra_tab['data'] && is_array($extra_tab['data'])) { $content = array_merge($content, $extra_tab['data']); } if ($extra_tab['preserve'] && is_array($extra_tab['preserve'])) { $preserve = array_merge($preserve, $extra_tab['preserve']); } if ($extra_tab['readonlys'] && is_array($extra_tab['readonlys'])) { $readonlys = array_merge($readonlys, $extra_tab['readonlys']); } // we must NOT add tabs and callbacks more then once! if (!$first_call) continue; if (!empty($extra_tab['pre_save_callback'])) { $preserve['pre_save_callbacks'][] = $extra_tab['pre_save_callback']; } if (!empty($extra_tab['post_save_callback'])) { $preserve['post_save_callbacks'][] = $extra_tab['post_save_callback']; } if (!empty($extra_tab['label']) && !empty($extra_tab['name'])) { $tabs[] = array( 'label' => $extra_tab['label'], 'template' => $extra_tab['name'], 'prepend' => $extra_tab['prepend'], ); } //error_log(__METHOD__."() changed tabs=".array2string($tabs)); } } return $this->tmpl->exec('addressbook.addressbook_ui.edit', $content, $sel_options, $readonlys, $preserve, 2); } /** * Check if user has right to share with / into given AB * * @param array $_data values for keys "shared_writable", "shared_values" and "contact" * @return array of entries removed from $shared_with because current user is not allowed to share into */ public function ajax_check_shared(array $_data) { $response = Api\Json\Response::get(); try { $shared = []; foreach($_data['shared_values'] as $value) { if (is_numeric($value)) { $shared[$value] = [ 'shared_with' => $value, 'shared_by' => $this->user, 'shared_writable' => (int)$_data['shared_writable'], ]; } else { $shared[$value] = array_combine(['shared_id', 'shared_with', 'shared_by', 'shared_writable'], explode(':', $value)); } $shared[$value]['contact'] = $_data['contact']; } if (($failed = $this->check_shared_with($shared, $error))) { $response->data(array_keys($failed)); $response->message($error ?: lang('You are not allowed to share into the addressbook of %1', implode(', ', array_map(function ($data) { return Api\Accounts::username($data['shared_with']); }, $failed))), 'error'); } } catch (\Exception $e) { $response->message($e->getMessage(), 'error'); } } /** * Set the readonlys for non-admins editing their own account * * @param array &$readonlys * @param int $id */ function _set_readonlys_for_own_account_acl(&$readonlys,$id) { // regular fields depending on the backend foreach($this->get_fields('supported',$id,0) as $field) { if (!$this->own_account_acl || !in_array($field,$this->own_account_acl)) { $readonlys[$field] = true; switch($field) { case 'tel_work': case 'tel_cell': case 'tel_home': $readonlys[$field.'2'] = true; break; case 'n_fileas': $readonlys['fileas_type'] = true; break; } } } // custom fields if ($this->customfields) { foreach(array_keys($this->customfields) as $name) { if (!$this->own_account_acl || !in_array('#'.$name,$this->own_account_acl)) { $readonlys['#'.$name] = true; } } } // links if (!$this->own_account_acl || !in_array('link_to',$this->own_account_acl)) { $readonlys['link_to'] = true; } } /** * Doublicate check: returns similar contacts: same email or 2 of name, firstname, org * * Also update/return fileas options, if necessary. * * @param array $values contact values from form * @param string $name name of changed value, eg. "email" * @param int $own_id =0 own contact id, to not check against it * @return array with keys 'msg' => "EMail address exists, do you want to open contact?" (or null if not existing) * 'data' => array of id => "full name (addressbook)" pairs * 'fileas_options' */ public function ajax_check_values($values, $name, $own_id=0) { if (!is_array($fields = $GLOBALS['egw_info']['user']['preferences']['addressbook']['duplicate_fields'] ?? [])) { $fields = explode(',', $fields); } $threshold = (int)$GLOBALS['egw_info']['user']['preferences']['addressbook']['duplicate_threshold']; $ret = array('doublicates' => array(), 'msg' => null); // set not returned n_fileas value, to keep custom fileas value $values['n_fileas'] = $this->fileas($values, $values['fileas_type']); // if email changed, check for doublicates if (in_array($name, array('email', 'email_home')) && in_array('contact_'.$name, $fields)) { if (preg_match(Etemplate\Widget\Url::EMAIL_PREG, $values[$name])) // only search for real email addresses, to not return to many contacts { $contacts = parent::search(array( 'email' => $values[$name], 'email_home' => $values[$name], ), false, '', '', '', false, 'OR'); } } else { // only set fileas-options if other then email changed $ret['fileas_options'] = array_values($this->fileas_options($values)); // Full options for et2 $ret['fileas_sel_options'] = $this->fileas_options($values); // if name, firstname or org changed and enough are specified, check for doublicates $specified_count = 0; foreach($fields as $field) { if($values[trim($field)]) { $specified_count++; } } if (in_array($name,$fields) && $specified_count >= $threshold) { $filter = array(); foreach($fields as $n) // use email too, to exclude obvious false positives { if (!empty($values[$n])) $filter[$n] = $values[$n]; } $contacts = parent::search('', false, '', '', '', false, 'AND', false, $filter); } } if ($contacts) { foreach($contacts as $contact) { if ($own_id && $contact['id'] == $own_id) continue; $ret['doublicates'][$contact['id']] = $this->fileas($contact).' ('. (!$contact['owner'] ? lang('Accounts') : ($contact['owner'] == $this->user ? ($contact['private'] ? lang('Private') : lang('Personal')) : Api\Accounts::username($contact['owner']))).')'; } if ($ret['doublicates']) { $ret['msg'] = lang('Similar contacts found:'). "\n\n".implode("\n", $ret['doublicates'])."\n\n". lang('Open for editing?'); } } //error_log(__METHOD__.'('.array2string($values).", '$name', $own_id) doublicates found ".array2string($ret['doublicates'])); Api\Json\Response::get()->data($ret); } /** * CRM view * * @param array $content */ function view(array $content=null) { // CRM list comes from content, request, or preference $crm_list = $content['crm_list'] ? $content['crm_list'] : ($_GET['crm_list'] ? $_GET['crm_list'] : $GLOBALS['egw_info']['user']['preferences']['addressbook']['crm_list']); if(!$crm_list || $crm_list == '~edit~') $crm_list = 'infolog'; if(is_array($content)) { $button = key($content['button'] ?? []); switch ($button) { case 'vcard': Egw::redirect_link('/index.php','menuaction=addressbook.uivcard.out&ab_id=' .$content['id']); case 'cancel': Egw::redirect_link('/index.php','menuaction=addressbook.addressbook_ui.index&ajax=true'); case 'delete': Egw::redirect_link('/index.php',array( 'menuaction' => 'addressbook.addressbook_ui.index', 'msg' => $this->delete($content) ? lang('Contact deleted') : lang('Error deleting the contact !!!'), )); case 'next': $inc = 1; // fall through case 'back': if (!isset($inc)) $inc = -1; // get next/previous contact in selection $query = Api\Cache::getSession('addressbook', 'index'); $query['start'] = $content['index'] + $inc; $query['num_rows'] = 1; $rows = $readonlys = array(); $num_rows = $this->get_rows($query, $rows, $readonlys, true); //error_log(__METHOD__."() get_rows()=$num_rows rows=".array2string($rows)); $contact_id = $rows[0]; if(!$contact_id || !is_array($content = $this->read($contact_id))) { Egw::redirect_link('/index.php',array( 'menuaction' => 'addressbook.addressbook_ui.index', 'msg' => $content, 'ajax' => 'true' )); } $content['index'] = $query['start']; // List nextmatch is already there, just update the filter if($contact_id && Api\Json\Request::isJSONRequest()) { switch($crm_list) { case 'infolog-organisation': $contact_id = $this->get_all_org_contacts($contact_id); // Fall through case 'infolog': case 'tracker': default: Api\Json\Response::get()->apply('app.addressbook.view_set_list',Array(Array('action'=>'addressbook', 'action_id' => $contact_id))); break; } // Clear contact_id, it's used as a flag to send the list unset($contact_id); } break; default: // No button, probably a refresh $content = $this->read($content['id']); break; } } else { // allow to search eg. for a phone number if (isset($_GET['search'])) { $query = Api\Cache::getSession('addressbook', 'index'); $query['search'] = $_GET['search']; unset($_GET['search']); // reset all filters unset($query['advanced_search']); $query['col_filter'] = array(); $query['filter'] = $query['filter2'] = $query['cat_id'] = ''; Api\Cache::setSession('addressbook', 'index', $query); $query['start'] = 0; $query['num_rows'] = 1; $rows = $readonlys = array(); $num_rows = $this->get_rows($query, $rows, $readonlys, true); $_GET['contact_id'] = array_shift($rows); $_GET['index'] = 0; } $contact_id = $_GET['contact_id'] ?? ((int)$_GET['account_id'] ? 'account:'.(int)$_GET['account_id'] : 0); if(!$contact_id || !is_array($content = $this->read($contact_id))) { Egw::redirect_link('/index.php',array( 'menuaction' => 'addressbook.addressbook_ui.index', 'msg' => $content, 'ajax' => 'true' )+(isset($_GET['search']) ? array('search' => $_GET['search']) : array())); } if (isset($_GET['index'])) { $content['index'] = (int)$_GET['index']; // get number of rows to determine if we can have a next button $query = Api\Cache::getSession('addressbook', 'index'); $query['start'] = $content['index']; $query['num_rows'] = 1; $rows = $readonlys = array(); $num_rows = $this->get_rows($query, $rows, $readonlys, true); } } $content['jpegphoto'] = !empty($content['jpegphoto']); // unused and messes up json encoding (not utf-8) // make everything not explicit mentioned readonly $readonlys['__ALL__'] = true; $readonlys['photo'] = $readonlys['button[copy]'] =false; foreach(array_keys($this->contact_fields) as $key) { if (in_array($key,array('tel_home','tel_work','tel_cell','tel_fax'))) { $content[$key.'2'] = $content[$key]; } } // respect category permissions if(!empty($content['cat_id'])) { $content['cat_id'] = $this->categories->check_list(Acl::READ,$content['cat_id']); } $content['cat_id_tree'] = $content['cat_id']; $content['view'] = true; $content['link_to'] = array( 'to_app' => 'addressbook', 'to_id' => $content['id'], ); // Links for deleted entries if($content['tid'] == self::DELETED_TYPE) { $content['link_to']['show_deleted'] = true; } $readonlys['button[delete]'] = !$content['owner'] || !$this->check_perms(Acl::DELETE,$content); $readonlys['button[edit]'] = !$this->check_perms(Acl::EDIT,$content); // how to display addresses $content['addr_format'] = $this->addr_format_by_country($content['adr_one_countryname']); $content['addr_format2'] = $this->addr_format_by_country($content['adr_two_countryname']); $sel_options['fileas_type'][$content['fileas_type']] = $this->fileas($content); $sel_options['owner'] = $this->get_addressbooks(); for($i = -23; $i<=23; $i++) { $tz[$i] = ($i > 0 ? '+' : '').$i; } $sel_options['tz'] = $tz; $content['tz'] = $content['tz'] ? $content['tz'] : 0; if (count($this->content_types) > 1) { foreach($this->content_types as $type => $data) { $sel_options['tid'][$type] = $data['name']; } $content['typegfx'] = Api\Html::image('addressbook',$this->content_types[$content['tid']]['options']['icon'],'',' width="16px" height="16px"'); } else { $content['no_tid'] = true; } $this->tmpl->read('addressbook.view'); /*if (!$this->tmpl->read($this->content_types[$content['tid']]['options']['template'] ? $this->content_types[$content['tid']]['options']['template'] : 'addressbook.edit')) { $content['msg'] = lang('WARNING: Template "%1" not found, using default template instead.', $this->content_types[$content['tid']]['options']['template'])."\n"; $content['msg'] .= lang('Please update the templatename in your customfields section!'); $this->tmpl->read('addressbook.edit'); }*/ if ($this->private_addressbook && $content['private'] && $content['owner'] == $this->user) { $content['owner'] .= 'p'; } $content['hide_change_org'] = true; // Prevent double countries - invalid code blanks it, disabling doesn't work $content['adr_one_countrycode'] = '-'; $content['adr_two_countrycode'] = '-'; // Enable history $this->setup_history($content, $sel_options); // disable not needed tabs $readonlys['tabs']['cats'] = !($content['cat_tab'] = $this->config['cat_tab']); $readonlys['tabs']['distribution_list'] = !$content['distrib_lists'];#false; $readonlys['tabs']['history'] = $this->contact_repository != 'sql' || !$content['id'] || $this->account_repository != 'sql' && $content['account_id']; if ($this->config['private_cf_tab']) $content['no_private_cfs'] = 0; // last and next calendar date if (!empty($content['id'])) $dates = current($this->read_calendar(array($content['account_id'] ? $content['account_id'] : 'c'.$content['id']),false)); if(is_array($dates)) $content += $dates; // Disable importexport $GLOBALS['egw_info']['flags']['disable_importexport']['export'] = true; $GLOBALS['egw_info']['flags']['disable_importexport']['merge'] = true; // set id for automatic linking via quick add $GLOBALS['egw_info']['flags']['currentid'] = $content['id']; // load app.css for addressbook explicit, as addressbook_view hooks changes currentapp! Framework::includeCSS('addressbook', 'app'); // dont show an app-header $GLOBALS['egw_info']['flags']['app_header'] = ''; // always show sidebox, as it contains contact-data unset($GLOBALS['egw_info']['user']['preferences']['common']['auto_hide_sidebox']); // need to load list's app.js now, as exec calls header before other app can include it // Framework::includeJS('/'.$crm_list.'/js/app.js'); // Load CRM code Framework::includeJS('.','CRM','addressbook'); $content['view_sidebox'] = addressbook_hooks::getViewDOMID($content['id'], $crm_list); $this->tmpl->exec('addressbook.addressbook_ui.view',$content,$sel_options,$readonlys,array( 'id' => $content['id'], 'index' => $content['index'], 'crm_list' => $crm_list )); // Only load this on first time - we're using AJAX, so it stays there through submits. // Sending it again (via ajax) will break the addressbook.view etemplate2 if($contact_id) { // Show for whole organisation, not just selected contact if($crm_list == 'infolog-organisation') { $crm_list = str_replace('-organisation','',$crm_list); $_query = Api\Cache::getSession('addressbook', 'index'); $content['id'] = $this->get_all_org_contacts($content['id']); } Api\Hooks::single(array( 'location' => 'addressbook_view', 'ab_id' => $content['id'] ),$crm_list); } } /** * Get all the contact IDs in the given contact's organisation * * @param int $contact_id * @param Array $query Optional base query * * @return array of contact IDs in the organisation */ function get_all_org_contacts($contact_id, $query = array()) { $contact = $this->read($contact_id); // No org name, early return with just the contact if(!$contact['org_name']) { return array($contact_id); } $query['num_rows'] = -1; $query['start'] = 0; if(!array_key_exists('filter', $query)) { $query['filter'] = ''; } if(!is_array($query['col_filter'])) { $query['col_filter'] = array(); } $query['grouped_view'] = 'org_name:'.$contact['org_name']; $org_contacts = array(); $readonlys = null; $this->get_rows($query,$org_contacts,$readonlys,true); // true = only return the id's return $org_contacts ? $org_contacts : array($contact_id); } /** * convert email-address in compose link * * @param string $email email-addresse * @return array/string array with get-params or mailto:$email, or '' or no mail addresse */ function email2link($email) { if (strpos($email,'@') == false) return ''; if($GLOBALS['egw_info']['user']['apps']['mail']) { return array( 'menuaction' => 'mail.mail_compose.compose', 'send_to' => base64_encode($email) ); } if($GLOBALS['egw_info']['user']['apps']['felamimail']) { return array( 'menuaction' => 'felamimail.uicompose.compose', 'send_to' => base64_encode($email) ); } if($GLOBALS['egw_info']['user']['apps']['email']) { return array( 'menuaction' => 'email.uicompose.compose', 'to' => $email, ); } return 'mailto:' . $email; } /** * Extended search * * @param array $_content * @return string */ function extSearch($_content=array()) { if(!empty($_content)) { $_content['cat_id'] = $this->config['cat_tab'] === 'Tree' ? $_content['cat_id_tree'] : $_content['cat_id']; $response = Api\Json\Response::get(); $query = Api\Cache::getSession('addressbook', 'index'); if ($_content['button']['cancelsearch']) { unset($query['advanced_search']); } else { $query['advanced_search'] = array_intersect_key( $_content, array_flip(array_merge($this->get_contact_columns(), array('operator', 'meth_select'))) ); foreach($query['advanced_search'] as $key => $value) { if(!$value) { unset($query['advanced_search'][$key]); } } // Skip n_fn, it causes problems in sql unset($query['advanced_search']['n_fn']); } $query['search'] = ''; // store the index state in the session Api\Cache::setSession('addressbook', 'index', $query); // store the advanced search in the session to call it again Api\Cache::setSession('addressbook', 'advanced_search', $query['advanced_search']); // Update client / nextmatch with filters, or clear $response->call("app.addressbook.adv_search", array('advanced_search' => $_content['button']['search'] ? $query['advanced_search'] : '')); if ($_content['button']['cancelsearch']) { Framework::window_close (); // No need to reload popup return; } } $GLOBALS['egw_info']['etemplate']['advanced_search'] = true; // initialize etemplate arrays $sel_options = $readonlys = array(); $this->tmpl->read('addressbook.edit'); $content = Api\Cache::getSession('addressbook', 'advanced_search'); $content['n_fn'] = $this->fullname($content); // Avoid ID conflict with tree & selectboxes $content['cat_id_tree'] = $content['cat_id']; for($i = -23; $i<=23; $i++) { $tz[$i] = ($i > 0 ? '+' : '').$i; } $sel_options['tz'] = $tz + array('' => lang('doesn\'t matter')); $sel_options['tid'][] = lang('all'); //foreach($this->content_types as $type => $data) $sel_options['tid'][$type] = $data['name']; // configure search options $sel_options['owner'] = $this->get_addressbooks(Acl::READ,lang('all')); $sel_options['operator'] = array( 'AND' => lang('AND'), 'OR' => lang('OR'), ); $sel_options['meth_select'] = array( '%' => lang('contains'), false => lang('exact'), ); if ($this->customfields) { foreach($this->customfields as $name => $data) { if(substr($data['type'], 0, 6) == 'select' && !($data['rows'] > 1)) { if(!isset($content['#' . $name])) { $content['#' . $name] = ''; } if(!isset($data['values'][''])) { $sel_options['#' . $name][''] = lang('Select one'); } } // Make them not required, otherwise you can't search $this->tmpl->setElementAttribute('#' . $name, 'required', FALSE); } } // configure edit template as search dialog $readonlys['change_photo'] = true; $readonlys['fileas_type'] = true; $readonlys['creator'] = true; // this setting will enable (and show) the search and cancel buttons, setting this to true will hide the before mentioned buttons completely $readonlys['button'] = false; // disable not needed tabs $readonlys['tabs']['cats'] = !($content['cat_tab'] = $this->config['cat_tab']); $readonlys['tabs']['links'] = true; $readonlys['tabs']['distribution_list'] = true; $readonlys['tabs']['history'] = true; // setting hidebuttons for content will hide the 'normal' addressbook edit dialog buttons $content['hidebuttons'] = true; $content['hide_change_org'] = true; $content['no_tid'] = true; $content['showsearchbuttons'] = true; // enable search operation and search buttons| they're disabled by default if ($this->config['private_cf_tab']) $content['no_private_cfs'] = 0; return $this->tmpl->exec('addressbook.addressbook_ui.extSearch',$content,$sel_options,$readonlys,array(),2); } /** * Check if there's a photo for given contact id. This is used for avatar widget * to set or unset delete button. If there's no uploaded photo it responses true. * * @param type $contact_id */ function ajax_noPhotoExists ($contact_id) { $response = Api\Json\Response::get(); $response->data((!($contact = $this->read($contact_id)) || empty($contact['photo']) && !(($contact['files'] & Api\Contacts::FILES_BIT_PHOTO) && ($size = filesize($url=Api\Link::vfs_path('addressbook', $contact_id, Api\Contacts::FILES_PHOTO)))))); } /** * Ajax method to update edited avatar photo via avatar widget * * @param string $etemplate_exec_id to update id, files, etag, ... * @param file string $file null means to delete */ function ajax_update_photo ($etemplate_exec_id, $file) { $et_request = Api\Etemplate\Request::read($etemplate_exec_id); $response = Api\Json\Response::get(); if ($file) { $filteredFile = substr($file, strpos($file, ",")+1); // resize photo if wider then default width of 240pixel (keeping aspect ratio) $decoded = $this->resize_photo(base64_decode($filteredFile)); } $response->data(true); // add photo into current eT2 request $et_request->preserv = array_merge($et_request->preserv, array( 'jpegphoto' => is_null($file) ? $file : $decoded, 'photo_unchanged' => false, // hint photo is changed )); } /** * Callback for vfs-upload widgets for PGP and S/Mime pubkey * * @param array $file * @param string $widget_id * @param Api\Etemplate\Request $request eT2 request eg. to access attribute $content * @param Api\Json\Response $response */ public function pubkey_uploaded(array $file, $widget_id, Api\Etemplate\Request $request, Api\Json\Response $response) { //error_log(__METHOD__."(".array2string($file).", ...) widget_id=$widget_id, id=".$request->content['id'].", files=".$request->content['files']); unset($file, $response); // not used, but required by function signature list(,,$path) = explode(':', $widget_id); $bit = $path === Api\Contacts::FILES_PGP_PUBKEY ? Api\Contacts::FILES_BIT_PGP_PUBKEY : Api\Contacts::FILES_BIT_SMIME_PUBKEY; if (!($request->content['files'] & $bit) && $this->check_perms(Acl::EDIT, $request->content)) { $content = $request->content; $content['files'] |= $bit; $content['photo_unchanged'] = true; // hint no need to store photo if ($this->save($content)) { $changed = array_diff_assoc($content, $request->content); //error_log(__METHOD__."() changed=".array2string($changed)); $request->content = $content; // need to update preserv, as edit stores content there too and we would get eg. an contact modified error when trying to store $request->preserv = array_merge($request->preserv, $changed); } } } /** * Migrate contacts to or from LDAP (called by Admin >> Addressbook >> Site configuration (Admin only) * */ function migrate2ldap($type=null) { $GLOBALS['egw_info']['flags']['app_header'] = lang('Addressbook').' - '.lang('Migration to LDAP'); echo $GLOBALS['egw']->framework->header(); echo $GLOBALS['egw']->framework->navbar(); if (!$this->is_admin()) { echo '

'.lang('Permission denied !!!')."

\n"; } else { parent::migrate2ldap($type ?? $_GET['type']); echo '

'.lang('Migration finished')."

\n"; } echo $GLOBALS['egw']->framework->footer(); } /** * Set n_fileas (and n_fn) in contacts of all users (called by Admin >> Addressbook >> Site configuration (Admin only) * * If $_GET[all] all fileas fields will be set, if !$_GET[all] only empty ones * */ function admin_set_fileas() { Api\Translation::add_app('admin'); $GLOBALS['egw_info']['flags']['app_header'] = lang('Addressbook').' - '.lang('Contact maintenance'); echo $GLOBALS['egw']->framework->header(); echo $GLOBALS['egw']->framework->navbar(); // check if user has admin rights AND if a valid fileas type is given (Security) if (!$this->is_admin() || $_GET['type'] != '' && !in_array($_GET['type'],$this->fileas_types)) { echo '

'.lang('Permission denied !!!')."

\n"; } else { $errors = null; $updated = parent::set_all_fileas($_GET['type'],(boolean)$_GET['all'],$errors,true); // true = ignore Acl echo '

'.lang('%1 contacts updated (%2 errors).',$updated,$errors)."

\n"; } echo $GLOBALS['egw']->framework->footer(); } /** * Cleanup all contacts of all users (called by Admin >> Addressbook >> Site configuration (Admin only) * */ function admin_set_all_cleanup() { Api\Translation::add_app('admin'); $GLOBALS['egw_info']['flags']['app_header'] = lang('Addressbook').' - '.lang('Contact maintenance'); echo $GLOBALS['egw']->framework->header(); echo $GLOBALS['egw']->framework->navbar(); // check if user has admin rights (Security) if (!$this->is_admin()) { echo '

'.lang('Permission denied !!!')."

\n"; } else { $errors = null; $updated = parent::set_all_cleanup($errors,true); // true = ignore Acl echo '

'.lang('%1 contacts updated (%2 errors).',$updated,$errors)."

\n"; } echo $GLOBALS['egw']->framework->footer(); } /** * Set up history log widget */ protected function setup_history(&$content, &$sel_options) { if ($this->contact_repository == 'ldap' || !$content['id'] || $this->account_repository == 'ldap' && $content['account_id']) { return; // no history for ldap as history table only allows integer id's } $content['history'] = array( 'id' => $content['id'], 'app' => 'addressbook', 'status-widgets' => array( 'owner' => 'select-account', 'creator' => 'select-account', 'created' => 'date-time', 'cat_id' => 'select-cat', 'adr_one_countrycode' => 'select-country', 'adr_two_countrycode' => 'select-country', ), ); foreach($this->content_types as $id => $settings) { $content['history']['status-widgets']['tid'][$id] = $settings['name']; } $sel_options['status'] = $this->contact_fields; // Addressbook also has an 'owner' field, which has different options. // If we don't put something here (just empty won't work), history log will use // those options instead of the select-account options. $sel_options['history']['owner'] = ['ignore' => 'me']; } }