427 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			427 lines
		
	
	
		
			13 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 <glib/gi18n-lib.h>
 | |
| 
 | |
| #include "action-editor.h"
 | |
| 
 | |
| #include "gtksizegroup.h"
 | |
| #include "gtktogglebutton.h"
 | |
| #include "gtkentry.h"
 | |
| #include "gtkbin.h"
 | |
| #include "gtklabel.h"
 | |
| 
 | |
| struct _GtkInspectorActionEditorPrivate
 | |
| {
 | |
|   GActionGroup *group;
 | |
|   gchar *prefix;
 | |
|   gchar *name;
 | |
|   gboolean enabled;
 | |
|   const GVariantType *parameter_type;
 | |
|   GVariantType *state_type;
 | |
|   GtkWidget *activate_button;
 | |
|   GtkWidget *parameter_entry;
 | |
|   GtkWidget *state_entry;
 | |
|   GtkSizeGroup *sg;
 | |
| };
 | |
| 
 | |
| enum
 | |
| {
 | |
|   PROP_0,
 | |
|   PROP_GROUP,
 | |
|   PROP_PREFIX,
 | |
|   PROP_NAME
 | |
| };
 | |
| 
 | |
| G_DEFINE_TYPE_WITH_PRIVATE (GtkInspectorActionEditor, gtk_inspector_action_editor, GTK_TYPE_BOX)
 | |
| 
 | |
| static void
 | |
| gtk_inspector_action_editor_init (GtkInspectorActionEditor *editor)
 | |
| {
 | |
|   editor->priv = gtk_inspector_action_editor_get_instance_private (editor);
 | |
|   g_object_set (editor,
 | |
|                 "orientation", GTK_ORIENTATION_VERTICAL,
 | |
|                 "spacing", 10,
 | |
|                 "margin", 10,
 | |
|                 NULL);
 | |
| }
 | |
| 
 | |
| typedef void (*VariantEditorChanged) (GtkWidget *editor, gpointer data);
 | |
| 
 | |
| typedef struct {
 | |
|   GtkWidget *editor;
 | |
|   VariantEditorChanged callback;
 | |
|   gpointer   data;
 | |
| } VariantEditorData;
 | |
| 
 | |
| static void
 | |
| variant_editor_changed_cb (GObject           *obj,
 | |
|                            GParamSpec        *pspec,
 | |
|                            VariantEditorData *data)
 | |
| {
 | |
|   data->callback (data->editor, data->data);
 | |
| }
 | |
| 
 | |
| static GtkWidget *
 | |
| variant_editor_new (const GVariantType   *type,
 | |
|                     VariantEditorChanged  callback,
 | |
|                     gpointer              data)
 | |
| {
 | |
|   GtkWidget *editor;
 | |
|   GtkWidget *label;
 | |
|   GtkWidget *entry;
 | |
|   VariantEditorData *d;
 | |
| 
 | |
|   d = g_new (VariantEditorData, 1);
 | |
|   d->callback = callback;
 | |
|   d->data = data;
 | |
| 
 | |
|   if (g_variant_type_equal (type, G_VARIANT_TYPE_BOOLEAN))
 | |
|     {
 | |
|       editor = gtk_toggle_button_new_with_label ("FALSE");
 | |
|       g_signal_connect (editor, "notify::active", G_CALLBACK (variant_editor_changed_cb), d);
 | |
|     }   
 | |
|   else if (g_variant_type_equal (type, G_VARIANT_TYPE_STRING))
 | |
|     {
 | |
|       editor = gtk_entry_new ();
 | |
|       g_signal_connect (editor, "notify::text", G_CALLBACK (variant_editor_changed_cb), d);
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       editor = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
 | |
|       entry = gtk_entry_new ();
 | |
|       gtk_container_add (GTK_CONTAINER (editor), entry);
 | |
|       label = gtk_label_new (g_variant_type_peek_string (type));
 | |
|       gtk_container_add (GTK_CONTAINER (editor), label);
 | |
|       g_signal_connect (entry, "notify::text", G_CALLBACK (variant_editor_changed_cb), d);
 | |
|     }
 | |
| 
 | |
|   g_object_set_data (G_OBJECT (editor), "type", (gpointer)type);
 | |
|   d->editor = editor;
 | |
|   g_object_set_data_full (G_OBJECT (editor), "callback", d, g_free);
 | |
| 
 | |
|   gtk_widget_show_all (editor);
 | |
| 
 | |
|   return editor;
 | |
| }
 | |
| 
 | |
| static void
 | |
| variant_editor_set_value (GtkWidget *editor,
 | |
|                           GVariant  *value)
 | |
| {
 | |
|   const GVariantType *type;
 | |
|   gpointer data;
 | |
| 
 | |
|   data = g_object_get_data (G_OBJECT (editor), "callback");
 | |
|   g_signal_handlers_block_by_func (editor, variant_editor_changed_cb, data);
 | |
| 
 | |
|   type = g_variant_get_type (value);
 | |
|   if (g_variant_type_equal (type, G_VARIANT_TYPE_BOOLEAN))
 | |
|     {
 | |
|       GtkToggleButton *tb = GTK_TOGGLE_BUTTON (editor);
 | |
|       GtkWidget *child;
 | |
| 
 | |
|       gtk_toggle_button_set_active (tb, g_variant_get_boolean (value));
 | |
|       child = gtk_bin_get_child (GTK_BIN (tb));
 | |
|       gtk_label_set_text (GTK_LABEL (child),
 | |
|                           g_variant_get_boolean (value) ? "TRUE" : "FALSE");    
 | |
|     }   
 | |
|   else if (g_variant_type_equal (type, G_VARIANT_TYPE_STRING))
 | |
|     {
 | |
|       GtkEntry *entry = GTK_ENTRY (editor);
 | |
|       gtk_entry_set_text (entry, g_variant_get_string (value, NULL));
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       GList *children;
 | |
|       GtkEntry *entry;
 | |
|       gchar *text;
 | |
| 
 | |
|       children = gtk_container_get_children (GTK_CONTAINER (editor));
 | |
|       entry = children->data;
 | |
|       g_list_free (children);
 | |
| 
 | |
|       text = g_variant_print (value, FALSE);
 | |
|       gtk_entry_set_text (entry, text);
 | |
|       g_free (text);
 | |
|     }
 | |
| 
 | |
|   g_signal_handlers_unblock_by_func (editor, variant_editor_changed_cb, data);
 | |
| }
 | |
| 
 | |
| static GVariant *
 | |
| variant_editor_get_value (GtkWidget *editor)
 | |
| {
 | |
|   const GVariantType *type;
 | |
|   GVariant *value;
 | |
| 
 | |
|   type = (const GVariantType *) g_object_get_data (G_OBJECT (editor), "type");
 | |
|   if (g_variant_type_equal (type, G_VARIANT_TYPE_BOOLEAN))
 | |
|     {
 | |
|       GtkToggleButton *tb = GTK_TOGGLE_BUTTON (editor);
 | |
|       value = g_variant_new_boolean (gtk_toggle_button_get_active (tb));
 | |
|     }
 | |
|   else if (g_variant_type_equal (type, G_VARIANT_TYPE_STRING))
 | |
|     {
 | |
|       GtkEntry *entry = GTK_ENTRY (editor);
 | |
|       value = g_variant_new_string (gtk_entry_get_text (entry));
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       GList *children;
 | |
|       GtkEntry *entry;
 | |
|       const gchar *text;
 | |
| 
 | |
|       children = gtk_container_get_children (GTK_CONTAINER (editor));
 | |
|       entry = children->data;
 | |
|       text = gtk_entry_get_text (entry);
 | |
|       g_list_free (children);
 | |
| 
 | |
|       value = g_variant_parse (type, text, NULL, NULL, NULL);
 | |
|     }
 | |
| 
 | |
|   return value;
 | |
| }
 | |
| 
 | |
| static void
 | |
| activate_action (GtkWidget                *button,
 | |
|                  GtkInspectorActionEditor *r)
 | |
| {
 | |
|   GVariant *parameter = NULL;
 | |
| 
 | |
|   if (r->priv->parameter_entry)
 | |
|     parameter = variant_editor_get_value (r->priv->parameter_entry);
 | |
|   g_action_group_activate_action (r->priv->group, r->priv->name, parameter);
 | |
| }
 | |
| 
 | |
| static void
 | |
| parameter_changed (GtkWidget *editor,
 | |
|                    gpointer   data)
 | |
| {
 | |
|   GtkInspectorActionEditor *r = data;
 | |
|   GVariant *value;
 | |
| 
 | |
|   value = variant_editor_get_value (editor);
 | |
|   gtk_widget_set_sensitive (r->priv->activate_button, r->priv->enabled && value != NULL);
 | |
|   if (value)
 | |
|     g_variant_unref (value);
 | |
| }
 | |
| 
 | |
| static void
 | |
| state_changed (GtkWidget *editor,
 | |
|                gpointer   data)
 | |
| {
 | |
|   GtkInspectorActionEditor *r = data;
 | |
|   GVariant *value;
 | |
| 
 | |
|   value = variant_editor_get_value (editor);
 | |
|   if (value)
 | |
|     g_action_group_change_action_state (r->priv->group, r->priv->name, value);
 | |
| }
 | |
| 
 | |
| static void
 | |
| action_enabled_changed_cb (GActionGroup             *group,
 | |
|                            const gchar              *action_name,
 | |
|                            gboolean                  enabled,
 | |
|                            GtkInspectorActionEditor *r)
 | |
| {
 | |
|   r->priv->enabled = enabled;
 | |
|   if (r->priv->parameter_entry)
 | |
|     {
 | |
|       gtk_widget_set_sensitive (r->priv->parameter_entry, enabled);
 | |
|       parameter_changed (r->priv->parameter_entry, r);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| action_state_changed_cb (GActionGroup             *group,
 | |
|                          const gchar              *action_name,
 | |
|                          GVariant                 *state,
 | |
|                          GtkInspectorActionEditor *r)
 | |
| {
 | |
|   if (r->priv->state_entry)
 | |
|     variant_editor_set_value (r->priv->state_entry, state);
 | |
| }
 | |
| 
 | |
| static void
 | |
| constructed (GObject *object)
 | |
| {
 | |
|   GtkInspectorActionEditor *r = GTK_INSPECTOR_ACTION_EDITOR (object);
 | |
|   GVariant *state;
 | |
|   gchar *fullname;
 | |
|   GtkWidget *row;
 | |
|   GtkWidget *label;
 | |
| 
 | |
|   r->priv->enabled = g_action_group_get_action_enabled (r->priv->group, r->priv->name);
 | |
|   state = g_action_group_get_action_state (r->priv->group, r->priv->name);
 | |
| 
 | |
|   fullname = g_strdup_printf ("%s.%s", r->priv->prefix, r->priv->name);
 | |
|   gtk_container_add (GTK_CONTAINER (r), gtk_label_new (fullname));
 | |
|   g_free (fullname);
 | |
| 
 | |
|   r->priv->sg = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
 | |
| 
 | |
|   row = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
 | |
| 
 | |
|   r->priv->activate_button = gtk_button_new_with_label (_("Activate"));
 | |
|   g_signal_connect (r->priv->activate_button, "clicked", G_CALLBACK (activate_action), r);
 | |
| 
 | |
|   gtk_size_group_add_widget (r->priv->sg, r->priv->activate_button);
 | |
|   gtk_widget_set_sensitive (r->priv->activate_button, r->priv->enabled);
 | |
|   gtk_container_add (GTK_CONTAINER (row), r->priv->activate_button);
 | |
| 
 | |
|   r->priv->parameter_type = g_action_group_get_action_parameter_type (r->priv->group, r->priv->name);
 | |
|   if (r->priv->parameter_type)
 | |
|     {
 | |
|       r->priv->parameter_entry = variant_editor_new (r->priv->parameter_type, parameter_changed, r);
 | |
|       gtk_widget_set_sensitive (r->priv->parameter_entry, r->priv->enabled);
 | |
|       gtk_container_add (GTK_CONTAINER (row), r->priv->parameter_entry);
 | |
|     }
 | |
| 
 | |
|   gtk_container_add (GTK_CONTAINER (r), row);
 | |
| 
 | |
|   if (state)
 | |
|     {
 | |
|       r->priv->state_type = g_variant_type_copy (g_variant_get_type (state));
 | |
|       row = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 10);
 | |
|       label = gtk_label_new (_("State"));
 | |
|       gtk_size_group_add_widget (r->priv->sg, label);
 | |
|       gtk_container_add (GTK_CONTAINER (row), label);
 | |
|       r->priv->state_entry = variant_editor_new (r->priv->state_type, state_changed, r);
 | |
|       variant_editor_set_value (r->priv->state_entry, state);
 | |
|       gtk_container_add (GTK_CONTAINER (row), r->priv->state_entry);
 | |
|       gtk_container_add (GTK_CONTAINER (r), row);
 | |
|     }
 | |
| 
 | |
|   g_signal_connect (r->priv->group, "action-enabled-changed",
 | |
|                     G_CALLBACK (action_enabled_changed_cb), r);
 | |
|   g_signal_connect (r->priv->group, "action-state-changed",
 | |
|                     G_CALLBACK (action_state_changed_cb), r);
 | |
| 
 | |
|   gtk_widget_show_all (GTK_WIDGET (r));
 | |
| }
 | |
| 
 | |
| static void
 | |
| finalize (GObject *object)
 | |
| {
 | |
|   GtkInspectorActionEditor *r = GTK_INSPECTOR_ACTION_EDITOR (object);
 | |
| 
 | |
|   g_free (r->priv->prefix);
 | |
|   g_free (r->priv->name);
 | |
|   g_object_unref (r->priv->sg);
 | |
|   if (r->priv->state_type)
 | |
|     g_variant_type_free (r->priv->state_type);
 | |
|   g_signal_handlers_disconnect_by_func (r->priv->group, action_enabled_changed_cb, r);
 | |
|   g_signal_handlers_disconnect_by_func (r->priv->group, action_state_changed_cb, r);
 | |
| 
 | |
|   G_OBJECT_CLASS (gtk_inspector_action_editor_parent_class)->finalize (object);
 | |
| }
 | |
| 
 | |
| static void
 | |
| get_property (GObject    *object,
 | |
|               guint       param_id,
 | |
|               GValue     *value,
 | |
|               GParamSpec *pspec)
 | |
| {
 | |
|   GtkInspectorActionEditor *r = GTK_INSPECTOR_ACTION_EDITOR (object);
 | |
| 
 | |
|   switch (param_id)
 | |
|     {
 | |
|     case PROP_GROUP:
 | |
|       g_value_set_object (value, r->priv->group);
 | |
|       break;
 | |
| 
 | |
|     case PROP_PREFIX:
 | |
|       g_value_set_string (value, r->priv->prefix);
 | |
|       break;
 | |
| 
 | |
|     case PROP_NAME:
 | |
|       g_value_set_string (value, r->priv->name);
 | |
|       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)
 | |
| {
 | |
|   GtkInspectorActionEditor *r = GTK_INSPECTOR_ACTION_EDITOR (object);
 | |
| 
 | |
|   switch (param_id)
 | |
|     {
 | |
|     case PROP_GROUP:
 | |
|       r->priv->group = g_value_get_object (value);
 | |
|       break;
 | |
| 
 | |
|     case PROP_PREFIX:
 | |
|       g_free (r->priv->prefix);
 | |
|       r->priv->prefix = g_value_dup_string (value);
 | |
|       break;
 | |
| 
 | |
|     case PROP_NAME:
 | |
|       g_free (r->priv->name);
 | |
|       r->priv->name = g_value_dup_string (value);
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       G_OBJECT_WARN_INVALID_PROPERTY_ID(object, param_id, pspec);
 | |
|       break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_inspector_action_editor_class_init (GtkInspectorActionEditorClass *klass)
 | |
| {
 | |
|   GObjectClass *object_class = G_OBJECT_CLASS (klass);
 | |
| 
 | |
|   object_class->constructed = constructed;
 | |
|   object_class->finalize = finalize;
 | |
|   object_class->get_property = get_property;
 | |
|   object_class->set_property = set_property;
 | |
| 
 | |
|   g_object_class_install_property (object_class, PROP_GROUP,
 | |
|       g_param_spec_object ("group", "Action Group", "The Action Group containing the action",
 | |
|                            G_TYPE_ACTION_GROUP, G_PARAM_READWRITE|G_PARAM_CONSTRUCT));
 | |
| 
 | |
|   g_object_class_install_property (object_class, PROP_PREFIX,
 | |
|       g_param_spec_string ("prefix", "Prefix", "The action name prefix",
 | |
|                            NULL, G_PARAM_READWRITE|G_PARAM_CONSTRUCT));
 | |
| 
 | |
|   g_object_class_install_property (object_class, PROP_NAME,
 | |
|       g_param_spec_string ("name", "Name", "The action name",
 | |
|                            NULL, G_PARAM_READWRITE|G_PARAM_CONSTRUCT));
 | |
| }
 | |
| 
 | |
| GtkWidget *
 | |
| gtk_inspector_action_editor_new (GActionGroup *group,
 | |
|                                  const gchar  *prefix,
 | |
|                                  const gchar  *name)
 | |
| {
 | |
|   return g_object_new (GTK_TYPE_INSPECTOR_ACTION_EDITOR,
 | |
|                        "group", group,
 | |
|                        "prefix", prefix,
 | |
|                        "name", name,
 | |
|                        NULL);
 | |
| }
 | 
