gtksearchbar/entry: Add [gs]et_key_capture_widget() API calls

This lets these widgets actively pull events from a widget, instead
of passively being fed events.
This commit is contained in:
Carlos Garnacho 2018-03-11 13:53:17 +01:00
parent ad5f0a63a3
commit 20c1e24b60
6 changed files with 295 additions and 47 deletions

View File

@ -2397,6 +2397,8 @@ gtk_search_bar_set_search_mode
gtk_search_bar_get_show_close_button gtk_search_bar_get_show_close_button
gtk_search_bar_set_show_close_button gtk_search_bar_set_show_close_button
gtk_search_bar_handle_event gtk_search_bar_handle_event
gtk_search_bar_set_key_capture_widget
gtk_search_bar_get_key_capture_widget
<SUBSECTION Standard> <SUBSECTION Standard>
GTK_TYPE_SEARCH_BAR GTK_TYPE_SEARCH_BAR
GTK_SEARCH_BAR GTK_SEARCH_BAR
@ -2414,6 +2416,8 @@ gtk_search_bar_get_type
GtkSearchEntry GtkSearchEntry
gtk_search_entry_new gtk_search_entry_new
gtk_search_entry_handle_event gtk_search_entry_handle_event
gtk_search_entry_set_key_capture_widget
gtk_search_entry_get_key_capture_widget
<SUBSECTION Standard> <SUBSECTION Standard>
GTK_TYPE_SEARCH_ENTRY GTK_TYPE_SEARCH_ENTRY
GTK_SEARCH_ENTRY GTK_SEARCH_ENTRY

View File

@ -37,6 +37,7 @@
#include "gtkrevealer.h" #include "gtkrevealer.h"
#include "gtksearchentryprivate.h" #include "gtksearchentryprivate.h"
#include "gtksnapshot.h" #include "gtksnapshot.h"
#include "gtkeventcontrollerkey.h"
/** /**
* SECTION:gtksearchbar * SECTION:gtksearchbar
@ -48,11 +49,12 @@
* built-in. The search bar would appear when a search is started through * built-in. The search bar would appear when a search is started through
* typing on the keyboard, or the applications search mode is toggled on. * typing on the keyboard, or the applications search mode is toggled on.
* *
* For keyboard presses to start a search, events will need to be * For keyboard presses to start a search, the search bar must be told
* forwarded from the top-level window that contains the search bar. * of a widget to capture key events from through
* See gtk_search_bar_handle_event() for example code. Common shortcuts * gtk_search_bar_set_key_capture_widget(). This widget will typically
* such as Ctrl+F should be handled as an application action, or through * be the top-level window, or a parent container of the search bar. Common
* the menu items. * shortcuts such as Ctrl+F should be handled as an application action, or
* through the menu items.
* *
* You will also need to tell the search bar about which entry you * You will also need to tell the search bar about which entry you
* are using as your search entry using gtk_search_bar_connect_entry(). * are using as your search entry using gtk_search_bar_connect_entry().
@ -86,6 +88,9 @@ typedef struct {
GtkWidget *entry; GtkWidget *entry;
gboolean reveal_child; gboolean reveal_child;
GtkWidget *capture_widget;
GtkEventController *capture_widget_controller;
} GtkSearchBarPrivate; } GtkSearchBarPrivate;
G_DEFINE_TYPE_WITH_PRIVATE (GtkSearchBar, gtk_search_bar, GTK_TYPE_BIN) G_DEFINE_TYPE_WITH_PRIVATE (GtkSearchBar, gtk_search_bar, GTK_TYPE_BIN)
@ -107,24 +112,6 @@ stop_search_cb (GtkWidget *entry,
gtk_revealer_set_reveal_child (GTK_REVEALER (priv->revealer), FALSE); gtk_revealer_set_reveal_child (GTK_REVEALER (priv->revealer), FALSE);
} }
static gboolean
entry_key_pressed_event_cb (GtkWidget *widget,
GdkEvent *event,
GtkSearchBar *bar)
{
guint keyval;
gdk_event_get_keyval (event, &keyval);
if (keyval == GDK_KEY_Escape)
{
stop_search_cb (widget, bar);
return GDK_EVENT_STOP;
}
else
return GDK_EVENT_PROPAGATE;
}
static void static void
preedit_changed_cb (GtkEntry *entry, preedit_changed_cb (GtkEntry *entry,
GtkWidget *popup, GtkWidget *popup,
@ -143,12 +130,12 @@ gtk_search_bar_handle_event_for_entry (GtkSearchBar *bar,
guint preedit_change_id; guint preedit_change_id;
gboolean res; gboolean res;
char *old_text, *new_text; char *old_text, *new_text;
guint keyval; guint keyval, state;
gdk_event_get_keyval (event, &keyval); gdk_event_get_keyval (event, &keyval);
gdk_event_get_state (event, &state);
if (gtk_search_entry_is_keynav (keyval, state) ||
if (gtk_search_entry_is_keynav_event (event) ||
keyval == GDK_KEY_space || keyval == GDK_KEY_space ||
keyval == GDK_KEY_Menu) keyval == GDK_KEY_Menu)
return GDK_EVENT_PROPAGATE; return GDK_EVENT_PROPAGATE;
@ -384,6 +371,7 @@ gtk_search_bar_dispose (GObject *object)
} }
gtk_search_bar_set_entry (bar, NULL); gtk_search_bar_set_entry (bar, NULL);
gtk_search_bar_set_key_capture_widget (bar, NULL);
G_OBJECT_CLASS (gtk_search_bar_parent_class)->dispose (object); G_OBJECT_CLASS (gtk_search_bar_parent_class)->dispose (object);
} }
@ -512,9 +500,10 @@ gtk_search_bar_set_entry (GtkSearchBar *bar,
if (priv->entry != NULL) if (priv->entry != NULL)
{ {
if (GTK_IS_SEARCH_ENTRY (priv->entry)) if (GTK_IS_SEARCH_ENTRY (priv->entry))
g_signal_handlers_disconnect_by_func (priv->entry, stop_search_cb, bar); {
else gtk_search_entry_set_key_capture_widget (GTK_SEARCH_ENTRY (priv->entry), NULL);
g_signal_handlers_disconnect_by_func (priv->entry, entry_key_pressed_event_cb, bar); g_signal_handlers_disconnect_by_func (priv->entry, stop_search_cb, bar);
}
g_object_remove_weak_pointer (G_OBJECT (priv->entry), (gpointer *) &priv->entry); g_object_remove_weak_pointer (G_OBJECT (priv->entry), (gpointer *) &priv->entry);
} }
@ -524,11 +513,13 @@ gtk_search_bar_set_entry (GtkSearchBar *bar,
{ {
g_object_add_weak_pointer (G_OBJECT (priv->entry), (gpointer *) &priv->entry); g_object_add_weak_pointer (G_OBJECT (priv->entry), (gpointer *) &priv->entry);
if (GTK_IS_SEARCH_ENTRY (priv->entry)) if (GTK_IS_SEARCH_ENTRY (priv->entry))
g_signal_connect (priv->entry, "stop-search", {
G_CALLBACK (stop_search_cb), bar); g_signal_connect (priv->entry, "stop-search",
else G_CALLBACK (stop_search_cb), bar);
g_signal_connect (priv->entry, "key-press-event", gtk_search_entry_set_key_capture_widget (GTK_SEARCH_ENTRY (priv->entry),
G_CALLBACK (entry_key_pressed_event_cb), bar); GTK_WIDGET (bar));
}
} }
} }
@ -632,3 +623,148 @@ gtk_search_bar_set_show_close_button (GtkSearchBar *bar,
g_object_notify (G_OBJECT (bar), "show-close-button"); g_object_notify (G_OBJECT (bar), "show-close-button");
} }
} }
static void
changed_cb (gboolean *changed)
{
*changed = TRUE;
}
static gboolean
capture_widget_key_handled (GtkEventControllerKey *controller,
guint keyval,
guint keycode,
GdkModifierType state,
GtkSearchBar *bar)
{
GtkSearchBarPrivate *priv = gtk_search_bar_get_instance_private (bar);
gboolean handled;
if (priv->reveal_child)
return GDK_EVENT_PROPAGATE;
if (priv->entry == NULL)
{
g_warning ("The search bar does not have an entry connected to it. Call gtk_search_bar_connect_entry() to connect one.");
return GDK_EVENT_PROPAGATE;
}
if (GTK_IS_SEARCH_ENTRY (priv->entry))
{
/* The search entry was told to listen to events from the search bar, so
* just forward the event to self, so the search entry has an opportunity
* to intercept those.
*/
handled = gtk_event_controller_key_forward (controller, GTK_WIDGET (bar));
}
else
{
gboolean preedit_changed, buffer_changed;
guint preedit_change_id, buffer_change_id;
gboolean res;
if (gtk_search_entry_is_keynav (keyval, state) ||
keyval == GDK_KEY_space ||
keyval == GDK_KEY_Menu)
return GDK_EVENT_PROPAGATE;
if (keyval == GDK_KEY_Escape)
{
if (gtk_revealer_get_reveal_child (GTK_REVEALER (priv->revealer)))
{
stop_search_cb (priv->entry, bar);
return GDK_EVENT_STOP;
}
return GDK_EVENT_PROPAGATE;
}
handled = GDK_EVENT_PROPAGATE;
preedit_changed = buffer_changed = FALSE;
preedit_change_id = g_signal_connect_swapped (priv->entry, "preedit-changed",
G_CALLBACK (changed_cb), &preedit_changed);
buffer_change_id = g_signal_connect_swapped (priv->entry, "changed",
G_CALLBACK (changed_cb), &buffer_changed);
res = gtk_event_controller_key_forward (controller, priv->entry);
g_signal_handler_disconnect (priv->entry, preedit_change_id);
g_signal_handler_disconnect (priv->entry, buffer_change_id);
if ((res && buffer_changed) || preedit_changed)
handled = GDK_EVENT_STOP;
}
if (handled == GDK_EVENT_STOP)
gtk_revealer_set_reveal_child (GTK_REVEALER (priv->revealer), TRUE);
return handled;
}
/**
* gtk_search_bar_set_key_capture_widget:
* @bar: a #GtkSearchBar
* @widget: (nullable) (transfer none): a #GtkWidget
*
* Sets @widget as the widget that @bar will capture key events from.
*
* If key events are handled by the search bar, the bar will
* be shown, and the entry populated with the entered text.
*
* Since: 3.94
**/
void
gtk_search_bar_set_key_capture_widget (GtkSearchBar *bar,
GtkWidget *widget)
{
GtkSearchBarPrivate *priv = gtk_search_bar_get_instance_private (bar);
g_return_if_fail (GTK_IS_SEARCH_BAR (bar));
g_return_if_fail (!widget || GTK_IS_WIDGET (widget));
if (priv->capture_widget == widget)
return;
if (priv->capture_widget)
{
g_clear_object (&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 (widget);
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), bar);
g_signal_connect (priv->capture_widget_controller, "key-released",
G_CALLBACK (capture_widget_key_handled), bar);
}
}
/**
* gtk_search_bar_get_key_capture_widget:
* @bar: a #GtkSearchBar
*
* Gets the widget that @bar is capturing key events from.
*
* Returns: The key capture widget.
*
* Since: 3.94
**/
GtkWidget *
gtk_search_bar_get_key_capture_widget (GtkSearchBar *bar)
{
GtkSearchBarPrivate *priv = gtk_search_bar_get_instance_private (bar);
g_return_val_if_fail (GTK_IS_SEARCH_BAR (bar), NULL);
return priv->capture_widget;
}

View File

@ -96,6 +96,11 @@ GDK_AVAILABLE_IN_ALL
gboolean gtk_search_bar_handle_event (GtkSearchBar *bar, gboolean gtk_search_bar_handle_event (GtkSearchBar *bar,
GdkEvent *event); GdkEvent *event);
GDK_AVAILABLE_IN_ALL
void gtk_search_bar_set_key_capture_widget (GtkSearchBar *bar,
GtkWidget *widget);
GtkWidget * gtk_search_bar_get_key_capture_widget (GtkSearchBar *bar);
G_END_DECLS G_END_DECLS
#endif /* __GTK_SEARCH_BAR_H__ */ #endif /* __GTK_SEARCH_BAR_H__ */

View File

@ -34,6 +34,7 @@
#include "gtkintl.h" #include "gtkintl.h"
#include "gtkmarshalers.h" #include "gtkmarshalers.h"
#include "gtkstylecontext.h" #include "gtkstylecontext.h"
#include "gtkeventcontrollerkey.h"
/** /**
* SECTION:gtksearchentry * SECTION:gtksearchentry
@ -63,7 +64,8 @@
* *
* Often, GtkSearchEntry will be fed events by means of being * Often, GtkSearchEntry will be fed events by means of being
* placed inside a #GtkSearchBar. If that is not the case, * placed inside a #GtkSearchBar. If that is not the case,
* you can use gtk_search_entry_handle_event() to pass events. * you can use gtk_search_entry_set_key_capture_widget() to let it
* capture key input from another widget.
*/ */
enum { enum {
@ -77,6 +79,9 @@ enum {
static guint signals[LAST_SIGNAL] = { 0 }; static guint signals[LAST_SIGNAL] = { 0 };
typedef struct { typedef struct {
GtkWidget *capture_widget;
GtkEventController *capture_widget_controller;
guint delayed_changed_id; guint delayed_changed_id;
gboolean content_changed; gboolean content_changed;
gboolean search_stopped; gboolean search_stopped;
@ -131,6 +136,8 @@ gtk_search_entry_finalize (GObject *object)
if (priv->delayed_changed_id > 0) if (priv->delayed_changed_id > 0)
g_source_remove (priv->delayed_changed_id); 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); G_OBJECT_CLASS (gtk_search_entry_parent_class)->finalize (object);
} }
@ -378,16 +385,9 @@ gtk_search_entry_new (void)
} }
gboolean gboolean
gtk_search_entry_is_keynav_event (GdkEvent *event) gtk_search_entry_is_keynav (guint keyval,
GdkModifierType state)
{ {
GdkModifierType state = 0;
guint keyval;
if (!gdk_event_get_keyval (event, &keyval))
return FALSE;
gdk_event_get_state (event, &state);
if (keyval == GDK_KEY_Tab || keyval == GDK_KEY_KP_Tab || if (keyval == GDK_KEY_Tab || keyval == GDK_KEY_KP_Tab ||
keyval == GDK_KEY_Up || keyval == GDK_KEY_KP_Up || keyval == GDK_KEY_Up || keyval == GDK_KEY_KP_Up ||
keyval == GDK_KEY_Down || keyval == GDK_KEY_KP_Down || keyval == GDK_KEY_Down || keyval == GDK_KEY_KP_Down ||
@ -433,14 +433,15 @@ gtk_search_entry_handle_event (GtkSearchEntry *entry,
{ {
GtkSearchEntryPrivate *priv = GET_PRIV (entry); GtkSearchEntryPrivate *priv = GET_PRIV (entry);
gboolean handled; gboolean handled;
guint keyval; guint keyval, state;
if (!gtk_widget_get_realized (GTK_WIDGET (entry))) if (!gtk_widget_get_realized (GTK_WIDGET (entry)))
gtk_widget_realize (GTK_WIDGET (entry)); gtk_widget_realize (GTK_WIDGET (entry));
gdk_event_get_keyval (event, &keyval); gdk_event_get_keyval (event, &keyval);
gdk_event_get_state (event, &state);
if (gtk_search_entry_is_keynav_event (event) || if (gtk_search_entry_is_keynav (keyval, state) ||
keyval == GDK_KEY_space || keyval == GDK_KEY_space ||
keyval == GDK_KEY_Menu) keyval == GDK_KEY_Menu)
return GDK_EVENT_PROPAGATE; return GDK_EVENT_PROPAGATE;
@ -452,3 +453,96 @@ gtk_search_entry_handle_event (GtkSearchEntry *entry,
return handled && priv->content_changed && !priv->search_stopped ? GDK_EVENT_STOP : GDK_EVENT_PROPAGATE; 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:
* @bar: 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.
*
* Since: 3.94
**/
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)
{
g_object_unref (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 (widget);
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_search_entry_get_key_capture_widget:
* @entry: a #GtkSearchEntry
*
* Gets the widget that @entry is capturing key events from.
*
* Returns: The key capture widget.
*
* Since: 3.94
**/
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;
}

View File

@ -71,6 +71,14 @@ GDK_AVAILABLE_IN_ALL
gboolean gtk_search_entry_handle_event (GtkSearchEntry *entry, gboolean gtk_search_entry_handle_event (GtkSearchEntry *entry,
GdkEvent *event); GdkEvent *event);
GDK_AVAILABLE_IN_ALL
void gtk_search_entry_set_key_capture_widget (GtkSearchEntry *entry,
GtkWidget *widget);
GDK_AVAILABLE_IN_ALL
GtkWidget* gtk_search_entry_get_key_capture_widget (GtkSearchEntry *entry);
G_END_DECLS G_END_DECLS
#endif /* __GTK_SEARCH_ENTRY_H__ */ #endif /* __GTK_SEARCH_ENTRY_H__ */

View File

@ -29,7 +29,8 @@
G_BEGIN_DECLS G_BEGIN_DECLS
gboolean gtk_search_entry_is_keynav_event (GdkEvent *event); gboolean gtk_search_entry_is_keynav (guint keyval,
GdkModifierType state);
G_END_DECLS G_END_DECLS