/* -*- 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 <clahey@helixcode.com>
*
* 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 <config.h>
#include <gnome.h>
#include "e-search-bar.h"
#include <gal/widgets/e-unicode.h>
#include <gal/widgets/e-gui-utils.h>
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;
}
}