/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * e-search-bar.c * Copyright (C) 2000 Helix Code, Inc. * Author: Chris Lahey * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * 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. */ #include #include #include "e-search-bar.h" #include #include static void e_search_bar_init (ESearchBar *card); static void e_search_bar_class_init (ESearchBarClass *klass); static void e_search_bar_set_arg (GtkObject *o, GtkArg *arg, guint arg_id); static void e_search_bar_get_arg (GtkObject *object, GtkArg *arg, guint arg_id); static void e_search_bar_destroy (GtkObject *object); enum { QUERY_CHANGED, MENU_ACTIVATED, LAST_SIGNAL }; static gint esb_signals [LAST_SIGNAL] = { 0, }; static GtkHBoxClass *parent_class = NULL; /* The arguments we take */ enum { ARG_0, ARG_OPTION_CHOICE, ARG_TEXT, }; GtkType e_search_bar_get_type (void) { static GtkType type = 0; if (!type) { static const GtkTypeInfo info = { "ESearchBar", sizeof (ESearchBar), sizeof (ESearchBarClass), (GtkClassInitFunc) e_search_bar_class_init, (GtkObjectInitFunc) e_search_bar_init, /* reserved_1 */ NULL, /* reserved_2 */ NULL, (GtkClassInitFunc) NULL, }; type = gtk_type_unique (gtk_hbox_get_type (), &info); } return type; } static void e_search_bar_class_init (ESearchBarClass *klass) { GtkObjectClass *object_class; object_class = GTK_OBJECT_CLASS(klass); parent_class = gtk_type_class (gtk_hbox_get_type ()); object_class->set_arg = e_search_bar_set_arg; object_class->get_arg = e_search_bar_get_arg; object_class->destroy = e_search_bar_destroy; gtk_object_add_arg_type ("ESearchBar::option_choice", GTK_TYPE_ENUM, GTK_ARG_READWRITE, ARG_OPTION_CHOICE); gtk_object_add_arg_type ("ESearchBar::text", GTK_TYPE_STRING, GTK_ARG_READWRITE, ARG_TEXT); esb_signals [QUERY_CHANGED] = gtk_signal_new ("query_changed", GTK_RUN_LAST, object_class->type, GTK_SIGNAL_OFFSET (ESearchBarClass, query_changed), gtk_marshal_NONE__NONE, GTK_TYPE_NONE, 0); esb_signals [MENU_ACTIVATED] = gtk_signal_new ("menu_activated", GTK_RUN_LAST, object_class->type, GTK_SIGNAL_OFFSET (ESearchBarClass, menu_activated), gtk_marshal_NONE__INT, GTK_TYPE_NONE, 1, GTK_TYPE_INT); gtk_object_class_add_signals (object_class, esb_signals, LAST_SIGNAL); } static void esb_query_changed(ESearchBar *esb) { gtk_signal_emit(GTK_OBJECT (esb), esb_signals [QUERY_CHANGED]); } static void esb_menu_activated(ESearchBar *esb, int item) { gtk_signal_emit(GTK_OBJECT (esb), esb_signals [MENU_ACTIVATED], item); } static void esb_menubar_activated(GtkWidget *widget, ESearchBar *esb) { int id = GPOINTER_TO_INT(gtk_object_get_data (GTK_OBJECT (widget), "EsbMenuId")); esb_menu_activated(esb, id); } static void esb_check_labels(GtkWidget *widget, gpointer data) { if (GTK_IS_LABEL(widget)) { char *text; gtk_object_get(GTK_OBJECT(widget), "label", &text, NULL); gtk_label_parse_uline(GTK_LABEL(widget), text); g_free(text); } } static void esb_pack_menubar(ESearchBar *esb, ESearchBarItem *items) { GtkWidget *menu, *menuitem; int i; menu = gtk_menu_new (); for (i = 0; items[i].id != -1; i++) { GtkWidget *item; if (items[i].text) item = gtk_menu_item_new_with_label (_(items[i].text)); else item = gtk_menu_item_new(); gtk_menu_append (GTK_MENU (menu), item); gtk_object_set_data (GTK_OBJECT (item), "EsbMenuId", GINT_TO_POINTER(items[i].id)); gtk_signal_connect (GTK_OBJECT (item), "activate", GTK_SIGNAL_FUNC (esb_menubar_activated), esb); } gtk_widget_show_all (menu); menuitem = gnome_stock_menu_item(GNOME_STOCK_MENU_SEARCH, _("Sear_ch")); e_container_foreach_leaf(GTK_CONTAINER(menuitem), esb_check_labels, NULL); gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), menu); gtk_widget_show (menuitem); gtk_menu_bar_append (GTK_MENU_BAR(esb->menubar), menuitem); gtk_widget_set_sensitive (esb->menubar, TRUE); } static void esb_option_activated(GtkWidget *widget, ESearchBar *esb) { int id = GPOINTER_TO_INT(gtk_object_get_data (GTK_OBJECT (widget), "EsbChoiceId")); esb->option_choice = id; esb_query_changed(esb); } static void esb_entry_activated(GtkWidget *widget, ESearchBar *esb) { esb_query_changed(esb); } static void esb_pack_option_menu(ESearchBar *esb, ESearchBarItem *items) { GtkWidget *menu; int i; menu = gtk_menu_new (); for (i = 0; items[i].id != -1; i++) { GtkWidget *item; if (items[i].text) item = gtk_menu_item_new_with_label (_(items[i].text)); else item = gtk_menu_item_new(); gtk_menu_append (GTK_MENU (menu), item); gtk_object_set_data (GTK_OBJECT (item), "EsbChoiceId", GINT_TO_POINTER(items[i].id)); gtk_signal_connect (GTK_OBJECT (item), "activate", GTK_SIGNAL_FUNC (esb_option_activated), esb); } gtk_widget_show_all (menu); gtk_option_menu_set_menu (GTK_OPTION_MENU (esb->option), menu); gtk_option_menu_set_history (GTK_OPTION_MENU (esb->option), 0); gtk_widget_set_sensitive (esb->option, TRUE); } static void e_search_bar_init (ESearchBar *esb) { GtkWidget *spacer; gtk_box_set_spacing(GTK_BOX(esb), 1); esb->menubar = gtk_menu_bar_new(); gtk_widget_show(esb->menubar); gtk_box_pack_start(GTK_BOX(esb), esb->menubar, FALSE, FALSE, 0); esb->option = gtk_option_menu_new(); gtk_widget_show(esb->option); gtk_box_pack_start(GTK_BOX(esb), esb->option, FALSE, FALSE, 0); esb->entry = gtk_entry_new(); gtk_signal_connect (GTK_OBJECT (esb->entry), "activate", GTK_SIGNAL_FUNC (esb_entry_activated), esb); gtk_widget_show(esb->entry); gtk_box_pack_start(GTK_BOX(esb), esb->entry, TRUE, TRUE, 0); esb->option_choice = 0; spacer = gtk_drawing_area_new(); gtk_widget_show(spacer); gtk_box_pack_start(GTK_BOX(esb), spacer, FALSE, FALSE, 0); gtk_widget_set_usize(spacer, 15, 1); } static void e_search_bar_destroy (GtkObject *object) { if (GTK_OBJECT_CLASS(parent_class)->destroy) GTK_OBJECT_CLASS(parent_class)->destroy(object); } GtkWidget * e_search_bar_new (ESearchBarItem *menu_items, ESearchBarItem *option_items) { GtkWidget *widget = GTK_WIDGET (gtk_type_new (e_search_bar_get_type ())); esb_pack_menubar(E_SEARCH_BAR(widget), menu_items); esb_pack_option_menu(E_SEARCH_BAR(widget), option_items); return widget; } static void e_search_bar_get_arg (GtkObject *object, GtkArg *arg, guint arg_id) { ESearchBar *esb = E_SEARCH_BAR(object); switch (arg_id) { case ARG_OPTION_CHOICE: GTK_VALUE_ENUM (*arg) = esb->option_choice; break; case ARG_TEXT: GTK_VALUE_STRING (*arg) = e_utf8_gtk_editable_get_text(GTK_EDITABLE(esb->entry)); break; default: arg->type = GTK_TYPE_INVALID; break; } } static void e_search_bar_set_arg (GtkObject *object, GtkArg *arg, guint arg_id) { ESearchBar *esb = E_SEARCH_BAR(object); switch (arg_id) { case ARG_OPTION_CHOICE: esb->option_choice = GTK_VALUE_ENUM (*arg); gtk_option_menu_set_history (GTK_OPTION_MENU (esb->option), esb->option_choice); esb_query_changed(esb); break; case ARG_TEXT: e_utf8_gtk_editable_set_text(GTK_EDITABLE(esb->entry), GTK_VALUE_STRING (*arg)); esb_query_changed(esb); break; default: break; } }