/* GTK - The GIMP Toolkit * Copyright (C) 2012 Red Hat, Inc. * * Authors: * - Bastien Nocera * * 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 2 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 . */ /* * Modified by the GTK+ Team and others 2012. See the AUTHORS * file for a list of people on the GTK+ Team. See the ChangeLog * files for a list of changes. These files are distributed with * GTK+ at ftp://ftp.gtk.org/pub/gtk/. */ #include "config.h" #include "gtksearchentryprivate.h" #include "gtkaccessible.h" #include "gtkbindings.h" #include "gtkentryprivate.h" #include "gtkintl.h" #include "gtkmarshalers.h" #include "gtkstylecontext.h" #include "gtkeventcontrollerkey.h" /** * SECTION:gtksearchentry * @Short_description: An entry which shows a search icon * @Title: GtkSearchEntry * * #GtkSearchEntry is a subclass of #GtkEntry that has been * tailored for use as a search entry. * * It will show an inactive symbolic “find” icon when the search * entry is empty, and a symbolic “clear” icon when there is text. * Clicking on the “clear” icon will empty the search entry. * * Note that the search/clear icon is shown using a secondary * icon, and thus does not work if you are using the secondary * icon position for some other purpose. * * To make filtering appear more reactive, it is a good idea to * not react to every change in the entry text immediately, but * only after a short delay. To support this, #GtkSearchEntry * emits the #GtkSearchEntry::search-changed signal which can * be used instead of the #GtkEditable::changed signal. * * The #GtkSearchEntry::previous-match, #GtkSearchEntry::next-match * and #GtkSearchEntry::stop-search signals can be used to implement * moving between search results and ending the search. * * Often, GtkSearchEntry will be fed events by means of being * placed inside a #GtkSearchBar. If that is not the case, * you can use gtk_search_entry_set_key_capture_widget() to let it * capture key input from another widget. */ enum { SEARCH_CHANGED, NEXT_MATCH, PREVIOUS_MATCH, STOP_SEARCH, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; typedef struct { GtkWidget *capture_widget; GtkEventController *capture_widget_controller; guint delayed_changed_id; gboolean content_changed; gboolean search_stopped; } GtkSearchEntryPrivate; static void gtk_search_entry_icon_release (GtkEntry *entry, GtkEntryIconPosition icon_pos); static void gtk_search_entry_changed (GtkEditable *editable); static void gtk_search_entry_editable_init (GtkEditableInterface *iface); static GtkEditableInterface *parent_editable_iface; G_DEFINE_TYPE_WITH_CODE (GtkSearchEntry, gtk_search_entry, GTK_TYPE_ENTRY, G_ADD_PRIVATE (GtkSearchEntry) G_IMPLEMENT_INTERFACE (GTK_TYPE_EDITABLE, gtk_search_entry_editable_init)) /* 150 mseconds of delay */ #define DELAYED_TIMEOUT_ID 150 /* This widget got created without a private structure, meaning * that we cannot now have one without breaking ABI */ #define GET_PRIV(e) ((GtkSearchEntryPrivate *) gtk_search_entry_get_instance_private ((GtkSearchEntry *) (e))) static void gtk_search_entry_preedit_changed (GtkEntry *entry, const gchar *preedit) { GtkSearchEntryPrivate *priv = GET_PRIV (entry); priv->content_changed = TRUE; } static void gtk_search_entry_notify (GObject *object, GParamSpec *pspec) { GtkSearchEntryPrivate *priv = GET_PRIV (object); if (strcmp (pspec->name, "text") == 0) priv->content_changed = TRUE; if (G_OBJECT_CLASS (gtk_search_entry_parent_class)->notify) G_OBJECT_CLASS (gtk_search_entry_parent_class)->notify (object, pspec); } static void gtk_search_entry_finalize (GObject *object) { GtkSearchEntryPrivate *priv = GET_PRIV (object); if (priv->delayed_changed_id > 0) g_source_remove (priv->delayed_changed_id); gtk_search_entry_set_key_capture_widget (GTK_SEARCH_ENTRY (object), NULL); G_OBJECT_CLASS (gtk_search_entry_parent_class)->finalize (object); } static void gtk_search_entry_stop_search (GtkSearchEntry *entry) { GtkSearchEntryPrivate *priv = GET_PRIV (entry); priv->search_stopped = TRUE; } static void gtk_search_entry_class_init (GtkSearchEntryClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkBindingSet *binding_set; object_class->finalize = gtk_search_entry_finalize; object_class->notify = gtk_search_entry_notify; klass->stop_search = gtk_search_entry_stop_search; g_signal_override_class_handler ("icon-release", GTK_TYPE_SEARCH_ENTRY, G_CALLBACK (gtk_search_entry_icon_release)); /** * GtkSearchEntry::search-changed: * @entry: the entry on which the signal was emitted * * The #GtkSearchEntry::search-changed signal is emitted with a short * delay of 150 milliseconds after the last change to the entry text. */ signals[SEARCH_CHANGED] = g_signal_new (I_("search-changed"), G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GtkSearchEntryClass, search_changed), NULL, NULL, NULL, G_TYPE_NONE, 0); /** * GtkSearchEntry::next-match: * @entry: the entry on which the signal was emitted * * The ::next-match signal is a [keybinding signal][GtkBindingSignal] * which gets emitted when the user initiates a move to the next match * for the current search string. * * Applications should connect to it, to implement moving between * matches. * * The default bindings for this signal is Ctrl-g. */ signals[NEXT_MATCH] = g_signal_new (I_("next-match"), G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (GtkSearchEntryClass, next_match), NULL, NULL, NULL, G_TYPE_NONE, 0); /** * GtkSearchEntry::previous-match: * @entry: the entry on which the signal was emitted * * The ::previous-match signal is a [keybinding signal][GtkBindingSignal] * which gets emitted when the user initiates a move to the previous match * for the current search string. * * Applications should connect to it, to implement moving between * matches. * * The default bindings for this signal is Ctrl-Shift-g. */ signals[PREVIOUS_MATCH] = g_signal_new (I_("previous-match"), G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (GtkSearchEntryClass, previous_match), NULL, NULL, NULL, G_TYPE_NONE, 0); /** * GtkSearchEntry::stop-search: * @entry: the entry on which the signal was emitted * * The ::stop-search signal is a [keybinding signal][GtkBindingSignal] * which gets emitted when the user stops a search via keyboard input. * * Applications should connect to it, to implement hiding the search * entry in this case. * * The default bindings for this signal is Escape. */ signals[STOP_SEARCH] = g_signal_new (I_("stop-search"), G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (GtkSearchEntryClass, stop_search), NULL, NULL, NULL, G_TYPE_NONE, 0); binding_set = gtk_binding_set_by_class (klass); gtk_binding_entry_add_signal (binding_set, GDK_KEY_g, GDK_CONTROL_MASK, "next-match", 0); gtk_binding_entry_add_signal (binding_set, GDK_KEY_g, GDK_SHIFT_MASK | GDK_CONTROL_MASK, "previous-match", 0); gtk_binding_entry_add_signal (binding_set, GDK_KEY_Escape, 0, "stop-search", 0); } static void gtk_search_entry_editable_init (GtkEditableInterface *iface) { parent_editable_iface = g_type_interface_peek_parent (iface); iface->do_insert_text = parent_editable_iface->do_insert_text; iface->do_delete_text = parent_editable_iface->do_delete_text; iface->insert_text = parent_editable_iface->insert_text; iface->delete_text = parent_editable_iface->delete_text; iface->get_chars = parent_editable_iface->get_chars; iface->set_selection_bounds = parent_editable_iface->set_selection_bounds; iface->get_selection_bounds = parent_editable_iface->get_selection_bounds; iface->set_position = parent_editable_iface->set_position; iface->get_position = parent_editable_iface->get_position; iface->changed = gtk_search_entry_changed; } static void gtk_search_entry_icon_release (GtkEntry *entry, GtkEntryIconPosition icon_pos) { if (icon_pos == GTK_ENTRY_ICON_SECONDARY) gtk_entry_set_text (entry, ""); } static gboolean gtk_search_entry_changed_timeout_cb (gpointer user_data) { GtkSearchEntry *entry = user_data; GtkSearchEntryPrivate *priv = GET_PRIV (entry); g_signal_emit (entry, signals[SEARCH_CHANGED], 0); priv->delayed_changed_id = 0; return G_SOURCE_REMOVE; } static void reset_timeout (GtkSearchEntry *entry) { GtkSearchEntryPrivate *priv = GET_PRIV (entry); if (priv->delayed_changed_id > 0) g_source_remove (priv->delayed_changed_id); priv->delayed_changed_id = g_timeout_add (DELAYED_TIMEOUT_ID, gtk_search_entry_changed_timeout_cb, entry); g_source_set_name_by_id (priv->delayed_changed_id, "[gtk] gtk_search_entry_changed_timeout_cb"); } static void gtk_search_entry_changed (GtkEditable *editable) { GtkSearchEntry *entry = GTK_SEARCH_ENTRY (editable); GtkSearchEntryPrivate *priv = GET_PRIV (entry); const char *str, *icon_name; gboolean cleared; /* Update the icons first */ str = gtk_entry_get_text (GTK_ENTRY (entry)); if (str == NULL || *str == '\0') { icon_name = NULL; cleared = TRUE; } else { icon_name = "edit-clear-symbolic"; cleared = FALSE; } g_object_set (entry, "secondary-icon-name", icon_name, "secondary-icon-activatable", !cleared, "secondary-icon-sensitive", !cleared, NULL); if (cleared) { if (priv->delayed_changed_id > 0) { g_source_remove (priv->delayed_changed_id); priv->delayed_changed_id = 0; } g_signal_emit (entry, signals[SEARCH_CHANGED], 0); } else { /* Queue up the timeout */ reset_timeout (entry); } } static void gtk_search_entry_init (GtkSearchEntry *entry) { AtkObject *atk_obj; g_object_set (entry, "primary-icon-name", "edit-find-symbolic", "primary-icon-activatable", FALSE, "primary-icon-sensitive", FALSE, NULL); atk_obj = gtk_widget_get_accessible (GTK_WIDGET (entry)); if (GTK_IS_ACCESSIBLE (atk_obj)) atk_object_set_name (atk_obj, _("Search")); gtk_style_context_add_class (gtk_widget_get_style_context (GTK_WIDGET (entry)), I_("search")); g_signal_connect (gtk_entry_get_text_widget (GTK_ENTRY (entry)), "preedit-changed", G_CALLBACK (gtk_search_entry_preedit_changed), NULL); } /** * gtk_search_entry_new: * * Creates a #GtkSearchEntry, with a find icon when the search field is * empty, and a clear icon when it isn't. * * Returns: a new #GtkSearchEntry */ GtkWidget * gtk_search_entry_new (void) { return GTK_WIDGET (g_object_new (GTK_TYPE_SEARCH_ENTRY, NULL)); } gboolean gtk_search_entry_is_keynav (guint keyval, GdkModifierType state) { if (keyval == GDK_KEY_Tab || keyval == GDK_KEY_KP_Tab || keyval == GDK_KEY_Up || keyval == GDK_KEY_KP_Up || keyval == GDK_KEY_Down || keyval == GDK_KEY_KP_Down || keyval == GDK_KEY_Left || keyval == GDK_KEY_KP_Left || keyval == GDK_KEY_Right || keyval == GDK_KEY_KP_Right || keyval == GDK_KEY_Home || keyval == GDK_KEY_KP_Home || keyval == GDK_KEY_End || keyval == GDK_KEY_KP_End || keyval == GDK_KEY_Page_Up || keyval == GDK_KEY_KP_Page_Up || keyval == GDK_KEY_Page_Down || keyval == GDK_KEY_KP_Page_Down || ((state & (GDK_CONTROL_MASK | GDK_MOD1_MASK)) != 0)) return TRUE; /* Other navigation events should get automatically * ignored as they will not change the content of the entry */ return FALSE; } /** * gtk_search_entry_handle_event: * @entry: a #GtkSearchEntry * @event: a key event * * This function should be called when the top-level window * which contains the search entry received a key event. If * the entry is part of a #GtkSearchBar, it is preferable * to call gtk_search_bar_handle_event() instead, which will * reveal the entry in addition to passing the event to this * function. * * If the key event is handled by the search entry and starts * or continues a search, %GDK_EVENT_STOP will be returned. * The caller should ensure that the entry is shown in this * case, and not propagate the event further. * * Returns: %GDK_EVENT_STOP if the key press event resulted * in a search beginning or continuing, %GDK_EVENT_PROPAGATE * otherwise. */ gboolean gtk_search_entry_handle_event (GtkSearchEntry *entry, GdkEvent *event) { GtkSearchEntryPrivate *priv = GET_PRIV (entry); gboolean handled; guint keyval, state; if (!gtk_widget_get_realized (GTK_WIDGET (entry))) gtk_widget_realize (GTK_WIDGET (entry)); gdk_event_get_keyval (event, &keyval); gdk_event_get_state (event, &state); if (gtk_search_entry_is_keynav (keyval, state) || keyval == GDK_KEY_space || keyval == GDK_KEY_Menu) return GDK_EVENT_PROPAGATE; priv->content_changed = FALSE; priv->search_stopped = FALSE; handled = gtk_widget_event (GTK_WIDGET (entry), event); return handled && priv->content_changed && !priv->search_stopped ? GDK_EVENT_STOP : GDK_EVENT_PROPAGATE; } static gboolean capture_widget_key_handled (GtkEventControllerKey *controller, guint keyval, guint keycode, GdkModifierType state, GtkWidget *entry) { GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (GTK_SEARCH_ENTRY (entry)); gboolean handled; if (gtk_search_entry_is_keynav (keyval, state) || keyval == GDK_KEY_space || keyval == GDK_KEY_Menu) return FALSE; priv->content_changed = FALSE; priv->search_stopped = FALSE; handled = gtk_event_controller_key_forward (controller, entry); return handled && priv->content_changed && !priv->search_stopped ? GDK_EVENT_STOP : GDK_EVENT_PROPAGATE; } /** * gtk_search_entry_set_key_capture_widget: * @entry: a #GtkSearchEntry * @widget: (nullable) (transfer none): a #GtkWidget * * Sets @widget as the widget that @entry will capture key events from. * * Key events are consumed by the search entry to start or * continue a search. * * If the entry is part of a #GtkSearchBar, it is preferable * to call gtk_search_bar_set_key_capture_widget() instead, which * will reveal the entry in addition to triggering the search entry. **/ void gtk_search_entry_set_key_capture_widget (GtkSearchEntry *entry, GtkWidget *widget) { GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry); g_return_if_fail (GTK_IS_SEARCH_ENTRY (entry)); g_return_if_fail (!widget || GTK_IS_WIDGET (widget)); if (priv->capture_widget) { gtk_widget_remove_controller (priv->capture_widget, priv->capture_widget_controller); g_object_remove_weak_pointer (G_OBJECT (priv->capture_widget), (gpointer *) &priv->capture_widget); } priv->capture_widget = widget; if (widget) { g_object_add_weak_pointer (G_OBJECT (priv->capture_widget), (gpointer *) &priv->capture_widget); priv->capture_widget_controller = gtk_event_controller_key_new (); gtk_event_controller_set_propagation_phase (priv->capture_widget_controller, GTK_PHASE_CAPTURE); g_signal_connect (priv->capture_widget_controller, "key-pressed", G_CALLBACK (capture_widget_key_handled), entry); g_signal_connect (priv->capture_widget_controller, "key-released", G_CALLBACK (capture_widget_key_handled), entry); gtk_widget_add_controller (widget, priv->capture_widget_controller); } } /** * gtk_search_entry_get_key_capture_widget: * @entry: a #GtkSearchEntry * * Gets the widget that @entry is capturing key events from. * * Returns: (transfer none): The key capture widget. **/ GtkWidget * gtk_search_entry_get_key_capture_widget (GtkSearchEntry *entry) { GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry); g_return_val_if_fail (GTK_IS_SEARCH_ENTRY (entry), NULL); return priv->capture_widget; }