diff --git a/gtk/inspector/style-prop-list.c b/gtk/inspector/style-prop-list.c index 3137d83e41..536661a009 100644 --- a/gtk/inspector/style-prop-list.c +++ b/gtk/inspector/style-prop-list.c @@ -31,7 +31,9 @@ #include "gtksettings.h" #include "gtktreeview.h" #include "gtktreeselection.h" -#include "gtksearchbar.h" +#include "gtkstack.h" +#include "gtksearchentry.h" +#include "gtklabel.h" enum { @@ -48,49 +50,134 @@ struct _GtkInspectorStylePropListPrivate GtkListStore *model; GtkWidget *widget; GtkWidget *tree; - GtkWidget *search_bar; GtkWidget *search_entry; + GtkWidget *search_stack; + GtkWidget *object_title; GHashTable *prop_iters; GtkTreeViewColumn *name_column; }; G_DEFINE_TYPE_WITH_PRIVATE (GtkInspectorStylePropList, gtk_inspector_style_prop_list, GTK_TYPE_BOX) +static void +search_close_clicked (GtkWidget *button, + GtkInspectorStylePropList *pl) +{ + gtk_entry_set_text (GTK_ENTRY (pl->priv->search_entry), ""); + gtk_stack_set_visible_child_name (GTK_STACK (pl->priv->search_stack), "title"); +} + +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; +} + +static void +preedit_changed_cb (GtkEntry *entry, + GtkWidget *popup, + gboolean *preedit_changed) +{ + *preedit_changed = TRUE; +} + static gboolean key_press_event (GtkWidget *window, GdkEvent *event, GtkInspectorStylePropList *pl) { - if (gtk_widget_get_mapped (GTK_WIDGET (pl))) - { - if (event->key.keyval == GDK_KEY_Return || - event->key.keyval == GDK_KEY_ISO_Enter || - event->key.keyval == GDK_KEY_KP_Enter) - { - GtkTreeSelection *selection; - GtkTreeModel *model; - GtkTreeIter iter; - GtkTreePath *path; + gboolean handled; + gboolean preedit_changed; + guint preedit_change_id; + gboolean res; + gchar *old_text, *new_text; - selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (pl->priv->tree)); - if (gtk_tree_selection_get_selected (selection, &model, &iter)) - { - path = gtk_tree_model_get_path (model, &iter); - gtk_tree_view_row_activated (GTK_TREE_VIEW (pl->priv->tree), - path, - pl->priv->name_column); - gtk_tree_path_free (path); - - return GDK_EVENT_STOP; - } - else - return GDK_EVENT_PROPAGATE; - } - - return gtk_search_bar_handle_event (GTK_SEARCH_BAR (pl->priv->search_bar), event); - } - else + if (!gtk_widget_get_mapped (GTK_WIDGET (pl))) return GDK_EVENT_PROPAGATE; + + if (is_keynav_event (event) || + event->key.keyval == GDK_KEY_space || + event->key.keyval == GDK_KEY_Menu) + return GDK_EVENT_PROPAGATE; + + if (event->key.keyval == GDK_KEY_Return || + event->key.keyval == GDK_KEY_ISO_Enter || + event->key.keyval == GDK_KEY_KP_Enter) + { + GtkTreeSelection *selection; + GtkTreeModel *model; + GtkTreeIter iter; + GtkTreePath *path; + + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (pl->priv->tree)); + if (gtk_tree_selection_get_selected (selection, &model, &iter)) + { + path = gtk_tree_model_get_path (model, &iter); + gtk_tree_view_row_activated (GTK_TREE_VIEW (pl->priv->tree), + path, + pl->priv->name_column); + gtk_tree_path_free (path); + + return GDK_EVENT_STOP; + } + else + return GDK_EVENT_PROPAGATE; + } + + if (event->key.keyval == GDK_KEY_Escape) + { + gtk_entry_set_text (GTK_ENTRY (pl->priv->search_entry), ""); + gtk_stack_set_visible_child_name (GTK_STACK (pl->priv->search_stack), "title"); + return GDK_EVENT_STOP; + } + + if (!gtk_widget_get_realized (pl->priv->search_entry)) + gtk_widget_realize (pl->priv->search_entry); + + handled = FALSE; + preedit_changed = FALSE; + preedit_change_id = g_signal_connect (pl->priv->search_entry, "preedit-changed", + G_CALLBACK (preedit_changed_cb), &preedit_changed); + + old_text = g_strdup (gtk_entry_get_text (GTK_ENTRY (pl->priv->search_entry))); + res = gtk_widget_event (pl->priv->search_entry, event); + new_text = g_strdup (gtk_entry_get_text (GTK_ENTRY (pl->priv->search_entry))); + + g_signal_handler_disconnect (pl->priv->search_entry, preedit_change_id); + + if ((res && g_strcmp0 (new_text, old_text) != 0) || preedit_changed) + { + gtk_stack_set_visible_child_name (GTK_STACK (pl->priv->search_stack), "search"); + handled = TRUE; + } + + g_free (old_text); + g_free (new_text); + + return handled ? GDK_EVENT_STOP : GDK_EVENT_PROPAGATE; } static void @@ -192,9 +279,11 @@ gtk_inspector_style_prop_list_class_init (GtkInspectorStylePropListClass *klass) gtk_widget_class_set_template_from_resource (widget_class, "/org/gtk/inspector/style-prop-list.ui"); gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStylePropList, model); gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStylePropList, tree); - gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStylePropList, search_bar); + gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStylePropList, search_stack); gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStylePropList, search_entry); + gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStylePropList, object_title); gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStylePropList, name_column); + gtk_widget_class_bind_template_callback (widget_class, search_close_clicked); } static gchar * @@ -369,6 +458,8 @@ void gtk_inspector_style_prop_list_set_object (GtkInspectorStylePropList *self, GObject *object) { + const gchar *title; + if (self->priv->widget == (GtkWidget *)object) { gtk_widget_hide (GTK_WIDGET (self)); @@ -388,6 +479,12 @@ gtk_inspector_style_prop_list_set_object (GtkInspectorStylePropList *self, return; } + title = (const gchar *)g_object_get_data (object, "gtk-inspector-object-title"); + gtk_label_set_label (GTK_LABEL (self->priv->object_title), title); + + gtk_entry_set_text (GTK_ENTRY (self->priv->search_entry), ""); + gtk_stack_set_visible_child_name (GTK_STACK (self->priv->search_stack), "title"); + self->priv->widget = (GtkWidget *)object; g_object_weak_ref (G_OBJECT (self), disconnect_each_other, object); g_object_weak_ref (G_OBJECT (object), disconnect_each_other, self); diff --git a/gtk/inspector/style-prop-list.ui b/gtk/inspector/style-prop-list.ui index 328426396a..5105cbfbb6 100644 --- a/gtk/inspector/style-prop-list.ui +++ b/gtk/inspector/style-prop-list.ui @@ -12,14 +12,57 @@