diff options
-rw-r--r-- | CHANGELOG | 4 | ||||
-rw-r--r-- | config/main.inc.php.dist | 5 | ||||
-rw-r--r-- | program/js/app.js | 163 | ||||
-rw-r--r-- | program/steps/mail/compose.inc | 50 | ||||
-rw-r--r-- | skins/default/common.css | 8 |
5 files changed, 99 insertions, 131 deletions
@@ -1,6 +1,10 @@ CHANGELOG RoundCube Webmail --------------------------- +2008/11/23 (thomasb) +---------- +- Do serverside addressbook queries for autocompletion (#1485531) + 2008/11/23 (alec) ---------- - Allow setting attachment col position in 'list_cols' option diff --git a/config/main.inc.php.dist b/config/main.inc.php.dist index c4a62b1f8..d9ef3d343 100644 --- a/config/main.inc.php.dist +++ b/config/main.inc.php.dist @@ -279,10 +279,13 @@ $rcmail_config['address_book_type'] = 'sql'; * 'sort' => 'cn', // The field to sort the listing by. * 'scope' => 'sub', // search mode: sub|base|list * 'filter' => '', // used for basic listing (if not empty) and will be &'d with search queries. example: status=act - * 'global_search' => true, // perform a global search for address auto-completion on compose * 'fuzzy_search' => true); // server allows wildcard search */ +// An ordered array of the ids of the addressbooks that should be searched +// when populating address autocomplete fields server-side. ex: array('sql','Verisign'); +$rcmail_config['autocomplete_addressbooks'] = array('sql'); + // don't allow these settings to be overriden by the user $rcmail_config['dont_override'] = array(); diff --git a/program/js/app.js b/program/js/app.js index 06978f60b..82687da3a 100644 --- a/program/js/app.js +++ b/program/js/app.js @@ -458,20 +458,13 @@ function rcube_webmail() this.init_address_input_events = function(obj) { var handler = function(e){ return ref.ksearch_keypress(e,this); }; - var handler2 = function(e){ return ref.ksearch_blur(e,this); }; if (obj.addEventListener) - { obj.addEventListener(bw.safari ? 'keydown' : 'keypress', handler, false); - obj.addEventListener('blur', handler2, false); - } else - { obj.onkeydown = handler; - obj.onblur = handler2; - } - obj.setAttribute('autocomplete', 'off'); + obj.setAttribute('autocomplete', 'off'); }; @@ -1173,6 +1166,9 @@ function rcube_webmail() this.contact_list.blur(); model = this.env.address_sources; } + else if (this.ksearch_value) { + this.ksearch_blur(); + } // handle mouse release when dragging if (this.drag_active && model) { @@ -2313,15 +2309,15 @@ function rcube_webmail() } if (this.gui_objects.search_filter) - addurl = '&_filter=' + this.gui_objects.search_filter.value; + addurl = '&_filter=' + this.gui_objects.search_filter.value; // reset vars this.env.current_page = 1; this.set_busy(true, 'searching'); this.http_request('search', '_q='+urlencode(value) - +(this.env.mailbox ? '&_mbox='+urlencode(this.env.mailbox) : '') - +(this.env.source ? '&_source='+urlencode(this.env.source) : '') - +(addurl ? addurl : ''), true); + + (this.env.mailbox ? '&_mbox='+urlencode(this.env.mailbox) : '') + + (this.env.source ? '&_source='+urlencode(this.env.source) : '') + + (addurl ? addurl : ''), true); } return true; }; @@ -2349,10 +2345,7 @@ function rcube_webmail() // handler for keyboard events on address-fields this.ksearch_keypress = function(e, obj) - { - if (typeof(this.env.contacts)!='object' || !this.env.contacts.length) - return true; - + { if (this.ksearch_timer) clearTimeout(this.ksearch_timer); @@ -2368,24 +2361,13 @@ function rcube_webmail() break; var dir = key==38 ? 1 : 0; - var next; highlight = document.getElementById('rcmksearchSelected'); if (!highlight) highlight = this.ksearch_pane.ul.firstChild; - if (highlight && (next = dir ? highlight.previousSibling : highlight.nextSibling)) - { - highlight.removeAttribute('id'); - this.set_classname(highlight, 'selected', false); - } - - if (next) - { - next.setAttribute('id', 'rcmksearchSelected'); - this.set_classname(next, 'selected', true); - this.ksearch_selected = next._rcm_id; - } + if (highlight) + this.ksearch_select(dir ? highlight.previousSibling : highlight.nextSibling); return rcube_event.cancel(e); @@ -2393,7 +2375,7 @@ function rcube_webmail() if(mod == SHIFT_KEY) break; - case 13: // enter + case 13: // enter if (this.ksearch_selected===null || !this.ksearch_input || !this.ksearch_value) break; @@ -2414,7 +2396,22 @@ function rcube_webmail() this.ksearch_input = obj; return true; - }; + }; + + this.ksearch_select = function(node) + { + var current = document.getElementById('rcmksearchSelected'); + if (current && node) { + current.removeAttribute('id'); + this.set_classname(current, 'selected', false); + } + + if (node) { + node.setAttribute('id', 'rcmksearchSelected'); + this.set_classname(node, 'selected', true); + this.ksearch_selected = node._rcm_id; + } + }; this.insert_recipient = function(id) { @@ -2440,10 +2437,13 @@ function rcube_webmail() // address search processor this.ksearch_get_results = function() - { + { var inp_value = this.ksearch_input ? this.ksearch_input.value : null; - if (inp_value===null) + if (inp_value === null) return; + + if (this.ksearch_pane && this.ksearch_pane.visible) + this.ksearch_pane.show(0); // get string from current cursor pos to last comma var cpos = this.get_caret_pos(this.ksearch_input); @@ -2453,45 +2453,45 @@ function rcube_webmail() // trim query string q = q.replace(/(^\s+|\s+$)/g, '').toLowerCase(); - if (!q.length || q==this.ksearch_value) - { - if (!q.length && this.ksearch_pane && this.ksearch_pane.visible) - this.ksearch_pane.show(0); - - return; - } + // Don't (re-)search if string is empty or if the last results are still active + if (!q.length || q == this.ksearch_value) + return; this.ksearch_value = q; - // start searching the contact list - var a_results = new Array(); - var a_result_ids = new Array(); + this.display_message('searching', 'loading', true); + this.http_post('autocomplete', '_search='+q); + }; + + this.ksearch_query_results = function(results) + { + this.hide_message(); + this.env.contacts = results ? results : []; + + var result_ids = new Array(); var c=0; - for (var i=0; i<this.env.contacts.length; i++) - { - if (this.env.contacts[i].toLowerCase().indexOf(q)>=0) - { - a_results[c] = this.env.contacts[i]; - a_result_ids[c++] = i; - - if (c==15) // limit search results - break; - } - } + for (var i=0; i < this.env.contacts.length; i++) { + result_ids[c++] = i; + if (c == 15) // limit search results + break; + } + + this.ksearch_display_results(this.env.contacts, result_ids, c); + }; + this.ksearch_display_results = function (a_results, a_result_ids, c) + { // display search results - if (c && a_results.length) - { + if (c && a_results.length) { var p, ul, li; // create results pane if not present - if (!this.ksearch_pane) - { + if (!this.ksearch_pane) { ul = document.createElement('UL'); this.ksearch_pane = new rcube_layer('rcmKSearchpane', {vis:0, zindex:30000}); this.ksearch_pane.elm.appendChild(ul); this.ksearch_pane.ul = ul; - } + } else ul = this.ksearch_pane.ul; @@ -2499,51 +2499,58 @@ function rcube_webmail() ul.innerHTML = ''; // add each result line to list - for (i=0; i<a_results.length; i++) - { + for (i=0; i<a_results.length; i++) { li = document.createElement('LI'); - li.innerHTML = a_results[i].replace(/</, '<').replace(/>/, '>'); + li.innerHTML = a_results[i].replace(/</, '<').replace(/>/, '>').replace(new RegExp('('+this.ksearch_value+')', 'ig'), '<b>$1</b>'); + li.onmouseover = function(){ ref.ksearch_select(this); }; + li.onclick = function(){ ref.ksearch_click(this) }; li._rcm_id = a_result_ids[i]; ul.appendChild(li); - } + } // check if last selected item is still in result list - if (this.ksearch_selected!==null) - { + if (this.ksearch_selected !== null) { p = find_in_array(this.ksearch_selected, a_result_ids); - if (p>=0 && ul.childNodes) - { + if (p >= 0 && ul.childNodes) { ul.childNodes[p].setAttribute('id', 'rcmksearchSelected'); this.set_classname(ul.childNodes[p], 'selected', true); - } + } else this.ksearch_selected = null; - } + } // if no item selected, select the first one - if (this.ksearch_selected===null) - { + if (this.ksearch_selected === null) { ul.firstChild.setAttribute('id', 'rcmksearchSelected'); this.set_classname(ul.firstChild, 'selected', true); this.ksearch_selected = a_result_ids[0]; - } + } // move the results pane right under the input box and make it visible var pos = rcube_get_object_pos(this.ksearch_input); this.ksearch_pane.move(pos.x, pos.y+this.ksearch_input.offsetHeight); - this.ksearch_pane.show(1); - } + this.ksearch_pane.show(1); + } // hide results pane else this.ksearch_hide(); - }; + }; + + this.ksearch_click = function(node) + { + this.insert_recipient(node._rcm_id); + this.ksearch_hide(); + + if (ref.ksearch_input) + this.ksearch_input.focus(); + }; - this.ksearch_blur = function(e, obj) + this.ksearch_blur = function() { if (this.ksearch_timer) clearTimeout(this.ksearch_timer); - this.ksearch_value = ''; + this.ksearch_value = ''; this.ksearch_input = null; this.ksearch_hide(); @@ -2555,7 +2562,7 @@ function rcube_webmail() this.ksearch_selected = null; if (this.ksearch_pane) - this.ksearch_pane.show(0); + this.ksearch_pane.show(0); }; diff --git a/program/steps/mail/compose.inc b/program/steps/mail/compose.inc index d4348f287..8a9c36a04 100644 --- a/program/steps/mail/compose.inc +++ b/program/steps/mail/compose.inc @@ -83,7 +83,7 @@ if (!is_array($_SESSION['compose']) || $_SESSION['compose']['id'] != get_input_v // add some labels to client $OUTPUT->add_label('nosubject', 'nosenderwarning', 'norecipientwarning', 'nosubjectwarning', 'nobodywarning', 'notsentwarning', 'savingmessage', 'sendingmessage', 'messagesaved', - 'converting', 'editorwarning'); + 'converting', 'editorwarning', 'searching'); // add config parameters to client script if (!empty($CONFIG['drafts_mbox'])) { @@ -913,54 +913,6 @@ $OUTPUT->add_handlers(array( 'storetarget' => 'rcmail_store_target_selection', )); -/****** get contacts for this user and add them to client scripts ********/ - -$CONTACTS = new rcube_contacts($DB, $USER->ID); -$CONTACTS->set_pagesize(1000); - -$a_contacts = array(); - -if ($result = $CONTACTS->list_records()) - { - while ($sql_arr = $result->iterate()) - if ($sql_arr['email']) - $a_contacts[] = format_email_recipient($sql_arr['email'], $sql_arr['name']); - } -if (!empty($CONFIG['ldap_public']) && is_array($CONFIG['ldap_public'])) - { - /* LDAP autocompletion */ - foreach ($CONFIG['ldap_public'] as $ldapserv_config) - { - if ($ldapserv_config['fuzzy_search'] != 1 || - $ldapserv_config['global_search'] != 1) - { - continue; - } - - $LDAP = new rcube_ldap($ldapserv_config); - $LDAP->connect(); - $LDAP->set_pagesize(1000); - - $results = $LDAP->search($ldapserv_config['mail_field'], ""); - - for ($i = 0; $i < $results->count; $i++) - { - if ($results->records[$i]['email'] != '') - { - $email = $results->records[$i]['email']; - $name = $results->records[$i]['name']; - - $a_contacts[] = format_email_recipient($email, $name); - } - } - $LDAP->close(); - } - } -if ($a_contacts) - { - $OUTPUT->set_env('contacts', $a_contacts); - } - $OUTPUT->send('compose'); ?> diff --git a/skins/default/common.css b/skins/default/common.css index 1225862f1..2ba57f55f 100644 --- a/skins/default/common.css +++ b/skins/default/common.css @@ -431,12 +431,14 @@ a.rcmContactAddress:hover #rcmKSearchpane ul li { + display: block; height: 16px; font-size: 11px; - padding-left: 8px; + padding-left: 6px; padding-top: 2px; - padding-right: 8px; - white-space: nowrap; + padding-right: 6px; + white-space: nowrap; + cursor: pointer; } #rcmKSearchpane ul li.selected |