[gtk/text: 10/14] search entry: Use a GtkText
- From: Matthias Clasen <matthiasc src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gtk/text: 10/14] search entry: Use a GtkText
- Date: Sun, 17 Feb 2019 05:03:16 +0000 (UTC)
commit 0b314159e02638d28e42b1ab7556d41029d74927
Author: Matthias Clasen <mclasen redhat com>
Date: Sat Feb 16 23:57:18 2019 -0500
search entry: Use a GtkText
gtk/gtksearchentry.c | 212 ++++++++++++++++++++++++++++----------------
gtk/gtksearchentry.h | 5 +-
gtk/gtksearchentryprivate.h | 2 +
3 files changed, 138 insertions(+), 81 deletions(-)
---
diff --git a/gtk/gtksearchentry.c b/gtk/gtksearchentry.c
index acbd5e110e..78508361f2 100644
--- a/gtk/gtksearchentry.c
+++ b/gtk/gtksearchentry.c
@@ -31,7 +31,11 @@
#include "gtkaccessible.h"
#include "gtkbindings.h"
-#include "gtkentryprivate.h"
+#include "gtkeditableprivate.h"
+#include "gtkbox.h"
+#include "gtkgesturemultipress.h"
+#include "gtktextprivate.h"
+#include "gtkimage.h"
#include "gtkintl.h"
#include "gtkmarshalers.h"
#include "gtkstylecontext.h"
@@ -83,19 +87,18 @@ typedef struct {
GtkWidget *capture_widget;
GtkEventController *capture_widget_controller;
+ GtkWidget *box;
+ GtkWidget *entry;
+ GtkWidget *icon;
+
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_DEFINE_TYPE_WITH_CODE (GtkSearchEntry, gtk_search_entry, GTK_TYPE_WIDGET,
G_ADD_PRIVATE (GtkSearchEntry)
G_IMPLEMENT_INTERFACE (GTK_TYPE_EDITABLE,
gtk_search_entry_editable_init))
@@ -103,36 +106,21 @@ G_DEFINE_TYPE_WITH_CODE (GtkSearchEntry, gtk_search_entry, GTK_TYPE_ENTRY,
/* 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)
+text_changed (GtkSearchEntry *entry)
{
- GtkSearchEntryPrivate *priv = GET_PRIV (entry);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (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);
+ GtkSearchEntry *entry = GTK_SEARCH_ENTRY (object);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
+
+ g_clear_pointer (&priv->box, gtk_widget_unparent);
if (priv->delayed_changed_id > 0)
g_source_remove (priv->delayed_changed_id);
@@ -145,25 +133,89 @@ gtk_search_entry_finalize (GObject *object)
static void
gtk_search_entry_stop_search (GtkSearchEntry *entry)
{
- GtkSearchEntryPrivate *priv = GET_PRIV (entry);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
priv->search_stopped = TRUE;
}
+static void
+gtk_search_entry_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ if (gtk_editable_delegate_set_property (object, prop_id, value, pspec))
+ return;
+
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+}
+
+static void
+gtk_search_entry_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ if (gtk_editable_delegate_get_property (object, prop_id, value, pspec))
+ return;
+
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+}
+
+static void
+gtk_search_entry_measure (GtkWidget *widget,
+ GtkOrientation orientation,
+ int for_size,
+ int *minimum,
+ int *natural,
+ int *minimum_baseline,
+ int *natural_baseline)
+{
+ GtkSearchEntry *entry = GTK_SEARCH_ENTRY (widget);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
+
+ gtk_widget_measure (priv->box, orientation, for_size,
+ minimum, natural,
+ minimum_baseline, natural_baseline);
+}
+
+static void
+gtk_search_entry_size_allocate (GtkWidget *widget,
+ int width,
+ int height,
+ int baseline)
+{
+ GtkSearchEntry *entry = GTK_SEARCH_ENTRY (widget);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
+
+ gtk_widget_size_allocate (priv->box,
+ &(GtkAllocation) { 0, 0, width, height },
+ baseline);
+}
+
static void
gtk_search_entry_class_init (GtkSearchEntryClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
GtkBindingSet *binding_set;
object_class->finalize = gtk_search_entry_finalize;
- object_class->notify = gtk_search_entry_notify;
+ object_class->get_property = gtk_search_entry_get_property;
+ object_class->set_property = gtk_search_entry_set_property;
+
+ widget_class->measure = gtk_search_entry_measure;
+ widget_class->size_allocate = gtk_search_entry_size_allocate;
klass->stop_search = gtk_search_entry_stop_search;
+#if 0
g_signal_override_class_handler ("icon-release",
GTK_TYPE_SEARCH_ENTRY,
G_CALLBACK (gtk_search_entry_icon_release));
+#endif
+
+ gtk_editable_install_properties (object_class);
/**
* GtkSearchEntry::search-changed:
@@ -254,37 +306,33 @@ gtk_search_entry_class_init (GtkSearchEntryClass *klass)
"previous-match", 0);
gtk_binding_entry_add_signal (binding_set, GDK_KEY_Escape, 0,
"stop-search", 0);
+
+ gtk_widget_class_set_css_name (widget_class, I_("entry"));
}
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;
+ gtk_editable_delegate_iface_init (iface);
}
static void
-gtk_search_entry_icon_release (GtkEntry *entry,
- GtkEntryIconPosition icon_pos)
+gtk_search_entry_icon_release (GtkGestureMultiPress *press,
+ int n_press,
+ double x,
+ double y,
+ GtkSearchEntry *entry)
{
- if (icon_pos == GTK_ENTRY_ICON_SECONDARY)
- gtk_entry_set_text (entry, "");
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
+
+ gtk_editable_set_text (GTK_EDITABLE (priv->entry), "");
}
static gboolean
gtk_search_entry_changed_timeout_cb (gpointer user_data)
{
GtkSearchEntry *entry = user_data;
- GtkSearchEntryPrivate *priv = GET_PRIV (entry);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
g_signal_emit (entry, signals[SEARCH_CHANGED], 0);
priv->delayed_changed_id = 0;
@@ -295,7 +343,7 @@ gtk_search_entry_changed_timeout_cb (gpointer user_data)
static void
reset_timeout (GtkSearchEntry *entry)
{
- GtkSearchEntryPrivate *priv = GET_PRIV (entry);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
if (priv->delayed_changed_id > 0)
g_source_remove (priv->delayed_changed_id);
@@ -306,35 +354,19 @@ reset_timeout (GtkSearchEntry *entry)
}
static void
-gtk_search_entry_changed (GtkEditable *editable)
+gtk_search_entry_changed (GtkEditable *editable,
+ GtkSearchEntry *entry)
{
- GtkSearchEntry *entry = GTK_SEARCH_ENTRY (editable);
- GtkSearchEntryPrivate *priv = GET_PRIV (entry);
- const char *str, *icon_name;
- gboolean cleared;
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
+ const char *str;
/* Update the icons first */
- str = gtk_entry_get_text (GTK_ENTRY (entry));
+ str = gtk_editable_get_text (GTK_EDITABLE (priv->entry));
if (str == NULL || *str == '\0')
{
- icon_name = NULL;
- cleared = TRUE;
- }
- else
- {
- icon_name = "edit-clear-symbolic";
- cleared = FALSE;
- }
+ gtk_widget_hide (priv->icon);
- 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);
@@ -344,6 +376,8 @@ gtk_search_entry_changed (GtkEditable *editable)
}
else
{
+ gtk_widget_show (priv->icon);
+
/* Queue up the timeout */
reset_timeout (entry);
}
@@ -352,22 +386,37 @@ gtk_search_entry_changed (GtkEditable *editable)
static void
gtk_search_entry_init (GtkSearchEntry *entry)
{
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
AtkObject *atk_obj;
+ GtkGesture *press;
+
+ gtk_widget_set_has_surface (GTK_WIDGET (entry), FALSE);
+
+ priv->box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
+ gtk_widget_set_parent (priv->box, GTK_WIDGET (entry));
+
+ priv->entry = gtk_text_new ();
+ gtk_widget_set_hexpand (priv->entry, TRUE);
+ gtk_widget_set_vexpand (priv->entry, TRUE);
+ gtk_container_add (GTK_CONTAINER (priv->box), GTK_WIDGET (priv->entry));
+ gtk_editable_set_delegate (GTK_EDITABLE (entry), GTK_EDITABLE (priv->entry));
+ g_signal_connect_swapped (priv->entry, "changed", G_CALLBACK (text_changed), entry);
+ g_signal_connect_after (priv->entry, "changed", G_CALLBACK (gtk_search_entry_changed), entry);
+ g_signal_connect_swapped (priv->entry, "preedit-changed", G_CALLBACK (text_changed), entry);
- g_object_set (entry,
- "primary-icon-name", "edit-find-symbolic",
- "primary-icon-activatable", FALSE,
- "primary-icon-sensitive", FALSE,
- NULL);
+ priv->icon = gtk_image_new_from_icon_name ("edit-clear-symbolic");
+ gtk_container_add (GTK_CONTAINER (priv->box), GTK_WIDGET (priv->icon));
+ gtk_widget_hide (priv->icon);
+
+ press = gtk_gesture_multi_press_new ();
+ g_signal_connect (press, "released", G_CALLBACK (gtk_search_entry_icon_release), entry);
+ gtk_widget_add_controller (priv->icon, GTK_EVENT_CONTROLLER (press));
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);
}
/**
@@ -431,7 +480,7 @@ gboolean
gtk_search_entry_handle_event (GtkSearchEntry *entry,
GdkEvent *event)
{
- GtkSearchEntryPrivate *priv = GET_PRIV (entry);
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
gboolean handled;
guint keyval, state;
@@ -544,3 +593,10 @@ gtk_search_entry_get_key_capture_widget (GtkSearchEntry *entry)
return priv->capture_widget;
}
+GtkEventController *
+gtk_search_entry_get_key_controller (GtkSearchEntry *entry)
+{
+ GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
+
+ return gtk_text_get_key_controller (GTK_TEXT (priv->entry));
+}
diff --git a/gtk/gtksearchentry.h b/gtk/gtksearchentry.h
index cdabb9fcda..8f33efb316 100644
--- a/gtk/gtksearchentry.h
+++ b/gtk/gtksearchentry.h
@@ -48,12 +48,12 @@ typedef struct _GtkSearchEntryClass GtkSearchEntryClass;
struct _GtkSearchEntry
{
- GtkEntry parent;
+ GtkWidget parent;
};
struct _GtkSearchEntryClass
{
- GtkEntryClass parent_class;
+ GtkWidgetClass parent_class;
void (*search_changed) (GtkSearchEntry *entry);
void (*next_match) (GtkSearchEntry *entry);
@@ -78,7 +78,6 @@ GDK_AVAILABLE_IN_ALL
GtkWidget* gtk_search_entry_get_key_capture_widget (GtkSearchEntry *entry);
-
G_END_DECLS
#endif /* __GTK_SEARCH_ENTRY_H__ */
diff --git a/gtk/gtksearchentryprivate.h b/gtk/gtksearchentryprivate.h
index 1b0706d412..46b5d9f5b9 100644
--- a/gtk/gtksearchentryprivate.h
+++ b/gtk/gtksearchentryprivate.h
@@ -32,6 +32,8 @@ G_BEGIN_DECLS
gboolean gtk_search_entry_is_keynav (guint keyval,
GdkModifierType state);
+GtkEventController * gtk_search_entry_get_key_controller (GtkSearchEntry *entry);
+
G_END_DECLS
#endif /* __GTK_SEARCH_ENTRY_PRIVATE_H__ */
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]