/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * e-search-bar.c * * Copyright (C) 2000, 2001 Ximian, Inc. * * Authors: * Chris Lahey * Ettore Perazzoli * Jon Trowbridge * * This library is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "e-icon-entry.h" #include "e-search-bar.h" #include "e-util/e-util-marshal.h" #include "e-util/e-icon-factory.h" enum { QUERY_CHANGED, MENU_ACTIVATED, SEARCH_ACTIVATED, SEARCH_CLEARED, LAST_SIGNAL }; static gint esb_signals [LAST_SIGNAL] = { 0, }; static GtkHBoxClass *parent_class = NULL; /* The arguments we take */ enum { PROP_0, PROP_ITEM_ID, PROP_SUBITEM_ID, PROP_TEXT, }; /* Forward decls. */ static int find_id (GtkWidget *menu, int idin, const char *type, GtkWidget **widget); static void emit_search_activated (ESearchBar *esb); static void emit_query_changed (ESearchBar *esb); /* Utility functions. */ static void set_find_now_sensitive (ESearchBar *search_bar, gboolean sensitive) { if (search_bar->ui_component != NULL) bonobo_ui_component_set_prop (search_bar->ui_component, "/commands/ESearchBar:FindNow", "sensitive", sensitive ? "1" : "0", NULL); } static void update_clear_menuitem_sensitive (ESearchBar *search_bar) { if (search_bar->ui_component != NULL) { gboolean sensitive = GTK_WIDGET_SENSITIVE (search_bar->clear_button) || search_bar->viewitem_id != 0; bonobo_ui_component_set_prop (search_bar->ui_component, "/commands/ESearchBar:Clear", "sensitive", sensitive ? "1" : "0", NULL); } } static void clear_button_state_changed (GtkWidget *clear_button, GtkStateType state, ESearchBar *search_bar) { g_assert (clear_button != NULL && search_bar != NULL); update_clear_menuitem_sensitive (search_bar); } static char * verb_name_from_id (int id) { return g_strdup_printf ("ESearchBar:Activate:%d", id); } /* This implements the "clear" action, i.e. clears the text and then emits * ::search_activated. */ static void clear_search (ESearchBar *esb) { e_search_bar_set_text (esb, ""); esb->block_search = TRUE; if (esb->item_id < 0) e_search_bar_set_item_id (esb, esb->last_search_option); e_search_bar_set_viewitem_id (esb, 0); esb->block_search = FALSE; emit_search_activated (esb); } static void free_menu_items (ESearchBar *esb) { GSList *p; if (esb->menu_items == NULL) return; for (p = esb->menu_items; p != NULL; p = p->next) { ESearchBarItem *item; item = (ESearchBarItem *) p->data; /* (No submitems for the menu_items, so no need to free that member.) */ g_free (item->text); g_free (item); } g_slist_free (esb->menu_items); esb->menu_items = NULL; } /* Signals. */ static void emit_query_changed (ESearchBar *esb) { g_signal_emit (esb, esb_signals [QUERY_CHANGED], 0); update_clear_menuitem_sensitive (esb); } static void emit_search_activated(ESearchBar *esb) { if (esb->pending_activate) { g_source_remove (esb->pending_activate); esb->pending_activate = 0; } g_signal_emit (esb, esb_signals [SEARCH_ACTIVATED], 0); set_find_now_sensitive (esb, FALSE); update_clear_menuitem_sensitive (esb); } static void emit_menu_activated (ESearchBar *esb, int item) { g_signal_emit (esb, esb_signals [MENU_ACTIVATED], 0, item); } /* Callbacks -- Standard verbs. */ static void search_now_verb_cb (BonoboUIComponent *ui_component, void *data, const char *verb_name) { ESearchBar *esb; GtkStyle *style = gtk_widget_get_default_style (); const char *text; esb = E_SEARCH_BAR (data); text = e_search_bar_get_text (esb); if (text && *text) { gtk_widget_modify_base (esb->entry, GTK_STATE_NORMAL, &(style->base[GTK_STATE_SELECTED])); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, &(style->text[GTK_STATE_SELECTED])); gtk_widget_modify_base (esb->icon_entry, GTK_STATE_NORMAL, &(style->base[GTK_STATE_SELECTED])); gtk_widget_modify_base (esb->viewoption, GTK_STATE_NORMAL, &(style->base[GTK_STATE_SELECTED])); } else { gtk_widget_modify_base (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_base (esb->icon_entry, GTK_STATE_NORMAL, NULL); } emit_search_activated (esb); } static void clear_verb_cb (BonoboUIComponent *ui_component, void *data, const char *verb_name) { ESearchBar *esb; esb = E_SEARCH_BAR (data); gtk_widget_modify_base (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_base (esb->icon_entry, GTK_STATE_NORMAL, NULL); clear_search (esb); gtk_entry_set_text (GTK_ENTRY (esb->entry), ""); gtk_widget_grab_focus (esb->entry); } static void setup_standard_verbs (ESearchBar *search_bar) { bonobo_ui_component_add_verb (search_bar->ui_component, "ESearchBar:Clear", clear_verb_cb, search_bar); bonobo_ui_component_add_verb (search_bar->ui_component, "ESearchBar:FindNow", search_now_verb_cb, search_bar); bonobo_ui_component_set (search_bar->ui_component, "/", ("" " " " " ""), NULL); /* Make sure the entries are created with the correct sensitivity. */ set_find_now_sensitive (search_bar, FALSE); update_clear_menuitem_sensitive (search_bar); } /* Callbacks -- The verbs for all the definable items. */ static void search_verb_cb (BonoboUIComponent *ui_component, void *data, const char *verb_name) { ESearchBar *esb; const char *p; int id; esb = E_SEARCH_BAR (data); p = strrchr (verb_name, ':'); g_assert (p != NULL); id = atoi (p + 1); emit_menu_activated (esb, id); } /* Get the selected menu item's label */ static const gchar * get_selected_item_label (GtkWidget *menu) { GtkWidget *label, *item; const gchar *text = NULL; item = gtk_menu_get_active ((GtkMenu *)menu); label = gtk_bin_get_child ((GtkBin *)item); if (GTK_IS_LABEL (label)) text = gtk_label_get_text ((GtkLabel *)label); return text; } static gboolean entry_focus_in_cb (GtkWidget *widget, GdkEventFocus *event, ESearchBar *esb) { GtkStyle *entry_style, *default_style; entry_style = gtk_widget_get_style (esb->entry); default_style = gtk_widget_get_default_style (); if (gdk_color_equal (&(entry_style->text[GTK_STATE_NORMAL]), &(default_style->text[GTK_STATE_INSENSITIVE]))) { gtk_entry_set_text (GTK_ENTRY (esb->entry), ""); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, NULL); } return FALSE; } static gboolean paint_search_text (GtkWidget *widget, ESearchBar *esb) { GtkStyle *style = gtk_widget_get_default_style (); const gchar *text = NULL; GtkWidget *menu_widget = esb->option_menu; text = gtk_entry_get_text (GTK_ENTRY (widget)); if (text && *text) return FALSE; if (!GTK_WIDGET_SENSITIVE (esb->option_button)) { menu_widget = esb->scopeoption_menu; text = g_object_get_data (G_OBJECT(gtk_menu_get_active ( GTK_MENU (esb->scopeoption_menu))),"string"); } else if (!GTK_IS_RADIO_MENU_ITEM (gtk_menu_get_active ( GTK_MENU (esb->option_menu)))) return FALSE; else /* no query in search entry .. so set the current option */ text = get_selected_item_label (menu_widget); if (text && *text) { if (!GTK_WIDGET_HAS_FOCUS(esb->entry)) { gtk_entry_set_text (GTK_ENTRY (esb->entry), text); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, &(style->text[GTK_STATE_INSENSITIVE])); } gtk_tooltips_set_tip (esb->tooltips, esb->option_button, text, "Search type"); gtk_widget_set_sensitive (esb->clear_button, FALSE); } return FALSE; } void e_search_bar_paint (ESearchBar *search_bar) { paint_search_text (search_bar->entry, search_bar); } static gboolean entry_focus_out_cb (GtkWidget *widget, GdkEventFocus *event, ESearchBar *esb) { return paint_search_text (widget, esb); } static void entry_activated_cb (GtkWidget *widget, ESearchBar *esb) { const char *text = gtk_entry_get_text (GTK_ENTRY (esb->entry)); GtkStyle *style = gtk_widget_get_default_style (); if (text && *text) { gtk_widget_modify_base (esb->entry, GTK_STATE_NORMAL, &(style->base[GTK_STATE_SELECTED])); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, &(style->text[GTK_STATE_SELECTED])); gtk_widget_modify_base (esb->icon_entry, GTK_STATE_NORMAL, &(style->base[GTK_STATE_SELECTED])); gtk_widget_modify_base (esb->viewoption, GTK_STATE_NORMAL, &(style->base[GTK_STATE_SELECTED])); } else { gtk_widget_modify_base (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_base (esb->icon_entry, GTK_STATE_NORMAL, NULL); } emit_search_activated (esb); } static void entry_changed_cb (GtkWidget *widget, ESearchBar *esb) { const char *text = gtk_entry_get_text (GTK_ENTRY (esb->entry)); GtkStyle *entry_style, *default_style; entry_style = gtk_widget_get_style (esb->entry); default_style = gtk_widget_get_default_style (); if (text && *text) if (gdk_color_equal (&(entry_style->text[GTK_STATE_NORMAL]), &(default_style->text[GTK_STATE_INSENSITIVE]))) gtk_widget_set_sensitive (esb->clear_button, FALSE); else gtk_widget_set_sensitive (esb->clear_button, TRUE); else gtk_widget_set_sensitive (esb->clear_button, FALSE); } static void viewitem_activated_cb(GtkWidget *widget, ESearchBar *esb) { gint viewid; GtkStyle *entry_style, *default_style; widget = gtk_menu_get_active (GTK_MENU (esb->viewoption_menu)); viewid = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (widget), "EsbItemId")); esb->viewitem_id = viewid; entry_style = gtk_widget_get_style (esb->entry); default_style = gtk_widget_get_default_style (); /* If the text is grayed, Its not the query string */ if (gdk_color_equal (&(entry_style->text[GTK_STATE_NORMAL]), &(default_style->text[GTK_STATE_INSENSITIVE]))) { gtk_entry_set_text (GTK_ENTRY (esb->entry), ""); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, NULL); } esb->block_search = TRUE; emit_search_activated (esb); esb->block_search = FALSE; } static void scopeitem_activated_cb(GtkWidget *widget, ESearchBar *esb) { gint scopeid; GtkStyle *entry_style, *default_style; widget = gtk_menu_get_active (GTK_MENU (esb->scopeoption_menu)); scopeid = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (widget), "EsbItemId")); esb->scopeitem_id = scopeid; entry_style = gtk_widget_get_style (esb->entry); default_style = gtk_widget_get_default_style (); /* If the text is grayed, Its not the query string */ if (gdk_color_equal (&(entry_style->text[GTK_STATE_NORMAL]), &(default_style->text[GTK_STATE_INSENSITIVE]))) { gtk_widget_grab_focus (esb->entry); gtk_entry_set_text (GTK_ENTRY (esb->entry), ""); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, NULL); } esb->block_search = TRUE; emit_search_activated (esb); esb->block_search = FALSE; } static char * string_without_underscores (const char *s) { char *new_string; const char *sp; char *dp; new_string = g_malloc (strlen (s) + 1); dp = new_string; for (sp = s; *sp != '\0'; sp ++) { if (*sp != '_') { *dp = *sp; dp ++; } else if (sp[1] == '_') { /* Translate "__" in "_". */ *dp = '_'; dp ++; sp ++; } } *dp = 0; return new_string; } static void option_activated_cb (GtkWidget *widget, ESearchBar *esb) { int id; const char *text; id = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (widget), "EsbItemId")); e_search_bar_set_item_id (esb, id); if (GTK_IS_RADIO_MENU_ITEM (gtk_menu_get_active ( GTK_MENU (esb->option_menu)))) { text = get_selected_item_label (esb->option_menu); if (text && *text) gtk_tooltips_set_tip (esb->tooltips, esb->option_button, text, "Search type"); } if (!esb->block_search) { emit_query_changed (esb); } if (!esb->block_search && id > 0) { emit_search_activated (esb); } } static void option_button_clicked_cb (GtkWidget *widget, GdkEventButton *event, ESearchBar *esb) { gtk_menu_popup (GTK_MENU (esb->option_menu), NULL, NULL, NULL, NULL,1,gtk_get_current_event_time()); gtk_widget_grab_focus (esb->entry); } static void clear_button_clicked_cb (GtkWidget *widget, GdkEventButton *event, ESearchBar *esb) { gtk_widget_modify_base (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_text (esb->entry, GTK_STATE_NORMAL, NULL); gtk_widget_modify_base (esb->icon_entry, GTK_STATE_NORMAL, NULL); clear_search (esb); gtk_entry_set_text (GTK_ENTRY (esb->entry), ""); gtk_widget_grab_focus (esb->entry); } static gboolean entry_key_press_cb (GtkWidget *widget, GdkEventKey *key_event, ESearchBar *esb) { if (((key_event->state & gtk_accelerator_get_default_mod_mask ()) == GDK_MOD1_MASK) && (key_event->keyval == GDK_Down)) { option_button_clicked_cb (NULL, NULL, esb); return TRUE; } return FALSE; } #if 0 static void scopeoption_changed_cb (GtkWidget *option_menu, ESearchBar *search_bar) { const gchar *text = NULL; text = e_search_bar_get_text (search_bar); if (!(text && *text)) gtk_widget_grab_focus (search_bar->entry); if(!search_bar->block_search) emit_query_changed (search_bar); } #endif /* Widgetry creation. */ #if 0 /* This function exists to fix the irreparable GtkOptionMenu stupidity. In fact, this lame-ass widget adds a 1-pixel-wide empty border around the button for no reason. So we have add a 1-pixel-wide border around the the buttons we have in the search bar to make things look right. This is done through an event box. */ static GtkWidget * put_in_spacer_widget (GtkWidget *widget) { GtkWidget *holder; holder = gtk_event_box_new (); gtk_container_set_border_width (GTK_CONTAINER (holder), 1); gtk_container_add (GTK_CONTAINER (holder), widget); return holder; } #endif static void append_xml_menu_item (GString *xml, const char *name, const char *label, const char *stock, const char *verb, const char *accelerator) { char *encoded_label; encoded_label = bonobo_ui_util_encode_str (label); g_string_append_printf (xml, ""); } static void remove_bonobo_menus (ESearchBar *esb) { if (bonobo_ui_component_get_container (esb->ui_component) == CORBA_OBJECT_NIL) return; bonobo_ui_component_rm (esb->ui_component, "/menu/SearchPlaceholder", NULL); } static void setup_bonobo_menus (ESearchBar *esb) { GString *xml; GSList *p; char *verb_name; char *encoded_title; xml = g_string_new (""); encoded_title = bonobo_ui_util_encode_str (_("_Search")); g_string_append_printf (xml, "", encoded_title); g_free (encoded_title); g_string_append (xml, ""); append_xml_menu_item (xml, "FindNow", _("_Find Now"), "gtk-find", "ESearchBar:FindNow", NULL); append_xml_menu_item (xml, "Clear", _("_Clear"), "gtk-clear", "ESearchBar:Clear", "*Control**Shift*q"); for (p = esb->menu_items; p != NULL; p = p->next) { const ESearchBarItem *item; item = (const ESearchBarItem *) p->data; verb_name = verb_name_from_id (item->id); bonobo_ui_component_add_verb (esb->ui_component, verb_name, search_verb_cb, esb); if (item->text == NULL) g_string_append (xml, ""); else append_xml_menu_item (xml, verb_name, item->text, NULL, verb_name, NULL); g_free (verb_name); } g_string_append (xml, ""); g_string_append (xml, ""); remove_bonobo_menus (esb); bonobo_ui_component_set (esb->ui_component, "/menu/SearchPlaceholder", xml->str, NULL); g_string_free (xml, TRUE); if (esb->clear_button) { g_signal_connect (esb->clear_button, "state-changed", G_CALLBACK (clear_button_state_changed), esb); } } static void update_bonobo_menus (ESearchBar *esb) { setup_bonobo_menus (esb); } static void set_menu (ESearchBar *esb, ESearchBarItem *items) { int i; free_menu_items (esb); if (items == NULL) return; for (i = 0; items[i].id != -1; i++) { ESearchBarItem *new_item; new_item = g_new (ESearchBarItem, 1); new_item->text = items[i].text ? g_strdup (_(items[i].text)) : NULL; new_item->id = items[i].id; new_item->type = items[i].type; esb->menu_items = g_slist_append (esb->menu_items, new_item); } if (esb->ui_component != NULL) update_bonobo_menus (esb); } /* /\* Callback used when an option item is destroyed. We have to destroy its */ /* * suboption items. */ /* *\/ */ /* static void */ /* option_item_destroy_cb (GtkObject *object, gpointer data) */ /* { */ /* /\* ESearchBarSubitem *subitems; *\/ */ /* /\* subitems = data; *\/ */ /* /\* g_assert (subitems != NULL); *\/ */ /* /\* free_subitems (subitems); *\/ */ /* /\* g_object_set_data (G_OBJECT (object), "EsbChoiceSubitems", NULL); *\/ */ /* } */ static void set_option (ESearchBar *esb, ESearchBarItem *items) { GtkWidget *menu; GSList *group = NULL; int i; if (esb->option_menu) gtk_widget_destroy (esb->option_menu); esb->option_menu = menu = gtk_menu_new (); for (i = 0; items[i].id != -1; i++) { GtkWidget *item; /* Create a new group */ if (items[i].id == 0) group = 0; if (items[i].text) { char *str; str = string_without_underscores (_(items[i].text)); switch (items[i].type) { case ESB_ITEMTYPE_NORMAL: item = gtk_menu_item_new_with_label (str); break; case ESB_ITEMTYPE_CHECK: item = gtk_check_menu_item_new_with_label (str); break; case ESB_ITEMTYPE_RADIO: item = gtk_radio_menu_item_new_with_label (group, str); group = gtk_radio_menu_item_get_group(GTK_RADIO_MENU_ITEM (item)); break; default: /* Fixme : this should be a normal item */ item = gtk_radio_menu_item_new_with_label (group, str); group = gtk_radio_menu_item_get_group(GTK_RADIO_MENU_ITEM (item)); break; } g_free (str); } else { item = gtk_menu_item_new (); gtk_widget_set_sensitive (item, FALSE); } gtk_menu_shell_append(GTK_MENU_SHELL(menu), item); g_object_set_data (G_OBJECT (item), "EsbItemId", GINT_TO_POINTER(items[i].id)); g_signal_connect (item, "activate", G_CALLBACK (option_activated_cb), esb); } gtk_widget_show_all (menu); g_object_set_data (G_OBJECT(esb->option_menu), "group", group); entry_focus_out_cb (esb->entry, NULL, esb); } static int find_id (GtkWidget *menu, int idin, const char *type, GtkWidget **widget) { GList *l = GTK_MENU_SHELL (menu)->children; int row = -1, i = 0, id; if (widget) *widget = NULL; while (l) { id = GPOINTER_TO_INT (g_object_get_data (l->data, type)); if (id == idin) { row = i; if (widget) *widget = l->data; break; } i++; l = l->next; } return row; } /* GtkObject methods. */ static void impl_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { ESearchBar *esb = E_SEARCH_BAR (object); switch (prop_id) { case PROP_ITEM_ID: g_value_set_int (value, e_search_bar_get_item_id (esb)); break; case PROP_TEXT: g_value_take_string (value, e_search_bar_get_text (esb)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void impl_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { ESearchBar *esb = E_SEARCH_BAR(object); switch (prop_id) { case PROP_ITEM_ID: e_search_bar_set_item_id (esb, g_value_get_int (value)); break; case PROP_TEXT: e_search_bar_set_text (esb, g_value_get_string (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void impl_dispose (GObject *object) { ESearchBar *esb = E_SEARCH_BAR (object); g_return_if_fail (object != NULL); g_return_if_fail (E_IS_SEARCH_BAR (object)); /* These three we do need to unref, because we explicitly hold references to them. */ if (esb->ui_component != NULL) { bonobo_object_unref (BONOBO_OBJECT (esb->ui_component)); esb->ui_component = NULL; } /* if (esb->entry) { */ /* g_object_unref (esb->entry); */ /* esb->entry = NULL; */ /* } */ if (esb->suboption) { g_object_unref (esb->suboption); esb->suboption = NULL; } if (esb->pending_activate) { g_source_remove (esb->pending_activate); esb->pending_activate = 0; } free_menu_items (esb); if (G_OBJECT_CLASS (parent_class)->dispose) G_OBJECT_CLASS (parent_class)->dispose (object); } static void class_init (ESearchBarClass *klass) { GObjectClass *object_class; object_class = G_OBJECT_CLASS (klass); parent_class = g_type_class_ref (gtk_hbox_get_type ()); object_class->set_property = impl_set_property; object_class->get_property = impl_get_property; object_class->dispose = impl_dispose; klass->set_menu = set_menu; klass->set_option = set_option; g_object_class_install_property (object_class, PROP_ITEM_ID, g_param_spec_int ("item_id", _("Item ID"), /*_( */"XXX blurb" /*)*/, 0, 0, 0, G_PARAM_READWRITE | G_PARAM_LAX_VALIDATION)); g_object_class_install_property (object_class, PROP_TEXT, g_param_spec_string ("text", _("Text"), /*_( */"XXX blurb" /*)*/, NULL, G_PARAM_READWRITE)); esb_signals [QUERY_CHANGED] = g_signal_new ("query_changed", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ESearchBarClass, query_changed), NULL, NULL, e_util_marshal_NONE__NONE, G_TYPE_NONE, 0); esb_signals [MENU_ACTIVATED] = g_signal_new ("menu_activated", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ESearchBarClass, menu_activated), NULL, NULL, e_util_marshal_NONE__INT, G_TYPE_NONE, 1, G_TYPE_INT); esb_signals [SEARCH_ACTIVATED] = g_signal_new ("search_activated", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ESearchBarClass, search_activated), NULL, NULL, e_util_marshal_NONE__NONE, G_TYPE_NONE, 0); esb_signals [SEARCH_CLEARED] = g_signal_new ("search_cleared", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ESearchBarClass, search_cleared), NULL, NULL, e_util_marshal_NONE__NONE, G_TYPE_NONE, 0); } static void init (ESearchBar *esb) { esb->ui_component = NULL; esb->menu_items = NULL; esb->option = NULL; esb->entry = NULL; esb->suboption = NULL; esb->option_menu = NULL; esb->suboption_menu = NULL; esb->option_button = NULL; esb->clear_button = NULL; esb->entry_box = NULL; esb->scopeoption_menu = NULL; esb->scopeoption = NULL; esb->scopeoption_box = NULL; esb->tooltips = NULL; esb->pending_activate = 0; esb->item_id = 0; esb->scopeitem_id = 0; esb->last_search_option = 0; esb->block_search = FALSE; } /* Object construction. */ static gint idle_activate_hack (gpointer ptr) { ESearchBar *esb = E_SEARCH_BAR (ptr); esb->pending_activate = 0; emit_search_activated (esb); return FALSE; } void e_search_bar_construct (ESearchBar *search_bar, ESearchBarItem *menu_items, ESearchBarItem *option_items) { GtkWidget *label, *hbox, *bighbox; g_return_if_fail (search_bar != NULL); g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); g_return_if_fail (option_items != NULL); gtk_box_set_spacing (GTK_BOX (search_bar), 3); gtk_box_set_homogeneous (GTK_BOX (search_bar), FALSE); search_bar->tooltips = gtk_tooltips_new (); bighbox = gtk_hbox_new (FALSE, 0); search_bar->entry_box = gtk_hbox_new (0, FALSE); search_bar->icon_entry = e_icon_entry_new (); search_bar->entry = e_icon_entry_get_entry (E_ICON_ENTRY (search_bar->icon_entry)); g_signal_connect (search_bar->entry, "changed", G_CALLBACK (entry_changed_cb), search_bar); g_signal_connect (search_bar->entry, "activate", G_CALLBACK (entry_activated_cb), search_bar); g_signal_connect (search_bar->entry, "focus-in-event", G_CALLBACK (entry_focus_in_cb), search_bar); g_signal_connect (search_bar->entry, "focus-out-event", G_CALLBACK (entry_focus_out_cb), search_bar); g_signal_connect (search_bar->entry, "key-press-event", G_CALLBACK (entry_key_press_cb), search_bar); search_bar->clear_button = e_icon_entry_create_button ("gtk-clear"); g_signal_connect (G_OBJECT (search_bar->clear_button), "button-press-event", G_CALLBACK(clear_button_clicked_cb), search_bar); e_icon_entry_pack_widget (E_ICON_ENTRY (search_bar->icon_entry), search_bar->clear_button, FALSE); search_bar->option_button = e_icon_entry_create_button ("gtk-find"); g_signal_connect (G_OBJECT (search_bar->option_button), "button-press-event", G_CALLBACK(option_button_clicked_cb), search_bar); e_icon_entry_pack_widget (E_ICON_ENTRY (search_bar->icon_entry), search_bar->option_button, TRUE); gtk_box_pack_start (GTK_BOX(search_bar->entry_box), search_bar->icon_entry, FALSE, FALSE, 0); gtk_widget_show_all (search_bar->entry_box); gtk_widget_set_sensitive (search_bar->clear_button, FALSE); /* Current View filter */ search_bar->viewoption_box = gtk_hbox_new (0, FALSE); /* To Translators: The "Show: " label is followed by the Quick Search Dropdown Menu where you can choose to display "All Messages", "Unread Messages", "Message with 'Important' Label" and so on... */ label = gtk_label_new_with_mnemonic (_("Sho_w: ")); gtk_widget_show (label); gtk_box_pack_start (GTK_BOX(search_bar->viewoption_box), label, FALSE, FALSE, 0); search_bar->viewoption = gtk_option_menu_new (); gtk_label_set_mnemonic_widget ((GtkLabel *)label, search_bar->viewoption); gtk_box_pack_start (GTK_BOX(search_bar->viewoption_box), search_bar->viewoption, FALSE, TRUE, 0); gtk_widget_show_all (search_bar->viewoption_box); gtk_box_pack_start (GTK_BOX(search_bar), search_bar->viewoption_box, FALSE, FALSE, 0); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX(search_bar), hbox, FALSE, FALSE, 0); /* Search entry */ hbox = gtk_hbox_new (FALSE, 0); /* To Translators: The "Show: " label is followed by the Quick Search Text input field where one enters the term to search for */ label = gtk_label_new_with_mnemonic (_("Sear_ch: ")); gtk_widget_show (label); gtk_box_pack_start (GTK_BOX(hbox), label, FALSE, FALSE, 0); gtk_box_pack_start (GTK_BOX(hbox), search_bar->entry_box, FALSE, FALSE, 0); gtk_widget_show (search_bar->entry_box); gtk_label_set_mnemonic_widget ((GtkLabel *)label, search_bar->entry); /* Search Scope Widgets */ search_bar->scopeoption_box = gtk_hbox_new (0, FALSE); gtk_box_set_spacing (GTK_BOX (search_bar->scopeoption_box), 3); /* To Translators: The " in " label is part of the Quick Search Bar, example: Search: | | in | Current Folder/All Accounts/Current Account */ label = gtk_label_new_with_mnemonic (_(" i_n ")); gtk_widget_show (label); gtk_box_pack_start (GTK_BOX(search_bar->scopeoption_box), label, FALSE, FALSE, 0); search_bar->scopeoption = gtk_option_menu_new (); /* g_signal_connect (GTK_OPTION_MENU (search_bar->scopeoption), "changed", scopeoption_changed_cb, search_bar); */ gtk_box_pack_start (GTK_BOX(search_bar->scopeoption_box), search_bar->scopeoption, FALSE, FALSE, 0); gtk_widget_show_all (search_bar->scopeoption_box); gtk_widget_hide (hbox); gtk_label_set_mnemonic_widget ((GtkLabel *)label, search_bar->scopeoption); gtk_box_pack_end (GTK_BOX(hbox), search_bar->scopeoption_box, FALSE, FALSE, 0); gtk_widget_hide (search_bar->scopeoption_box); gtk_box_pack_end (GTK_BOX(search_bar), hbox, FALSE, FALSE, 0); gtk_widget_show (hbox); /* Set the menu */ e_search_bar_set_menu (search_bar, menu_items); e_search_bar_set_option (search_bar, option_items); /* * If the default choice for the option menu has subitems, then we need to * activate the search immediately. However, the developer won't have * connected to the activated signal until after the object is constructed, * so we can't emit here. Thus we launch a one-shot idle function that will * emit the changed signal, so that the proper callback will get invoked. */ search_bar->pending_activate = g_idle_add (idle_activate_hack, search_bar); } void e_search_bar_set_menu (ESearchBar *search_bar, ESearchBarItem *menu_items) { g_return_if_fail (search_bar != NULL); g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); ((ESearchBarClass *) GTK_OBJECT_GET_CLASS (search_bar))->set_menu (search_bar, menu_items); } void e_search_bar_add_menu (ESearchBar *search_bar, ESearchBarItem *menu_item) { g_return_if_fail (search_bar != NULL); g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); set_menu (search_bar, menu_item); } void e_search_bar_set_option (ESearchBar *search_bar, ESearchBarItem *option_items) { g_return_if_fail (search_bar != NULL); g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); g_return_if_fail (option_items != NULL); ((ESearchBarClass *) GTK_OBJECT_GET_CLASS (search_bar))->set_option (search_bar, option_items); } void e_search_bar_set_viewoption_menufunc (ESearchBar *search_bar, ESearchBarMenuFunc *menu_gen_func, void *data) { g_signal_connect (search_bar->viewoption, "button_press_event", G_CALLBACK (menu_gen_func), data); } /** * e_search_bar_set_viewoption_menu: * @search_bar: A search bar. * @option_id: Identifier of the main option menu item under which the subitems * are to be set. * @subitems: Array of subitem information. * * Sets the items for the secondary option menu of a search bar. **/ void e_search_bar_set_viewoption_menu (ESearchBar *search_bar, GtkWidget *menu) { if (search_bar->viewoption_menu != NULL) gtk_option_menu_remove_menu (GTK_OPTION_MENU (search_bar->viewoption)); search_bar->viewoption_menu = menu; gtk_option_menu_set_menu (GTK_OPTION_MENU (search_bar->viewoption), search_bar->viewoption_menu); g_signal_connect (search_bar->viewoption_menu, "selection-done", G_CALLBACK (viewitem_activated_cb), search_bar); } GtkWidget * e_search_bar_get_selected_viewitem (ESearchBar *search_bar) { GtkWidget *widget = NULL; widget = gtk_menu_get_active (GTK_MENU (search_bar->viewoption_menu)); return widget; } /** * e_search_bar_set_viewoption: * @search_bar: A search bar. * @option_id: Identifier of the main option menu item under which the subitems * are to be set. * @subitems: Array of subitem information. * * Sets the items for the secondary option menu of a search bar. **/ void e_search_bar_set_viewoption (ESearchBar *search_bar, int option_id, ESearchBarItem *subitems) { GtkWidget *menu; GtkWidget *menu_item; gint i; /* Create the menu if it is not there. right scenario ????*/ if (search_bar->viewoption_menu == NULL) { search_bar->viewoption_menu = menu = gtk_menu_new (); } else { gtk_option_menu_remove_menu (GTK_OPTION_MENU (search_bar->viewoption)); search_bar->viewoption_menu = menu = gtk_menu_new (); } /* Create the items */ for (i = 0; subitems[i].id != -1; ++i) { if (subitems[i].text) { char *str = NULL; str = string_without_underscores (subitems[i].text); menu_item = gtk_menu_item_new_with_label (str); g_free (str); } else { menu_item = gtk_menu_item_new (); gtk_widget_set_sensitive (menu_item, FALSE); } g_object_set_data (G_OBJECT (menu_item), "EsbItemId", GINT_TO_POINTER (subitems[i].id)); g_signal_connect (menu_item, "activate", G_CALLBACK (viewitem_activated_cb), search_bar); gtk_widget_show (menu_item); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menu_item); } gtk_option_menu_set_menu (GTK_OPTION_MENU (search_bar->viewoption), menu); } /** * e_search_bar_set_scopeoption: * @search_bar: A search bar. * are to be set. * @scopeitems: Array of scope information. * * Sets the items for the search scope option menu of a search bar. **/ void e_search_bar_set_scopeoption (ESearchBar *search_bar, ESearchBarItem *scopeitems) { GtkWidget *menu; GtkWidget *menu_item; gint i; gtk_widget_show (search_bar->scopeoption_box); if (search_bar->scopeoption_menu != NULL) { gtk_option_menu_remove_menu (GTK_OPTION_MENU (search_bar->scopeoption)); } search_bar->scopeoption_menu = menu = gtk_menu_new (); /* Generate items */ for (i = 0; scopeitems[i].id != -1; ++i) { if (scopeitems[i].text) { char *str; str = string_without_underscores (_(scopeitems[i].text)); menu_item = gtk_menu_item_new_with_label (str); g_object_set_data_full (G_OBJECT (menu_item), "string",str, g_free); } else { menu_item = gtk_menu_item_new (); gtk_widget_set_sensitive (menu_item, FALSE); } g_object_set_data (G_OBJECT (menu_item), "EsbItemId", GINT_TO_POINTER (scopeitems[i].id)); g_signal_connect (menu_item, "activate", G_CALLBACK (scopeitem_activated_cb), search_bar); gtk_widget_show (menu_item); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menu_item); } gtk_option_menu_set_menu (GTK_OPTION_MENU (search_bar->scopeoption), menu); } /** * e_search_bar_set_scopeoption_menu: * @search_bar: A search bar. * @menu: the scope option menu * * Sets the items for the secondary option menu of a search bar. **/ void e_search_bar_set_scopeoption_menu (ESearchBar *search_bar, GtkMenu *menu) { if (search_bar->scopeoption_menu != NULL) gtk_option_menu_remove_menu (GTK_OPTION_MENU (search_bar->scopeoption)); search_bar->scopeoption_menu = GTK_WIDGET (menu); gtk_option_menu_set_menu (GTK_OPTION_MENU (search_bar->scopeoption), search_bar->scopeoption_menu); g_signal_connect (search_bar->scopeoption_menu, "selection-done", G_CALLBACK (scopeitem_activated_cb), search_bar); } GtkWidget * e_search_bar_new (ESearchBarItem *menu_items, ESearchBarItem *option_items) { GtkWidget *widget; g_return_val_if_fail (option_items != NULL, NULL); widget = g_object_new (e_search_bar_get_type (), NULL); e_search_bar_construct (E_SEARCH_BAR (widget), menu_items, option_items); return widget; } void e_search_bar_set_ui_component (ESearchBar *search_bar, BonoboUIComponent *ui_component) { g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); if (search_bar->ui_component != NULL) { remove_bonobo_menus (search_bar); bonobo_object_unref (BONOBO_OBJECT (search_bar->ui_component)); } search_bar->ui_component = ui_component; if (ui_component != NULL) { bonobo_object_ref (BONOBO_OBJECT (ui_component)); setup_standard_verbs (search_bar); setup_bonobo_menus (search_bar); } } void e_search_bar_set_menu_sensitive (ESearchBar *search_bar, int id, gboolean state) { char *verb_name; char *path; verb_name = verb_name_from_id (id); path = g_strconcat ("/commands/", verb_name, NULL); g_free (verb_name); bonobo_ui_component_set_prop (search_bar->ui_component, path, "sensitive", state ? "1" : "0", NULL); g_free (path); } GType e_search_bar_get_type (void) { static GType type = 0; if (!type) { static const GTypeInfo info = { sizeof (ESearchBarClass), NULL, /* base_init */ NULL, /* base_finalize */ (GClassInitFunc) class_init, NULL, /* class_finalize */ NULL, /* class_data */ sizeof (ESearchBar), 0, /* n_preallocs */ (GInstanceInitFunc) init, }; type = g_type_register_static (gtk_hbox_get_type (), "ESearchBar", &info, 0); } return type; } void e_search_bar_set_viewitem_id (ESearchBar *search_bar, int id) { int row; g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); row = find_id (search_bar->viewoption_menu, id, "EsbItemId", NULL); if (row == -1) return; search_bar->viewitem_id = id; gtk_option_menu_set_history (GTK_OPTION_MENU (search_bar->viewoption), row); emit_query_changed (search_bar); } /** * e_search_bar_set_item_id: * @search_bar: A search bar. * @id: Identifier of the item to set. * * Sets the active item in the options menu of a search bar. **/ void e_search_bar_set_item_id (ESearchBar *search_bar, int id) { int row; g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); row = find_id (search_bar->option_menu, id, "EsbItemId", NULL); if (row == -1) return; if (id>=0) search_bar->last_search_option = id; search_bar->item_id = id; gtk_menu_set_active ((GtkMenu *)search_bar->option_menu, row); if (!search_bar->block_search) emit_query_changed (search_bar); update_clear_menuitem_sensitive (search_bar); } void e_search_bar_set_item_menu (ESearchBar *search_bar, int id) { int row; GtkWidget *item; g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); row = find_id (search_bar->option_menu, id, "EsbItemId", &item); if (row == -1) return; gtk_menu_set_active ((GtkMenu *)search_bar->option_menu, row); if (id>=0) gtk_check_menu_item_set_active ((GtkCheckMenuItem *)item, TRUE); } /** * e_search_bar_set_search_scope: * @search_bar: A search bar. * @id: Identifier of the item to set. * * Sets the active item in the options menu of a search bar. **/ void e_search_bar_set_search_scope (ESearchBar *search_bar, int id) { int row; g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); row = find_id (search_bar->scopeoption_menu, id, "EsbItemId", NULL); if (row == -1) return; search_bar->scopeitem_id = id; gtk_option_menu_set_history (GTK_OPTION_MENU (search_bar->scopeoption), row); if (!search_bar->block_search) emit_query_changed (search_bar); } /** * e_search_bar_get_item_id: * @search_bar: A search bar. * * Queries the currently selected item in the options menu of a search bar. * * Return value: Identifier of the selected item in the options menu. **/ int e_search_bar_get_item_id (ESearchBar *search_bar) { GtkWidget *menu_item; gint item_id; g_return_val_if_fail (search_bar != NULL, -1); g_return_val_if_fail (E_IS_SEARCH_BAR (search_bar), -1); menu_item = gtk_menu_get_active (GTK_MENU (search_bar->option_menu)); item_id = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (menu_item), "EsbItemId")); search_bar->item_id = item_id; return search_bar->item_id; } /** * e_search_bar_get_search_scope: * @search_bar: A search bar. * * Queries the currently selected search type in the options menu of a search bar. * * Return value: Identifier of the selected item in the options menu. **/ int e_search_bar_get_search_scope (ESearchBar *search_bar) { GtkWidget *menu_item; gint scopeitem_id; g_return_val_if_fail (search_bar != NULL, -1); g_return_val_if_fail (E_IS_SEARCH_BAR (search_bar), -1); menu_item = gtk_menu_get_active (GTK_MENU (search_bar->scopeoption_menu)); scopeitem_id = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (menu_item), "EsbItemId")); search_bar->scopeitem_id = scopeitem_id; return search_bar->scopeitem_id; } /** * e_search_bar_get_viewitem_id: * @search_bar: A search bar. * * Queries the currently selected item in the viewoptions menu of a search bar. * * Return value: Identifier of the selected item in the viewoptions menu. * If the search bar currently contains an entry rather than a a viewoption menu, * a value less than zero is returned. **/ int e_search_bar_get_viewitem_id (ESearchBar *search_bar) { GtkWidget *menu_item; gint viewitem_id; g_return_val_if_fail (search_bar != NULL, -1); g_return_val_if_fail (E_IS_SEARCH_BAR (search_bar), -1); menu_item = gtk_menu_get_active (GTK_MENU (search_bar->viewoption_menu)); viewitem_id = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (menu_item), "EsbItemId")); search_bar->viewitem_id = viewitem_id; return search_bar->viewitem_id; } /** * e_search_bar_set_ids: * @search_bar: A search bar. * @item_id: Identifier of the item to set. * @subitem_id: Identifier of the subitem to set. * * Sets the item and subitem ids for a search bar. This is intended to switch * to an item that has subitems. **/ void e_search_bar_set_ids (ESearchBar *search_bar, int item_id, int subitem_id) { int item_row; GtkWidget *item_widget; g_return_if_fail (search_bar != NULL); g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); item_row = find_id (search_bar->option_menu, item_id, "EsbChoiceId", &item_widget); if (item_row == -1 || !item_widget) return; search_bar->item_id = item_id; gtk_option_menu_set_history (GTK_OPTION_MENU (search_bar->option), item_row); } /** * e_search_bar_set_text: * @search_bar: A search bar. * @text: Text to set in the search bar's entry line. * * Sets the text string inside the entry line of a search bar. **/ void e_search_bar_set_text (ESearchBar *search_bar, const char *text) { g_return_if_fail (E_IS_SEARCH_BAR (search_bar)); gtk_entry_set_text (GTK_ENTRY (search_bar->entry), text); } /** * e_search_bar_get_text: * @search_bar: A search bar. * * Queries the text of the entry line in a search bar. * * Return value: The text string that is in the entry line of the search bar. * This must be freed using g_free(). If a suboption menu is active instead * of an entry, NULL is returned. **/ char * e_search_bar_get_text (ESearchBar *search_bar) { GtkStyle *entry_style, *default_style; g_return_val_if_fail (search_bar != NULL, NULL); g_return_val_if_fail (E_IS_SEARCH_BAR (search_bar), NULL); entry_style = gtk_widget_get_style (search_bar->entry); default_style = gtk_widget_get_default_style (); if (gdk_color_equal (&(entry_style->text[GTK_STATE_NORMAL]), &(default_style->text[GTK_STATE_INSENSITIVE]))) return g_strdup (""); return g_strdup (gtk_entry_get_text (GTK_ENTRY (search_bar->entry))); } void e_search_bar_scope_enable (ESearchBar *esb, int did, gboolean state) { GtkWidget *widget=NULL; GList *l = GTK_MENU_SHELL (esb->scopeoption_menu)->children; int row = -1, i = 0, id; while (l) { id = GPOINTER_TO_INT (g_object_get_data (l->data, "EsbItemId")); if (id == did) { row = i; widget = l->data; break; } i++; l = l->next; } if (widget) gtk_widget_set_sensitive (widget, state); }