forked from AuroraMiddleware/gtk
Change the way GtkSearchEntry does delayed change notification
We add a GtkSearchEntry::search-changed signal which gets emitted with a 150 millisecond delay. The ::change signal goes back to its expected semantics. https://bugzilla.gnome.org/show_bug.cgi?id=700229
This commit is contained in:
parent
1b135b28c0
commit
398f9e8b5b
@ -28,6 +28,8 @@
|
|||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
|
||||||
#include "gtksearchentry.h"
|
#include "gtksearchentry.h"
|
||||||
|
#include "gtkmarshalers.h"
|
||||||
|
#include "gtkintl.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* SECTION:gtksearchentry
|
* SECTION:gtksearchentry
|
||||||
@ -46,15 +48,37 @@
|
|||||||
* icon, and thus does not work if you are using the secondary
|
* icon, and thus does not work if you are using the secondary
|
||||||
* icon position for some other purpose.
|
* 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.
|
||||||
|
*
|
||||||
* Since: 3.6
|
* Since: 3.6
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
enum {
|
||||||
|
SEARCH_CHANGED,
|
||||||
|
LAST_SIGNAL
|
||||||
|
};
|
||||||
|
|
||||||
|
static guint signals[LAST_SIGNAL] = { 0 };
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
guint delayed_changed_id;
|
guint delayed_changed_id;
|
||||||
gboolean in_timeout;
|
|
||||||
} GtkSearchEntryPrivate;
|
} GtkSearchEntryPrivate;
|
||||||
|
|
||||||
G_DEFINE_TYPE_WITH_PRIVATE (GtkSearchEntry, gtk_search_entry, GTK_TYPE_ENTRY)
|
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 */
|
/* 150 mseconds of delay */
|
||||||
#define DELAYED_TIMEOUT_ID 150
|
#define DELAYED_TIMEOUT_ID 150
|
||||||
@ -74,14 +98,6 @@ 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
|
|
||||||
search_entry_clear_cb (GtkEntry *entry,
|
|
||||||
GtkEntryIconPosition icon_pos)
|
|
||||||
{
|
|
||||||
if (icon_pos == GTK_ENTRY_ICON_SECONDARY)
|
|
||||||
gtk_entry_set_text (entry, "");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gtk_search_entry_class_init (GtkSearchEntryClass *klass)
|
gtk_search_entry_class_init (GtkSearchEntryClass *klass)
|
||||||
{
|
{
|
||||||
@ -91,7 +107,49 @@ gtk_search_entry_class_init (GtkSearchEntryClass *klass)
|
|||||||
|
|
||||||
g_signal_override_class_handler ("icon-release",
|
g_signal_override_class_handler ("icon-release",
|
||||||
GTK_TYPE_SEARCH_ENTRY,
|
GTK_TYPE_SEARCH_ENTRY,
|
||||||
G_CALLBACK (search_entry_clear_cb));
|
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.
|
||||||
|
*
|
||||||
|
* Since: 3.10
|
||||||
|
*/
|
||||||
|
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,
|
||||||
|
_gtk_marshal_VOID__VOID,
|
||||||
|
G_TYPE_NONE, 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
|
static gboolean
|
||||||
@ -100,10 +158,8 @@ gtk_search_entry_changed_timeout_cb (gpointer user_data)
|
|||||||
GtkSearchEntry *entry = user_data;
|
GtkSearchEntry *entry = user_data;
|
||||||
GtkSearchEntryPrivate *priv = GET_PRIV (entry);
|
GtkSearchEntryPrivate *priv = GET_PRIV (entry);
|
||||||
|
|
||||||
priv->in_timeout = TRUE;
|
g_signal_emit (entry, signals[SEARCH_CHANGED], 0);
|
||||||
g_signal_emit_by_name (entry, "changed");
|
|
||||||
priv->delayed_changed_id = 0;
|
priv->delayed_changed_id = 0;
|
||||||
priv->in_timeout = FALSE;
|
|
||||||
|
|
||||||
return G_SOURCE_REMOVE;
|
return G_SOURCE_REMOVE;
|
||||||
}
|
}
|
||||||
@ -121,13 +177,12 @@ reset_timeout (GtkSearchEntry *entry)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
search_entry_changed_cb (GtkSearchEntry *entry,
|
gtk_search_entry_changed (GtkEditable *editable)
|
||||||
gpointer user_data)
|
|
||||||
{
|
{
|
||||||
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (editable);
|
||||||
GtkSearchEntryPrivate *priv = GET_PRIV (entry);
|
GtkSearchEntryPrivate *priv = GET_PRIV (entry);
|
||||||
const char *str, *icon_name;
|
const char *str, *icon_name;
|
||||||
gboolean active;
|
gboolean cleared;
|
||||||
gboolean cleared = FALSE;
|
|
||||||
|
|
||||||
/* Update the icons first */
|
/* Update the icons first */
|
||||||
str = gtk_entry_get_text (GTK_ENTRY (entry));
|
str = gtk_entry_get_text (GTK_ENTRY (entry));
|
||||||
@ -135,7 +190,6 @@ search_entry_changed_cb (GtkSearchEntry *entry,
|
|||||||
if (str == NULL || *str == '\0')
|
if (str == NULL || *str == '\0')
|
||||||
{
|
{
|
||||||
icon_name = NULL;
|
icon_name = NULL;
|
||||||
active = FALSE;
|
|
||||||
cleared = TRUE;
|
cleared = TRUE;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -144,36 +198,34 @@ search_entry_changed_cb (GtkSearchEntry *entry,
|
|||||||
icon_name = "edit-clear-rtl-symbolic";
|
icon_name = "edit-clear-rtl-symbolic";
|
||||||
else
|
else
|
||||||
icon_name = "edit-clear-symbolic";
|
icon_name = "edit-clear-symbolic";
|
||||||
active = TRUE;
|
cleared = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
g_object_set (entry,
|
g_object_set (entry,
|
||||||
"secondary-icon-name", icon_name,
|
"secondary-icon-name", icon_name,
|
||||||
"secondary-icon-activatable", active,
|
"secondary-icon-activatable", !cleared,
|
||||||
"secondary-icon-sensitive", active,
|
"secondary-icon-sensitive", !cleared,
|
||||||
NULL);
|
NULL);
|
||||||
|
|
||||||
/* Don't stop the emission if it's the timeout
|
|
||||||
* emitting the signal, otherwise we'll get in a loop */
|
|
||||||
if (priv->in_timeout)
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* Don't emit the signal in a timeout if we've cleared
|
|
||||||
* the entry, we don't want a delay */
|
|
||||||
if (cleared)
|
if (cleared)
|
||||||
return;
|
{
|
||||||
|
if (priv->delayed_changed_id > 0)
|
||||||
/* Queue up the timeout */
|
{
|
||||||
reset_timeout (entry);
|
g_source_remove (priv->delayed_changed_id);
|
||||||
g_signal_stop_emission_by_name (entry, "changed");
|
priv->delayed_changed_id = 0;
|
||||||
|
}
|
||||||
|
g_signal_emit (entry, signals[SEARCH_CHANGED], 0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Queue up the timeout */
|
||||||
|
reset_timeout (entry);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gtk_search_entry_init (GtkSearchEntry *entry)
|
gtk_search_entry_init (GtkSearchEntry *entry)
|
||||||
{
|
{
|
||||||
g_signal_connect (entry, "changed",
|
|
||||||
G_CALLBACK (search_entry_changed_cb), NULL);
|
|
||||||
|
|
||||||
g_object_set (entry,
|
g_object_set (entry,
|
||||||
"primary-icon-name", "edit-find-symbolic",
|
"primary-icon-name", "edit-find-symbolic",
|
||||||
"primary-icon-activatable", FALSE,
|
"primary-icon-activatable", FALSE,
|
||||||
|
@ -55,11 +55,12 @@ struct _GtkSearchEntryClass
|
|||||||
{
|
{
|
||||||
GtkEntryClass parent_class;
|
GtkEntryClass parent_class;
|
||||||
|
|
||||||
|
void (*search_changed) (GtkSearchEntry *entry);
|
||||||
|
|
||||||
/* Padding for future expansion */
|
/* Padding for future expansion */
|
||||||
void (*_gtk_reserved1) (void);
|
void (*_gtk_reserved1) (void);
|
||||||
void (*_gtk_reserved2) (void);
|
void (*_gtk_reserved2) (void);
|
||||||
void (*_gtk_reserved3) (void);
|
void (*_gtk_reserved3) (void);
|
||||||
void (*_gtk_reserved4) (void);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
GDK_AVAILABLE_IN_3_6
|
GDK_AVAILABLE_IN_3_6
|
||||||
|
Loading…
Reference in New Issue
Block a user