GtkSearchEntry: Add more API
Add ::next-match, ::previous-match and ::stop-search keybinding signals that are bound to Ctrl-g, Ctrl-Shift-g and Escape. Also add a gtk_search_entry_handle_event() function to handle key events.
This commit is contained in:
@ -3198,6 +3198,7 @@ gtk_search_bar_get_type
|
|||||||
<TITLE>GtkSearchEntry</TITLE>
|
<TITLE>GtkSearchEntry</TITLE>
|
||||||
GtkSearchEntry
|
GtkSearchEntry
|
||||||
gtk_search_entry_new
|
gtk_search_entry_new
|
||||||
|
gtk_search_entry_handle_event
|
||||||
<SUBSECTION Standard>
|
<SUBSECTION Standard>
|
||||||
GTK_TYPE_SEARCH_ENTRY
|
GTK_TYPE_SEARCH_ENTRY
|
||||||
GTK_SEARCH_ENTRY
|
GTK_SEARCH_ENTRY
|
||||||
|
|||||||
@ -31,19 +31,19 @@
|
|||||||
#include "gtksearchentry.h"
|
#include "gtksearchentry.h"
|
||||||
#include "gtkmarshalers.h"
|
#include "gtkmarshalers.h"
|
||||||
#include "gtkintl.h"
|
#include "gtkintl.h"
|
||||||
|
#include "gtkbindings.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* SECTION:gtksearchentry
|
* SECTION:gtksearchentry
|
||||||
* @Short_description: An entry which shows a search icon
|
* @Short_description: An entry which shows a search icon
|
||||||
* @Title: GtkSearchEntry
|
* @Title: GtkSearchEntry
|
||||||
*
|
*
|
||||||
* #GtkSearchEntry is a subclass of #GtkEntry that has
|
* #GtkSearchEntry is a subclass of #GtkEntry that has been
|
||||||
* been tailored for use as a search entry.
|
* tailored for use as a search entry.
|
||||||
*
|
*
|
||||||
* It will show an inactive symbolic “find” icon when the
|
* It will show an inactive symbolic “find” icon when the search
|
||||||
* search entry is empty, and a symbolic “clear” icon when
|
* entry is empty, and a symbolic “clear” icon when there is text.
|
||||||
* there is text. Clicking on the “clear” icon will empty
|
* Clicking on the “clear” icon will empty the search entry.
|
||||||
* the search entry.
|
|
||||||
*
|
*
|
||||||
* Note that the search/clear icon is shown using a secondary
|
* Note that the search/clear icon is shown using a secondary
|
||||||
* icon, and thus does not work if you are using the secondary
|
* icon, and thus does not work if you are using the secondary
|
||||||
@ -55,11 +55,22 @@
|
|||||||
* emits the #GtkSearchEntry::search-changed signal which can
|
* emits the #GtkSearchEntry::search-changed signal which can
|
||||||
* be used instead of the #GtkEditable::changed signal.
|
* be used instead of the #GtkEditable::changed signal.
|
||||||
*
|
*
|
||||||
|
* The #GtkSearchEntry::previous-match, #GtkSearchEntry::next-match
|
||||||
|
* and #GtkSearchEntry::stop-search signals can be uesd to implement
|
||||||
|
* moving between search results and ending the search.
|
||||||
|
*
|
||||||
|
* Often, GtkSearchEntry will be fed events by means of being
|
||||||
|
* placed inside a #GtkSearchEntry. If that is not the case,
|
||||||
|
* you can use gtk_search_entry_handle_event() to pass events.
|
||||||
|
*
|
||||||
* Since: 3.6
|
* Since: 3.6
|
||||||
*/
|
*/
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
SEARCH_CHANGED,
|
SEARCH_CHANGED,
|
||||||
|
NEXT_MATCH,
|
||||||
|
PREVIOUS_MATCH,
|
||||||
|
STOP_SEARCH,
|
||||||
LAST_SIGNAL
|
LAST_SIGNAL
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -67,6 +78,8 @@ static guint signals[LAST_SIGNAL] = { 0 };
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
guint delayed_changed_id;
|
guint delayed_changed_id;
|
||||||
|
gboolean content_changed;
|
||||||
|
gboolean search_stopped;
|
||||||
} GtkSearchEntryPrivate;
|
} GtkSearchEntryPrivate;
|
||||||
|
|
||||||
static void gtk_search_entry_icon_release (GtkEntry *entry,
|
static void gtk_search_entry_icon_release (GtkEntry *entry,
|
||||||
@ -88,6 +101,28 @@ G_DEFINE_TYPE_WITH_CODE (GtkSearchEntry, gtk_search_entry, GTK_TYPE_ENTRY,
|
|||||||
* that we cannot now have one without breaking ABI */
|
* that we cannot now have one without breaking ABI */
|
||||||
#define GET_PRIV(e) ((GtkSearchEntryPrivate *) gtk_search_entry_get_instance_private ((GtkSearchEntry *) (e)))
|
#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
|
static void
|
||||||
gtk_search_entry_finalize (GObject *object)
|
gtk_search_entry_finalize (GObject *object)
|
||||||
{
|
{
|
||||||
@ -99,17 +134,33 @@ gtk_search_entry_finalize (GObject *object)
|
|||||||
G_OBJECT_CLASS (gtk_search_entry_parent_class)->finalize (object);
|
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
|
static void
|
||||||
gtk_search_entry_class_init (GtkSearchEntryClass *klass)
|
gtk_search_entry_class_init (GtkSearchEntryClass *klass)
|
||||||
{
|
{
|
||||||
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||||||
|
GtkBindingSet *binding_set;
|
||||||
|
|
||||||
object_class->finalize = gtk_search_entry_finalize;
|
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",
|
g_signal_override_class_handler ("icon-release",
|
||||||
GTK_TYPE_SEARCH_ENTRY,
|
GTK_TYPE_SEARCH_ENTRY,
|
||||||
G_CALLBACK (gtk_search_entry_icon_release));
|
G_CALLBACK (gtk_search_entry_icon_release));
|
||||||
|
|
||||||
|
g_signal_override_class_handler ("preedit-changed",
|
||||||
|
GTK_TYPE_SEARCH_ENTRY,
|
||||||
|
G_CALLBACK (gtk_search_entry_preedit_changed));
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GtkSearchEntry::search-changed:
|
* GtkSearchEntry::search-changed:
|
||||||
* @entry: the entry on which the signal was emitted
|
* @entry: the entry on which the signal was emitted
|
||||||
@ -127,6 +178,86 @@ gtk_search_entry_class_init (GtkSearchEntryClass *klass)
|
|||||||
NULL, NULL,
|
NULL, NULL,
|
||||||
_gtk_marshal_VOID__VOID,
|
_gtk_marshal_VOID__VOID,
|
||||||
G_TYPE_NONE, 0);
|
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.
|
||||||
|
*
|
||||||
|
* Since: 3.16
|
||||||
|
*/
|
||||||
|
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,
|
||||||
|
_gtk_marshal_VOID__VOID,
|
||||||
|
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.
|
||||||
|
*
|
||||||
|
* Since: 3.16
|
||||||
|
*/
|
||||||
|
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,
|
||||||
|
_gtk_marshal_VOID__VOID,
|
||||||
|
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.
|
||||||
|
*
|
||||||
|
* Since: 3.16
|
||||||
|
*/
|
||||||
|
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,
|
||||||
|
_gtk_marshal_VOID__VOID,
|
||||||
|
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
|
static void
|
||||||
@ -253,3 +384,78 @@ gtk_search_entry_new (void)
|
|||||||
{
|
{
|
||||||
return GTK_WIDGET (g_object_new (GTK_TYPE_SEARCH_ENTRY, NULL));
|
return GTK_WIDGET (g_object_new (GTK_TYPE_SEARCH_ENTRY, NULL));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
is_keynav_event (GdkEvent *event)
|
||||||
|
{
|
||||||
|
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 ||
|
||||||
|
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.
|
||||||
|
*
|
||||||
|
* Since: 3.16
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
gtk_search_entry_handle_event (GtkSearchEntry *entry,
|
||||||
|
GdkEvent *event)
|
||||||
|
{
|
||||||
|
GtkSearchEntryPrivate *priv = GET_PRIV (entry);
|
||||||
|
gboolean handled;
|
||||||
|
|
||||||
|
if (!gtk_widget_get_realized (GTK_WIDGET (entry)))
|
||||||
|
gtk_widget_realize (GTK_WIDGET (entry));
|
||||||
|
|
||||||
|
if (is_keynav_event (event) ||
|
||||||
|
event->key.keyval == GDK_KEY_space ||
|
||||||
|
event->key.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;
|
||||||
|
}
|
||||||
|
|||||||
@ -56,11 +56,9 @@ struct _GtkSearchEntryClass
|
|||||||
GtkEntryClass parent_class;
|
GtkEntryClass parent_class;
|
||||||
|
|
||||||
void (*search_changed) (GtkSearchEntry *entry);
|
void (*search_changed) (GtkSearchEntry *entry);
|
||||||
|
void (*next_match) (GtkSearchEntry *entry);
|
||||||
/* Padding for future expansion */
|
void (*previous_match) (GtkSearchEntry *entry);
|
||||||
void (*_gtk_reserved1) (void);
|
void (*stop_search) (GtkSearchEntry *entry);
|
||||||
void (*_gtk_reserved2) (void);
|
|
||||||
void (*_gtk_reserved3) (void);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
GDK_AVAILABLE_IN_3_6
|
GDK_AVAILABLE_IN_3_6
|
||||||
@ -69,6 +67,10 @@ GType gtk_search_entry_get_type (void) G_GNUC_CONST;
|
|||||||
GDK_AVAILABLE_IN_3_6
|
GDK_AVAILABLE_IN_3_6
|
||||||
GtkWidget* gtk_search_entry_new (void);
|
GtkWidget* gtk_search_entry_new (void);
|
||||||
|
|
||||||
|
GDK_AVAILABLE_IN_3_16
|
||||||
|
gboolean gtk_search_entry_handle_event (GtkSearchEntry *entry,
|
||||||
|
GdkEvent *event);
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
#endif /* __GTK_SEARCH_ENTRY_H__ */
|
#endif /* __GTK_SEARCH_ENTRY_H__ */
|
||||||
|
|||||||
Reference in New Issue
Block a user