[gtk/prop-list: 55/73] inspector: Use a column view for the resource list
- From: Matthias Clasen <matthiasc src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gtk/prop-list: 55/73] inspector: Use a column view for the resource list
- Date: Sun, 15 Dec 2019 22:04:51 +0000 (UTC)
commit 21fce3ca2c8cadaf1fb591fc3bdec89ed7543e3b
Author: Matthias Clasen <mclasen redhat com>
Date: Sat Dec 7 22:26:35 2019 -0500
inspector: Use a column view for the resource list
A conversion from tree view to column view.
gtk/inspector/meson.build | 3 +-
gtk/inspector/resource-holder.c | 106 ++++++
gtk/inspector/resource-holder.h | 24 ++
gtk/inspector/resource-list.c | 713 +++++++++++++++++++++++-----------------
gtk/inspector/resource-list.ui | 60 ++--
5 files changed, 568 insertions(+), 338 deletions(-)
---
diff --git a/gtk/inspector/meson.build b/gtk/inspector/meson.build
index f297fe2d0a..ea5d77c09e 100644
--- a/gtk/inspector/meson.build
+++ b/gtk/inspector/meson.build
@@ -35,5 +35,6 @@ inspector_sources = files(
'updatesoverlay.c',
'visual.c',
'window.c',
- 'prop-holder.c'
+ 'prop-holder.c',
+ 'resource-holder.c'
)
diff --git a/gtk/inspector/resource-holder.c b/gtk/inspector/resource-holder.c
new file mode 100644
index 0000000000..d9aee9bed2
--- /dev/null
+++ b/gtk/inspector/resource-holder.c
@@ -0,0 +1,106 @@
+#include "resource-holder.h"
+
+struct _ResourceHolder {
+ GObject instance;
+
+ char *name;
+ char *path;
+ int count;
+ gsize size;
+ GListModel *children;
+ ResourceHolder *parent;
+};
+
+G_DEFINE_TYPE (ResourceHolder, resource_holder, G_TYPE_OBJECT)
+
+static void
+resource_holder_init (ResourceHolder *holder)
+{
+}
+
+static void
+resource_holder_finalize (GObject *object)
+{
+ ResourceHolder *holder = RESOURCE_HOLDER (object);
+
+ g_free (holder->name);
+ g_free (holder->path);
+ g_clear_object (&holder->children);
+
+ G_OBJECT_CLASS (resource_holder_parent_class)->finalize (object);
+}
+
+static void
+resource_holder_class_init (ResourceHolderClass *class)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (class);
+
+ object_class->finalize = resource_holder_finalize;
+}
+
+ResourceHolder *
+resource_holder_new (const char *name,
+ const char *path,
+ int count,
+ gsize size,
+ GListModel *children)
+{
+ ResourceHolder *holder;
+
+ holder = g_object_new (RESOURCE_TYPE_HOLDER, NULL);
+
+ holder->name = g_strdup (name);
+ holder->path = g_strdup (path);
+ holder->count = count;
+ holder->size = size;
+ g_set_object (&holder->children, children);
+
+ if (children)
+ {
+ int i;
+ for (i = 0; i < g_list_model_get_n_items (children); i++)
+ {
+ ResourceHolder *child = g_list_model_get_item (children, i);
+ child->parent = holder;
+ g_object_unref (child);
+ }
+ }
+
+ return holder;
+}
+
+const char *
+resource_holder_get_name (ResourceHolder *holder)
+{
+ return holder->name;
+}
+
+const char *
+resource_holder_get_path (ResourceHolder *holder)
+{
+ return holder->path;
+}
+
+int
+resource_holder_get_count (ResourceHolder *holder)
+{
+ return holder->count;
+}
+
+gsize
+resource_holder_get_size (ResourceHolder *holder)
+{
+ return holder->size;
+}
+
+GListModel *
+resource_holder_get_children (ResourceHolder *holder)
+{
+ return holder->children;
+}
+
+ResourceHolder *
+resource_holder_get_parent (ResourceHolder *holder)
+{
+ return holder->parent;
+}
diff --git a/gtk/inspector/resource-holder.h b/gtk/inspector/resource-holder.h
new file mode 100644
index 0000000000..bb847f887f
--- /dev/null
+++ b/gtk/inspector/resource-holder.h
@@ -0,0 +1,24 @@
+#ifndef __RESOURCE_HOLDER_H__
+#define __RESOURCE_HOLDER_H__
+
+#include <gtk/gtk.h>
+
+#define RESOURCE_TYPE_HOLDER (resource_holder_get_type ())
+
+G_DECLARE_FINAL_TYPE (ResourceHolder, resource_holder, RESOURCE, HOLDER, GObject)
+
+ResourceHolder * resource_holder_new (const char *name,
+ const char *path,
+ int count,
+ gsize size,
+ GListModel *children);
+
+const char *resource_holder_get_name (ResourceHolder *holder);
+const char *resource_holder_get_path (ResourceHolder *holder);
+int resource_holder_get_count (ResourceHolder *holder);
+gsize resource_holder_get_size (ResourceHolder *holder);
+GListModel *resource_holder_get_children (ResourceHolder *holder);
+ResourceHolder *resource_holder_get_parent (ResourceHolder *holder);
+
+#endif /* __RESOURCE_HOLDER_H__ */
+
diff --git a/gtk/inspector/resource-list.c b/gtk/inspector/resource-list.c
index 9f413d0ef1..9be6f7387f 100644
--- a/gtk/inspector/resource-list.c
+++ b/gtk/inspector/resource-list.c
@@ -19,8 +19,6 @@
#include "resource-list.h"
-#include "treewalk.h"
-
#include "gtkbutton.h"
#include "gtklabel.h"
#include "gtksearchbar.h"
@@ -30,6 +28,7 @@
#include "gtktreeselection.h"
#include "gtktreestore.h"
#include "gtkeventcontrollerkey.h"
+#include "resource-holder.h"
#include <glib/gi18n-lib.h>
@@ -39,14 +38,6 @@ enum
PROP_BUTTONS
};
-enum
-{
- COLUMN_NAME,
- COLUMN_PATH,
- COLUMN_COUNT,
- COLUMN_SIZE
-};
-
struct _GtkInspectorResourceListPrivate
{
GtkTreeStore *model;
@@ -59,35 +50,34 @@ struct _GtkInspectorResourceListPrivate
GtkWidget *size_label;
GtkWidget *info_grid;
GtkWidget *stack;
- GtkWidget *tree;
GtkWidget *buttons;
GtkWidget *open_details_button;
GtkWidget *close_details_button;
- GtkTreeViewColumn *path_column;
- GtkTreeViewColumn *count_column;
- GtkCellRenderer *count_renderer;
- GtkTreeViewColumn *size_column;
- GtkCellRenderer *size_renderer;
GtkWidget *search_bar;
GtkWidget *search_entry;
- GtkTreeWalk *walk;
- gint search_length;
+
+ GtkWidget *list;
+ GtkColumnViewColumn *path;
+ GtkColumnViewColumn *count;
+ GtkColumnViewColumn *size;
+
+ GtkTreeListModel *tree_model;
+ GtkSingleSelection *selection;
};
G_DEFINE_TYPE_WITH_PRIVATE (GtkInspectorResourceList, gtk_inspector_resource_list, GTK_TYPE_BOX)
-static void
-load_resources_recurse (GtkInspectorResourceList *sl,
- GtkTreeIter *parent,
- const gchar *path,
- gint *count_out,
- gsize *size_out)
+static GListModel *
+load_resources_recurse (const char *path,
+ int *count_out,
+ gsize *size_out)
{
gchar **names;
gint i;
- GtkTreeIter iter;
- guint64 stored_size;
+ GListStore *result;
+
+ result = g_list_store_new (RESOURCE_TYPE_HOLDER);
names = g_resources_enumerate_children (path, 0, NULL);
for (i = 0; names[i]; i++)
@@ -97,6 +87,8 @@ load_resources_recurse (GtkInspectorResourceList *sl,
gboolean has_slash;
gint count;
gsize size;
+ GListModel *children;
+ ResourceHolder *holder;
p = g_strconcat (path, names[i], NULL);
@@ -106,18 +98,13 @@ load_resources_recurse (GtkInspectorResourceList *sl,
if (has_slash)
names[i][len - 1] = '\0';
- gtk_tree_store_append (sl->priv->model, &iter, parent);
- gtk_tree_store_set (sl->priv->model, &iter,
- COLUMN_NAME, names[i],
- COLUMN_PATH, p,
- -1);
-
count = 0;
size = 0;
if (has_slash)
{
- load_resources_recurse (sl, &iter, p, &count, &size);
+ children = load_resources_recurse (p, &count, &size);
+
*count_out += count;
*size_out += size;
}
@@ -129,53 +116,41 @@ load_resources_recurse (GtkInspectorResourceList *sl,
*count_out += 1;
*size_out += size;
}
+ children = NULL;
}
- stored_size = size;
- gtk_tree_store_set (sl->priv->model, &iter,
- COLUMN_COUNT, count,
- COLUMN_SIZE, stored_size,
- -1);
+ holder = resource_holder_new (names[i], p, count, size, children);
+ g_clear_object (&children);
+ g_list_store_append (result, holder);
+ g_object_unref (holder);
g_free (p);
}
+
g_strfreev (names);
+ return G_LIST_MODEL (result);
}
static gboolean
populate_details (GtkInspectorResourceList *rl,
- GtkTreePath *tree_path)
+ ResourceHolder *holder)
{
- GtkTreeIter iter;
- gchar *path;
- gchar *name;
+ const char *path;
+ const char *name;
GBytes *bytes;
gchar *type;
gconstpointer data;
- gint count;
gsize size;
- guint64 stored_size;
GError *error = NULL;
gchar *markup;
-
- gtk_tree_model_get_iter (GTK_TREE_MODEL (rl->priv->model), &iter, tree_path);
-
- gtk_tree_model_get (GTK_TREE_MODEL (rl->priv->model), &iter,
- COLUMN_PATH, &path,
- COLUMN_NAME, &name,
- COLUMN_COUNT, &count,
- COLUMN_SIZE, &stored_size,
- -1);
- size = stored_size;
+ path = resource_holder_get_path (holder);
+ name = resource_holder_get_name (holder);
+ size = resource_holder_get_size (holder);
if (g_str_has_suffix (path, "/"))
- {
- g_free (path);
- g_free (name);
- return FALSE;
- }
+ return FALSE;
markup = g_strconcat ("<span face='Monospace' size='small'>", path, "</span>", NULL);
gtk_label_set_markup (GTK_LABEL (rl->priv->name_label), markup);
@@ -231,237 +206,145 @@ populate_details (GtkInspectorResourceList *rl,
g_free (content_text);
}
- g_free (path);
- g_free (name);
-
return TRUE;
}
static void
-row_activated (GtkTreeView *treeview,
- GtkTreePath *path,
- GtkTreeViewColumn *column,
- GtkInspectorResourceList *sl)
+on_row_activated (GtkColumnView *view,
+ guint position,
+ GtkInspectorResourceList *rl)
{
- if (!populate_details (sl, path))
- return;
+ gpointer item;
+ ResourceHolder *holder;
- gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->stack), "details");
- gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->buttons), "details");
+ item = g_list_model_get_item (G_LIST_MODEL (rl->priv->selection), position);
+ holder = gtk_tree_list_row_get_item (item);
+ g_object_unref (item);
+ if (populate_details (rl, holder))
+ {
+ gtk_stack_set_visible_child_name (GTK_STACK (rl->priv->stack), "details");
+ gtk_stack_set_visible_child_name (GTK_STACK (rl->priv->buttons), "details");
+ }
+ g_object_unref (holder);
}
static gboolean
can_show_details (GtkInspectorResourceList *rl)
{
- GtkTreeSelection *selection;
- GtkTreeModel *model;
- GtkTreeIter iter;
- gchar *path;
+ gpointer item;
+ ResourceHolder *holder;
+ const char *path;
- selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (rl->priv->tree));
- if (!gtk_tree_selection_get_selected (selection, &model, &iter))
+ item = gtk_single_selection_get_selected_item (rl->priv->selection);
+ holder = gtk_tree_list_row_get_item (item);
+ if (holder == NULL)
return FALSE;
-
- gtk_tree_model_get (GTK_TREE_MODEL (rl->priv->model), &iter,
- COLUMN_PATH, &path,
- -1);
-
- if (g_str_has_suffix (path, "/"))
- {
- g_free (path);
- return FALSE;
- }
-
- g_free (path);
- return TRUE;
+ path = resource_holder_get_path (holder);
+ g_object_unref (holder);
+ return !g_str_has_suffix (path, "/");
}
static void
-on_selection_changed (GtkTreeSelection *selection,
+on_selection_changed (GtkSelectionModel *selection,
+ guint position,
+ guint n_items,
GtkInspectorResourceList *rl)
{
- GtkTreeIter iter;
-
- if (gtk_tree_selection_get_selected (selection, NULL, &iter))
- gtk_tree_walk_reset (rl->priv->walk, &iter);
- else
- gtk_tree_walk_reset (rl->priv->walk, NULL);
-
gtk_widget_set_sensitive (rl->priv->open_details_button, can_show_details (rl));
}
static void
open_details (GtkWidget *button,
- GtkInspectorResourceList *sl)
+ GtkInspectorResourceList *rl)
{
- GtkTreeSelection *selection;
- GtkTreeModel *model;
- GtkTreeIter iter;
- GtkTreePath *path;
+ gpointer item;
+ ResourceHolder *holder;
- selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (sl->priv->tree));
- if (!gtk_tree_selection_get_selected (selection, &model, &iter))
+ item = gtk_single_selection_get_selected_item (rl->priv->selection);
+ holder = gtk_tree_list_row_get_item (item);
+ if (holder == NULL)
return;
-
- path = gtk_tree_model_get_path (model, &iter);
- if (populate_details (sl, path))
+ if (populate_details (rl, holder))
{
- gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->stack), "details");
- gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->buttons), "details");
+ gtk_stack_set_visible_child_name (GTK_STACK (rl->priv->stack), "details");
+ gtk_stack_set_visible_child_name (GTK_STACK (rl->priv->buttons), "details");
}
-
- gtk_tree_path_free (path);
+ g_object_unref (holder);
}
static void
close_details (GtkWidget *button,
- GtkInspectorResourceList *sl)
+ GtkInspectorResourceList *rl)
{
- gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->stack), "list");
- gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->buttons), "list");
+ gtk_stack_set_visible_child_name (GTK_STACK (rl->priv->stack), "list");
+ gtk_stack_set_visible_child_name (GTK_STACK (rl->priv->buttons), "list");
}
-static void
-load_resources (GtkInspectorResourceList *sl)
+static GListModel *
+load_resources (void)
{
gint count = 0;
gsize size = 0;
- load_resources_recurse (sl, NULL, "/", &count, &size);
-}
-
-static void
-count_data_func (GtkTreeViewColumn *col,
- GtkCellRenderer *cell,
- GtkTreeModel *model,
- GtkTreeIter *iter,
- gpointer data)
-{
- gint count;
- gchar *text;
-
- gtk_tree_model_get (model, iter, COLUMN_COUNT, &count, -1);
- if (count > 0)
- {
- text = g_strdup_printf ("%d", count);
- g_object_set (cell, "text", text, NULL);
- g_free (text);
- }
- else
- g_object_set (cell, "text", "", NULL);
-}
-
-static void
-size_data_func (GtkTreeViewColumn *col,
- GtkCellRenderer *cell,
- GtkTreeModel *model,
- GtkTreeIter *iter,
- gpointer data)
-{
- gsize size;
- guint64 stored_size;
- gchar *text;
-
- gtk_tree_model_get (model, iter, COLUMN_SIZE, &stored_size, -1);
- size = stored_size;
- text = g_format_size (size);
- g_object_set (cell, "text", text, NULL);
- g_free (text);
+ return load_resources_recurse ("/", &count, &size);
}
static void
on_map (GtkWidget *widget)
{
- GtkInspectorResourceList *sl = GTK_INSPECTOR_RESOURCE_LIST (widget);
+ GtkInspectorResourceList *rl = GTK_INSPECTOR_RESOURCE_LIST (widget);
- gtk_tree_view_expand_all (GTK_TREE_VIEW (sl->priv->tree));
- gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->stack), "list");
- gtk_widget_set_sensitive (sl->priv->open_details_button, can_show_details (sl));
+ gtk_stack_set_visible_child_name (GTK_STACK (rl->priv->stack), "list");
+ gtk_widget_set_sensitive (rl->priv->open_details_button, can_show_details (rl));
}
-static void
-move_search_to_row (GtkInspectorResourceList *sl,
- GtkTreeIter *iter)
-{
- GtkTreeSelection *selection;
- GtkTreePath *path;
-
- selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (sl->priv->tree));
- path = gtk_tree_model_get_path (GTK_TREE_MODEL (sl->priv->model), iter);
- gtk_tree_view_expand_to_path (GTK_TREE_VIEW (sl->priv->tree), path);
- gtk_tree_selection_select_path (selection, path);
- gtk_tree_view_scroll_to_cell (GTK_TREE_VIEW (sl->priv->tree), path, NULL, TRUE, 0.5, 0.0);
- gtk_tree_path_free (path);
-}
+static gboolean search (GtkInspectorResourceList *rl,
+ gboolean forward,
+ gboolean force_progress);
static gboolean
key_pressed (GtkEventController *controller,
guint keyval,
guint keycode,
GdkModifierType state,
- GtkInspectorResourceList *sl)
+ GtkInspectorResourceList *rl)
{
- if (gtk_widget_get_mapped (GTK_WIDGET (sl)))
+ if (gtk_widget_get_mapped (GTK_WIDGET (rl)))
{
GdkModifierType default_accel;
gboolean search_started;
- search_started = gtk_search_bar_get_search_mode (GTK_SEARCH_BAR (sl->priv->search_bar));
- default_accel = gtk_widget_get_modifier_mask (GTK_WIDGET (sl),
GDK_MODIFIER_INTENT_PRIMARY_ACCELERATOR);
+ search_started = gtk_search_bar_get_search_mode (GTK_SEARCH_BAR (rl->priv->search_bar));
+ default_accel = gtk_widget_get_modifier_mask (GTK_WIDGET (rl),
GDK_MODIFIER_INTENT_PRIMARY_ACCELERATOR);
if (search_started &&
(keyval == GDK_KEY_Return ||
keyval == GDK_KEY_ISO_Enter ||
keyval == GDK_KEY_KP_Enter))
{
- GtkTreeSelection *selection;
- GtkTreeModel *model;
- GtkTreeIter iter;
- GtkTreePath *path;
-
- selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (sl->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 (sl->priv->tree),
- path,
- sl->priv->path_column);
- gtk_tree_path_free (path);
-
- return GDK_EVENT_STOP;
- }
- else
- return GDK_EVENT_PROPAGATE;
+ gtk_widget_activate (GTK_WIDGET (rl->priv->list));
+ return GDK_EVENT_PROPAGATE;
}
else if (search_started &&
(keyval == GDK_KEY_Escape))
{
- gtk_search_bar_set_search_mode (GTK_SEARCH_BAR (sl->priv->search_bar), FALSE);
+ gtk_search_bar_set_search_mode (GTK_SEARCH_BAR (rl->priv->search_bar), FALSE);
return GDK_EVENT_STOP;
}
else if (search_started &&
((state & (default_accel | GDK_SHIFT_MASK)) == (default_accel | GDK_SHIFT_MASK)) &&
(keyval == GDK_KEY_g || keyval == GDK_KEY_G))
{
- GtkTreeIter iter;
- if (gtk_tree_walk_next_match (sl->priv->walk, TRUE, TRUE, &iter))
- move_search_to_row (sl, &iter);
- else
- gtk_widget_error_bell (GTK_WIDGET (sl));
-
+ if (!search (rl, FALSE, TRUE))
+ gtk_widget_error_bell (GTK_WIDGET (rl));
return GDK_EVENT_STOP;
}
else if (search_started &&
((state & (default_accel | GDK_SHIFT_MASK)) == default_accel) &&
(keyval == GDK_KEY_g || keyval == GDK_KEY_G))
{
- GtkTreeIter iter;
-
- if (gtk_tree_walk_next_match (sl->priv->walk, TRUE, FALSE, &iter))
- move_search_to_row (sl, &iter);
- else
- gtk_widget_error_bell (GTK_WIDGET (sl));
-
+ if (!search (rl, TRUE, TRUE))
+ gtk_widget_error_bell (GTK_WIDGET (rl));
return GDK_EVENT_STOP;
}
}
@@ -478,7 +361,7 @@ destroy_controller (GtkEventController *controller)
static void
root (GtkWidget *widget)
{
- GtkInspectorResourceList *sl = GTK_INSPECTOR_RESOURCE_LIST (widget);
+ GtkInspectorResourceList *rl = GTK_INSPECTOR_RESOURCE_LIST (widget);
GtkEventController *controller;
GtkWidget *toplevel;
@@ -491,7 +374,7 @@ root (GtkWidget *widget)
g_signal_connect (controller, "key-pressed", G_CALLBACK (key_pressed), widget);
gtk_widget_add_controller (toplevel, controller);
- gtk_search_bar_set_key_capture_widget (GTK_SEARCH_BAR (sl->priv->search_bar), toplevel);
+ gtk_search_bar_set_key_capture_widget (GTK_SEARCH_BAR (rl->priv->search_bar), toplevel);
}
static void
@@ -505,27 +388,6 @@ unroot (GtkWidget *widget)
GTK_WIDGET_CLASS (gtk_inspector_resource_list_parent_class)->unroot (widget);
}
-static void
-on_search_changed (GtkSearchEntry *entry,
- GtkInspectorResourceList *sl)
-{
- GtkTreeIter iter;
- gint length;
- gboolean backwards;
-
- length = strlen (gtk_editable_get_text (GTK_EDITABLE (entry)));
- backwards = length < sl->priv->search_length;
- sl->priv->search_length = length;
-
- if (length == 0)
- return;
-
- if (gtk_tree_walk_next_match (sl->priv->walk, backwards, backwards, &iter))
- move_search_to_row (sl, &iter);
- else if (!backwards)
- gtk_widget_error_bell (GTK_WIDGET (sl));
-}
-
static gboolean
match_string (const gchar *string,
const gchar *text)
@@ -544,107 +406,247 @@ match_string (const gchar *string,
}
static gboolean
-match_row (GtkTreeModel *model,
- GtkTreeIter *iter,
- gpointer data)
+match_object (GObject *object,
+ const char *text)
{
- GtkInspectorResourceList *sl = data;
- gchar *name, *path;
- const gchar *text;
- gboolean match;
+ const char *name = resource_holder_get_name (RESOURCE_HOLDER (object));
- text = gtk_editable_get_text (GTK_EDITABLE (sl->priv->search_entry));
- gtk_tree_model_get (model, iter,
- COLUMN_NAME, &name,
- COLUMN_PATH, &path,
- -1);
+ if (match_string (name, text))
+ return TRUE;
- match = (match_string (name, text) ||
- match_string (path, text));
+ return FALSE;
+}
- g_free (name);
- g_free (path);
+static GObject *
+search_children (GObject *object,
+ const char *text,
+ gboolean forward)
+{
+ GListModel *children;
+ GObject *child, *result;
+ guint i, n;
- return match;
+ children = resource_holder_get_children (RESOURCE_HOLDER (object));
+ if (children == NULL)
+ return NULL;
+
+ n = g_list_model_get_n_items (children);
+ for (i = 0; i < n; i++)
+ {
+ child = g_list_model_get_item (children, forward ? i : n - i - 1);
+ if (match_object (child, text))
+ return child;
+
+ result = search_children (child, text, forward);
+ g_object_unref (child);
+ if (result)
+ return result;
+ }
+
+ return NULL;
+}
+
+static guint
+model_get_item_index (GListModel *model,
+ gpointer item)
+{
+ gpointer cmp;
+ guint i;
+
+ for (i = 0; (cmp = g_list_model_get_item (model, i)); i++)
+ {
+ if (cmp == item)
+ {
+ g_object_unref (cmp);
+ return i;
+ }
+ g_object_unref (cmp);
+ }
+
+ return G_MAXUINT;
+}
+
+static GtkTreeListRow *
+find_and_expand_object (GtkTreeListModel *model,
+ GObject *object)
+{
+ GtkTreeListRow *result;
+ GObject *parent;
+ guint pos;
+
+ parent = G_OBJECT (resource_holder_get_parent (RESOURCE_HOLDER (object)));
+ if (parent)
+ {
+ GtkTreeListRow *parent_row = find_and_expand_object (model, parent);
+ if (parent_row == NULL)
+ return NULL;
+
+ gtk_tree_list_row_set_expanded (parent_row, TRUE);
+ pos = model_get_item_index (gtk_tree_list_row_get_children (parent_row), object);
+ result = gtk_tree_list_row_get_child_row (parent_row, pos);
+ g_object_unref (parent_row);
+ }
+ else
+ {
+ pos = model_get_item_index (gtk_tree_list_model_get_model (model), object);
+ result = gtk_tree_list_model_get_child_row (model, pos);
+ }
+
+ return result;
}
static void
-search_mode_changed (GObject *search_bar,
- GParamSpec *pspec,
- GtkInspectorResourceList *sl)
+select_object (GtkInspectorResourceList *rl,
+ GObject *object)
+{
+ GtkTreeListRow *row_item;
+
+ row_item = find_and_expand_object (rl->priv->tree_model, object);
+ if (row_item == NULL)
+ return;
+
+ gtk_single_selection_set_selected (rl->priv->selection,
+ gtk_tree_list_row_get_position (row_item));
+}
+
+static gboolean
+search (GtkInspectorResourceList *rl,
+ gboolean forward,
+ gboolean force_progress)
{
- if (!gtk_search_bar_get_search_mode (GTK_SEARCH_BAR (search_bar)))
+ GListModel *model = G_LIST_MODEL (rl->priv->tree_model);
+ GtkTreeListRow *row_item;
+ GObject *child, *result;
+ guint i, selected, n, row;
+ const char *text;
+
+ text = gtk_editable_get_text (GTK_EDITABLE (rl->priv->search_entry));
+ selected = gtk_single_selection_get_selected (rl->priv->selection);
+ n = g_list_model_get_n_items (model);
+ if (selected >= n)
+ selected = 0;
+
+ for (i = 0; i < n; i++)
{
- gtk_tree_walk_reset (sl->priv->walk, NULL);
- sl->priv->search_length = 0;
+ row = (selected + (forward ? i : n - i - 1)) % n;
+ row_item = g_list_model_get_item (model, row);
+ child = gtk_tree_list_row_get_item (row_item);
+ if (i > 0 || !force_progress)
+ {
+ if (match_object (child, text))
+ {
+ gtk_single_selection_set_selected (rl->priv->selection, row);
+ g_object_unref (child);
+ g_object_unref (row_item);
+ return TRUE;
+ }
+ }
+
+ if (!gtk_tree_list_row_get_expanded (row_item))
+ {
+ result = search_children (child, text, forward);
+ if (result)
+ {
+ select_object (rl, result);
+ g_object_unref (result);
+ g_object_unref (child);
+ g_object_unref (row_item);
+ return TRUE;
+ }
+ }
+ g_object_unref (child);
+ g_object_unref (row_item);
}
+
+ return FALSE;
+}
+
+static void
+on_search_changed (GtkSearchEntry *entry,
+ GtkInspectorResourceList *rl)
+{
+ if (!search (rl, TRUE, FALSE))
+ gtk_widget_error_bell (GTK_WIDGET (rl));
}
static void
next_match (GtkButton *button,
- GtkInspectorResourceList *sl)
+ GtkInspectorResourceList *rl)
{
- if (gtk_search_bar_get_search_mode (GTK_SEARCH_BAR (sl->priv->search_bar)))
+ if (gtk_search_bar_get_search_mode (GTK_SEARCH_BAR (rl->priv->search_bar)))
{
- GtkTreeIter iter;
-
- if (gtk_tree_walk_next_match (sl->priv->walk, TRUE, FALSE, &iter))
- move_search_to_row (sl, &iter);
- else
- gtk_widget_error_bell (GTK_WIDGET (sl));
+ if (!search (rl, TRUE, TRUE))
+ gtk_widget_error_bell (GTK_WIDGET (rl));
}
}
static void
previous_match (GtkButton *button,
- GtkInspectorResourceList *sl)
+ GtkInspectorResourceList *rl)
{
- if (gtk_search_bar_get_search_mode (GTK_SEARCH_BAR (sl->priv->search_bar)))
+ if (gtk_search_bar_get_search_mode (GTK_SEARCH_BAR (rl->priv->search_bar)))
{
- GtkTreeIter iter;
-
- if (gtk_tree_walk_next_match (sl->priv->walk, TRUE, TRUE, &iter))
- move_search_to_row (sl, &iter);
- else
- gtk_widget_error_bell (GTK_WIDGET (sl));
+ if (!search (rl, FALSE, TRUE))
+ gtk_widget_error_bell (GTK_WIDGET (rl));
}
}
static void
-gtk_inspector_resource_list_init (GtkInspectorResourceList *sl)
+stop_search (GtkWidget *entry,
+ GtkInspectorResourceList *rl)
{
- sl->priv = gtk_inspector_resource_list_get_instance_private (sl);
+ gtk_editable_set_text (GTK_EDITABLE (rl->priv->search_entry), "");
+ gtk_search_bar_set_search_mode (GTK_SEARCH_BAR (rl->priv->search_bar), FALSE);
+}
+
+static void
+gtk_inspector_resource_list_init (GtkInspectorResourceList *rl)
+{
+ rl->priv = gtk_inspector_resource_list_get_instance_private (rl);
- gtk_widget_init_template (GTK_WIDGET (sl));
+ gtk_widget_init_template (GTK_WIDGET (rl));
- gtk_tree_view_column_set_cell_data_func (sl->priv->count_column,
- sl->priv->count_renderer,
- count_data_func, sl, NULL);
- gtk_tree_view_column_set_cell_data_func (sl->priv->size_column,
- sl->priv->size_renderer,
- size_data_func, sl, NULL);
+ g_signal_connect (rl, "map", G_CALLBACK (on_map), NULL);
- g_signal_connect (sl, "map", G_CALLBACK (on_map), NULL);
+ gtk_search_bar_connect_entry (GTK_SEARCH_BAR (rl->priv->search_bar),
+ GTK_EDITABLE (rl->priv->search_entry));
+}
- gtk_search_bar_connect_entry (GTK_SEARCH_BAR (sl->priv->search_bar),
- GTK_EDITABLE (sl->priv->search_entry));
+static GListModel *
+create_model_for_object (gpointer item, gpointer data)
+{
+ GListModel *model = resource_holder_get_children (RESOURCE_HOLDER (item));
- g_signal_connect (sl->priv->search_bar, "notify::search-mode-enabled",
- G_CALLBACK (search_mode_changed), sl);
- sl->priv->walk = gtk_tree_walk_new (GTK_TREE_MODEL (sl->priv->model), match_row, sl, NULL);
+ if (model)
+ return g_object_ref (model);
+
+ return NULL;
}
static void
constructed (GObject *object)
{
GtkInspectorResourceList *rl = GTK_INSPECTOR_RESOURCE_LIST (object);
-
+ GListModel *root_model;
+
g_signal_connect (rl->priv->open_details_button, "clicked",
G_CALLBACK (open_details), rl);
g_signal_connect (rl->priv->close_details_button, "clicked",
G_CALLBACK (close_details), rl);
-
- load_resources (rl);
+
+ root_model = load_resources ();
+ rl->priv->tree_model = gtk_tree_list_model_new (FALSE,
+ root_model,
+ FALSE,
+ create_model_for_object,
+ NULL,
+ NULL);
+ rl->priv->selection = gtk_single_selection_new (G_LIST_MODEL (rl->priv->tree_model));
+ g_object_unref (root_model);
+
+ gtk_column_view_set_model (GTK_COLUMN_VIEW (rl->priv->list), G_LIST_MODEL (rl->priv->selection));
+
+ g_signal_connect (rl->priv->selection, "selection-changed", G_CALLBACK (on_selection_changed), rl);
}
static void
@@ -692,13 +694,120 @@ set_property (GObject *object,
static void
finalize (GObject *object)
{
- GtkInspectorResourceList *sl = GTK_INSPECTOR_RESOURCE_LIST (object);
+ GtkInspectorResourceList *rl = GTK_INSPECTOR_RESOURCE_LIST (object);
- gtk_tree_walk_free (sl->priv->walk);
+ g_object_unref (rl->priv->selection);
+ g_object_unref (rl->priv->tree_model);
G_OBJECT_CLASS (gtk_inspector_resource_list_parent_class)->finalize (object);
}
+static void
+setup_name_cb (GtkSignalListItemFactory *factory,
+ GtkListItem *list_item)
+{
+ GtkWidget *expander;
+ GtkWidget *label;
+
+ expander = gtk_tree_expander_new ();
+ gtk_list_item_set_child (list_item, expander);
+
+ label = gtk_label_new (NULL);
+ gtk_widget_set_margin_start (label, 5);
+ gtk_widget_set_margin_end (label, 5);
+ gtk_label_set_xalign (GTK_LABEL (label), 0.0);
+ gtk_tree_expander_set_child (GTK_TREE_EXPANDER (expander), label);
+}
+
+static void
+bind_name_cb (GtkSignalListItemFactory *factory,
+ GtkListItem *list_item)
+{
+ GtkTreeListRow *list_row;
+ GtkWidget *expander;
+ GtkWidget *label;
+ gpointer item;
+
+ list_row = gtk_list_item_get_item (list_item);
+ expander = gtk_list_item_get_child (list_item);
+ gtk_tree_expander_set_list_row (GTK_TREE_EXPANDER (expander), list_row);
+ item = gtk_tree_list_row_get_item (list_row);
+ label = gtk_tree_expander_get_child (GTK_TREE_EXPANDER (expander));
+
+ gtk_label_set_label (GTK_LABEL (label), resource_holder_get_name (RESOURCE_HOLDER (item)));
+ g_object_unref (item);
+}
+
+static void
+setup_size_cb (GtkSignalListItemFactory *factory,
+ GtkListItem *list_item)
+{
+ GtkWidget *label;
+
+ label = gtk_label_new (NULL);
+ gtk_widget_set_margin_start (label, 5);
+ gtk_widget_set_margin_end (label, 5);
+ gtk_label_set_xalign (GTK_LABEL (label), 1.0);
+ gtk_list_item_set_child (list_item, label);
+}
+
+static void
+bind_size_cb (GtkSignalListItemFactory *factory,
+ GtkListItem *list_item)
+{
+ GObject *item;
+ GtkWidget *label;
+ gsize size;
+ char *text;
+
+ item = gtk_tree_list_row_get_item (gtk_list_item_get_item (list_item));
+ label = gtk_list_item_get_child (list_item);
+
+ size = resource_holder_get_size (RESOURCE_HOLDER (item));
+ text = g_format_size (size);
+ gtk_label_set_label (GTK_LABEL (label), text);
+ g_free (text);
+
+ g_object_unref (item);
+}
+
+static void
+setup_count_cb (GtkSignalListItemFactory *factory,
+ GtkListItem *list_item)
+{
+ GtkWidget *label;
+
+ label = gtk_label_new (NULL);
+ gtk_widget_set_margin_start (label, 5);
+ gtk_widget_set_margin_end (label, 5);
+ gtk_label_set_xalign (GTK_LABEL (label), 1.0);
+ gtk_list_item_set_child (list_item, label);
+}
+
+static void
+bind_count_cb (GtkSignalListItemFactory *factory,
+ GtkListItem *list_item)
+{
+ GObject *item;
+ GtkWidget *label;
+ int count;
+ char *text;
+
+ item = gtk_tree_list_row_get_item (gtk_list_item_get_item (list_item));
+ label = gtk_list_item_get_child (list_item);
+
+ count = resource_holder_get_count (RESOURCE_HOLDER (item));
+ if (count > 0)
+ {
+ text = g_strdup_printf ("%d", count);
+ gtk_label_set_label (GTK_LABEL (label), text);
+ g_free (text);
+ }
+ else
+ gtk_label_set_label (GTK_LABEL (label), "");
+ g_object_unref (item);
+}
+
static void
gtk_inspector_resource_list_class_init (GtkInspectorResourceListClass *klass)
{
@@ -727,21 +836,25 @@ gtk_inspector_resource_list_class_init (GtkInspectorResourceListClass *klass)
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, type);
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, size_label);
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, info_grid);
- gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, count_column);
- gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, count_renderer);
- gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, size_column);
- gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, size_renderer);
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, stack);
- gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, tree);
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, search_bar);
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, search_entry);
- gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, path_column);
+ gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, list);
+ gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, path);
+ gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, count);
+ gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorResourceList, size);
- gtk_widget_class_bind_template_callback (widget_class, row_activated);
- gtk_widget_class_bind_template_callback (widget_class, on_selection_changed);
gtk_widget_class_bind_template_callback (widget_class, on_search_changed);
+ gtk_widget_class_bind_template_callback (widget_class, on_row_activated);
gtk_widget_class_bind_template_callback (widget_class, next_match);
gtk_widget_class_bind_template_callback (widget_class, previous_match);
+ gtk_widget_class_bind_template_callback (widget_class, stop_search);
+ gtk_widget_class_bind_template_callback (widget_class, setup_name_cb);
+ gtk_widget_class_bind_template_callback (widget_class, bind_name_cb);
+ gtk_widget_class_bind_template_callback (widget_class, setup_count_cb);
+ gtk_widget_class_bind_template_callback (widget_class, bind_count_cb);
+ gtk_widget_class_bind_template_callback (widget_class, setup_size_cb);
+ gtk_widget_class_bind_template_callback (widget_class, bind_size_cb);
}
// vim: set et sw=2 ts=2:
diff --git a/gtk/inspector/resource-list.ui b/gtk/inspector/resource-list.ui
index 7493d98f3a..d6db726f54 100644
--- a/gtk/inspector/resource-list.ui
+++ b/gtk/inspector/resource-list.ui
@@ -30,6 +30,9 @@
<object class="GtkSearchEntry" id="search_entry">
<property name="max-width-chars">40</property>
<signal name="search-changed" handler="on_search_changed"/>
+ <signal name="next-match" handler="next_match"/>
+ <signal name="previous-match" handler="previous_match"/>
+ <signal name="stop-search" handler="stop_search"/>
</object>
</child>
<child>
@@ -53,56 +56,39 @@
<property name="expand">1</property>
<property name="hscrollbar-policy">never</property>
<child>
- <object class="GtkTreeView" id="tree">
- <property name="model">model</property>
- <property name="enable-search">0</property>
- <property name="enable-grid-lines">vertical</property>
- <signal name="row-activated" handler="row_activated"/>
- <child internal-child="selection">
- <object class="GtkTreeSelection">
- <signal name="changed" handler="on_selection_changed"/>
- </object>
- </child>
+ <object class="GtkColumnView" id="list">
+ <signal name="activate" handler="on_row_activated"/>
<child>
- <object class="GtkTreeViewColumn" id="path_column">
+ <object class="GtkColumnViewColumn" id="path">
<property name="title" translatable="yes">Path</property>
- <property name="resizable">1</property>
- <property name="sort-column-id">0</property>
- <child>
- <object class="GtkCellRendererText">
- <property name="scale">0.8</property>
- <property name="ellipsize">end</property>
- <property name="width-chars">10</property>
- <property name="max-width-chars">5</property>
+ <property name="factory">
+ <object class="GtkSignalListItemFactory">
+ <signal name="setup" handler="setup_name_cb"/>
+ <signal name="bind" handler="bind_name_cb"/>
</object>
- <attributes>
- <attribute name="text">0</attribute>
- </attributes>
- </child>
+ </property>
</object>
</child>
<child>
- <object class="GtkTreeViewColumn" id="count_column">
+ <object class="GtkColumnViewColumn" id="count">
<property name="title" translatable="yes">Count</property>
- <property name="resizable">1</property>
- <property name="sort-column-id">1</property>
- <child>
- <object class="GtkCellRendererText" id="count_renderer">
- <property name="scale">0.8</property>
+ <property name="factory">
+ <object class="GtkSignalListItemFactory">
+ <signal name="setup" handler="setup_count_cb"/>
+ <signal name="bind" handler="bind_count_cb"/>
</object>
- </child>
+ </property>
</object>
</child>
<child>
- <object class="GtkTreeViewColumn" id="size_column">
+ <object class="GtkColumnViewColumn" id="size">
<property name="title" translatable="yes">Size</property>
- <property name="resizable">1</property>
- <property name="sort-column-id">2</property>
- <child>
- <object class="GtkCellRendererText" id="size_renderer">
- <property name="scale">0.8</property>
+ <property name="factory">
+ <object class="GtkSignalListItemFactory">
+ <signal name="setup" handler="setup_size_cb"/>
+ <signal name="bind" handler="bind_size_cb"/>
</object>
- </child>
+ </property>
</object>
</child>
</object>
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]