diff --git a/plugins/kolab_addressbook/kolab_addressbook.js b/plugins/kolab_addressbook/kolab_addressbook.js index 82b6d9dd..c519a673 100644 --- a/plugins/kolab_addressbook/kolab_addressbook.js +++ b/plugins/kolab_addressbook/kolab_addressbook.js @@ -87,6 +87,91 @@ if (window.rcmail) { } }); } + + // append search form for address books + if (rcmail.gui_objects.folderlist) { + var container = $(rcmail.gui_objects.folderlist); + $('') + .insertBefore(container.parent()); + + $('' + + rcmail.gettext('findaddressbooks', 'kolab_addressbook') + '') + .appendTo('#directorylistbox h2.boxtitle') + .click(function(e){ + var box = $('#directorylistbox .listsearchbox'), + dir = box.is(':visible') ? -1 : 1; + + box.slideToggle({ + duration: 160, + progress: function(animation, progress) { + if (dir < 0) progress = 1 - progress; + $('#directorylistbox .scroller').css('top', (34 + 34 * progress) + 'px'); + }, + complete: function() { + box.toggleClass('expanded'); + if (box.is(':visible')) { + box.find('input[type=text]').focus(); + } + else { + $('#directorylistsearch-reset').click(); + } + } + }); + }); + + + // remove event handlers set by the regular treelist widget + rcmail.treelist.container.off('click mousedown focusin focusout'); + + // re-initialize folderlist widget + // copy form app.js with additional parameters + var widget_class = window.kolab_folderlist || rcube_treelist_widget; + rcmail.treelist = new widget_class(rcmail.gui_objects.folderlist, { + selectable: true, + id_prefix: 'rcmli', + id_encode: rcmail.html_identifier_encode, + id_decode: rcmail.html_identifier_decode, + searchbox: '#addressbooksearch', + search_action: 'plugin.book-search', + search_sources: [ 'folders', 'users' ], + search_title: rcmail.gettext('listsearchresults','kolab_addressbook'), + check_droptarget: function(node) { return !node.virtual && rcmail.check_droptarget(node.id) } + }); + + rcmail.treelist + .addEventListener('collapse', function(node) { rcmail.folder_collapsed(node) }) + .addEventListener('expand', function(node) { rcmail.folder_collapsed(node) }) + .addEventListener('select', function(node) { rcmail.triggerEvent('selectfolder', { folder:node.id, prefix:'rcmli' }) }) + .addEventListener('subscribe', function(node) { + var source; + if ((source = rcmail.env.address_sources[node.id])) { + source.subscribed = node.subscribed || false; + rcmail.http_post('plugin.book-subscribe', { _source:node.id, _permanent:source.subscribed?1:0 }); + } + }) + .addEventListener('insert-item', function(data) { + // register new address source + rcmail.env.address_sources[data.id] = rcmail.env.contactfolders[data.id] = data.data; + if (data.data.subscribed) + rcmail.http_post('plugin.book-subscribe', { _source:data.id, _permanent:1 }); + // TODO: load groups and add them to the list + }) + .addEventListener('search-complete', function(data) { + if (data.length) + rcmail.display_message(rcmail.gettext('nraddressbooksfound','kolab_addressbook').replace('$nr', data.length), 'voice'); + else + rcmail.display_message(rcmail.gettext('noaddressbooksfound','kolab_addressbook'), 'info'); + }); + } }); rcmail.addEventListener('listupdate', function() { rcmail.set_book_actions(); diff --git a/plugins/kolab_addressbook/kolab_addressbook.php b/plugins/kolab_addressbook/kolab_addressbook.php index 530580bb..7bada56b 100644 --- a/plugins/kolab_addressbook/kolab_addressbook.php +++ b/plugins/kolab_addressbook/kolab_addressbook.php @@ -66,6 +66,8 @@ class kolab_addressbook extends rcube_plugin // Plugin actions $this->register_action('plugin.book', array($this, 'book_actions')); $this->register_action('plugin.book-save', array($this, 'book_save')); + $this->register_action('plugin.book-search', array($this, 'book_search')); + $this->register_action('plugin.book-subscribe', array($this, 'book_subscribe')); // Load UI elements if ($this->api->output->type == 'html') { @@ -133,19 +135,34 @@ class kolab_addressbook extends rcube_plugin */ protected function abook_prop($id, $abook) { - return array( - 'id' => $id, - 'name' => $abook->get_name(), - 'listname' => $abook->get_foldername(), - 'readonly' => $abook->readonly, - 'editable' => $abook->editable, - 'groups' => $abook->groups, - 'undelete' => $abook->undelete && $this->rc->config->get('undo_timeout'), - 'realname' => rcube_charset::convert($abook->get_realname(), 'UTF7-IMAP'), // IMAP folder name - 'group' => $abook->get_namespace(), - 'carddavurl' => $abook->get_carddav_url(), - 'kolab' => true, - ); + if ($abook->virtual) { + return array( + 'id' => $id, + 'name' => $abook->get_name(), + 'listname' => $abook->get_foldername(), + 'group' => $abook instanceof kolab_storage_folder_user ? 'user' : $abook->get_namespace(), + 'readonly' => true, + 'editable' => false, + 'kolab' => true, + 'virtual' => true, + ); + } + else { + return array( + 'id' => $id, + 'name' => $abook->get_name(), + 'listname' => $abook->get_foldername(), + 'readonly' => $abook->readonly, + 'editable' => $abook->editable, + 'groups' => $abook->groups, + 'undelete' => $abook->undelete && $this->rc->config->get('undo_timeout'), + 'realname' => rcube_charset::convert($abook->get_realname(), 'UTF7-IMAP'), // IMAP folder name + 'group' => $abook->get_namespace(), + 'subscribed' => $abook->is_subscribed(), + 'carddavurl' => $abook->get_carddav_url(), + 'kolab' => true, + ); + } } /** @@ -167,7 +184,8 @@ class kolab_addressbook extends rcube_plugin kolab_storage::folder_hierarchy($this->folders, $tree); $out .= $this->folder_tree_html($tree, $sources, $jsdata); - $this->rc->output->set_env('contactgroups', $jsdata); + $this->rc->output->set_env('contactgroups', array_filter($jsdata, function($src){ return $src['type'] == 'group'; })); + $this->rc->output->set_env('address_sources', array_filter($jsdata, function($src){ return $src['type'] != 'group'; })); $args['content'] = html::tag('ul', $args, $out, html::$common_attrib); return $args; @@ -185,16 +203,11 @@ class kolab_addressbook extends rcube_plugin $is_collapsed = strpos($this->rc->config->get('collapsed_abooks',''), '&'.rawurlencode($id).'&') !== false; if ($folder->virtual) { - $source = array( - 'id' => $folder->id, - 'name' => $folder->get_name(), - 'listname' => $folder->get_foldername(), - 'group' => $folder->get_namespace(), - 'readonly' => true, - 'editable' => false, - 'kolab' => true, - 'virtual' => true, - ); + $source = $this->abook_prop($folder->id, $folder); + } + else if (empty($source)) { + $this->sources[$id] = new rcube_kolab_contacts($folder->name); + $source = $this->abook_prop($id, $this->sources[$id]); } $content = $this->addressbook_list_item($id, $source, $jsdata); @@ -219,11 +232,15 @@ class kolab_addressbook extends rcube_plugin /** * */ - protected function addressbook_list_item($id, $source, &$jsdata, $checkbox = false) + protected function addressbook_list_item($id, $source, &$jsdata, $search_mode = false) { $folder = $this->folders[$id]; $current = rcube_utils::get_input_value('_source', rcube_utils::INPUT_GPC); + if (!$source['virtual']) { + $jsdata[$id] = $source; + } + // set class name(s) $classes = array($source['group'] ?: '000', 'addressbook'); if ($current === $id) @@ -236,19 +253,48 @@ class kolab_addressbook extends rcube_plugin $classes[] = $source['class_name']; $name = !empty($source['listname']) ? $source['listname'] : (!empty($source['name']) ? $source['name'] : $id); + $label_id = 'kabt:' . $id; + $inner = ($source['virtual'] ? + html::a(array('tabindex' => '0'), $name) : + html::a(array( + 'href' => $this->rc->url(array('_source' => $id)), + 'rel' => $source['id'], + 'id' => $label_id, + 'onclick' => "return " . rcmail_output::JS_OBJECT_NAME.".command('list','" . rcube::JQ($id) . "',this)", + ), $name) + ); + + if (isset($source['subscribed'])) { + $inner .= html::span(array( + 'class' => 'subscribed', + 'title' => $this->gettext('foldersubscribe'), + 'role' => 'checkbox', + 'aria-checked' => $source['subscribed'] ? 'true' : 'false', + ), ''); + } + + // don't wrap in
  • but add a checkbox for search results listing + if ($search_mode) { + $jsdata[$id]['group'] = join(' ', $classes); + + if (!$source['virtual']) { + $inner .= html::tag('input', array( + 'type' => 'checkbox', + 'name' => '_source[]', + 'value' => $id, + 'checked' => $prop['active'], + 'aria-labelledby' => $label_id, + )); + } + return html::div(null, $inner); + } + $out .= html::tag('li', array( 'id' => 'rcmli' . rcube_utils::html_identifier($id, true), 'class' => join(' ', $classes), 'noclose' => true, ), - ($source['virtual'] ? - html::a(array('tabindex' => '0'), $name) : - html::a(array( - 'href' => $this->rc->url(array('_source' => $id)), - 'rel' => $source['id'], - 'onclick' => "return " . rcmail_output::JS_OBJECT_NAME.".command('list','" . rcube::JQ($id) . "',this)", - ), $name) - ) + html::div($source['subscribed'] ? 'subscribed' : null, $inner) ); $groupdata = array('out' => '', 'jsdata' => $jsdata, 'source' => $id); @@ -596,6 +642,119 @@ class kolab_addressbook extends rcube_plugin $this->ui->book_edit(); } + /** + * + */ + public function book_search() + { + $results = array(); + $query = rcube_utils::get_input_value('q', RCUBE_INPUT_GPC); + $source = rcube_utils::get_input_value('source', RCUBE_INPUT_GPC); + + kolab_storage::$encode_ids = true; + $search_more_results = false; + $this->sources = array(); + $this->folders = array(); + + // find unsubscribed IMAP folders that have "event" type + if ($source == 'folders') { + foreach ((array)kolab_storage::search_folders('contact', $query, array('other')) as $folder) { + $this->folders[$folder->id] = $folder; + $this->sources[$folder->id] = new rcube_kolab_contacts($folder->name); + } + } + // search other user's namespace via LDAP + else if ($source == 'users') { + $limit = $this->rc->config->get('autocomplete_max', 15) * 2; // we have slightly more space, so display twice the number + foreach (kolab_storage::search_users($query, 0, array(), $limit * 10) as $user) { + $folders = array(); + // search for contact folders shared by this user + foreach (kolab_storage::list_user_folders($user, 'contact', false) as $foldername) { + $folders[] = new kolab_storage_folder($foldername, 'contact'); + } + + if (count($folders)) { + $userfolder = new kolab_storage_folder_user($user['kolabtargetfolder'], '', $user); + $this->folders[$userfolder->id] = $userfolder; + $this->sources[$userfolder->id] = $userfolder; + + foreach ($folders as $folder) { + $this->folders[$folder->id] = $folder; + $this->sources[$folder->id] = new rcube_kolab_contacts($folder->name);; + $count++; + } + } + + if ($count >= $limit) { + $search_more_results = true; + break; + } + } + } + + $delim = $this->rc->get_storage()->get_hierarchy_delimiter(); + + // build results list + foreach ($this->sources as $id => $source) { + $folder = $this->folders[$id]; + $imap_path = explode($delim, $folder->name); + + // find parent + do { + array_pop($imap_path); + $parent_id = kolab_storage::folder_id(join($delim, $imap_path)); + } + while (count($imap_path) > 1 && !$this->folders[$parent_id]); + + // restore "real" parent ID + if ($parent_id && !$this->folders[$parent_id]) { + $parent_id = kolab_storage::folder_id($folder->get_parent()); + } + + $prop = $this->abook_prop($id, $source); + $prop['parent'] = $parent_id; + unset($prop['group']); + + $html = $this->addressbook_list_item($id, $prop, $jsdata, true); + $prop += (array)$jsdata[$id]; + $prop['html'] = $html; + + $results[] = $prop; + } + + // report more results available + if ($search_more_results) { + $this->rc->output->show_message('autocompletemore', 'info'); + } + + $this->rc->output->command('multi_thread_http_response', $results, rcube_utils::get_input_value('_reqid', RCUBE_INPUT_GPC)); + } + + /** + * + */ + public function book_subscribe() + { + $success = false; + $id = rcube_utils::get_input_value('_source', RCUBE_INPUT_GPC); + + if ($id && ($folder = kolab_storage::get_folder(kolab_storage::id_decode($id)))) { + if (isset($_POST['_permanent'])) + $success |= $folder->subscribe(intval($_POST['_permanent'])); + if (isset($_POST['_active'])) + $success |= $folder->activate(intval($_POST['_active'])); + } + + if ($success) { + $this->rc->output->show_message('successfullysaved', 'confirmation'); + } + else { + $this->rc->output->show_message($this->gettext('errorsaving'), 'error'); + } + + $this->rc->output->send(); + } + /** * Handler for address book delete action (AJAX) diff --git a/plugins/kolab_addressbook/lib/kolab_addressbook_ui.php b/plugins/kolab_addressbook/lib/kolab_addressbook_ui.php index c7c1a7f5..898dd8cd 100644 --- a/plugins/kolab_addressbook/lib/kolab_addressbook_ui.php +++ b/plugins/kolab_addressbook/lib/kolab_addressbook_ui.php @@ -54,6 +54,11 @@ class kolab_addressbook_ui // Include stylesheet (for directorylist) $this->plugin->include_stylesheet($this->plugin->local_skin_path().'/kolab_addressbook.css'); + // include kolab folderlist widget if available + if (is_readable($this->plugin->api->dir . 'libkolab/js/folderlist.js')) { + $this->plugin->api->include_script('libkolab/js/folderlist.js'); + } + // Add actions on address books $options = array('book-create', 'book-edit', 'book-delete'); $idx = 0; @@ -91,7 +96,14 @@ class kolab_addressbook_ui 'kolab_addressbook.carddavurldescription', 'kolab_addressbook.bookedit', 'kolab_addressbook.bookdelete', - 'kolab_addressbook.bookshowurl'); + 'kolab_addressbook.bookshowurl', + 'kolab_addressbook.findaddressbooks', + 'kolab_addressbook.searchterms', + 'kolab_addressbook.foldersearchform', + 'kolab_addressbook.listsearchresults', + 'kolab_addressbook.nraddressbooksfound', + 'kolab_addressbook.noaddressbooksfound', + 'resetsearch'); } // book create/edit form else { diff --git a/plugins/kolab_addressbook/lib/rcube_kolab_contacts.php b/plugins/kolab_addressbook/lib/rcube_kolab_contacts.php index 2e93d462..8bd18865 100644 --- a/plugins/kolab_addressbook/lib/rcube_kolab_contacts.php +++ b/plugins/kolab_addressbook/lib/rcube_kolab_contacts.php @@ -198,6 +198,16 @@ class rcube_kolab_contacts extends rcube_addressbook return $this->storagefolder->get_parent(); } + /** + * Check subscription status of this folder + * + * @return boolean True if subscribed, false if not + */ + public function is_subscribed() + { + return kolab_storage::folder_is_subscribed($this->imap_folder); + } + /** * Compose an URL for CardDAV access to this address book (if configured) */ diff --git a/plugins/kolab_addressbook/localization/en_US.inc b/plugins/kolab_addressbook/localization/en_US.inc index c1ab0f5e..11a45817 100644 --- a/plugins/kolab_addressbook/localization/en_US.inc +++ b/plugins/kolab_addressbook/localization/en_US.inc @@ -34,6 +34,14 @@ $labels['globalfirst'] = 'Global address book(s) first'; $labels['personalonly'] = 'Personal address book(s) only'; $labels['globalonly'] = 'Global address book(s) only'; +$labels['findaddressbooks'] = 'Find address books'; +$labels['searchterms'] = 'Search terms'; +$labels['listsearchresults'] = 'Additional address books'; +$labels['foldersearchform'] = 'Address book search form'; +$labels['foldersubscribe'] = 'List permanently'; +$labels['nraddressbooksfound'] = '$nr address books found'; +$labels['noaddressbooksfound'] = 'No address books found'; + $messages['bookdeleteconfirm'] = 'Do you really want to delete the selected address book and all contacts in it?'; $messages['bookdeleting'] = 'Deleting address book...'; $messages['booksaving'] = 'Saving address book...'; diff --git a/plugins/kolab_addressbook/skins/classic/kolab_addressbook.css b/plugins/kolab_addressbook/skins/classic/kolab_addressbook.css index da3abed2..30b95939 100644 --- a/plugins/kolab_addressbook/skins/classic/kolab_addressbook.css +++ b/plugins/kolab_addressbook/skins/classic/kolab_addressbook.css @@ -1,6 +1,19 @@ +#directorylistbox ul.treelist li.virtual { + background-image: none !important; +} + +#directorylistbox ul.treelist li.virtual > div a { + color: #aaa; + background-image: none; +} + +#directorylistbox ul.treelist li.virtual > .treetoggle { + top: -2px !important; +} + +/* #directorylist li.addressbook.readonly { - /* don't use 'background' to not reset background color */ background-image: url(kolab_folders.png); background-position: 5px 0px; background-repeat: no-repeat; @@ -33,3 +46,4 @@ background-position: 5px -36px; background-repeat: no-repeat; } +*/ \ No newline at end of file diff --git a/plugins/kolab_addressbook/skins/larry/folder_icons.png b/plugins/kolab_addressbook/skins/larry/folder_icons.png index 62805ad2..07674ab0 100644 Binary files a/plugins/kolab_addressbook/skins/larry/folder_icons.png and b/plugins/kolab_addressbook/skins/larry/folder_icons.png differ diff --git a/plugins/kolab_addressbook/skins/larry/kolab_addressbook.css b/plugins/kolab_addressbook/skins/larry/kolab_addressbook.css index 2cfac3b6..00f34e8a 100644 --- a/plugins/kolab_addressbook/skins/larry/kolab_addressbook.css +++ b/plugins/kolab_addressbook/skins/larry/kolab_addressbook.css @@ -1,4 +1,79 @@ +#directorylistbox .listsearchbox + .scroller { + top: 34px; +} + +#directorylistbox .listsearchbox.expanded + .scroller { + top: 68px; +} + +#directorylistbox .boxtitle a.iconbutton.search { + position: absolute; + top: 8px; + right: 8px; + width: 16px; + cursor: pointer; + background-position: -2px -317px; +} + +#directorylistbox ul.treelist li.virtual { + background-image: none !important; +} + +#directorylistbox ul.treelist li.virtual > div a { + color: #aaa; + background-image: none; + height: 16px; + padding-top: 3px; + padding-bottom: 3px; +} + +#directorylistbox ul.treelist li.virtual > .treetoggle { + top: 6px !important; +} + +#directorylistbox ul.treelist li input { + position: absolute; + top: 4px; + left: 10px; +} + +#directorylistbox ul.treelist ul li input { + left: 36px; +} + +#directorylist li input { + display: none; +} + +#directorylistbox ul.treelist div span.subscribed { + display: inline-block; + position: absolute; + top: 4px; + right: 5px; + height: 16px; + width: 16px; + padding: 0; + background: url(folder_icons.png) -100px 0 no-repeat; + overflow: hidden; + text-indent: -5000px; + cursor: pointer; +} + +#directorylistbox ul.treelist div > span.subscribed:focus, +#directorylistbox ul.treelist div:hover > span.subscribed { + background-position: 2px -160px; +} + +#directorylistbox ul.treelist div.subscribed span.subscribed { + background-position: -16px -160px; +} + +#directorylistbox ul.treelist div span.subscribed:focus { + border-radius: 3px; + outline: 2px solid rgba(30,150,192, 0.5); +} + #directorylist li.addressbook.readonly, #directorylist li.addressbook.shared, #directorylist li.addressbook.other { @@ -15,6 +90,7 @@ background-position: 98% -52px; } +#directorylist li.addressbook.personal.readonly, #directorylist li.addressbook.other.readonly { background-position: 98% -77px; } @@ -27,6 +103,11 @@ background-position: 98% -130px; } +#directorylist li.addressbook.virtual.user { + background-image: url(folder_icons.png) !important; + background-position: 98% -52px; +} + #directorylist li.addressbook.readonly a, #directorylist li.addressbook.shared a, #directorylist li.addressbook.other a {