mirror of
https://github.com/varun-r-mallya/sysprof.git
synced 2025-12-31 20:36:25 +00:00
This is a major redesign a modernization of Sysprof. The core data structures and design are largely the same, but it has been ported to Gtk3 and has lots of additions that should make your profiling experience smoother. Especially for those that are new to profiling. There are some very simple help docs added, but we really need the experts to come in and write some documentation here.
858 lines
28 KiB
C
858 lines
28 KiB
C
/* sp-profiler-menu-button.c
|
|
*
|
|
* Copyright (C) 2016 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/>.
|
|
*/
|
|
|
|
#include <glib/gi18n.h>
|
|
#include <string.h>
|
|
|
|
#include "sp-model-filter.h"
|
|
#include "sp-process-model.h"
|
|
#include "sp-process-model-item.h"
|
|
#include "sp-process-model-row.h"
|
|
#include "sp-profiler-menu-button.h"
|
|
|
|
typedef struct
|
|
{
|
|
SpProfiler *profiler;
|
|
SpModelFilter *process_filter;
|
|
|
|
/* Gtk template widgets */
|
|
GtkLabel *label;
|
|
GtkPopover *popover;
|
|
GtkEntry *process_filter_entry;
|
|
GtkListBox *process_list_box;
|
|
SpProcessModel *process_model;
|
|
GtkBox *processes_box;
|
|
GtkEntry *spawn_entry;
|
|
GtkStack *stack;
|
|
GtkSwitch *whole_system_switch;
|
|
GtkTreeView *env_tree_view;
|
|
GtkTreeViewColumn *env_key_column;
|
|
GtkTreeViewColumn *env_value_column;
|
|
GtkCellRenderer *key_cell;
|
|
GtkCellRenderer *value_cell;
|
|
GtkCheckButton *inherit_environ;
|
|
|
|
/* Property Bindings */
|
|
GBinding *inherit_binding;
|
|
GBinding *list_sensitive_binding;
|
|
GBinding *mutable_binding;
|
|
GBinding *whole_system_binding;
|
|
|
|
/* Signal handlers */
|
|
gulong notify_whole_system_handler;
|
|
|
|
/* GSources */
|
|
guint save_env_source;
|
|
} SpProfilerMenuButtonPrivate;
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (SpProfilerMenuButton, sp_profiler_menu_button, GTK_TYPE_MENU_BUTTON)
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_PROFILER,
|
|
N_PROPS
|
|
};
|
|
|
|
static GParamSpec *properties [N_PROPS];
|
|
|
|
GtkWidget *
|
|
sp_profiler_menu_button_new (void)
|
|
{
|
|
return g_object_new (SP_TYPE_PROFILER_MENU_BUTTON, NULL);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_update_label (SpProfilerMenuButton *self)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
g_autofree gchar *str = NULL;
|
|
const gchar *visible_child;
|
|
const GPid *pids;
|
|
guint n_pids = 0;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
|
|
if (priv->profiler == NULL)
|
|
{
|
|
gtk_label_set_label (priv->label, "");
|
|
return;
|
|
}
|
|
|
|
visible_child = gtk_stack_get_visible_child_name (priv->stack);
|
|
|
|
if (g_strcmp0 (visible_child, "spawn") == 0)
|
|
{
|
|
const gchar *text;
|
|
|
|
text = gtk_entry_get_text (priv->spawn_entry);
|
|
|
|
if (text && *text)
|
|
gtk_label_set_label (priv->label, text);
|
|
else if (sp_profiler_get_whole_system (priv->profiler))
|
|
gtk_label_set_label (priv->label, _("All Processes"));
|
|
else
|
|
gtk_label_set_label (priv->label, _("New Process"));
|
|
|
|
sp_profiler_set_spawn (priv->profiler, text && *text);
|
|
|
|
return;
|
|
}
|
|
|
|
sp_profiler_set_spawn (priv->profiler, FALSE);
|
|
|
|
pids = sp_profiler_get_pids (priv->profiler, &n_pids);
|
|
|
|
if (n_pids == 0 || sp_profiler_get_whole_system (priv->profiler))
|
|
{
|
|
gtk_label_set_label (priv->label, _("All Processes"));
|
|
return;
|
|
}
|
|
|
|
if (n_pids == 1)
|
|
{
|
|
str = g_strdup_printf (_("Process %d"), pids[0]);
|
|
gtk_label_set_label (priv->label, str);
|
|
return;
|
|
}
|
|
|
|
str = g_strdup_printf (_("%u Processes"), n_pids);
|
|
gtk_label_set_label (priv->label, str);
|
|
}
|
|
|
|
static void
|
|
clear_selected_flags (GtkWidget *widget,
|
|
gpointer user_data)
|
|
{
|
|
sp_process_model_row_set_selected (SP_PROCESS_MODEL_ROW (widget), FALSE);
|
|
}
|
|
|
|
static void
|
|
add_binding (GBinding **binding,
|
|
gpointer src,
|
|
const gchar *src_property,
|
|
gpointer dst,
|
|
const gchar *dst_property,
|
|
GBindingFlags flags)
|
|
{
|
|
g_assert (binding != NULL);
|
|
g_assert (*binding == NULL);
|
|
g_assert (src != NULL);
|
|
g_assert (src_property != NULL);
|
|
g_assert (dst != NULL);
|
|
g_assert (dst_property != NULL);
|
|
|
|
*binding = g_object_bind_property (src, src_property,
|
|
dst, dst_property,
|
|
flags);
|
|
g_object_add_weak_pointer (G_OBJECT (*binding), (gpointer *)binding);
|
|
}
|
|
|
|
static void
|
|
clear_binding (GBinding **binding)
|
|
{
|
|
g_assert (binding != NULL);
|
|
g_assert (!*binding || G_IS_BINDING (*binding));
|
|
|
|
if (*binding != NULL)
|
|
{
|
|
g_object_remove_weak_pointer (G_OBJECT (*binding), (gpointer *)binding);
|
|
g_binding_unbind (*binding);
|
|
*binding = NULL;
|
|
}
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_connect (SpProfilerMenuButton *self)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (SP_IS_PROFILER (priv->profiler));
|
|
|
|
add_binding (&priv->mutable_binding,
|
|
priv->profiler, "is-mutable",
|
|
self, "sensitive",
|
|
G_BINDING_SYNC_CREATE);
|
|
|
|
add_binding (&priv->whole_system_binding,
|
|
priv->profiler, "whole-system",
|
|
priv->whole_system_switch, "active",
|
|
G_BINDING_SYNC_CREATE | G_BINDING_BIDIRECTIONAL);
|
|
|
|
add_binding (&priv->list_sensitive_binding,
|
|
priv->profiler, "whole-system",
|
|
priv->processes_box, "sensitive",
|
|
G_BINDING_SYNC_CREATE | G_BINDING_INVERT_BOOLEAN);
|
|
|
|
add_binding (&priv->inherit_binding,
|
|
priv->inherit_environ, "active",
|
|
priv->profiler, "spawn-inherit-environ",
|
|
G_BINDING_SYNC_CREATE | G_BINDING_BIDIRECTIONAL);
|
|
|
|
priv->notify_whole_system_handler =
|
|
g_signal_connect_object (priv->profiler,
|
|
"notify::whole-system",
|
|
G_CALLBACK (sp_profiler_menu_button_update_label),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
sp_profiler_menu_button_update_label (self);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_disconnect (SpProfilerMenuButton *self)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (SP_IS_PROFILER (priv->profiler));
|
|
|
|
clear_binding (&priv->mutable_binding);
|
|
clear_binding (&priv->whole_system_binding);
|
|
clear_binding (&priv->list_sensitive_binding);
|
|
clear_binding (&priv->inherit_binding);
|
|
|
|
g_signal_handler_disconnect (priv->profiler, priv->notify_whole_system_handler);
|
|
priv->notify_whole_system_handler = 0;
|
|
|
|
gtk_widget_set_sensitive (GTK_WIDGET (self), FALSE);
|
|
|
|
g_clear_object (&priv->profiler);
|
|
|
|
gtk_container_foreach (GTK_CONTAINER (priv->process_list_box),
|
|
clear_selected_flags,
|
|
NULL);
|
|
|
|
sp_profiler_menu_button_update_label (self);
|
|
}
|
|
|
|
/**
|
|
* sp_profiler_menu_button_get_profiler:
|
|
* @self: An #SpProfilerMenuButton
|
|
*
|
|
* Gets the profiler instance that is being configured.
|
|
*
|
|
* Returns: (nullable) (transfer none): An #SpProfiler or %NULL.
|
|
*/
|
|
SpProfiler *
|
|
sp_profiler_menu_button_get_profiler (SpProfilerMenuButton *self)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
g_return_val_if_fail (SP_IS_PROFILER_MENU_BUTTON (self), NULL);
|
|
|
|
return priv->profiler;
|
|
}
|
|
|
|
void
|
|
sp_profiler_menu_button_set_profiler (SpProfilerMenuButton *self,
|
|
SpProfiler *profiler)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
g_return_if_fail (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_return_if_fail (!profiler || SP_IS_PROFILER (profiler));
|
|
|
|
if (priv->profiler != profiler)
|
|
{
|
|
if (priv->profiler != NULL)
|
|
sp_profiler_menu_button_disconnect (self);
|
|
|
|
if (profiler != NULL)
|
|
{
|
|
priv->profiler = g_object_ref (profiler);
|
|
sp_profiler_menu_button_connect (self);
|
|
}
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_PROFILER]);
|
|
}
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_row_activated (SpProfilerMenuButton *self,
|
|
SpProcessModelRow *row,
|
|
GtkListBox *list_box)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
gboolean selected;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (SP_IS_PROCESS_MODEL_ROW (row));
|
|
g_assert (GTK_IS_LIST_BOX (list_box));
|
|
|
|
selected = !sp_process_model_row_get_selected (row);
|
|
sp_process_model_row_set_selected (row, selected);
|
|
|
|
if (priv->profiler != NULL)
|
|
{
|
|
SpProcessModelItem *item = sp_process_model_row_get_item (row);
|
|
GPid pid = sp_process_model_item_get_pid (item);
|
|
|
|
if (selected)
|
|
sp_profiler_add_pid (priv->profiler, pid);
|
|
else
|
|
sp_profiler_remove_pid (priv->profiler, pid);
|
|
}
|
|
|
|
sp_profiler_menu_button_update_label (self);
|
|
}
|
|
|
|
static GtkWidget *
|
|
sp_profiler_menu_button_create_row (gpointer itemptr,
|
|
gpointer user_data)
|
|
{
|
|
SpProcessModelItem *item = itemptr;
|
|
SpProfilerMenuButton *self = user_data;
|
|
|
|
g_assert (SP_IS_PROCESS_MODEL_ITEM (item));
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
|
|
return g_object_new (SP_TYPE_PROCESS_MODEL_ROW,
|
|
"item", item,
|
|
"visible", TRUE,
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_clicked (GtkButton *button)
|
|
{
|
|
SpProfilerMenuButton *self = (SpProfilerMenuButton *)button;
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
|
|
sp_process_model_queue_reload (priv->process_model);
|
|
|
|
GTK_BUTTON_CLASS (sp_profiler_menu_button_parent_class)->clicked (button);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_filter_changed (SpProfilerMenuButton *self,
|
|
GtkEntry *entry)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (GTK_IS_ENTRY (entry));
|
|
|
|
sp_model_filter_invalidate (priv->process_filter);
|
|
}
|
|
|
|
static gboolean
|
|
sp_profiler_menu_button_filter_func (GObject *object,
|
|
gpointer user_data)
|
|
{
|
|
SpProfilerMenuButton *self = user_data;
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
const gchar *cmdline;
|
|
const gchar *text;
|
|
gboolean ret;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (SP_IS_PROCESS_MODEL_ITEM (object));
|
|
|
|
text = gtk_entry_get_text (priv->process_filter_entry);
|
|
if (!text || !*text)
|
|
return TRUE;
|
|
|
|
cmdline = sp_process_model_item_get_command_line (SP_PROCESS_MODEL_ITEM (object));
|
|
if (!cmdline)
|
|
return FALSE;
|
|
|
|
ret = (strstr (cmdline, text) != NULL);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_constructed (GObject *object)
|
|
{
|
|
SpProfilerMenuButton *self = (SpProfilerMenuButton *)object;
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
|
|
priv->process_filter = sp_model_filter_new (G_LIST_MODEL (priv->process_model));
|
|
sp_model_filter_set_filter_func (priv->process_filter,
|
|
sp_profiler_menu_button_filter_func,
|
|
self, NULL);
|
|
gtk_list_box_bind_model (priv->process_list_box,
|
|
G_LIST_MODEL (priv->process_filter),
|
|
sp_profiler_menu_button_create_row,
|
|
self, NULL);
|
|
|
|
G_OBJECT_CLASS (sp_profiler_menu_button_parent_class)->constructed (object);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_realize (GtkWidget *widget)
|
|
{
|
|
SpProfilerMenuButton *self = (SpProfilerMenuButton *)widget;
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
g_autoptr(GSettings) settings = NULL;
|
|
g_auto(GStrv) env = NULL;
|
|
|
|
GTK_WIDGET_CLASS (sp_profiler_menu_button_parent_class)->realize (widget);
|
|
|
|
settings = g_settings_new ("org.gnome.sysprof2");
|
|
|
|
env = g_settings_get_strv (settings, "last-spawn-env");
|
|
|
|
g_settings_bind (settings, "last-spawn-argv",
|
|
priv->spawn_entry, "text",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
g_settings_bind (settings, "last-spawn-inherit-env",
|
|
priv->inherit_environ, "active",
|
|
G_SETTINGS_BIND_DEFAULT);
|
|
|
|
if (env)
|
|
{
|
|
GtkTreeModel *model;
|
|
GtkTreeIter iter;
|
|
guint i;
|
|
|
|
model = gtk_tree_view_get_model (priv->env_tree_view);
|
|
gtk_list_store_clear (GTK_LIST_STORE (model));
|
|
|
|
for (i = 0; env [i]; i++)
|
|
{
|
|
const gchar *key = env [i];
|
|
const gchar *value = NULL;
|
|
gchar *eq = strchr (env[i], '=');
|
|
|
|
if (eq)
|
|
{
|
|
*eq = '\0';
|
|
value = eq + 1;
|
|
}
|
|
|
|
gtk_list_store_append (GTK_LIST_STORE (model), &iter);
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter,
|
|
0, key,
|
|
1, value,
|
|
-1);
|
|
}
|
|
|
|
gtk_list_store_append (GTK_LIST_STORE (model), &iter);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
save_environ_to_gsettings (gpointer data)
|
|
{
|
|
SpProfilerMenuButton *self = data;
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
g_autoptr(GPtrArray) ar = NULL;
|
|
g_autoptr(GSettings) settings = NULL;
|
|
GtkTreeModel *model;
|
|
GtkTreeIter iter;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
|
|
priv->save_env_source = 0;
|
|
|
|
settings = g_settings_new ("org.gnome.sysprof2");
|
|
|
|
model = gtk_tree_view_get_model (priv->env_tree_view);
|
|
|
|
ar = g_ptr_array_new_with_free_func (g_free);
|
|
|
|
if (gtk_tree_model_get_iter_first (model, &iter))
|
|
{
|
|
do
|
|
{
|
|
g_autofree gchar *key = NULL;
|
|
g_autofree gchar *value = NULL;
|
|
|
|
gtk_tree_model_get (model, &iter,
|
|
0, &key,
|
|
1, &value,
|
|
-1);
|
|
|
|
if (!key || !*key)
|
|
continue;
|
|
|
|
g_ptr_array_add (ar, g_strdup_printf ("%s=%s", key, value ? value : ""));
|
|
}
|
|
while (gtk_tree_model_iter_next (model, &iter));
|
|
}
|
|
|
|
g_ptr_array_add (ar, NULL);
|
|
|
|
g_settings_set_strv (settings, "last-spawn-env", (const gchar * const *)ar->pdata);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
|
|
static void
|
|
sp_profiler_menu_button_destroy (GtkWidget *widget)
|
|
{
|
|
SpProfilerMenuButton *self = (SpProfilerMenuButton *)widget;
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
if (priv->profiler != NULL)
|
|
{
|
|
sp_profiler_menu_button_disconnect (self);
|
|
g_clear_object (&priv->profiler);
|
|
}
|
|
|
|
if (priv->save_env_source)
|
|
save_environ_to_gsettings (self);
|
|
|
|
g_clear_object (&priv->process_filter);
|
|
|
|
GTK_WIDGET_CLASS (sp_profiler_menu_button_parent_class)->destroy (widget);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
SpProfilerMenuButton *self = SP_PROFILER_MENU_BUTTON (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_PROFILER:
|
|
g_value_set_object (value, sp_profiler_menu_button_get_profiler (self));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
SpProfilerMenuButton *self = SP_PROFILER_MENU_BUTTON (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_PROFILER:
|
|
sp_profiler_menu_button_set_profiler (self, g_value_get_object (value));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_class_init (SpProfilerMenuButtonClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
GtkButtonClass *button_class = GTK_BUTTON_CLASS (klass);
|
|
|
|
object_class->constructed = sp_profiler_menu_button_constructed;
|
|
object_class->get_property = sp_profiler_menu_button_get_property;
|
|
object_class->set_property = sp_profiler_menu_button_set_property;
|
|
|
|
widget_class->destroy = sp_profiler_menu_button_destroy;
|
|
widget_class->realize = sp_profiler_menu_button_realize;
|
|
|
|
button_class->clicked = sp_profiler_menu_button_clicked;
|
|
|
|
properties [PROP_PROFILER] =
|
|
g_param_spec_object ("profiler",
|
|
"Profiler",
|
|
"Profiler",
|
|
SP_TYPE_PROFILER,
|
|
(G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_properties (object_class, N_PROPS, properties);
|
|
|
|
gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/sysprof/ui/sp-profiler-menu-button.ui");
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, env_key_column);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, env_tree_view);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, env_value_column);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, inherit_environ);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, key_cell);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, label);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, popover);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, process_filter_entry);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, process_list_box);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, process_model);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, processes_box);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, spawn_entry);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, stack);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, value_cell);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SpProfilerMenuButton, whole_system_switch);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_env_row_changed (SpProfilerMenuButton *self,
|
|
GtkTreePath *tree_path,
|
|
GtkTreeIter *iter,
|
|
gpointer user_data)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
g_autoptr(GPtrArray) env = NULL;
|
|
GtkTreeModel *model;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (tree_path != NULL);
|
|
g_assert (iter != NULL);
|
|
|
|
/* queue saving settings to gsettings */
|
|
if (priv->save_env_source)
|
|
g_source_remove (priv->save_env_source);
|
|
priv->save_env_source = g_timeout_add_seconds (1, save_environ_to_gsettings, self);
|
|
|
|
/* sync the environ to the profiler */
|
|
env = g_ptr_array_new_with_free_func (g_free);
|
|
model = gtk_tree_view_get_model (priv->env_tree_view);
|
|
if (gtk_tree_model_get_iter_first (model, iter))
|
|
{
|
|
do
|
|
{
|
|
g_autofree gchar *key = NULL;
|
|
g_autofree gchar *value = NULL;
|
|
|
|
gtk_tree_model_get (model, iter,
|
|
0, &key,
|
|
1, &value,
|
|
-1);
|
|
|
|
if (key && *key)
|
|
g_ptr_array_add (env, g_strdup_printf ("%s=%s", key, value));
|
|
}
|
|
while (gtk_tree_model_iter_next (model, iter));
|
|
}
|
|
g_ptr_array_add (env, NULL);
|
|
sp_profiler_set_spawn_env (priv->profiler, (const gchar * const *)env->pdata);
|
|
}
|
|
|
|
static void
|
|
on_backspace (SpProfilerMenuButton *self,
|
|
GtkEntry *entry)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
if (g_object_get_data (G_OBJECT (entry), "CELL_WAS_EMPTY"))
|
|
{
|
|
GtkTreeModel *model;
|
|
GtkTreeSelection *selection;
|
|
GtkTreeIter iter;
|
|
|
|
model = gtk_tree_view_get_model (priv->env_tree_view);
|
|
selection = gtk_tree_view_get_selection (priv->env_tree_view);
|
|
|
|
if (gtk_tree_selection_get_selected (selection, NULL, &iter))
|
|
{
|
|
gtk_cell_renderer_stop_editing (priv->key_cell, TRUE);
|
|
gtk_list_store_remove (GTK_LIST_STORE (model), &iter);
|
|
}
|
|
}
|
|
else
|
|
g_object_set_data (G_OBJECT (entry), "CELL_WAS_EMPTY",
|
|
GINT_TO_POINTER (*gtk_entry_get_text (entry) == '\0'));
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_env_key_editing_started (SpProfilerMenuButton *self,
|
|
GtkCellEditable *editable,
|
|
const gchar *path,
|
|
GtkCellRenderer *cell)
|
|
{
|
|
g_signal_connect_object (editable,
|
|
"backspace",
|
|
G_CALLBACK (on_backspace),
|
|
self,
|
|
G_CONNECT_AFTER | G_CONNECT_SWAPPED);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_env_key_edited (SpProfilerMenuButton *self,
|
|
const gchar *path,
|
|
const gchar *new_text,
|
|
GtkCellRendererText *cell)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
GtkTreeModel *model;
|
|
GtkTreePath *tree_path;
|
|
GtkTreeIter iter;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (path != NULL);
|
|
g_assert (new_text != NULL);
|
|
g_assert (GTK_IS_CELL_RENDERER_TEXT (cell));
|
|
|
|
model = gtk_tree_view_get_model (priv->env_tree_view);
|
|
|
|
tree_path = gtk_tree_path_new_from_string (path);
|
|
|
|
if (gtk_tree_model_get_iter (model, &iter, tree_path))
|
|
{
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter,
|
|
0, new_text,
|
|
-1);
|
|
|
|
if (!gtk_tree_model_iter_next (model, &iter))
|
|
gtk_list_store_append (GTK_LIST_STORE (model), &iter);
|
|
|
|
gtk_tree_view_set_cursor_on_cell (priv->env_tree_view,
|
|
tree_path,
|
|
priv->env_value_column,
|
|
priv->value_cell,
|
|
TRUE);
|
|
}
|
|
|
|
gtk_tree_path_free (tree_path);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_env_value_edited (SpProfilerMenuButton *self,
|
|
const gchar *path,
|
|
const gchar *new_text,
|
|
GtkCellRendererText *cell)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
GtkTreeModel *model;
|
|
GtkTreePath *tree_path;
|
|
GtkTreeIter iter;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (path != NULL);
|
|
g_assert (new_text != NULL);
|
|
g_assert (GTK_IS_CELL_RENDERER_TEXT (cell));
|
|
|
|
model = gtk_tree_view_get_model (priv->env_tree_view);
|
|
|
|
tree_path = gtk_tree_path_new_from_string (path);
|
|
|
|
if (gtk_tree_model_get_iter (model, &iter, tree_path))
|
|
{
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter,
|
|
1, new_text,
|
|
-1);
|
|
|
|
if (!gtk_tree_model_iter_next (model, &iter))
|
|
gtk_list_store_append (GTK_LIST_STORE (model), &iter);
|
|
|
|
gtk_tree_path_next (tree_path);
|
|
|
|
gtk_tree_view_set_cursor_on_cell (priv->env_tree_view,
|
|
tree_path,
|
|
priv->env_key_column,
|
|
priv->key_cell,
|
|
TRUE);
|
|
}
|
|
|
|
gtk_tree_path_free (tree_path);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_validate_spawn (SpProfilerMenuButton *self,
|
|
GtkEntry *entry)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
g_auto(GStrv) argv = NULL;
|
|
const gchar *icon_name = NULL;
|
|
const gchar *text;
|
|
gint argc;
|
|
|
|
g_assert (SP_IS_PROFILER_MENU_BUTTON (self));
|
|
g_assert (GTK_IS_ENTRY (entry));
|
|
|
|
text = gtk_entry_get_text (entry);
|
|
|
|
if (!g_shell_parse_argv (text, &argc, &argv, NULL))
|
|
{
|
|
icon_name = "dialog-warning-symbolic";
|
|
sp_profiler_set_spawn_argv (priv->profiler, NULL);
|
|
}
|
|
else
|
|
sp_profiler_set_spawn_argv (priv->profiler, (const gchar * const *)argv);
|
|
|
|
gtk_entry_set_icon_from_icon_name (entry,
|
|
GTK_ENTRY_ICON_SECONDARY,
|
|
icon_name);
|
|
}
|
|
|
|
static void
|
|
sp_profiler_menu_button_init (SpProfilerMenuButton *self)
|
|
{
|
|
SpProfilerMenuButtonPrivate *priv = sp_profiler_menu_button_get_instance_private (self);
|
|
|
|
gtk_widget_init_template (GTK_WIDGET (self));
|
|
|
|
g_signal_connect_object (priv->process_filter_entry,
|
|
"changed",
|
|
G_CALLBACK (sp_profiler_menu_button_filter_changed),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (priv->spawn_entry,
|
|
"changed",
|
|
G_CALLBACK (sp_profiler_menu_button_update_label),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (priv->spawn_entry,
|
|
"changed",
|
|
G_CALLBACK (sp_profiler_menu_button_validate_spawn),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (priv->stack,
|
|
"notify::visible-child",
|
|
G_CALLBACK (sp_profiler_menu_button_update_label),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (priv->process_list_box,
|
|
"row-activated",
|
|
G_CALLBACK (sp_profiler_menu_button_row_activated),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (priv->key_cell,
|
|
"edited",
|
|
G_CALLBACK (sp_profiler_menu_button_env_key_edited),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (priv->value_cell,
|
|
"edited",
|
|
G_CALLBACK (sp_profiler_menu_button_env_value_edited),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (gtk_tree_view_get_model (priv->env_tree_view),
|
|
"row-changed",
|
|
G_CALLBACK (sp_profiler_menu_button_env_row_changed),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_signal_connect_object (priv->key_cell,
|
|
"editing-started",
|
|
G_CALLBACK (sp_profiler_menu_button_env_key_editing_started),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
gtk_widget_set_sensitive (GTK_WIDGET (self), FALSE);
|
|
}
|