* @author Ralf Becker * @copyright (c) 2005/6 by Cornelius Weiss and Ralf Becker * @package addressbook * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @version $Id$ */ require_once(EGW_INCLUDE_ROOT.'/addressbook/inc/class.bocontacts.inc.php'); /** * General user interface object of the adressbook * * @package addressbook * @author Cornelius Weiss * @author Ralf Becker * @copyright (c) 2005/6 by Cornelius Weiss and Ralf Becker * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License */ class uicontacts extends bocontacts { var $public_functions = array( 'search' => True, 'edit' => True, 'view' => True, 'index' => True, 'photo' => True, 'emailpopup'=> True, 'migrate2ldap' => True, ); /** * use a separate private addressbook (former private flag), for contacts not shareable via regular read acl * * @var boolean */ var $private_addressbook = false; var $org_views; var $config; function uicontacts($contact_app='addressbook') { $this->bocontacts($contact_app); foreach(array( 'tmpl' => 'etemplate.etemplate', ) as $my => $app_class) { list(,$class) = explode('.',$app_class); if (!is_object($GLOBALS['egw']->$class)) { $GLOBALS['egw']->$class =& CreateObject($app_class); } $this->$my = &$GLOBALS['egw']->$class; } $this->private_addressbook = $this->contact_repository == 'sql' && $this->prefs['private_addressbook']; $this->org_views = array( 'org_name' => lang('Organisations'), 'org_name,adr_one_locality' => lang('Organisations by location'), 'org_name,org_unit' => lang('Organisations by departments'), ); // our javascript // to be moved in a seperate file if rewrite is over $GLOBALS['egw_info']['flags']['java_script'] .= $this->js(); $this->config =& $GLOBALS['egw_info']['server']; } /** * List contacts of an addressbook * * @param array $content=null submitted content * @param string $msg=null message to show * @param boolean $do_email=false do an email-selection popup or the regular index-page */ function index($content=null,$msg=null,$do_email=false) { //echo "

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

\n"; if (($re_submit = is_array($content))) { $do_email = $content['do_email']; if (isset($content['nm']['rows']['delete'])) // handle a single delete like delete with the checkboxes { list($id) = @each($content['nm']['rows']['delete']); $content['action'] = 'delete'; $content['nm']['rows']['checked'] = array($id); } if ($content['action'] !== '') { if (!count($content['nm']['rows']['checked']) && !$content['use_all']) { $msg = lang('You need to select some contacts first'); } else { if ($this->action($content['action'],$content['nm']['rows']['checked'],$content['use_all'], $success,$failed,$action_msg,$content['do_email'] ? 'email' : 'index')) { $msg .= lang('%1 contact(s) %2',$success,$action_msg); } else { $msg .= lang('%1 contact(s) %2, %3 failed because of insufficent rights !!!',$success,$action_msg,$failed); } } } if ($content['nm']['rows']['infolog']) { list($org) = each($content['nm']['rows']['infolog']); return $this->infolog_org_view($org); } if ($content['nm']['rows']['view']) // show all contacts of an organisation { list($org_view) = each($content['nm']['rows']['view']); } else { $org_view = $content['nm']['org_view']; } } $preserv = array( 'do_email' => $do_email, ); $content = array( 'msg' => $msg ? $msg : $_GET['msg'], ); $content['nm'] = $GLOBALS['egw']->session->appsession($do_email ? 'email' : 'index','addressbook'); if (!is_array($content['nm'])) { $content['nm'] = array( 'get_rows' => 'addressbook.uicontacts.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 'options-cat_id' => array(lang('none')), '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) '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' => '', // IO filter2, if not 'no_filter2' => True 'filter2_no_lang'=> True, // I set no_lang for filter2 (=dont translate the options) 'lettersearch' => true, 'do_email' => $do_email, ); // use the state of the last session stored in the user prefs if (($state = @unserialize($this->prefs[$do_email ? 'email_state' : 'index_state']))) { $content['nm'] = array_merge($content['nm'],$state); } } if ($do_email) { if (!$re_submit) { $content['nm']['to'] = 'to'; $content['nm']['search'] = '@'; } $content['nm']['header_left'] = 'addressbook.email.left'; } // Organisation stuff is not (yet) availible with ldap elseif($GLOBALS['egw_info']['server']['contact_repository'] != 'ldap') { $content['nm']['header_left'] = 'addressbook.index.left'; } $sel_options = array( 'filter' => $this->get_addressbooks(EGW_ACL_READ,lang('All')), 'to' => array( 'to' => 'To', 'cc' => 'Cc', 'bcc' => 'Bcc', ), ); $sel_options['action'] = array(); if ($do_email) { $sel_options['action'] = array( 'email' => lang('Add %1',lang('business email')), 'email_home' => lang('Add %1',lang('home email')), ); } $sel_options['action'] += array( 'delete' => lang('Delete'), 'csv' => lang('Export as CSV'), 'vcard' => lang('Export as VCard'), // ToDo: move this to importexport framework ); if ($GLOBALS['egw_info']['user']['apps']['infolog']) { $sel_options['action']['infolog'] = lang('View linked InfoLog entries'); } $sel_options['action'] += $this->get_addressbooks(EGW_ACL_ADD); if (!array_key_exists('importexport',$GLOBALS['egw_info']['user']['apps'])) unset($sel_options['action']['export']); // dont show tid-selection if we have only one content_type if (count($this->content_types) <= 1) { $content['nm']['col_filter']['tid'] = 'n'; } else { $content['nm']['header_right'] = 'addressbook.index.right'; foreach($this->content_types as $tid => $data) { $sel_options['col_filter[tid]'][$tid] = $data['name']; } } // get the availible org-views plus the label of the contacts view of one org $sel_options['org_view'] = $this->org_views; if (isset($org_view)) $content['nm']['org_view'] = $org_view; if (!isset($sel_options['org_view'][(string) $content['nm']['org_view']])) { $org_name = array(); foreach(explode('|||',$content['nm']['org_view']) as $part) { list(,$name) = explode(':',$part,2); if ($name) $org_name[] = $name; } $org_name = implode(': ',$org_name); $sel_options['org_view'][(string) $content['nm']['org_view']] = $org_name; } $content['nm']['org_view_label'] = $sel_options['org_view'][(string) $content['nm']['org_view']]; $this->tmpl->read(/*$do_email ? 'addressbook.email' :*/ 'addressbook.index'); return $this->tmpl->exec($do_email ? 'addressbook.uicontacts.emailpopup' : 'addressbook.uicontacts.index', $content,$sel_options,$readonlys,$preserv,$do_email ? 2 : 0); } /** * Email address-selection popup * * @param array $content=null submitted content * @param string $msg=null message to show */ function emailpopup($content=null,$msg=null) { if ($_GET['compat']) // 1.2 felamimail or old email { $handler = "if (opener.document.doit[to].value != '') { opener.document.doit[to].value += ','; } opener.document.doit[to].value += email"; } else // 1.3+ felamimail { $handler = 'opener.addEmail(to,email)'; } $GLOBALS['egw_info']['flags']['java_script'] .= " "; return $this->index($content,$msg,true); } /** * Show the infologs of an whole organisation * * @param string $org */ function infolog_org_view($org) { $query = $GLOBALS['egw']->session->appsession('index','addressbook'); $query['num_rows'] = -1; // all $query['org_view'] = $org; $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(); 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 } $GLOBALS['egw']->redirect_link('/index.php',array( 'menuaction' => 'infolog.uiinfolog.index', 'action' => 'addressbook', 'action_id' => implode(',',$checked), 'action_title' => $org, )); } /** * 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 $session_name 'index' or 'email' depending if we are in the main list or the popup * @return boolean true if all actions succeded, false otherwise */ function action($action,$checked,$use_all,&$success,&$failed,&$action_msg,$session_name) { //echo "

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

\n"; $success = $failed = 0; if ($use_all) { // get the whole selection $query = $GLOBALS['egw']->session->appsession($session_name,'addressbook'); $query['num_rows'] = -1; // all $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 $org_contacts = array(); foreach($checked as $n => $id) { if (substr($id,0,9) == 'org_name:') { if (count($checked) == 1) { return $this->infolog_org_view($id); // uses the org-name, instead of 'selected contacts' } unset($checked[$n]); $query = $GLOBALS['egw']->session->appsession($session_name,'addressbook'); $query['num_rows'] = -1; // all $query['org_view'] = $id; $this->get_rows($query,$extra,$readonlys,true); // true = only return the id's if ($extra[0]) $org_contacts = array_merge($org_contacts,$extra); } } if ($org_contacts) $checked = array_unique($checked ? array_merge($checked,$org_contacts) : $org_contacts); //_debug_array($checked); return false; switch($action) { case 'csv': $action_msg = lang('exported'); $csv_export =& CreateObject('addressbook.csv_export',$this,$this->prefs['csv_charset']); switch ($this->prefs['csv_fields']) { case 'business': $fields = $this->business_contact_fields; break; case 'home': $fields = $this-home_contact_fields; break; default: $fields = $this->contact_fields; foreach($this->customfields as $name => $data) { $fields['#'.$name] = $data['label']; } break; } $csv_export->export($checked,$fields); // does not return! $Ok = true; break; case 'vcard': $action_msg = lang('exported'); ExecMethod('addressbook.vcaladdressbook.export',$checked); // does not return! $Ok = false; break; case 'infolog': $GLOBALS['egw']->redirect_link('/index.php',array( 'menuaction' => 'infolog.uiinfolog.index', 'action' => 'addressbook', 'action_id' => implode(',',$checked), 'action_title' => count($checked) > 1 ? lang('selected contacts') : '', )); break; } foreach($checked as $id) { switch($action) { case 'delete': $action_msg = lang('deleted'); if (($Ok = !!($contact = $this->read($id)) && $this->check_perms(EGW_ACL_DELETE,$contact))) { if ($contact['owner']) // regular contact { $Ok = $this->delete($id); } // delete single account --> redirect to admin elseif (count($checked) == 1 && $contact['account_id']) { $GLOBALS['egw']->redirect_link('/index.php',array( 'menuaction' => 'admin.uiaccounts.delete_user', 'account_id' => $contact['account_id'], )); // this does NOT return! } else // no mass delete of accounts { $Ok = false; } } break; case 'email': case 'email_home': $action_msg = lang('%1 added',$action=='email'?lang('Business email') : lang('Home email')); if (($Ok = !!($contact = $this->read($id)) && strstr($contact[$action],'@'))) { if(!@is_object($GLOBALS['egw']->js)) $GLOBALS['egw']->js =& CreateObject('phpgwapi.javascript'); $GLOBALS['egw']->js->set_onload("addEmail('".addslashes( $contact['n_fn'] ? $contact['n_fn'].' <'.$contact[$action].'>' : $contact[$action])."');"); } break; default: // move to an other addressbook if (!is_numeric($action) || !($this->grants[(string) (int) $action] & EGW_ACL_EDIT)) // might be ADD in the future { return false; } $action_msg = lang('moved'); if (($Ok = !!($contact = $this->read($id)) && $this->check_perms(EGW_ACL_DELETE,$contact))) { if (!$contact['owner']) // no mass-change of accounts { $Ok = false; } elseif ($contact['owner'] != $action) // no need to change { $contact['owner'] = (int) $action; $contact['private'] = 0; $Ok = $this->save($contact); } } break; } if ($Ok) { ++$success; } elseif ($action != 'email' && $action != 'email_home') { ++$failed; } } return !$failed; } /** * 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) { $do_email = $query['do_email']; $old_state = $GLOBALS['egw']->session->appsession($do_email ? 'email' : 'index','addressbook'); if (isset($this->org_views[(string) $query['org_view']])) // we have an org view, reset the advanced search { if (is_array($query['search'])) unset($query['search']); unset($query['advanced_search']); } elseif (is_array($query['search'])) // new advanced search, store it in the session { $query['advanced_search'] = array_intersect_key($query['search'],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]); } $query['start'] = 0; $query['search'] = ''; // store the advanced search in the session to call it again $GLOBALS['egw']->session->appsession('advanced_search','addressbook',$query['advanced_search']); } elseif(!$query['search'] && $old_state['advanced_search']) // eg. paging in an advanced search { $query['advanced_search'] = $old_state['advanced_search']; } if ($do_email && $GLOBALS['egw_info']['etemplate']['loop'] && is_object($GLOBALS['egw']->js)) { // remove previous addEmail() calls, otherwise they will be run again $GLOBALS['egw']->js->body['onLoad'] = preg_replace('/addEmail\([^)]+\);/','',$GLOBALS['egw']->js->body['onLoad']); } //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['org_view']) { if ($old_state['filter'] === '0') // user changed to org_view { $query['filter'] = ''; // --> change filter to all contacts } else // user changed to accounts { $query['org_view'] = ''; // --> change to regular contacts view } } if ($query['org_view'] && isset($this->org_views[$old_state['org_view']]) && !isset($this->org_views[$query['org_view']])) { $query['searchletter'] = ''; // reset lettersearch if viewing the contacts of one organisation } $GLOBALS['egw']->session->appsession($do_email ? 'email' : 'index','addressbook',$query); // 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']), 'org_view' => $query['org_view'], )); if ($state != $this->prefs['index_state']) { $GLOBALS['egw']->preferences->add('addressbook',$do_email ? 'email_state' : 'index_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; } } if (isset($this->org_views[(string) $query['org_view']])) // we have an org view { $query['template'] = 'addressbook.index.org_rows'; if ($query['order'] != 'org_name') { $query['sort'] = 'ASC'; $query['order'] = 'org_name'; } $rows = parent::organisations($query); } else // contacts view { $query['template'] = $do_email ? 'addressbook.email.rows' : 'addressbook.index.rows'; if ($query['org_view']) // view the contacts of one organisation only { foreach(explode('|||',$query['org_view']) as $part) { list($name,$value) = explode(':',$part,2); $query['col_filter'][$name] = $value; } } // 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 = "org_name!='' DESC,org_name $sort,n_family $sort,n_given $sort"; break; default: $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; } if ($query['searchletter']) // only show contacts if the order-criteria starts with the given letter { $query['col_filter'][] = $query['order'].' LIKE '.$GLOBALS['egw']->db->quote($query['searchletter'].'%'); } $wildcard = '%'; $op = 'OR'; if ($query['advanced_search']) { $op = $query['advanced_search']['operator']; unset($query['advanced_search']['operator']); $wildcard = $query['advanced_search']['meth_select']; unset($query['advanced_search']['meth_select']); } $rows = parent::search($query['advanced_search'] ? $query['advanced_search'] : $query['search'],false, $order,'',$wildcard,false,$op,array((int)$query['start'],(int) $query['num_rows']),$query['col_filter']); // do we need the custom fields if (!$id_only && $this->prefs['custom_colum'] != 'never' && $rows && $this->customfields) { foreach((array) $rows as $n => $val) { if ($val && (int)$val['id']) $ids[] = $val['id']; } if ($ids) $customfields = $this->read_customfields($ids); } } 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']; $readonlys = array(); $photos = $homeaddress = false; foreach($rows as $n => $val) { $row =& $rows[$n]; $given = $row['n_given'] ? $row['n_given'] : ($row['n_prefix'] ? $row['n_prefix'] : ''); switch($order) { 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->org_views[(string) $query['org_view']])) { $row['type'] = 'home'; $row['type_label'] = lang('Organisation'); $readonlys["delete[$row[id]]"] = $query['filter'] && !($this->grants[(int)$query['filter']] & EGW_ACL_DELETE); $readonlys["infolog[$row[id]]"] = !$GLOBALS['egw_info']['user']['apps']['infolog']; } else { $this->type_icon($row['owner'],$row['private'],$row['tid'],$row['type'],$row['type_label']); static $tel2show = array('tel_work','tel_cell','tel_home'); foreach($tel2show as $name) { $this->call_link($row[$name],$row[$name.'_link']); $row[$name] .= ' '.($row['tel_prefer'] == $name ? '♥' : ''); // .' ' to NOT remove the field } // allways show the prefered phone, if not already shown if (!in_array($row['tel_prefer'],$tel2show) && $row[$row['tel_prefer']]) { $this->call_link($row[$row['tel_prefer']],$row['tel_prefered_link']); $row['tel_prefered'] = $row[$row['tel_prefer']].' ♥'; } foreach(array('email','email_home') as $name) { if ($row[$name]) { $row[$name.'_link'] = $this->email2link($row[$name]); if ($GLOBALS['egw_info']['user']['apps']['felamimail']) { $row[$name.'_popup'] = '700x750'; } } else { $row[$name] = ' '; // to NOT remove the field } } $readonlys["delete[$row[id]]"] = !$this->check_perms(EGW_ACL_DELETE,$row); $readonlys["edit[$row[id]]"] = !$this->check_perms(EGW_ACL_EDIT,$row); if ($row['photo']) $photos = true; if (isset($customfields[$row['id']])) { foreach($this->customfields as $name => $data) { $row['customfields'][] = $customfields[$row['id']][$name] ? $customfields[$row['id']][$name] : ' '; } } if ($this->prefs['home_column'] != 'never' && !$homeaddress) { foreach(array('adr_two_countryname','adr_two_locality','adr_two_postalcode','adr_two_street','adr_two_street2') as $name) { if ($row[$name]) $homeaddress = true; } } } } // disable photo column, if view contains no photo(s) if (!$photos || $this->prefs['photo_column'] == 'never') $rows['no_photo'] = '1'; // disable customfields column, if we have no customefield(s) if (!$customfields || $this->prefs['custom_column'] == 'never') $rows['no_customfields'] = '1'; // disable homeaddress column, if we have no homeaddress(es) if ($homeaddress && !$this->prefs['home_column'] || $this->prefs['home_column'] == 'always') $rows['show_home'] = '1'; $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 $GLOBALS['egw_info']['flags']['app_header'] = lang('addressbook'); if ($query['filter'] !== '' && !isset($this->org_views[$query['org_view']])) { $GLOBALS['egw_info']['flags']['app_header'] .= ' '.($query['filter'] == '0' ? lang('accounts') : ($GLOBALS['egw']->accounts->get_type($query['filter']) == 'g' ? lang('Group %1',$GLOBALS['egw']->accounts->id2name($query['filter'])) : $GLOBALS['egw']->common->grab_owner_name((int)$query['filter']). (substr($query['filter'],-1) == 'p' ? ' ('.lang('private').')' : ''))); } if ($query['org_view']) { $GLOBALS['egw_info']['flags']['app_header'] .= ': '.$query['org_view_label']; } if($query['advanced_search']) { $GLOBALS['egw_info']['flags']['app_header'] .= ': '.lang('Advanced search'); } if ($query['cat_id']) { if (!is_object($GLOBALS['egw']->categories)) { $GLOBALS['egw']->categories =& CreateObject('phpgwapi.categories'); } $GLOBALS['egw_info']['flags']['app_header'] .= ': '.lang('Category').' '.$GLOBALS['egw']->categories->id2name($query['cat_id']); } if ($query['searchletter']) { $order = $order == 'org_name' ? lang('company name') : ($order == 'n_given' ? lang('first name') : lang('last name')); $GLOBALS['egw_info']['flags']['app_header'] .= ' - '.lang("%1 starts with '%2'",$order,$query['searchletter']); } if ($query['search']) { $GLOBALS['egw_info']['flags']['app_header'] .= ' - '.lang("Search for '%1'",$query['search']); } return $this->total; } /** * Get addressbook type icon from owner, private and tid * * @param int $owner user- or group-id or 0 for 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 ($row['private']) { $icon = 'private'; $label = lang('private'); } elseif ($tid != 'n') { // ToDo Conny: tid-icons $icon = ''; $label = $tid; } 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') : $GLOBALS['egw']->common->grab_owner_name($owner); } } /** * Get the availible addressbooks of the user * * @param int $required=EGW_ACL_READ required rights on the addressbook * @param string $extra_label first label if given (already translated) * @return array with owner => label pairs */ function get_addressbooks($required=EGW_ACL_READ,$extra_label=null) { //echo "uicontacts::get_addressbooks($required,$include_all) grants="; _debug_array($this->grants); $addressbooks = array(); if ($extra_label) $addressbooks[''] = $extra_label; $addressbooks[$this->user] = lang('Personal'); // add all group addressbooks the user has the necessary rights too foreach($this->grants as $uid => $rights) { if (($rights & $required) && $GLOBALS['egw']->accounts->get_type($uid) == 'g') { $addressbooks[$uid] = lang('Group %1',$GLOBALS['egw']->accounts->id2name($uid)); } } if ($this->grants[0] & $required) { $addressbooks[0] = lang('Accounts'); } // add all other user addressbooks the user has the necessary rights too foreach($this->grants as $uid => $rights) { if ($uid != $this->user && ($rights & $required) && $GLOBALS['egw']->accounts->get_type($uid) == 'u') { $addressbooks[$uid] = $GLOBALS['egw']->common->grab_owner_name($uid); } } if ($this->private_addressbook) { $addressbooks[$this->user.'p'] = lang('Private'); } //_debug_array($addressbooks); return $addressbooks; } /** * Edit a contact * * @param array $content=null submitted content * @param int $_GET['contact_id'] contact_id manly for popup use * @param bool $_GET['makecp'] ture if you want do copy the contact given by $_GET['contact_id'] */ function edit($content=null) { if (!is_object($this->link)) { if (!is_object($GLOBALS['egw']->link)) { $GLOBALS['egw']->link =& CreateObject('phpgwapi.bolink'); } $this->link =& $GLOBALS['egw']->link; } if (is_array($content)) { list($button) = @each($content['button']); unset($content['button']); $content['private'] = (int) ($content['owner'] && substr($content['owner'],-1) == 'p'); $content['owner'] = (string) (int) $content['owner']; switch($button) { case 'save': case 'apply': if ($content['delete_photo']) $content['jpegphoto'] = null; if (is_array($content['upload_photo']) && !empty($content['upload_photo']['tmp_name']) && $content['upload_photo']['tmp_name'] != 'none') { $content['jpegphoto'] = $this->resize_photo($content['upload_photo']); } $links = false; if (!$content['id'] && is_array($content['link_to']['to_id'])) { $links = $content['link_to']['to_id']; } if ($content['id'] && $content['org_name'] && $content['change_org']) { $old_org_entry = $this->read($content['id']); } if ($this->save($content)) { $content['msg'] = lang('Contact saved'); 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); } } } 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']) { $this->link->link('addressbook',$content['id'],$links); } if ($button == 'save') { echo "\n"; /* $link = $GLOBALS['egw']->link('/index.php',array( 'menuaction' => 'addressbook.uicontacts.view', 'contact_id' => $content['id'], )); echo "\n"; */ $GLOBALS['egw']->common->egw_exit(); } $content['link_to']['to_id'] = $content['id']; $GLOBALS['egw_info']['flags']['java_script'] .= ""; break; case 'delete': if($this->delete($content)) { echo "\n"; $GLOBALS['egw']->common->egw_exit(); } else { $content['msg'] = lang('Error deleting the contact !!!'); } break; } // type change } else { $content = array(); $contact_id = $_GET['contact_id'] ? $_GET['contact_id'] : ((int)$_GET['account_id'] ? 'account:'.(int)$_GET['account_id'] : 0); $view = $_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" } else // not found { $state = $GLOBALS['egw']->session->appsession('index','addressbook'); // check if we create the new contact in an existing org if ($_GET['org']) { $content = $this->read_org($_GET['org']); } elseif ($state['org_view'] && !isset($this->org_views[$state['org_view']])) { $content = $this->read_org($state['org_view']); } if (isset($_GET['owner']) && $_GET['owner'] !== '') { $content['owner'] = $_GET['owner']; } else { $content['owner'] = $state['filter']; } $content['private'] = (int) ($content['owner'] && substr($content['owner'],-1) == 'p'); if (!($this->grants[$content['owner'] = (string) (int) $content['owner']] & EGW_ACL_ADD)) { $content['owner'] = $this->prefs['add_default']; $content['private'] = (int) ($content['owner'] && substr($content['owner'],-1) == 'p'); if (!($this->grants[$content['owner'] = (string) (int) $content['owner']] & EGW_ACL_ADD)) { $content['owner'] = (string) $this->user; $content['private'] = 0; } } $new_type = array_keys($this->content_types); $content['tid'] = $_GET['typeid'] ? $_GET['typeid'] : $new_type[0]; foreach($this->get_contact_columns() as $field) { if ($_GET['presets'][$field]) $content[$field] = $_GET['presets'][$field]; } $content['creator'] = $this->user; $content['created'] = $this->now_su; unset($state); } if($content && $_GET['makecp']) // copy the contact { $content['link_to']['to_id'] = 0; $this->link->link('addressbook',$content['link_to']['to_id'],'addressbook',$content['id'], lang('Copied by %1, from record #%2.',$GLOBALS['egw']->common->display_fullname('', $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('id','modified','modifier') as $key) unset($content[$key]); $content['owner'] = $this->prefs['add_default'] ? $this->prefs['add_default'] : $this->user; $content['creator'] = $this->user; $content['created'] = $this->now_su; $content['msg'] = lang('Contact copied'); } else { $content['link_to']['to_id'] = $contact_id; } } $content['disable_change_org'] = $view || !$content['org_name']; //_debug_array($content); $readonlys['button[delete]'] = !$content['owner'] || !$this->check_perms(EGW_ACL_DELETE,$content); $readonlys['button[copy]'] = $readonlys['button[edit]'] = $readonlys['button[vcard]'] = true; $sel_options['fileas_type'] = $this->fileas_options($content); $sel_options['owner'] = $this->get_addressbooks(EGW_ACL_ADD); if ((string) $content['owner'] !== '') { if (!isset($sel_options['owner'][(int)$content['owner']])) { $sel_options['owner'][(int)$content['owner']] = !$content['owner'] ? lang('Accounts') : $GLOBALS['egw']->common->grab_owner_name($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(EGW_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 the own account (by a non-admin), enable only the fields allowed via the "onw_account_acl" if (!$content['owner'] && !$this->is_admin($content)) { foreach($this->get_fields('supported',$content['id'],$content['owner']) as $field) { if (!$this->own_account_acl || !in_array($field,$this->own_account_acl)) $readonlys[$field] = true; } } 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'] = $GLOBALS['egw']->html->image('addressbook',$this->content_types[$content['tid']]['options']['icon'],'',' width="16px" height="16px"'); } else { $content['no_tid'] = true; } $content['link_to'] = array( 'to_app' => 'addressbook', 'to_id' => $content['link_to']['to_id'], ); $content['photo'] = $this->photo_src($content['id'],$content['jpegphoto'],'template'); if ($content['private']) $content['owner'] .= 'p'; $GLOBALS['egw_info']['flags']['include_xajax'] = true; $this->tmpl->read($this->content_types[$content['tid']]['options']['template']); return $this->tmpl->exec('addressbook.uicontacts.edit',$content,$sel_options,$readonlys,$content, 2); } function ajax_setFileasOptions($n_prefix,$n_given,$n_middle,$n_family,$n_suffix,$org_name) { $names = array( 'n_prefix' => $n_prefix, 'n_given' => $n_given, 'n_middle' => $n_middle, 'n_family' => $n_family, 'n_suffix' => $n_suffix, 'org_name' => $org_name, ); $response =& new xajaxResponse(); $response->addScript("setOptions('".addslashes(implode("\b",$this->fileas_options($names)))."');"); return $response->getXML(); } /** * resizes the uploaded photo to 60*80 pixel and returns it * * @param array $file info uploaded file * @return string with resized jpeg photo */ function resize_photo($file) { switch($file['type']) { case 'image/gif': $upload = imagecreatefromgif($file['tmp_name']); break; case 'image/jpeg': case 'image/pjpeg': $upload = imagecreatefromjpeg($file['tmp_name']); break; case 'image/png': case 'image/x-png': $upload = imagecreatefrompng($file['tmp_name']); break; default: return null; } if (!$upload) return null; list($src_w,$src_h) = getimagesize($file['tmp_name']); // scale the image to a width of 60 and a height according to the proportion of the source image $photo = imagecreatetruecolor($dst_w = 60,$dst_h = round($src_h * 60 / $src_w)); imagecopyresized($photo,$upload,0,0,0,0,$dst_w,$dst_h,$src_w,$src_h); //echo "

imagecopyresized(\$photo,\$upload,0,0,0,0,$dst_w,$dst_h,$src_w,$src_h);

\n"; ob_start(); imagejpeg($photo,'',90); $jpeg = ob_get_contents(); ob_end_clean(); imagedestroy($photo); imagedestroy($upload); return $jpeg; } function view($content=null) { if(is_array($content)) { list($button) = each($content['button']); switch ($button) { case 'vcard': $GLOBALS['egw']->redirect_link('/index.php','menuaction=addressbook.uivcard.out&ab_id=' .$content['id']); case 'cancel': $GLOBALS['egw']->redirect_link('/index.php','menuaction=addressbook.uicontacts.index'); case 'delete': $GLOBALS['egw']->redirect_link('/index.php',array( 'menuaction' => 'addressbook.uicontacts.index', 'msg' => $this->delete($content) ? lang('Error deleting the contact !!!') : lang('Contact deleted'), )); } } else { if(!$_GET['contact_id'] || !is_array($content = $this->read($_GET['contact_id']))) { $GLOBALS['egw']->redirect_link('/index.php',array( 'menuaction' => 'addressbook.uicontacts.index', 'msg' => $content, )); } } foreach(array_keys($this->contact_fields) as $key) { $readonlys[$key] = true; if (in_array($key,array('tel_home','tel_work','tel_cell'))) { $readonlys[$key.'2'] = true; $content[$key.'2'] = $content[$key]; } } $content['view'] = true; $content['link_to'] = array( 'to_app' => 'addressbook', 'to_id' => $content['id'], ); $readonlys['link_to'] = $readonlys['customfields'] = $readonlys['fileas_type'] = true; $readonlys['button[save]'] = $readonlys['button[apply]'] = $readonlys['change_photo'] = true; $readonlys['button[delete]'] = !$content['owner'] || !$this->check_perms(EGW_ACL_DELETE,$content); $readonlys['button[edit]'] = !$this->check_perms(EGW_ACL_EDIT,$content); // ToDo: fix vCard export $readonlys['button[vcard]'] = true; $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'] = $GLOBALS['egw']->html->image('addressbook',$this->content_types[$content['tid']]['options']['icon'],'',' width="16px" height="16px"'); } else { $content['no_tid'] = true; } /* Conny: what's that? foreach(explode(',',$content['published_groups']) as $id) { $sel_options['published_groups'][$id] = $GLOBALS['egw']->accounts->id2name($id); } */ $this->tmpl->read($this->content_types[$content['tid']]['options']['template']); foreach(array('email','email_home','url','url_home') as $name) { if ($content[$name] ) { $url = substr($name,0,3) == 'url' ? $content[$name] : $this->email2link($content[$name]); if (!is_array($url)) { $this->tmpl->set_cell_attribute($name,'size','b,,1'); } elseif ($url) { $content[$name.'_link'] = $url; $this->tmpl->set_cell_attribute($name,'size','b,@'.$name.'_link,,,_blank'. ($GLOBALS['egw_info']['user']['apps']['felamimail']?',700x750':'')); } $this->tmpl->set_cell_attribute($name,'type','label'); $this->tmpl->set_cell_attribute($name,'no_lang',true); } } $this->tmpl->exec('addressbook.uicontacts.view',$content,$sel_options,$readonlys,array('id' => $content['id'])); $GLOBALS['egw']->hooks->process(array( 'location' => 'addressbook_view', 'ab_id' => $content['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 (!strstr($email,'@')) return ''; 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; } function search($_content=array()) { if(!empty($_content)) { $response = new xajaxResponse(); $index_nm = $GLOBALS['egw']->session->appsession($do_email ? 'email' : 'index','addressbook'); $index_nm['search'] = $_content; $GLOBALS['egw']->session->appsession($do_email ? 'email' : 'index','addressbook',$index_nm); $response->addScript(" var link = opener.location.href; link = link.replace(/#/,''); opener.location.href=link.replace(/\#/,''); xajax_eT_wrapper(); "); return $response->getXML(); } else { } $GLOBALS['egw_info']['flags']['include_xajax'] = true; $GLOBALS['egw_info']['flags']['java_script'] .= $this->js(); $GLOBALS['egw_info']['flags']['java_script'] .= ""; $GLOBALS['egw_info']['etemplate']['advanced_search'] = true; // initialize etemplate arrays $sel_options = $readonlys = $preserv = array(); $content = $GLOBALS['egw']->session->appsession('advanced_search','addressbook'); 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(EGW_ACL_READ,lang('all')); $sel_options['operator'] = array( 'OR' => 'OR', 'AND' => 'AND' ); $sel_options['meth_select'] = array( '%' => lang('contains'), false => lang('exact'), ); // configure edit template as search dialog $readonlys['change_photo'] = true; $readonlys['fileas_type'] = true; $readonlys['creator'] = true; $readonlys['button'] = true; $readonlys['personal|organisation|home|details|links']['links'] = true; $content['hidebuttons'] = true; $content['no_tid'] = true; $content['disable_change_org'] = true; $this->tmpl->read('addressbook.search'); return $this->tmpl->exec('addressbook.uicontacts.search',$content,$sel_options,$readonlys,$preserv,2); } /** * download photo of the given ($_GET['contact_id'] or $_GET['account_id']) contact */ function photo() { ob_start(); $contact_id = isset($_GET['contact_id']) ? $_GET['contact_id'] : (isset($_GET['account_id']) ? 'account:'.$_GET['account_id'] : 0); if (substr($contact_id,0,8) == 'account:') { $contact_id = $GLOBALS['egw']->accounts->id2name(substr($contact_id,8),'person_id'); } if (!($contact = $this->read($contact_id)) || !$contact['jpegphoto']) { $GLOBALS['egw']->redirect($GLOBALS['egw']->common->image('addressbook','photo')); } if (!ob_get_contents()) { header('Content-type: image/jpeg'); header('Content-length: '.(extension_loaded(mbstring) ? mb_strlen($contact['jpegphoto'],'ascii') : strlen($contact['jpegphoto']))); echo $contact['jpegphoto']; exit; } } /** * returns link to call the given phonenumer * * @param string $number phone number * @param string &$link returns the link * @return boolean true if we have a link, false if not */ function call_link($number,&$link) { if (!$number || !$this->config['call_link']) return false; $link = str_replace('%1',urlencode($number),$this->config['call_link']); } function js() { return ''; } function migrate2ldap() { $GLOBALS['egw_info']['flags']['app_header'] = lang('Addressbook').' - '.lang('Migration to LDAP'); $GLOBALS['egw']->common->egw_header(); parse_navbar(); if (!$this->is_admin()) { echo '

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

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

'.lang('Migration finished')."

\n"; } $GLOBALS['egw']->common->egw_footer(); } } if (!function_exists('array_intersect_key')) // php5.1 function { function array_intersect_key($array1,$array2) { $intersection = $keys = array(); foreach(func_get_args() as $arr) { $keys[] = array_keys((array)$arr); } foreach(call_user_func_array('array_intersect',$keys) as $key) { $intersection[$key] = $array1[$key]; } return $intersection; } }