[gnome-builder/search] search-box: add GbSearchBox widget to contain search functionality
- From: Christian Hergert <chergert src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-builder/search] search-box: add GbSearchBox widget to contain search functionality
- Date: Mon, 15 Dec 2014 23:56:32 +0000 (UTC)
commit 609a4964f4d8ef5acbcb63b6104d3ec45d7e38c9
Author: Christian Hergert <christian hergert me>
Date: Mon Dec 15 15:56:10 2014 -0800
search-box: add GbSearchBox widget to contain search functionality
This will let us keep Workbench from getting too complicated.
src/gnome-builder.mk | 2 +
src/resources/gnome-builder.gresource.xml | 1 +
src/resources/ui/gb-search-box.ui | 43 ++++
src/search/gb-search-box.c | 334 +++++++++++++++++++++++++++++
src/search/gb-search-box.h | 61 ++++++
5 files changed, 441 insertions(+), 0 deletions(-)
---
diff --git a/src/gnome-builder.mk b/src/gnome-builder.mk
index 78f2443..46f12b8 100644
--- a/src/gnome-builder.mk
+++ b/src/gnome-builder.mk
@@ -159,6 +159,8 @@ libgnome_builder_la_SOURCES = \
src/log/gb-log.h \
src/nautilus/nautilus-floating-bar.c \
src/nautilus/nautilus-floating-bar.h \
+ src/search/gb-search-box.c \
+ src/search/gb-search-box.h \
src/search/gb-search-context.c \
src/search/gb-search-context.h \
src/search/gb-search-display.c \
diff --git a/src/resources/gnome-builder.gresource.xml b/src/resources/gnome-builder.gresource.xml
index 7b943f7..85fe5af 100644
--- a/src/resources/gnome-builder.gresource.xml
+++ b/src/resources/gnome-builder.gresource.xml
@@ -34,6 +34,7 @@
<file>ui/gb-preferences-page-editor.ui</file>
<file>ui/gb-preferences-page-git.ui</file>
<file>ui/gb-preferences-page-language.ui</file>
+ <file>ui/gb-search-box.ui</file>
<file>ui/gb-search-display.ui</file>
<file>ui/gb-workbench.ui</file>
</gresource>
diff --git a/src/resources/ui/gb-search-box.ui b/src/resources/ui/gb-search-box.ui
new file mode 100644
index 0000000..0c4ceed
--- /dev/null
+++ b/src/resources/ui/gb-search-box.ui
@@ -0,0 +1,43 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<interface>
+ <!-- interface-requires gtk+ 3.14 -->
+ <template class="GbSearchBox" parent="GtkBox">
+ <property name="orientation">horizontal</property>
+ <style>
+ <class name="linked"/>
+ </style>
+ <child>
+ <object class="GtkSearchEntry" id="entry">
+ <property name="visible">true</property>
+ <property name="hexpand">true</property>
+ </object>
+ </child>
+ <child>
+ <object class="GtkMenuButton" id="button">
+ <property name="visible">true</property>
+ <property name="popover">popover</property>
+ <style>
+ <class name="image-button"/>
+ <class name="text-button"/>
+ </style>
+ <child>
+ <object class="GtkImage">
+ <property name="visible">True</property>
+ <property name="icon_name">open-menu-symbolic</property>
+ <property name="valign">baseline</property>
+ </object>
+ </child>
+ </object>
+ </child>
+ </template>
+ <object class="GtkPopover" id="popover">
+ <property name="visible">true</property>
+ <property name="modal">false</property>
+ <property name="relative-to">entry</property>
+ <child>
+ <object class="GbSearchDisplay" id="display">
+ <property name="visible">true</property>
+ </object>
+ </child>
+ </object>
+</interface>
diff --git a/src/search/gb-search-box.c b/src/search/gb-search-box.c
new file mode 100644
index 0000000..090b773
--- /dev/null
+++ b/src/search/gb-search-box.c
@@ -0,0 +1,334 @@
+/* gb-search-box.c
+ *
+ * Copyright (C) 2014 Christian Hergert <christian hergert me>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#define G_LOG_DOMAIN "search-box"
+
+#include <glib/gi18n.h>
+
+#include "gb-search-box.h"
+#include "gb-search-display.h"
+#include "gb-search-manager.h"
+#include "gb-widget.h"
+
+#define DELAY_TIMEOUT_MSEC 250
+
+struct _GbSearchBoxPrivate
+{
+ GbSearchManager *search_manager;
+
+ GtkMenuButton *button;
+ GbSearchDisplay *display;
+ GtkSearchEntry *entry;
+ GtkPopover *popover;
+
+ guint delay_timeout;
+};
+
+G_DEFINE_TYPE_WITH_PRIVATE (GbSearchBox, gb_search_box, GTK_TYPE_BOX)
+
+enum {
+ PROP_0,
+ PROP_SEARCH_MANAGER,
+ LAST_PROP
+};
+
+static GParamSpec *gParamSpecs [LAST_PROP];
+
+GtkWidget *
+gb_search_box_new (void)
+{
+ return g_object_new (GB_TYPE_SEARCH_BOX, NULL);
+}
+
+GbSearchManager *
+gb_search_box_get_search_manager (GbSearchBox *box)
+{
+ g_return_val_if_fail (GB_IS_SEARCH_BOX (box), NULL);
+
+ return box->priv->search_manager;
+}
+
+void
+gb_search_box_set_search_manager (GbSearchBox *box,
+ GbSearchManager *search_manager)
+{
+ g_return_if_fail (GB_IS_SEARCH_BOX (box));
+ g_return_if_fail (!search_manager || GB_IS_SEARCH_MANAGER (search_manager));
+
+ if (box->priv->search_manager != search_manager)
+ {
+ g_clear_object (&box->priv->search_manager);
+
+ if (search_manager)
+ box->priv->search_manager = g_object_ref (search_manager);
+
+ g_object_notify_by_pspec (G_OBJECT (box),
+ gParamSpecs [PROP_SEARCH_MANAGER]);
+ }
+}
+
+static gboolean
+gb_search_box_delay_cb (gpointer user_data)
+{
+ GbSearchBox *box = user_data;
+ GbSearchContext *context;
+ const gchar *search_text;
+
+ g_return_val_if_fail (GB_IS_SEARCH_BOX (box), G_SOURCE_REMOVE);
+
+ box->priv->delay_timeout = 0;
+
+ context = gb_search_display_get_context (box->priv->display);
+ if (context)
+ gb_search_context_cancel (context);
+
+ if (!box->priv->search_manager)
+ return G_SOURCE_REMOVE;
+
+ search_text = gtk_entry_get_text (GTK_ENTRY (box->priv->entry));
+ if (!search_text)
+ return G_SOURCE_REMOVE;
+
+ context = gb_search_manager_search (box->priv->search_manager, search_text);
+ gb_search_display_set_context (box->priv->display, context);
+ g_object_unref (context);
+
+ return G_SOURCE_REMOVE;
+}
+
+static void
+gb_search_box_popover_closed (GbSearchBox *box,
+ GtkPopover *popover)
+{
+ g_return_if_fail (GB_IS_SEARCH_BOX (box));
+ g_return_if_fail (GTK_IS_POPOVER (popover));
+
+}
+
+static gboolean
+gb_search_box_entry_focus_in (GbSearchBox *box,
+ GdkEventFocus *focus,
+ GtkWidget *entry)
+{
+ g_return_val_if_fail (GB_IS_SEARCH_BOX (box), FALSE);
+ g_return_val_if_fail (focus, FALSE);
+ g_return_val_if_fail (GTK_IS_SEARCH_ENTRY (entry), FALSE);
+
+ gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (box->priv->button), TRUE);
+
+ return GDK_EVENT_PROPAGATE;
+}
+
+static void
+gb_search_box_entry_changed (GbSearchBox *box,
+ GtkSearchEntry *entry)
+{
+ const gchar *search_text;
+
+ g_return_if_fail (GB_IS_SEARCH_BOX (box));
+ g_return_if_fail (GTK_IS_SEARCH_ENTRY (entry));
+
+ if (box->priv->delay_timeout)
+ {
+ g_source_remove (box->priv->delay_timeout);
+ box->priv->delay_timeout = 0;
+ }
+
+ search_text = gtk_entry_get_text (GTK_ENTRY (entry));
+
+ if (search_text)
+ box->priv->delay_timeout = g_timeout_add (DELAY_TIMEOUT_MSEC,
+ gb_search_box_delay_cb,
+ box);
+}
+
+static gboolean
+gb_search_box_entry_key_press_event (GbSearchBox *box,
+ GdkEventKey *key,
+ GtkSearchEntry *entry)
+{
+ g_return_val_if_fail (GB_IS_SEARCH_BOX (box), GDK_EVENT_PROPAGATE);
+ g_return_val_if_fail (key, GDK_EVENT_PROPAGATE);
+ g_return_val_if_fail (GTK_IS_SEARCH_ENTRY (entry), GDK_EVENT_PROPAGATE);
+
+ if (key->keyval == GDK_KEY_Escape)
+ {
+ GtkWidget *toplevel;
+
+ gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (box->priv->button),
+ FALSE);
+
+ toplevel = gtk_widget_get_toplevel (GTK_WIDGET (entry));
+ gtk_window_set_focus (GTK_WINDOW (toplevel), NULL);
+
+ return GDK_EVENT_STOP;
+ }
+
+ return GDK_EVENT_PROPAGATE;
+}
+
+static void
+gb_search_box_grab_focus (GtkWidget *widget)
+{
+ GbSearchBox *box = (GbSearchBox *)widget;
+
+ g_return_if_fail (GB_IS_SEARCH_BOX (box));
+
+ gtk_widget_grab_focus (GTK_WIDGET (box->priv->entry));
+}
+
+static void
+gb_search_box_constructed (GObject *object)
+{
+ GbSearchBoxPrivate *priv;
+ GbSearchBox *self = (GbSearchBox *)object;
+
+ g_return_if_fail (GB_IS_SEARCH_BOX (self));
+
+ priv = self->priv;
+
+ G_OBJECT_CLASS (gb_search_box_parent_class)->constructed (object);
+
+ gtk_popover_set_relative_to (priv->popover, GTK_WIDGET (priv->entry));
+
+ g_signal_connect_object (priv->popover,
+ "closed",
+ G_CALLBACK (gb_search_box_popover_closed),
+ self,
+ G_CONNECT_SWAPPED);
+ g_signal_connect_object (priv->entry,
+ "focus-in-event",
+ G_CALLBACK (gb_search_box_entry_focus_in),
+ self,
+ G_CONNECT_SWAPPED);
+ g_signal_connect_object (priv->entry,
+ "changed",
+ G_CALLBACK (gb_search_box_entry_changed),
+ self,
+ G_CONNECT_SWAPPED);
+ g_signal_connect_object (priv->entry,
+ "key-press-event",
+ G_CALLBACK (gb_search_box_entry_key_press_event),
+ self,
+ G_CONNECT_SWAPPED);
+}
+
+static void
+gb_search_box_finalize (GObject *object)
+{
+ GbSearchBoxPrivate *priv = GB_SEARCH_BOX (object)->priv;
+
+ if (priv->delay_timeout)
+ {
+ g_source_remove (priv->delay_timeout);
+ priv->delay_timeout = 0;
+ }
+
+ g_clear_object (&priv->search_manager);
+
+ G_OBJECT_CLASS (gb_search_box_parent_class)->finalize (object);
+}
+
+static void
+gb_search_box_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GbSearchBox *self = GB_SEARCH_BOX (object);
+
+ switch (prop_id)
+ {
+ case PROP_SEARCH_MANAGER:
+ g_value_set_object (value, gb_search_box_get_search_manager (self));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+gb_search_box_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ GbSearchBox *self = GB_SEARCH_BOX (object);
+
+ switch (prop_id)
+ {
+ case PROP_SEARCH_MANAGER:
+ gb_search_box_set_search_manager (self, g_value_get_object (value));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+gb_search_box_class_init (GbSearchBoxClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+
+ object_class->constructed = gb_search_box_constructed;
+ object_class->finalize = gb_search_box_finalize;
+ object_class->get_property = gb_search_box_get_property;
+ object_class->set_property = gb_search_box_set_property;
+
+ widget_class->grab_focus = gb_search_box_grab_focus;
+
+ gParamSpecs [PROP_SEARCH_MANAGER] =
+ g_param_spec_object ("search-manager",
+ _("Search Manager"),
+ _("The search manager for the search box."),
+ GB_TYPE_SEARCH_MANAGER,
+ (G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
+ g_object_class_install_property (object_class, PROP_SEARCH_MANAGER,
+ gParamSpecs [PROP_SEARCH_MANAGER]);
+
+ GB_WIDGET_CLASS_TEMPLATE (klass, "gb-search-box.ui");
+ GB_WIDGET_CLASS_BIND (klass, GbSearchBox, button);
+ GB_WIDGET_CLASS_BIND (klass, GbSearchBox, display);
+ GB_WIDGET_CLASS_BIND (klass, GbSearchBox, entry);
+ GB_WIDGET_CLASS_BIND (klass, GbSearchBox, popover);
+
+ g_type_ensure (GB_TYPE_SEARCH_DISPLAY);
+}
+
+static void
+gb_search_box_init (GbSearchBox *self)
+{
+ self->priv = gb_search_box_get_instance_private (self);
+
+ gtk_widget_init_template (GTK_WIDGET (self));
+
+ /*
+ * WORKAROUND:
+ *
+ * The GtkWidget template things that popover is a child of ours. When in
+ * reality it is a child of the GtkMenuButton (since it owns the "popover"
+ * property. Both our widget and the menu button try to call
+ * gtk_widget_destroy() on it.
+ *
+ * https://bugzilla.gnome.org/show_bug.cgi?id=741529
+ */
+ g_object_ref (self->priv->popover);
+}
diff --git a/src/search/gb-search-box.h b/src/search/gb-search-box.h
new file mode 100644
index 0000000..6f3e32e
--- /dev/null
+++ b/src/search/gb-search-box.h
@@ -0,0 +1,61 @@
+/* gb-search-box.h
+ *
+ * Copyright (C) 2014 Christian Hergert <christian hergert me>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef GB_SEARCH_BOX_H
+#define GB_SEARCH_BOX_H
+
+#include <gtk/gtk.h>
+
+#include "gb-search-manager.h"
+
+G_BEGIN_DECLS
+
+#define GB_TYPE_SEARCH_BOX (gb_search_box_get_type())
+#define GB_SEARCH_BOX(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GB_TYPE_SEARCH_BOX, GbSearchBox))
+#define GB_SEARCH_BOX_CONST(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GB_TYPE_SEARCH_BOX, GbSearchBox
const))
+#define GB_SEARCH_BOX_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GB_TYPE_SEARCH_BOX,
GbSearchBoxClass))
+#define GB_IS_SEARCH_BOX(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GB_TYPE_SEARCH_BOX))
+#define GB_IS_SEARCH_BOX_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GB_TYPE_SEARCH_BOX))
+#define GB_SEARCH_BOX_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GB_TYPE_SEARCH_BOX,
GbSearchBoxClass))
+
+typedef struct _GbSearchBox GbSearchBox;
+typedef struct _GbSearchBoxClass GbSearchBoxClass;
+typedef struct _GbSearchBoxPrivate GbSearchBoxPrivate;
+
+struct _GbSearchBox
+{
+ GtkBox parent;
+
+ /*< private >*/
+ GbSearchBoxPrivate *priv;
+};
+
+struct _GbSearchBoxClass
+{
+ GtkBoxClass parent;
+};
+
+GType gb_search_box_get_type (void);
+GtkWidget *gb_search_box_new (void);
+GbSearchManager *gb_search_box_get_search_manager (GbSearchBox *box);
+void gb_search_box_set_search_manager (GbSearchBox *box,
+ GbSearchManager *search_manager);
+
+G_END_DECLS
+
+#endif /* GB_SEARCH_BOX_H */
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]