forked from AuroraMiddleware/gtk
496 lines
14 KiB
C
496 lines
14 KiB
C
/*
|
|
* Copyright (c) 2014 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "statistics.h"
|
|
|
|
#include "graphdata.h"
|
|
|
|
#include "gtkcelllayout.h"
|
|
#include "gtkcellrenderertext.h"
|
|
#include "gtklabel.h"
|
|
#include "gtksearchbar.h"
|
|
#include "gtkstack.h"
|
|
#include "gtktogglebutton.h"
|
|
#include "gtktreeselection.h"
|
|
#include "gtktreeview.h"
|
|
#include "gtkeventcontrollerkey.h"
|
|
#include "gtkmain.h"
|
|
#include "gtkliststore.h"
|
|
|
|
#include <glib/gi18n-lib.h>
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_BUTTON
|
|
};
|
|
|
|
struct _GtkInspectorStatisticsPrivate
|
|
{
|
|
GtkWidget *stack;
|
|
GtkWidget *excuse;
|
|
GtkTreeModel *model;
|
|
GtkTreeView *view;
|
|
GtkWidget *button;
|
|
GHashTable *data;
|
|
GtkTreeViewColumn *column_self1;
|
|
GtkCellRenderer *renderer_self1;
|
|
GtkTreeViewColumn *column_cumulative1;
|
|
GtkCellRenderer *renderer_cumulative1;
|
|
GtkTreeViewColumn *column_self2;
|
|
GtkCellRenderer *renderer_self2;
|
|
GtkTreeViewColumn *column_cumulative2;
|
|
GtkCellRenderer *renderer_cumulative2;
|
|
GHashTable *counts;
|
|
guint update_source_id;
|
|
GtkWidget *search_entry;
|
|
GtkWidget *search_bar;
|
|
};
|
|
|
|
typedef struct {
|
|
GType type;
|
|
GtkTreeIter treeiter;
|
|
GtkGraphData *self;
|
|
GtkGraphData *cumulative;
|
|
} TypeData;
|
|
|
|
enum
|
|
{
|
|
COLUMN_TYPE,
|
|
COLUMN_TYPE_NAME,
|
|
COLUMN_SELF1,
|
|
COLUMN_CUMULATIVE1,
|
|
COLUMN_SELF2,
|
|
COLUMN_CUMULATIVE2,
|
|
COLUMN_SELF_DATA,
|
|
COLUMN_CUMULATIVE_DATA
|
|
};
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (GtkInspectorStatistics, gtk_inspector_statistics, GTK_TYPE_BOX)
|
|
|
|
static int
|
|
add_type_count (GtkInspectorStatistics *sl, GType type)
|
|
{
|
|
int cumulative;
|
|
int self;
|
|
GType *children;
|
|
guint n_children;
|
|
int i;
|
|
TypeData *data;
|
|
|
|
cumulative = 0;
|
|
|
|
children = g_type_children (type, &n_children);
|
|
for (i = 0; i < n_children; i++)
|
|
cumulative += add_type_count (sl, children[i]);
|
|
|
|
data = g_hash_table_lookup (sl->priv->counts, GSIZE_TO_POINTER (type));
|
|
if (!data)
|
|
{
|
|
data = g_new0 (TypeData, 1);
|
|
data->type = type;
|
|
data->self = gtk_graph_data_new (60);
|
|
data->cumulative = gtk_graph_data_new (60);
|
|
gtk_list_store_append (GTK_LIST_STORE (sl->priv->model), &data->treeiter);
|
|
gtk_list_store_set (GTK_LIST_STORE (sl->priv->model), &data->treeiter,
|
|
COLUMN_TYPE, data->type,
|
|
COLUMN_TYPE_NAME, g_type_name (data->type),
|
|
COLUMN_SELF_DATA, data->self,
|
|
COLUMN_CUMULATIVE_DATA, data->cumulative,
|
|
-1);
|
|
g_hash_table_insert (sl->priv->counts, GSIZE_TO_POINTER (type), data);
|
|
}
|
|
|
|
self = g_type_get_instance_count (type);
|
|
cumulative += self;
|
|
|
|
gtk_graph_data_prepend_value (data->self, self);
|
|
gtk_graph_data_prepend_value (data->cumulative, cumulative);
|
|
|
|
gtk_list_store_set (GTK_LIST_STORE (sl->priv->model), &data->treeiter,
|
|
COLUMN_SELF1, (int) gtk_graph_data_get_value (data->self, 1),
|
|
COLUMN_CUMULATIVE1, (int) gtk_graph_data_get_value (data->cumulative, 1),
|
|
COLUMN_SELF2, (int) gtk_graph_data_get_value (data->self, 0),
|
|
COLUMN_CUMULATIVE2, (int) gtk_graph_data_get_value (data->cumulative, 0),
|
|
-1);
|
|
return cumulative;
|
|
}
|
|
|
|
static gboolean
|
|
update_type_counts (gpointer data)
|
|
{
|
|
GtkInspectorStatistics *sl = data;
|
|
GType type;
|
|
|
|
for (type = G_TYPE_INTERFACE; type <= G_TYPE_FUNDAMENTAL_MAX; type += (1 << G_TYPE_FUNDAMENTAL_SHIFT))
|
|
{
|
|
if (!G_TYPE_IS_INSTANTIATABLE (type))
|
|
continue;
|
|
|
|
add_type_count (sl, type);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
toggle_record (GtkToggleButton *button,
|
|
GtkInspectorStatistics *sl)
|
|
{
|
|
if (gtk_toggle_button_get_active (button) == (sl->priv->update_source_id != 0))
|
|
return;
|
|
|
|
if (gtk_toggle_button_get_active (button))
|
|
{
|
|
sl->priv->update_source_id = g_timeout_add_seconds (1, update_type_counts, sl);
|
|
update_type_counts (sl);
|
|
}
|
|
else
|
|
{
|
|
g_source_remove (sl->priv->update_source_id);
|
|
sl->priv->update_source_id = 0;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
has_instance_counts (void)
|
|
{
|
|
return g_type_get_instance_count (GTK_TYPE_LABEL) > 0;
|
|
}
|
|
|
|
static gboolean
|
|
instance_counts_enabled (void)
|
|
{
|
|
const char *string;
|
|
guint flags = 0;
|
|
|
|
string = g_getenv ("GOBJECT_DEBUG");
|
|
if (string != NULL)
|
|
{
|
|
GDebugKey debug_keys[] = {
|
|
{ "objects", 1 },
|
|
{ "instance-count", 2 },
|
|
{ "signals", 4 }
|
|
};
|
|
|
|
flags = g_parse_debug_string (string, debug_keys, G_N_ELEMENTS (debug_keys));
|
|
}
|
|
|
|
return (flags & 2) != 0;
|
|
}
|
|
|
|
static void
|
|
cell_data_data (GtkCellLayout *layout,
|
|
GtkCellRenderer *cell,
|
|
GtkTreeModel *model,
|
|
GtkTreeIter *iter,
|
|
gpointer data)
|
|
{
|
|
int column;
|
|
int count;
|
|
char *text;
|
|
|
|
column = GPOINTER_TO_INT (data);
|
|
|
|
gtk_tree_model_get (model, iter, column, &count, -1);
|
|
|
|
text = g_strdup_printf ("%d", count);
|
|
g_object_set (cell, "text", text, NULL);
|
|
g_free (text);
|
|
}
|
|
|
|
static void
|
|
cell_data_delta (GtkCellLayout *layout,
|
|
GtkCellRenderer *cell,
|
|
GtkTreeModel *model,
|
|
GtkTreeIter *iter,
|
|
gpointer data)
|
|
{
|
|
int column;
|
|
int count1;
|
|
int count2;
|
|
char *text;
|
|
|
|
column = GPOINTER_TO_INT (data);
|
|
|
|
gtk_tree_model_get (model, iter, column - 2, &count1, column, &count2, -1);
|
|
|
|
if (count2 > count1)
|
|
text = g_strdup_printf ("%d (↗ %d)", count2, count2 - count1);
|
|
else if (count2 < count1)
|
|
text = g_strdup_printf ("%d (↘ %d)", count2, count1 - count2);
|
|
else
|
|
text = g_strdup_printf ("%d", count2);
|
|
g_object_set (cell, "text", text, NULL);
|
|
g_free (text);
|
|
}
|
|
|
|
static void
|
|
type_data_free (gpointer data)
|
|
{
|
|
TypeData *type_data = data;
|
|
|
|
g_object_unref (type_data->self);
|
|
g_object_unref (type_data->cumulative);
|
|
|
|
g_free (type_data);
|
|
}
|
|
|
|
static gboolean
|
|
key_pressed (GtkEventController *controller,
|
|
guint keyval,
|
|
guint keycode,
|
|
GdkModifierType state,
|
|
GtkInspectorStatistics *sl)
|
|
{
|
|
if (gtk_widget_get_mapped (GTK_WIDGET (sl)))
|
|
{
|
|
if (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 (sl->priv->view);
|
|
if (gtk_tree_selection_get_selected (selection, &model, &iter))
|
|
{
|
|
path = gtk_tree_model_get_path (model, &iter);
|
|
gtk_tree_view_row_activated (sl->priv->view, path, NULL);
|
|
gtk_tree_path_free (path);
|
|
|
|
return GDK_EVENT_STOP;
|
|
}
|
|
}
|
|
}
|
|
|
|
return GDK_EVENT_PROPAGATE;
|
|
}
|
|
|
|
static gboolean
|
|
match_string (const char *string,
|
|
const char *text)
|
|
{
|
|
char *lower;
|
|
gboolean match = FALSE;
|
|
|
|
if (string)
|
|
{
|
|
lower = g_ascii_strdown (string, -1);
|
|
match = g_str_has_prefix (lower, text);
|
|
g_free (lower);
|
|
}
|
|
|
|
return match;
|
|
}
|
|
|
|
static gboolean
|
|
match_row (GtkTreeModel *model,
|
|
int column,
|
|
const char *key,
|
|
GtkTreeIter *iter,
|
|
gpointer data)
|
|
{
|
|
char *type;
|
|
gboolean match;
|
|
|
|
gtk_tree_model_get (model, iter, column, &type, -1);
|
|
|
|
match = match_string (type, key);
|
|
|
|
g_free (type);
|
|
|
|
return !match;
|
|
}
|
|
|
|
static void
|
|
destroy_controller (GtkEventController *controller)
|
|
{
|
|
gtk_widget_remove_controller (gtk_event_controller_get_widget (controller), controller);
|
|
}
|
|
|
|
static void
|
|
root (GtkWidget *widget)
|
|
{
|
|
GtkInspectorStatistics *sl = GTK_INSPECTOR_STATISTICS (widget);
|
|
GtkEventController *controller;
|
|
GtkWidget *toplevel;
|
|
|
|
GTK_WIDGET_CLASS (gtk_inspector_statistics_parent_class)->root (widget);
|
|
|
|
toplevel = GTK_WIDGET (gtk_widget_get_root (widget));
|
|
|
|
controller = gtk_event_controller_key_new ();
|
|
g_object_set_data_full (G_OBJECT (toplevel), "statistics-controller", controller, (GDestroyNotify)destroy_controller);
|
|
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);
|
|
}
|
|
|
|
static void
|
|
unroot (GtkWidget *widget)
|
|
{
|
|
GtkWidget *toplevel;
|
|
|
|
toplevel = GTK_WIDGET (gtk_widget_get_root (widget));
|
|
g_object_set_data (G_OBJECT (toplevel), "statistics-controller", NULL);
|
|
|
|
GTK_WIDGET_CLASS (gtk_inspector_statistics_parent_class)->unroot (widget);
|
|
}
|
|
|
|
static void
|
|
gtk_inspector_statistics_init (GtkInspectorStatistics *sl)
|
|
{
|
|
sl->priv = gtk_inspector_statistics_get_instance_private (sl);
|
|
gtk_widget_init_template (GTK_WIDGET (sl));
|
|
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (sl->priv->column_self1),
|
|
sl->priv->renderer_self1,
|
|
cell_data_data,
|
|
GINT_TO_POINTER (COLUMN_SELF1), NULL);
|
|
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (sl->priv->column_cumulative1),
|
|
sl->priv->renderer_cumulative1,
|
|
cell_data_data,
|
|
GINT_TO_POINTER (COLUMN_CUMULATIVE1), NULL);
|
|
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (sl->priv->column_self2),
|
|
sl->priv->renderer_self2,
|
|
cell_data_delta,
|
|
GINT_TO_POINTER (COLUMN_SELF2), NULL);
|
|
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (sl->priv->column_cumulative2),
|
|
sl->priv->renderer_cumulative2,
|
|
cell_data_delta,
|
|
GINT_TO_POINTER (COLUMN_CUMULATIVE2), NULL);
|
|
sl->priv->counts = g_hash_table_new_full (NULL, NULL, NULL, type_data_free);
|
|
|
|
gtk_tree_view_set_search_entry (sl->priv->view, GTK_EDITABLE (sl->priv->search_entry));
|
|
gtk_tree_view_set_search_equal_func (sl->priv->view, match_row, sl, NULL);
|
|
}
|
|
|
|
static void
|
|
constructed (GObject *object)
|
|
{
|
|
GtkInspectorStatistics *sl = GTK_INSPECTOR_STATISTICS (object);
|
|
|
|
g_signal_connect (sl->priv->button, "toggled",
|
|
G_CALLBACK (toggle_record), sl);
|
|
|
|
if (has_instance_counts ())
|
|
update_type_counts (sl);
|
|
else
|
|
{
|
|
if (instance_counts_enabled ())
|
|
gtk_label_set_text (GTK_LABEL (sl->priv->excuse), _("GLib must be configured with -Dbuildtype=debug"));
|
|
gtk_stack_set_visible_child_name (GTK_STACK (sl->priv->stack), "excuse");
|
|
gtk_widget_set_sensitive (sl->priv->button, FALSE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
finalize (GObject *object)
|
|
{
|
|
GtkInspectorStatistics *sl = GTK_INSPECTOR_STATISTICS (object);
|
|
|
|
if (sl->priv->update_source_id)
|
|
g_source_remove (sl->priv->update_source_id);
|
|
|
|
g_hash_table_unref (sl->priv->counts);
|
|
|
|
G_OBJECT_CLASS (gtk_inspector_statistics_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
get_property (GObject *object,
|
|
guint param_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkInspectorStatistics *sl = GTK_INSPECTOR_STATISTICS (object);
|
|
|
|
switch (param_id)
|
|
{
|
|
case PROP_BUTTON:
|
|
g_value_take_object (value, sl->priv->button);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
set_property (GObject *object,
|
|
guint param_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkInspectorStatistics *sl = GTK_INSPECTOR_STATISTICS (object);
|
|
|
|
switch (param_id)
|
|
{
|
|
case PROP_BUTTON:
|
|
sl->priv->button = g_value_get_object (value);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_inspector_statistics_class_init (GtkInspectorStatisticsClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
object_class->get_property = get_property;
|
|
object_class->set_property = set_property;
|
|
object_class->constructed = constructed;
|
|
object_class->finalize = finalize;
|
|
|
|
widget_class->root = root;
|
|
widget_class->unroot = unroot;
|
|
|
|
g_object_class_install_property (object_class, PROP_BUTTON,
|
|
g_param_spec_object ("button", NULL, NULL,
|
|
GTK_TYPE_WIDGET, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
gtk_widget_class_set_template_from_resource (widget_class, "/org/gtk/libgtk/inspector/statistics.ui");
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, view);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, stack);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, model);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, column_self1);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, renderer_self1);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, column_cumulative1);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, renderer_cumulative1);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, column_self2);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, renderer_self2);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, column_cumulative2);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, renderer_cumulative2);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, search_entry);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, search_bar);
|
|
gtk_widget_class_bind_template_child_private (widget_class, GtkInspectorStatistics, excuse);
|
|
|
|
}
|
|
|
|
// vim: set et sw=2 ts=2:
|