1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
3 /* e-name-selector-list.c - Single-line text entry widget for EDestinations.
5 * Copyright (C) 2005 Novell, Inc.
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of version 2 of the GNU General Public
9 * License as published by the Free Software Foundation.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
16 * You should have received a copy of the GNU General Public
17 * License along with this library; if not, write to the
18 * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19 * Boston, MA 02110-1301, USA.
21 * Authors: Srinivasa Ragavan <sragavan@novell.com>
22 * : Devashish Sharma <sdevashish@novell.com>
27 #include <gdk/gdkkeysyms.h>
28 #include <gtk/gtkmain.h>
29 #include <gtk/gtkentry.h>
30 #include <gtk/gtkentrycompletion.h>
31 #include <gtk/gtkcelllayout.h>
32 #include <gtk/gtkcellrenderertext.h>
33 #include <gtk/gtkframe.h>
34 #include <gtk/gtkmenuitem.h>
35 #include <gtk/gtkradiomenuitem.h>
36 #include <gtk/gtkseparatormenuitem.h>
37 #include <gtk/gtkscrolledwindow.h>
38 #include <gtk/gtktreeselection.h>
39 #include <gtk/gtktreemodel.h>
40 #include <gtk/gtkvbox.h>
41 #include <glib/gi18n-lib.h>
43 #include <libebook/e-book.h>
44 #include <libebook/e-contact.h>
45 #include <libebook/e-destination.h>
46 #include <libedataserverui/e-book-auth-util.h>
47 #include "libedataserver/e-sexp.h"
48 #include <libedataserverui/e-data-server-ui-marshal.h>
49 #include <libedataserverui/e-name-selector-entry.h>
50 #include "e-name-selector-list.h"
54 G_DEFINE_TYPE (ENameSelectorList, e_name_selector_list, E_TYPE_NAME_SELECTOR_ENTRY);
56 static void e_name_selector_list_class_init (ENameSelectorListClass *name_selector_list_class);
57 static void e_name_selector_list_init (ENameSelectorList *name_selector_list);
58 static void e_name_selector_list_dispose (GObject *object);
59 static void e_name_selector_list_finalize (GObject *object);
64 enl_popup_size (ENameSelectorList *list)
66 int height = 0, count;
67 GtkTreeViewColumn *column = NULL;
69 column = gtk_tree_view_get_column ( GTK_TREE_VIEW (list->tree_view), 0);
71 gtk_tree_view_column_cell_get_size (column, NULL, NULL, NULL, NULL, &height);
73 /* Show a maximum of 10 rows in the popup list view */
80 gtk_widget_set_size_request (list->tree_view, ((GtkWidget *)list)->allocation.width - 3 , height * count);
84 enl_popup_position (ENameSelectorList *list)
88 enl_popup_size (list);
89 gdk_window_get_origin (((GtkWidget *)list)->window, &x, &y);
90 y = y +((GtkWidget *)list)->allocation.height;
92 gtk_window_move (list->popup, x, y);
96 enl_popup_grab (ENameSelectorList *list)
100 gtk_grab_add (GTK_WIDGET (list->popup));
102 gdk_pointer_grab (((GtkWidget *)list->popup)->window, TRUE,
103 GDK_BUTTON_PRESS_MASK |
104 GDK_BUTTON_RELEASE_MASK |
105 GDK_POINTER_MOTION_MASK,
106 NULL, NULL, GDK_CURRENT_TIME);
108 gdk_keyboard_grab (((GtkWidget *)list->popup)->window, TRUE, GDK_CURRENT_TIME);
109 gtk_widget_grab_focus ((GtkWidget *)list);
111 /* Build the listview from the model */
112 gtk_tree_view_set_model (GTK_TREE_VIEW (list->tree_view), GTK_TREE_MODEL(((ENameSelectorEntry *)list)->destination_store));
114 /* If any selection of text is present, unselect it */
115 len = strlen(gtk_entry_get_text(GTK_ENTRY(list)));
116 gtk_editable_select_region (GTK_EDITABLE(list), len, -1);
120 enl_popup_ungrab (ENameSelectorList *list)
122 if (!GTK_WIDGET_HAS_GRAB(list->popup))
125 gdk_pointer_ungrab (GDK_CURRENT_TIME);
126 gtk_grab_remove ( GTK_WIDGET (list->popup));
127 gdk_keyboard_ungrab (GDK_CURRENT_TIME);
131 enl_entry_focus_in (ENameSelectorList *list, GdkEventFocus *event, gpointer dummy)
135 /* FIXME: Dont select every thing by default- Code is there but still it does */
136 len = strlen(gtk_entry_get_text(GTK_ENTRY(list)));
137 gtk_editable_select_region (GTK_EDITABLE(list), len, -1);
143 enl_entry_focus_out (ENameSelectorList *list, GdkEventFocus *event, gpointer dummy)
145 /* When we lose focus and popup is still present hide it. Dont do it, when we click the popup. Look for grab */
146 if (GTK_WIDGET_VISIBLE (list->popup) && !GTK_WIDGET_HAS_GRAB(list->popup)) {
147 enl_popup_ungrab (list);
148 gtk_widget_hide ((GtkWidget *)list->popup);
157 enl_popup_button_press (GtkWidget *widget,
158 GdkEventButton *event,
159 ENameSelectorList *list)
161 if (!GTK_WIDGET_MAPPED (widget))
163 /* if we come here, it's usually time to popdown */
164 gtk_widget_hide ((GtkWidget *)list->popup);
170 enl_popup_focus_out (GtkWidget *w,
171 GdkEventFocus *event,
172 ENameSelectorList *list)
174 /* Just ungrab. We lose focus on button press event */
175 enl_popup_ungrab(list);
180 enl_popup_enter_notify (GtkWidget *widget,
181 GdkEventCrossing *event,
182 ENameSelectorList *list)
184 if (event->type == GDK_ENTER_NOTIFY && !GTK_WIDGET_HAS_GRAB (list->popup))
185 enl_popup_grab (list);
191 enl_tree_select_node (ENameSelectorList *list,
194 GtkTreeSelection *selection;
198 selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (list->tree_view));
199 iter.stamp = ((ENameSelectorEntry *) list)->destination_store->stamp;
200 iter.user_data = GINT_TO_POINTER (n-1);
202 gtk_tree_selection_unselect_all (selection);
203 gtk_tree_selection_select_iter (selection, &iter);
205 path = e_destination_store_get_path (GTK_TREE_MODEL(((ENameSelectorEntry *) list)->destination_store), &iter);
206 gtk_tree_view_scroll_to_cell ( GTK_TREE_VIEW (list->tree_view), path, gtk_tree_view_get_column( GTK_TREE_VIEW (list->tree_view), 0), FALSE, 0, 0);
207 gtk_tree_view_set_cursor ( GTK_TREE_VIEW (list->tree_view), path, gtk_tree_view_get_column( GTK_TREE_VIEW (list->tree_view), 0), FALSE);
208 gtk_widget_grab_focus (list->tree_view);
209 /*Fixme: We should grab the focus to the column. How? */
211 gtk_tree_path_free (path);
215 enl_entry_key_press_event (ENameSelectorList *list,
219 if ( (event->state & GDK_CONTROL_MASK) && (event->keyval == GDK_Down)) {
220 enl_popup_position (list);
221 gtk_widget_show_all (GTK_WIDGET (list->popup));
222 enl_popup_grab (list);
223 list->rows = e_destination_store_get_destination_count (((ENameSelectorEntry *) list)->destination_store);
224 enl_popup_size (list);
225 enl_tree_select_node (list, 1);
232 delete_row (GtkTreePath *path, ENameSelectorList *list)
236 GtkTreeSelection *selection;
238 if (!gtk_tree_model_get_iter (GTK_TREE_MODEL (E_NAME_SELECTOR_ENTRY (list)->destination_store), &iter, path))
241 selection = gtk_tree_view_get_selection ( GTK_TREE_VIEW (list->tree_view));
242 len = e_destination_store_get_destination_count (E_NAME_SELECTOR_ENTRY (list)->destination_store);
243 n = GPOINTER_TO_INT (iter.user_data);
245 e_destination_store_remove_destination_nth (((ENameSelectorEntry *) list)->destination_store, n);
247 /* If the last one is deleted select the last but one or the deleted +1 */
251 /* We deleted the last entry */
253 enl_popup_ungrab (list);
255 gtk_menu_popdown(GTK_MENU (list->menu));
256 gtk_widget_hide ( GTK_WIDGET (list->popup));
260 iter.stamp = ((ENameSelectorEntry *) list)->destination_store->stamp;
261 iter.user_data = GINT_TO_POINTER (n);
263 gtk_tree_selection_unselect_all (selection);
264 gtk_tree_selection_select_iter (selection, &iter);
266 gtk_tree_path_free (path);
268 list->rows = e_destination_store_get_destination_count (((ENameSelectorEntry *) list)->destination_store);
269 enl_popup_size (list);
275 popup_activate_email (ENameSelectorEntry *name_selector_entry, GtkWidget *menu_item)
277 EDestination *destination;
281 destination = name_selector_entry->popup_destination;
285 contact = e_destination_get_contact (destination);
289 email_num = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (menu_item), "order"));
290 e_destination_set_contact (destination, contact, email_num);
294 popup_activate_list (EDestination *destination, GtkWidget *item)
296 gboolean status = gtk_check_menu_item_get_active (GTK_CHECK_MENU_ITEM (item));
298 e_destination_set_ignored (destination, !status);
302 destination_set_list (GtkWidget *item, EDestination *destination)
305 gboolean status = gtk_check_menu_item_get_active (GTK_CHECK_MENU_ITEM (item));
307 contact = e_destination_get_contact (destination);
311 e_destination_set_ignored (destination, !status);
315 destination_set_email (GtkWidget *item, EDestination *destination)
320 if (!gtk_check_menu_item_get_active (GTK_CHECK_MENU_ITEM (item)))
322 contact = e_destination_get_contact (destination);
326 email_num = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (item), "order"));
327 e_destination_set_contact (destination, contact, email_num);
331 ENameSelectorList *list;
336 popup_delete_row(GtkWidget *w, PopupDeleteRowInfo *row_info)
338 delete_row(row_info->path, row_info->list);
343 menu_deactivate (GtkMenuShell *junk, ENameSelectorList *list)
345 enl_popup_grab (list);
349 enl_tree_button_press_event (GtkWidget *widget,
350 GdkEventButton *event,
351 ENameSelectorList *list)
354 EDestination *destination;
355 ENameSelectorEntry *name_selector_entry;
357 GtkWidget *menu_item;
358 GList *email_list = NULL, *l;
362 GSList *group = NULL;
364 gboolean show_menu = FALSE;
365 GtkTreeSelection *selection;
367 PopupDeleteRowInfo *row_info;
370 if ( !GTK_WIDGET_HAS_GRAB (list->popup))
371 enl_popup_grab (list);
373 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW (list->tree_view), event->x, event->y, &path, NULL);
374 selection = gtk_tree_view_get_selection ( GTK_TREE_VIEW (list->tree_view));
375 if (!gtk_tree_model_get_iter (GTK_TREE_MODEL (E_NAME_SELECTOR_ENTRY (list)->destination_store), &iter, path))
378 gtk_tree_selection_unselect_all (selection);
379 gtk_tree_selection_select_iter (selection, &iter);
381 if (event->button != 3) {
385 name_selector_entry = E_NAME_SELECTOR_ENTRY (list);
387 destination = e_destination_store_get_destination ( ((ENameSelectorEntry *)list)->destination_store, &iter);
392 contact = e_destination_get_contact (destination);
397 gtk_menu_popdown (GTK_MENU (list->menu));
399 menu = gtk_menu_new ();
400 g_signal_connect (menu, "deactivate", G_CALLBACK(menu_deactivate), list);
402 gtk_menu_popup (GTK_MENU (menu), NULL, NULL, NULL, NULL, event->button, gtk_get_current_event_time());
404 email_num = e_destination_get_email_num (destination);
407 is_list = e_contact_get (contact, E_CONTACT_IS_LIST) ? TRUE : FALSE;
409 const GList *dests = e_destination_list_get_dests (destination);
411 int length = g_list_length ((GList *)dests);
413 for (iters = (GList *)dests; iters; iters = iters->next) {
414 EDestination *dest = (EDestination *) iters->data;
415 const char *email = e_destination_get_email (dest);
417 if (!email || *email == '\0')
421 menu_item = gtk_check_menu_item_new_with_label (email);
422 g_signal_connect (menu_item, "toggled", G_CALLBACK (destination_set_list), dest);
424 menu_item = gtk_menu_item_new_with_label (email);
427 gtk_widget_show (menu_item);
428 gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menu_item);
432 gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (menu_item), !e_destination_is_ignored(dest));
433 g_signal_connect_swapped (menu_item, "activate", G_CALLBACK (popup_activate_list),
439 email_list = e_contact_get (contact, E_CONTACT_EMAIL);
440 len = g_list_length (email_list);
442 for (l = email_list, i = 0; l; l = g_list_next (l), i++) {
443 gchar *email = l->data;
445 if (!email || *email == '\0')
449 menu_item = gtk_radio_menu_item_new_with_label (group, email);
450 group = gtk_radio_menu_item_group (GTK_RADIO_MENU_ITEM (menu_item));
451 g_signal_connect (menu_item, "toggled", G_CALLBACK (destination_set_email), destination);
453 menu_item = gtk_menu_item_new_with_label (email);
456 gtk_widget_show (menu_item);
457 gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menu_item);
459 g_object_set_data (G_OBJECT (menu_item), "order", GINT_TO_POINTER (i));
461 if ( i == email_num && len > 1 ) {
462 gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (menu_item), TRUE);
463 g_signal_connect_swapped (menu_item, "activate", G_CALLBACK (popup_activate_email),
464 name_selector_entry);
467 g_list_foreach (email_list, (GFunc) g_free, NULL);
468 g_list_free (email_list);
474 menu_item = gtk_separator_menu_item_new ();
475 gtk_widget_show (menu_item);
476 gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menu_item);
479 delete_label = g_strdup_printf(_("_Delete %s"), (char *)e_contact_get_const (contact, E_CONTACT_FILE_AS));
480 menu_item = gtk_menu_item_new_with_mnemonic (delete_label);
481 g_free (delete_label);
482 gtk_widget_show (menu_item);
483 gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menu_item);
485 row_info = g_new (PopupDeleteRowInfo, 1);
486 row_info->list = list;
487 row_info->path = path;
489 g_signal_connect (menu_item, "activate", G_CALLBACK (popup_delete_row),
498 enl_tree_key_press_event (GtkWidget *w,
500 ENameSelectorList *list)
502 if (event->keyval == GDK_Escape) {
503 enl_popup_ungrab (list);
504 gtk_widget_hide ( GTK_WIDGET (list->popup));
506 } else if (event->keyval == GDK_Delete) {
507 GtkTreeSelection *selection;
510 selection = gtk_tree_view_get_selection ( GTK_TREE_VIEW (list->tree_view));
511 paths = gtk_tree_selection_get_selected_rows (selection, (GtkTreeModel **)&(E_NAME_SELECTOR_ENTRY (list)->destination_store));
512 paths = g_list_reverse (paths);
513 g_list_foreach (paths, (GFunc) delete_row, list);
515 } else if (event->keyval != GDK_Up && event->keyval != GDK_Down
516 && event->keyval != GDK_Shift_R && event->keyval != GDK_Shift_L
517 && event->keyval != GDK_Control_R && event->keyval != GDK_Control_L){
519 enl_popup_ungrab (list);
520 gtk_widget_hide ( GTK_WIDGET (list->popup));
521 gtk_widget_event (GTK_WIDGET (list), (GdkEvent *)event);
529 e_name_selector_list_expand_clicked(ENameSelectorList *list)
532 if (!GTK_WIDGET_VISIBLE (list->popup)) {
533 enl_popup_position (list);
534 gtk_widget_show_all (GTK_WIDGET (list->popup));
535 enl_popup_grab (list);
536 list->rows = e_destination_store_get_destination_count (((ENameSelectorEntry *) list)->destination_store);
537 enl_popup_size (list);
538 enl_tree_select_node (list, 1);
541 enl_popup_ungrab (list);
543 gtk_menu_popdown(GTK_MENU (list->menu));
544 gtk_widget_hide (GTK_WIDGET (list->popup));
549 e_name_selector_list_dispose (GObject *object)
551 if (G_OBJECT_CLASS (e_name_selector_list_parent_class)->dispose)
552 G_OBJECT_CLASS (e_name_selector_list_parent_class)->dispose (object);
556 e_name_selector_list_finalize (GObject *object)
558 if (G_OBJECT_CLASS (e_name_selector_list_parent_class)->finalize)
559 G_OBJECT_CLASS (e_name_selector_list_parent_class)->finalize (object);
563 e_name_selector_list_realize (GtkWidget *widget)
565 ENameSelectorList *list = (ENameSelectorList *)widget;
566 GTK_WIDGET_CLASS (e_name_selector_list_parent_class)->realize (widget);
568 gtk_tree_view_set_model ( GTK_TREE_VIEW (list->tree_view), GTK_TREE_MODEL(((ENameSelectorEntry *)list)->destination_store));
572 e_name_selector_list_class_init (ENameSelectorListClass *name_selector_list_class)
574 GObjectClass *object_class = G_OBJECT_CLASS (name_selector_list_class);
575 GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (name_selector_list_class);
577 object_class->dispose = e_name_selector_list_dispose;
578 object_class->finalize = e_name_selector_list_finalize;
580 widget_class->realize = e_name_selector_list_realize;
582 /* Install properties */
584 /* Install signals */
589 e_name_selector_list_init (ENameSelectorList *list)
591 GtkCellRenderer *renderer;
592 GtkWidget *scroll, *popup_frame, *vbox;
593 GtkTreeSelection *selection;
594 GtkTreeViewColumn *column;
595 ENameSelectorEntry *entry = E_NAME_SELECTOR_ENTRY (list);
596 GtkEntryCompletion *completion;
598 list->store = e_destination_store_new ();
601 list->tree_view = GTK_WIDGET (gtk_tree_view_new_with_model (GTK_TREE_MODEL(entry->destination_store)));
602 gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (list->tree_view), FALSE);
603 gtk_tree_view_set_hover_selection (GTK_TREE_VIEW (list->tree_view), FALSE);
605 selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (list->tree_view));
606 gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE);
607 gtk_tree_selection_unselect_all (selection);
608 gtk_tree_view_set_enable_search (GTK_TREE_VIEW (list->tree_view), FALSE);
610 completion = gtk_entry_get_completion (GTK_ENTRY(list));
611 gtk_entry_completion_set_inline_completion (completion, TRUE);
612 gtk_entry_completion_set_popup_completion (completion, TRUE);
614 renderer = gtk_cell_renderer_text_new ();
615 column = gtk_tree_view_column_new_with_attributes ("Name", renderer, "text", E_DESTINATION_STORE_COLUMN_ADDRESS, NULL);
616 gtk_tree_view_append_column (GTK_TREE_VIEW (list->tree_view), column);
617 gtk_tree_view_column_set_clickable (column, TRUE);
619 scroll = gtk_scrolled_window_new (NULL, NULL);
620 gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scroll),
622 GTK_POLICY_AUTOMATIC);
623 gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scroll),
625 gtk_widget_set_size_request (GTK_SCROLLED_WINDOW (scroll)->vscrollbar, -1, 0);
627 list->popup = GTK_WINDOW (gtk_window_new (GTK_WINDOW_POPUP));
628 gtk_window_set_resizable (GTK_WINDOW (list->popup), FALSE);
630 popup_frame = gtk_frame_new (NULL);
631 gtk_frame_set_shadow_type (GTK_FRAME (popup_frame),
632 GTK_SHADOW_ETCHED_IN);
634 gtk_container_add (GTK_CONTAINER (list->popup), popup_frame);
636 vbox = gtk_vbox_new (FALSE, 0);
637 gtk_container_add (GTK_CONTAINER (popup_frame), vbox);
639 gtk_container_add (GTK_CONTAINER (scroll), list->tree_view);
640 gtk_box_pack_start (GTK_BOX (vbox), scroll,
643 g_signal_connect_after (GTK_WIDGET (list), "focus-in-event", G_CALLBACK(enl_entry_focus_in), NULL);
644 g_signal_connect (GTK_WIDGET (list), "focus-out-event", G_CALLBACK(enl_entry_focus_out), NULL);
645 g_signal_connect (GTK_WIDGET (list), "key-press-event", G_CALLBACK(enl_entry_key_press_event), NULL);
647 g_signal_connect_after (list->tree_view, "key-press-event", G_CALLBACK(enl_tree_key_press_event), list);
648 g_signal_connect (list->tree_view, "button-press-event", G_CALLBACK (enl_tree_button_press_event), list);
650 g_signal_connect (GTK_WIDGET (list->popup), "button-press-event", G_CALLBACK(enl_popup_button_press), list);
651 g_signal_connect (GTK_WIDGET (list->popup), "focus-out-event", G_CALLBACK(enl_popup_focus_out), list);
652 g_signal_connect (GTK_WIDGET (list->popup), "enter-notify-event", G_CALLBACK (enl_popup_enter_notify), list);
657 e_name_selector_list_new (void)
659 return g_object_new (e_name_selector_list_get_type (), NULL);