mirror of
https://github.com/roundcube/roundcubemail.git
synced 2026-03-03 23:04:01 +01:00
Before the search worked only in Addressbook, not also in Compose. The point of the change is also to align group searches with contact searches in that it now uses the same set of attributes. Previously groups in Compose were searched by name only.
195 lines
7.9 KiB
PHP
195 lines
7.9 KiB
PHP
<?php
|
|
|
|
/**
|
|
+-----------------------------------------------------------------------+
|
|
| program/steps/mail/autocomplete.inc |
|
|
| |
|
|
| This file is part of the Roundcube Webmail client |
|
|
| Copyright (C) 2008-2013, Roundcube Dev Team |
|
|
| Copyright (C) 2011-2013, Kolab Systems AG |
|
|
| |
|
|
| Licensed under the GNU General Public License version 3 or |
|
|
| any later version with exceptions for skins & plugins. |
|
|
| See the README file for a full license statement. |
|
|
| |
|
|
| PURPOSE: |
|
|
| Perform a search on configured address books for the address |
|
|
| autocompletion of the message compose screen |
|
|
+-----------------------------------------------------------------------+
|
|
| Author: Thomas Bruederli <roundcube@gmail.com> |
|
|
+-----------------------------------------------------------------------+
|
|
*/
|
|
|
|
if ($RCMAIL->action == 'group-expand') {
|
|
$abook = $RCMAIL->get_address_book(rcube_utils::get_input_value('_source', rcube_utils::INPUT_GPC));
|
|
if ($gid = rcube_utils::get_input_value('_gid', rcube_utils::INPUT_GPC)) {
|
|
$abook->set_group($gid);
|
|
$abook->set_pagesize(1000); // TODO: limit number of group members by config
|
|
|
|
$separator = trim($RCMAIL->config->get('recipients_separator', ',')) . ' ';
|
|
$result = $abook->list_records($RCMAIL->config->get('contactlist_fields'));
|
|
$members = array();
|
|
|
|
while ($result && ($sql_arr = $result->iterate())) {
|
|
$emails = (array) $abook->get_col_values('email', $sql_arr, true);
|
|
if (!empty($emails) && ($email = array_shift($emails))) {
|
|
$members[] = format_email_recipient($email, rcube_addressbook::compose_list_name($sql_arr));
|
|
}
|
|
}
|
|
|
|
$OUTPUT->command('replace_group_recipients', $gid, join($separator, array_unique($members)));
|
|
}
|
|
|
|
$OUTPUT->send();
|
|
}
|
|
|
|
|
|
$MAXNUM = (int) $RCMAIL->config->get('autocomplete_max', 15);
|
|
$mode = (int) $RCMAIL->config->get('addressbook_search_mode');
|
|
$single = (bool) $RCMAIL->config->get('autocomplete_single');
|
|
$search = rcube_utils::get_input_value('_search', rcube_utils::INPUT_GPC, true);
|
|
$source = rcube_utils::get_input_value('_source', rcube_utils::INPUT_GPC);
|
|
$reqid = rcube_utils::get_input_value('_reqid', rcube_utils::INPUT_GPC);
|
|
|
|
if (strlen($source)) {
|
|
$book_types = array($source);
|
|
}
|
|
else {
|
|
$book_types = (array) $RCMAIL->config->get('autocomplete_addressbooks', 'sql');
|
|
}
|
|
|
|
$contacts = array();
|
|
if (!empty($book_types) && strlen($search)) {
|
|
$sort_keys = array();
|
|
$books_num = count($book_types);
|
|
$search_lc = mb_strtolower($search);
|
|
$mode |= rcube_addressbook::SEARCH_GROUPS;
|
|
|
|
foreach ($book_types as $abook_id) {
|
|
$abook = $RCMAIL->get_address_book($abook_id);
|
|
$abook->set_pagesize($MAXNUM);
|
|
|
|
if ($result = $abook->search($RCMAIL->config->get('contactlist_fields'), $search, $mode, true, true, 'email')) {
|
|
while ($sql_arr = $result->iterate()) {
|
|
// Contact can have more than one e-mail address
|
|
$email_arr = (array)$abook->get_col_values('email', $sql_arr, true);
|
|
$email_cnt = count($email_arr);
|
|
$idx = 0;
|
|
|
|
foreach ($email_arr as $email) {
|
|
if (empty($email)) {
|
|
continue;
|
|
}
|
|
|
|
$name = rcube_addressbook::compose_list_name($sql_arr);
|
|
$contact = format_email_recipient($email, $name);
|
|
|
|
// skip entries that don't match
|
|
if ($email_cnt > 1 && strpos(mb_strtolower($contact), $search_lc) === false) {
|
|
continue;
|
|
}
|
|
|
|
$index = $contact;
|
|
|
|
// skip duplicates
|
|
if (empty($contacts[$index])) {
|
|
$contact = array(
|
|
'name' => $contact,
|
|
'type' => $sql_arr['_type'],
|
|
'id' => $sql_arr['contact_id'],
|
|
'source' => $abook_id,
|
|
);
|
|
|
|
if (($display = rcube_addressbook::compose_search_name($sql_arr, $email, $name)) && $display != $contact['name']) {
|
|
$contact['display'] = $display;
|
|
}
|
|
|
|
$contacts[$index] = $contact;
|
|
$sort_keys[$index] = sprintf('%s %03d', $contact['display'] ?: $name, $idx++);
|
|
|
|
if (count($contacts) >= $MAXNUM) {
|
|
break 2;
|
|
}
|
|
}
|
|
|
|
// skip redundant entries (show only first email address)
|
|
if ($single) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// also list matching contact groups
|
|
if ($abook->groups && count($contacts) < $MAXNUM) {
|
|
foreach ($abook->list_groups($search, $mode) as $group) {
|
|
$abook->reset();
|
|
$abook->set_group($group['ID']);
|
|
|
|
$group_prop = $abook->get_group($group['ID']);
|
|
|
|
// group (distribution list) with email address(es)
|
|
if ($group_prop['email']) {
|
|
$idx = 0;
|
|
foreach ((array)$group_prop['email'] as $email) {
|
|
$index = format_email_recipient($email, $group['name']);
|
|
|
|
if (empty($contacts[$index])) {
|
|
$sort_keys[$index] = sprintf('%s %03d', $group['name'] , $idx++);
|
|
$contacts[$index] = array(
|
|
'name' => $index,
|
|
'email' => $email,
|
|
'type' => 'group',
|
|
'id' => $group['ID'],
|
|
'source' => $abook_id,
|
|
);
|
|
|
|
if (count($contacts) >= $MAXNUM) {
|
|
break 2;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// show group with count
|
|
else if (($result = $abook->count()) && $result->count) {
|
|
if (empty($contacts[$group['name']])) {
|
|
$sort_keys[$group['name']] = $group['name'];
|
|
$contacts[$group['name']] = array(
|
|
'name' => $group['name'] . ' (' . intval($result->count) . ')',
|
|
'type' => 'group',
|
|
'id' => $group['ID'],
|
|
'source' => $abook_id,
|
|
);
|
|
|
|
if (count($contacts) >= $MAXNUM) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (count($contacts)) {
|
|
// sort contacts index
|
|
asort($sort_keys, SORT_LOCALE_STRING);
|
|
// re-sort contacts according to index
|
|
foreach ($sort_keys as $idx => $val) {
|
|
$sort_keys[$idx] = $contacts[$idx];
|
|
}
|
|
$contacts = array_values($sort_keys);
|
|
}
|
|
}
|
|
|
|
|
|
// Allow autocomplete result optimization via plugin
|
|
$pluginResult = $RCMAIL->plugins->exec_hook('contacts_autocomplete_after', array(
|
|
'search' => $search,
|
|
'contacts' => $contacts, // Provide already-found contacts to plugin if they are required
|
|
));
|
|
$contacts = $pluginResult['contacts'];
|
|
|
|
|
|
$OUTPUT->command('ksearch_query_results', $contacts, $search, $reqid);
|
|
$OUTPUT->send();
|