/* LIBGIMP - The GIMP Library * Copyright (C) 1995-1997 Peter Mattis and Spencer Kimball * * gimpbrowser.c * Copyright (C) 2005 Michael Natterer * * This library is free software: you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . */ #include "config.h" #include #include #include #include "gimpwidgetstypes.h" #include "gimpwidgets.h" #include "gimpwidgetsmarshal.h" #include "libgimp/libgimp-intl.h" /** * SECTION: gimpbrowser * @title: GimpBrowser * @short_description: A base class for a documentation browser. * * A base class for a documentation browser. **/ enum { SEARCH, LAST_SIGNAL }; struct _GimpBrowser { GtkPaned parent_instance; GtkWidget *left_vbox; GtkWidget *search_entry; guint search_timeout_id; GtkWidget *search_type_combo; gint search_type; GtkWidget *count_label; GtkWidget *right_vbox; GtkWidget *right_widget; }; static void gimp_browser_dispose (GObject *object); static void gimp_browser_combo_changed (GtkComboBox *combo, GimpBrowser *browser); static void gimp_browser_entry_changed (GtkEntry *entry, GimpBrowser *browser); static void gimp_browser_entry_icon_press (GtkEntry *entry, GtkEntryIconPosition icon_pos, GdkEvent *event, GimpBrowser *browser); static gboolean gimp_browser_search_timeout (gpointer data); G_DEFINE_TYPE (GimpBrowser, gimp_browser, GTK_TYPE_PANED) #define parent_class gimp_browser_parent_class static guint browser_signals[LAST_SIGNAL] = { 0 }; static void gimp_browser_class_init (GimpBrowserClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); browser_signals[SEARCH] = g_signal_new ("search", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, 0, NULL, NULL, _gimp_widgets_marshal_VOID__STRING_INT, G_TYPE_NONE, 2, G_TYPE_STRING, G_TYPE_INT); object_class->dispose = gimp_browser_dispose; } static void gimp_browser_init (GimpBrowser *browser) { GtkWidget *hbox; GtkWidget *label; GtkWidget *scrolled_window; GtkWidget *viewport; gtk_orientable_set_orientation (GTK_ORIENTABLE (browser), GTK_ORIENTATION_HORIZONTAL); browser->search_type = -1; browser->left_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 6); gtk_paned_pack1 (GTK_PANED (browser), browser->left_vbox, TRUE, FALSE); gtk_widget_show (browser->left_vbox); /* search entry */ hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6); gtk_box_pack_start (GTK_BOX (browser->left_vbox), hbox, FALSE, FALSE, 0); gtk_widget_show (hbox); label = gtk_label_new_with_mnemonic (_("_Search:")); gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0); gtk_widget_show (label); browser->search_entry = gtk_entry_new (); gtk_box_pack_start (GTK_BOX (hbox), browser->search_entry, TRUE, TRUE, 0); gtk_widget_show (browser->search_entry); gtk_label_set_mnemonic_widget (GTK_LABEL (label), browser->search_entry); g_signal_connect (browser->search_entry, "changed", G_CALLBACK (gimp_browser_entry_changed), browser); gtk_entry_set_icon_from_icon_name (GTK_ENTRY (browser->search_entry), GTK_ENTRY_ICON_SECONDARY, "edit-clear"); gtk_entry_set_icon_activatable (GTK_ENTRY (browser->search_entry), GTK_ENTRY_ICON_SECONDARY, TRUE); gtk_entry_set_icon_sensitive (GTK_ENTRY (browser->search_entry), GTK_ENTRY_ICON_SECONDARY, FALSE); g_signal_connect (browser->search_entry, "icon-press", G_CALLBACK (gimp_browser_entry_icon_press), browser); /* count label */ browser->count_label = gtk_label_new (_("No matches")); gtk_label_set_xalign (GTK_LABEL (browser->count_label), 0.0); gimp_label_set_attributes (GTK_LABEL (browser->count_label), PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC, -1); gtk_box_pack_end (GTK_BOX (browser->left_vbox), browser->count_label, FALSE, FALSE, 0); gtk_widget_show (browser->count_label); /* scrolled window */ scrolled_window = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_paned_pack2 (GTK_PANED (browser), scrolled_window, TRUE, FALSE); gtk_widget_show (scrolled_window); viewport = gtk_viewport_new (NULL, NULL); gtk_container_add (GTK_CONTAINER (scrolled_window), viewport); gtk_widget_show (viewport); browser->right_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); gtk_container_set_border_width (GTK_CONTAINER (browser->right_vbox), 12); gtk_container_add (GTK_CONTAINER (viewport), browser->right_vbox); gtk_widget_show (browser->right_vbox); gtk_widget_grab_focus (browser->search_entry); } static void gimp_browser_dispose (GObject *object) { GimpBrowser *browser = GIMP_BROWSER (object); if (browser->search_timeout_id) { g_source_remove (browser->search_timeout_id); browser->search_timeout_id = 0; } G_OBJECT_CLASS (parent_class)->dispose (object); } /* public functions */ /** * gimp_browser_new: * * Create a new #GimpBrowser widget. * * Returns: a newly created #GimpBrowser. * * Since: 2.4 **/ GtkWidget * gimp_browser_new (void) { return g_object_new (GIMP_TYPE_BROWSER, NULL); } /** * gimp_browser_add_search_types: (skip) * @browser: a #GimpBrowser widget * @first_type_label: the label of the first search type * @first_type_id: an integer that identifies the first search type * @...: a %NULL-terminated list of more labels and ids. * * Populates the #GtkComboBox with search types. * * Since: 2.4 **/ void gimp_browser_add_search_types (GimpBrowser *browser, const gchar *first_type_label, gint first_type_id, ...) { g_return_if_fail (GIMP_IS_BROWSER (browser)); g_return_if_fail (first_type_label != NULL); if (! browser->search_type_combo) { GtkWidget *combo; va_list args; va_start (args, first_type_id); combo = gimp_int_combo_box_new_valist (first_type_label, first_type_id, args); va_end (args); gtk_widget_set_focus_on_click (combo, FALSE); browser->search_type_combo = combo; browser->search_type = first_type_id; gtk_box_pack_end (GTK_BOX (gtk_widget_get_parent (browser->search_entry)), combo, FALSE, FALSE, 0); gtk_widget_show (combo); gimp_int_combo_box_connect (GIMP_INT_COMBO_BOX (combo), browser->search_type, G_CALLBACK (gimp_int_combo_box_get_active), &browser->search_type, NULL); g_signal_connect (combo, "changed", G_CALLBACK (gimp_browser_combo_changed), browser); } else { gimp_int_combo_box_append (GIMP_INT_COMBO_BOX (browser->search_type_combo), first_type_label, first_type_id, NULL); } } /** * gimp_browser_get_left_vbox: * @browser: a #GimpBrowser widget * * Returns: (transfer none) (type GtkBox): The left vbox. * * Since: 3.0 **/ GtkWidget * gimp_browser_get_left_vbox (GimpBrowser *browser) { g_return_val_if_fail (GIMP_IS_BROWSER (browser), NULL); return browser->left_vbox; } /** * gimp_browser_get_right_vbox: * @browser: a #GimpBrowser widget * * Returns: (transfer none) (type GtkBox): The right vbox. * * Since: 3.0 **/ GtkWidget * gimp_browser_get_right_vbox (GimpBrowser *browser) { g_return_val_if_fail (GIMP_IS_BROWSER (browser), NULL); return browser->right_vbox; } /** * gimp_browser_set_search_summary: * @browser: a #GimpBrowser widget * @summary: a string describing the search result * * Sets the search summary text. * * Since: 3.0 **/ void gimp_browser_set_search_summary (GimpBrowser *browser, const gchar *summary) { g_return_if_fail (GIMP_IS_BROWSER (browser)); g_return_if_fail (summary != NULL); gtk_label_set_text (GTK_LABEL (browser->count_label), summary); } /** * gimp_browser_set_widget: * @browser: a #GimpBrowser widget * @widget: a #GtkWidget * * Sets the widget to appear on the right side of the @browser. * * Since: 2.4 **/ void gimp_browser_set_widget (GimpBrowser *browser, GtkWidget *widget) { g_return_if_fail (GIMP_IS_BROWSER (browser)); g_return_if_fail (widget == NULL || GTK_IS_WIDGET (widget)); if (widget == browser->right_widget) return; if (browser->right_widget) gtk_container_remove (GTK_CONTAINER (browser->right_vbox), browser->right_widget); browser->right_widget = widget; if (widget) { gtk_box_pack_start (GTK_BOX (browser->right_vbox), widget, FALSE, FALSE, 0); gtk_widget_show (widget); } } /** * gimp_browser_show_message: * @browser: a #GimpBrowser widget * @message: text message * * Displays @message in the right side of the @browser. Unless the right * side already contains a #GtkLabel, the widget previously added with * gimp_browser_set_widget() is removed and replaced by a #GtkLabel. * * Since: 2.4 **/ void gimp_browser_show_message (GimpBrowser *browser, const gchar *message) { g_return_if_fail (GIMP_IS_BROWSER (browser)); g_return_if_fail (message != NULL); if (GTK_IS_LABEL (browser->right_widget)) { gtk_label_set_text (GTK_LABEL (browser->right_widget), message); } else { GtkWidget *label = gtk_label_new (message); gimp_label_set_attributes (GTK_LABEL (label), PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC, -1); gimp_browser_set_widget (browser, label); } while (gtk_events_pending ()) gtk_main_iteration (); } /* private functions */ static void gimp_browser_queue_search (GimpBrowser *browser) { if (browser->search_timeout_id) g_source_remove (browser->search_timeout_id); browser->search_timeout_id = g_timeout_add (100, gimp_browser_search_timeout, browser); } static void gimp_browser_combo_changed (GtkComboBox *combo, GimpBrowser *browser) { gimp_browser_queue_search (browser); } static void gimp_browser_entry_changed (GtkEntry *entry, GimpBrowser *browser) { gimp_browser_queue_search (browser); gtk_entry_set_icon_sensitive (entry, GTK_ENTRY_ICON_SECONDARY, gtk_entry_get_text_length (entry) > 0); } static void gimp_browser_entry_icon_press (GtkEntry *entry, GtkEntryIconPosition icon_pos, GdkEvent *event, GimpBrowser *browser) { GdkEventButton *bevent = (GdkEventButton *) event; if (icon_pos == GTK_ENTRY_ICON_SECONDARY && bevent->button == 1) { gtk_entry_set_text (entry, ""); } } static gboolean gimp_browser_search_timeout (gpointer data) { GimpBrowser *browser = GIMP_BROWSER (data); const gchar *search_string; search_string = gtk_entry_get_text (GTK_ENTRY (browser->search_entry)); if (! search_string) search_string = ""; g_signal_emit (data, browser_signals[SEARCH], 0, search_string, browser->search_type); browser->search_timeout_id = 0; return FALSE; }