783 lines
22 KiB
C
783 lines
22 KiB
C
|
/*
|
||
|
* mooedit-actions.c
|
||
|
*
|
||
|
* Copyright (C) 2004-2006 by Yevgen Muntyan <muntyan@math.tamu.edu>
|
||
|
*
|
||
|
* 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 2 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* See COPYING file that comes with this distribution.
|
||
|
*/
|
||
|
|
||
|
#define MOOEDIT_COMPILATION
|
||
|
#include "mooedit/mooedit-actions.h"
|
||
|
#include "mooedit/mooedit-private.h"
|
||
|
#include "mooutils/mooutils-gobject.h"
|
||
|
#include "mooutils/mootoggleaction.h"
|
||
|
#include "mooutils/moocompat.h"
|
||
|
#include <string.h>
|
||
|
#include <gobject/gvaluecollector.h>
|
||
|
|
||
|
|
||
|
static void moo_edit_add_action (MooEdit *edit,
|
||
|
MooAction *action);
|
||
|
static void moo_edit_remove_action (MooEdit *edit,
|
||
|
const char *action_id);
|
||
|
static void moo_edit_action_check (MooEditAction *action);
|
||
|
|
||
|
|
||
|
#define MOO_EDIT_ACTIONS_QUARK (moo_edit_get_actions_quark ())
|
||
|
|
||
|
typedef struct {
|
||
|
MooObjectFactory *action;
|
||
|
char **conditions;
|
||
|
} ActionInfo;
|
||
|
|
||
|
|
||
|
static GQuark
|
||
|
moo_edit_get_actions_quark (void)
|
||
|
{
|
||
|
static GQuark q;
|
||
|
|
||
|
if (!q)
|
||
|
q = g_quark_from_static_string ("moo-edit-actions");
|
||
|
|
||
|
return q;
|
||
|
}
|
||
|
|
||
|
|
||
|
static ActionInfo*
|
||
|
action_info_new (MooObjectFactory *action,
|
||
|
char **conditions)
|
||
|
{
|
||
|
ActionInfo *info;
|
||
|
|
||
|
g_return_val_if_fail (MOO_IS_OBJECT_FACTORY (action), NULL);
|
||
|
|
||
|
info = g_new0 (ActionInfo, 1);
|
||
|
info->action = g_object_ref (action);
|
||
|
info->conditions = g_strdupv (conditions);
|
||
|
|
||
|
return info;
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
action_info_free (ActionInfo *info)
|
||
|
{
|
||
|
if (info)
|
||
|
{
|
||
|
g_object_unref (info->action);
|
||
|
g_strfreev (info->conditions);
|
||
|
g_free (info);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_add_action (MooEdit *edit,
|
||
|
MooAction *action)
|
||
|
{
|
||
|
MooActionGroup *group;
|
||
|
|
||
|
g_return_if_fail (MOO_IS_EDIT (edit));
|
||
|
g_return_if_fail (MOO_IS_ACTION (action));
|
||
|
|
||
|
g_object_set (action, "group-name", "MooEdit", NULL);
|
||
|
|
||
|
group = moo_edit_get_actions (edit);
|
||
|
moo_action_group_add (group, action);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_remove_action (MooEdit *edit,
|
||
|
const char *action_id)
|
||
|
{
|
||
|
MooActionGroup *group;
|
||
|
|
||
|
g_return_if_fail (MOO_IS_EDIT (edit));
|
||
|
g_return_if_fail (action_id != NULL);
|
||
|
|
||
|
group = moo_edit_get_actions (edit);
|
||
|
moo_action_group_remove_action (group, action_id);
|
||
|
}
|
||
|
|
||
|
|
||
|
static MooAction*
|
||
|
create_action (const char *action_id,
|
||
|
ActionInfo *info,
|
||
|
MooEdit *edit)
|
||
|
{
|
||
|
MooAction *action;
|
||
|
|
||
|
g_return_val_if_fail (info != NULL, NULL);
|
||
|
g_return_val_if_fail (MOO_IS_OBJECT_FACTORY (info->action), NULL);
|
||
|
g_return_val_if_fail (action_id && action_id[0], NULL);
|
||
|
|
||
|
action = moo_object_factory_create_object (info->action, edit,
|
||
|
"closure-object", edit,
|
||
|
NULL);
|
||
|
g_return_val_if_fail (action != NULL, NULL);
|
||
|
g_object_set (action, "id", action_id, NULL);
|
||
|
|
||
|
if (g_type_is_a (info->action->object_type, MOO_TYPE_TOGGLE_ACTION))
|
||
|
g_object_set (action, "toggled-object", edit, NULL);
|
||
|
|
||
|
if (g_type_is_a (info->action->object_type, MOO_TYPE_EDIT_ACTION))
|
||
|
g_object_set (action, "doc", edit, NULL);
|
||
|
|
||
|
if (info->conditions)
|
||
|
{
|
||
|
char **p;
|
||
|
|
||
|
for (p = info->conditions; *p != NULL; p += 2)
|
||
|
{
|
||
|
if (p[1][0] == '!')
|
||
|
moo_bind_bool_property (action, p[0], edit, p[1] + 1, TRUE);
|
||
|
else
|
||
|
moo_bind_bool_property (action, p[0], edit, p[1], FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return action;
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
moo_edit_class_new_action (MooEditClass *klass,
|
||
|
const char *id,
|
||
|
const char *first_prop_name,
|
||
|
...)
|
||
|
{
|
||
|
va_list args;
|
||
|
va_start (args, first_prop_name);
|
||
|
moo_edit_class_new_actionv (klass, id, first_prop_name, args);
|
||
|
va_end (args);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_class_install_action (MooEditClass *klass,
|
||
|
const char *action_id,
|
||
|
MooObjectFactory *action,
|
||
|
char **conditions)
|
||
|
{
|
||
|
GHashTable *actions;
|
||
|
ActionInfo *info;
|
||
|
GType type;
|
||
|
GSList *l;
|
||
|
|
||
|
g_return_if_fail (MOO_IS_EDIT_CLASS (klass));
|
||
|
g_return_if_fail (MOO_IS_OBJECT_FACTORY (action));
|
||
|
g_return_if_fail (action_id && action_id[0]);
|
||
|
|
||
|
type = G_OBJECT_CLASS_TYPE (klass);
|
||
|
actions = g_type_get_qdata (type, MOO_EDIT_ACTIONS_QUARK);
|
||
|
|
||
|
if (!actions)
|
||
|
{
|
||
|
actions = g_hash_table_new_full (g_str_hash, g_str_equal,
|
||
|
g_free, (GDestroyNotify) action_info_free);
|
||
|
g_type_set_qdata (type, MOO_EDIT_ACTIONS_QUARK, actions);
|
||
|
}
|
||
|
|
||
|
if (g_hash_table_lookup (actions, action_id))
|
||
|
moo_edit_class_remove_action (klass, action_id);
|
||
|
|
||
|
info = action_info_new (action, conditions);
|
||
|
g_hash_table_insert (actions, g_strdup (action_id), info);
|
||
|
|
||
|
for (l = _moo_edit_instances; l != NULL; l = l->next)
|
||
|
{
|
||
|
if (g_type_is_a (G_OBJECT_TYPE (l->data), type))
|
||
|
{
|
||
|
MooAction *action = create_action (action_id, info, l->data);
|
||
|
|
||
|
if (action)
|
||
|
moo_edit_add_action (l->data, action);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
moo_edit_class_new_actionv (MooEditClass *klass,
|
||
|
const char *action_id,
|
||
|
const char *first_prop_name,
|
||
|
va_list var_args)
|
||
|
{
|
||
|
const char *name;
|
||
|
GType action_type = 0;
|
||
|
GObjectClass *action_class = NULL;
|
||
|
GArray *action_params = NULL;
|
||
|
GPtrArray *conditions = NULL;
|
||
|
|
||
|
g_return_if_fail (MOO_IS_EDIT_CLASS (klass));
|
||
|
g_return_if_fail (first_prop_name != NULL);
|
||
|
g_return_if_fail (action_id != NULL);
|
||
|
|
||
|
action_params = g_array_new (FALSE, TRUE, sizeof (GParameter));
|
||
|
conditions = g_ptr_array_new ();
|
||
|
|
||
|
name = first_prop_name;
|
||
|
while (name)
|
||
|
{
|
||
|
GParameter param = {NULL, {0, {{0}, {0}}}};
|
||
|
GParamSpec *pspec;
|
||
|
char *err = NULL;
|
||
|
|
||
|
/* ignore id property */
|
||
|
if (!strcmp (name, "id"))
|
||
|
{
|
||
|
g_critical ("%s: id property specified", G_STRLOC);
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
if (!strcmp (name, "action-type::") || !strcmp (name, "action_type::"))
|
||
|
{
|
||
|
g_value_init (¶m.value, MOO_TYPE_GTYPE);
|
||
|
G_VALUE_COLLECT (¶m.value, var_args, 0, &err);
|
||
|
|
||
|
if (err)
|
||
|
{
|
||
|
g_warning ("%s: %s", G_STRLOC, err);
|
||
|
g_free (err);
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
action_type = moo_value_get_gtype (¶m.value);
|
||
|
|
||
|
if (!g_type_is_a (action_type, MOO_TYPE_ACTION))
|
||
|
{
|
||
|
g_warning ("%s: invalid action type", G_STRLOC);
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
action_class = g_type_class_ref (action_type);
|
||
|
}
|
||
|
else if (!strncmp (name, "condition::", strlen ("condition::")))
|
||
|
{
|
||
|
const char *suffix = strstr (name, "::");
|
||
|
|
||
|
if (!suffix || !suffix[1] || !suffix[2])
|
||
|
{
|
||
|
g_warning ("%s: invalid condition name '%s'", G_STRLOC, name);
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
g_ptr_array_add (conditions, g_strdup (suffix + 2));
|
||
|
|
||
|
name = va_arg (var_args, gchar*);
|
||
|
|
||
|
if (!name)
|
||
|
{
|
||
|
g_warning ("%s: unterminated '%s' property",
|
||
|
G_STRLOC,
|
||
|
(char*) g_ptr_array_index (conditions, conditions->len - 1));
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
g_ptr_array_add (conditions, g_strdup (name));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (!action_class)
|
||
|
{
|
||
|
if (!action_type)
|
||
|
action_type = MOO_TYPE_EDIT_ACTION;
|
||
|
action_class = g_type_class_ref (action_type);
|
||
|
}
|
||
|
|
||
|
pspec = g_object_class_find_property (action_class, name);
|
||
|
|
||
|
if (!pspec)
|
||
|
{
|
||
|
g_warning ("%s: no property '%s' in class '%s'",
|
||
|
G_STRLOC, name, g_type_name (action_type));
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
g_value_init (¶m.value, G_PARAM_SPEC_VALUE_TYPE (pspec));
|
||
|
G_VALUE_COLLECT (¶m.value, var_args, 0, &err);
|
||
|
|
||
|
if (err)
|
||
|
{
|
||
|
g_warning ("%s: %s", G_STRLOC, err);
|
||
|
g_free (err);
|
||
|
g_value_unset (¶m.value);
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
param.name = g_strdup (name);
|
||
|
g_array_append_val (action_params, param);
|
||
|
}
|
||
|
|
||
|
name = va_arg (var_args, gchar*);
|
||
|
}
|
||
|
|
||
|
G_STMT_START
|
||
|
{
|
||
|
MooObjectFactory *action_factory = NULL;
|
||
|
|
||
|
action_factory = moo_object_factory_new_a (action_type,
|
||
|
(GParameter*) action_params->data,
|
||
|
action_params->len);
|
||
|
|
||
|
if (!action_factory)
|
||
|
{
|
||
|
g_warning ("%s: error in moo_object_factory_new_a()", G_STRLOC);
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
g_array_free (action_params, FALSE);
|
||
|
action_params = NULL;
|
||
|
|
||
|
g_ptr_array_add (conditions, NULL);
|
||
|
|
||
|
moo_edit_class_install_action (klass,
|
||
|
action_id,
|
||
|
action_factory,
|
||
|
(char**) conditions->pdata);
|
||
|
|
||
|
g_strfreev ((char**) conditions->pdata);
|
||
|
g_ptr_array_free (conditions, FALSE);
|
||
|
|
||
|
if (action_class)
|
||
|
g_type_class_unref (action_class);
|
||
|
if (action_factory)
|
||
|
g_object_unref (action_factory);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
G_STMT_END;
|
||
|
|
||
|
error:
|
||
|
if (action_params)
|
||
|
{
|
||
|
guint i;
|
||
|
GParameter *params = (GParameter*) action_params->data;
|
||
|
|
||
|
for (i = 0; i < action_params->len; ++i)
|
||
|
{
|
||
|
g_value_unset (¶ms[i].value);
|
||
|
g_free ((char*) params[i].name);
|
||
|
}
|
||
|
|
||
|
g_array_free (action_params, TRUE);
|
||
|
}
|
||
|
|
||
|
if (conditions)
|
||
|
{
|
||
|
guint i;
|
||
|
for (i = 0; i < conditions->len; ++i)
|
||
|
g_free (g_ptr_array_index (conditions, i));
|
||
|
g_ptr_array_free (conditions, TRUE);
|
||
|
}
|
||
|
|
||
|
if (action_class)
|
||
|
g_type_class_unref (action_class);
|
||
|
}
|
||
|
|
||
|
|
||
|
static GObject*
|
||
|
custom_action_factory_func (MooEdit *edit,
|
||
|
MooObjectFactory *factory)
|
||
|
{
|
||
|
const char *action_id;
|
||
|
MooEditActionFunc func;
|
||
|
gpointer func_data;
|
||
|
MooAction *action;
|
||
|
|
||
|
g_return_val_if_fail (MOO_IS_EDIT (edit), NULL);
|
||
|
|
||
|
action_id = g_object_get_data (G_OBJECT (factory), "moo-edit-class-action-id");
|
||
|
func = g_object_get_data (G_OBJECT (factory), "moo-edit-class-action-func");
|
||
|
func_data = g_object_get_data (G_OBJECT (factory), "moo-edit-class-action-func-data");
|
||
|
|
||
|
g_return_val_if_fail (action_id != NULL, NULL);
|
||
|
g_return_val_if_fail (func != NULL, NULL);
|
||
|
|
||
|
action = func (edit, func_data);
|
||
|
|
||
|
return action ? G_OBJECT (action) : NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
moo_edit_class_new_action_custom (MooEditClass *klass,
|
||
|
const char *action_id,
|
||
|
MooEditActionFunc func,
|
||
|
gpointer data,
|
||
|
GDestroyNotify notify)
|
||
|
{
|
||
|
MooObjectFactory *action_factory;
|
||
|
|
||
|
g_return_if_fail (MOO_IS_EDIT_CLASS (klass));
|
||
|
g_return_if_fail (action_id && action_id[0]);
|
||
|
g_return_if_fail (func != NULL);
|
||
|
|
||
|
action_factory = moo_object_factory_new_func ((MooObjectFactoryFunc) custom_action_factory_func, NULL);
|
||
|
g_object_set_data (G_OBJECT (action_factory), "moo-edit-class", klass);
|
||
|
g_object_set_data_full (G_OBJECT (action_factory), "moo-edit-class-action-id",
|
||
|
g_strdup (action_id), g_free);
|
||
|
g_object_set_data (G_OBJECT (action_factory), "moo-edit-class-action-func", func);
|
||
|
g_object_set_data_full (G_OBJECT (action_factory), "moo-edit-class-action-func-data",
|
||
|
data, notify);
|
||
|
|
||
|
moo_edit_class_install_action (klass, action_id, action_factory, NULL);
|
||
|
g_object_unref (action_factory);
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
moo_edit_class_remove_action (MooEditClass *klass,
|
||
|
const char *action_id)
|
||
|
{
|
||
|
GHashTable *actions;
|
||
|
GType type;
|
||
|
GSList *l;
|
||
|
|
||
|
g_return_if_fail (MOO_IS_EDIT_CLASS (klass));
|
||
|
|
||
|
type = G_OBJECT_CLASS_TYPE (klass);
|
||
|
actions = g_type_get_qdata (type, MOO_EDIT_ACTIONS_QUARK);
|
||
|
|
||
|
if (actions)
|
||
|
g_hash_table_remove (actions, action_id);
|
||
|
|
||
|
for (l = _moo_edit_instances; l != NULL; l = l->next)
|
||
|
if (g_type_is_a (G_OBJECT_TYPE (l->data), type))
|
||
|
moo_edit_remove_action (l->data, action_id);
|
||
|
}
|
||
|
|
||
|
|
||
|
MooActionGroup *
|
||
|
moo_edit_get_actions (MooEdit *edit)
|
||
|
{
|
||
|
g_return_val_if_fail (MOO_IS_EDIT (edit), NULL);
|
||
|
return edit->priv->actions;
|
||
|
}
|
||
|
|
||
|
|
||
|
MooAction *
|
||
|
moo_edit_get_action_by_id (MooEdit *edit,
|
||
|
const char *action_id)
|
||
|
{
|
||
|
MooActionGroup *actions;
|
||
|
|
||
|
g_return_val_if_fail (MOO_IS_EDIT (edit), NULL);
|
||
|
g_return_val_if_fail (action_id != NULL, NULL);
|
||
|
|
||
|
actions = moo_edit_get_actions (edit);
|
||
|
return moo_action_group_get_action (actions, action_id);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
add_action (const char *id,
|
||
|
ActionInfo *info,
|
||
|
MooEdit *edit)
|
||
|
{
|
||
|
MooAction *action = create_action (id, info, edit);
|
||
|
|
||
|
if (action)
|
||
|
moo_edit_add_action (edit, action);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
_moo_edit_add_class_actions (MooEdit *edit)
|
||
|
{
|
||
|
GType type;
|
||
|
|
||
|
g_return_if_fail (MOO_IS_EDIT (edit));
|
||
|
|
||
|
type = G_OBJECT_TYPE (edit);
|
||
|
|
||
|
while (TRUE)
|
||
|
{
|
||
|
GHashTable *actions;
|
||
|
|
||
|
actions = g_type_get_qdata (type, MOO_EDIT_ACTIONS_QUARK);
|
||
|
|
||
|
if (actions)
|
||
|
g_hash_table_foreach (actions, (GHFunc) add_action, edit);
|
||
|
|
||
|
if (type == MOO_TYPE_EDIT)
|
||
|
break;
|
||
|
|
||
|
type = g_type_parent (type);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static gboolean
|
||
|
check_action (G_GNUC_UNUSED MooActionGroup *group,
|
||
|
MooAction *action,
|
||
|
G_GNUC_UNUSED gpointer data)
|
||
|
{
|
||
|
if (MOO_IS_EDIT_ACTION (action))
|
||
|
moo_edit_action_check (MOO_EDIT_ACTION (action));
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
_moo_edit_check_actions (MooEdit *edit)
|
||
|
{
|
||
|
moo_action_group_foreach (edit->priv->actions,
|
||
|
check_action, NULL);
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
_moo_edit_class_init_actions (MooEditClass *klass)
|
||
|
{
|
||
|
moo_edit_class_new_action (klass, "Undo",
|
||
|
"name", "Undo",
|
||
|
"label", "_Undo",
|
||
|
"tooltip", "Undo",
|
||
|
"icon-stock-id", GTK_STOCK_UNDO,
|
||
|
"closure-signal", "undo",
|
||
|
"condition::sensitive", "can-undo",
|
||
|
NULL);
|
||
|
|
||
|
moo_edit_class_new_action (klass, "Redo",
|
||
|
"name", "Redo",
|
||
|
"label", "_Redo",
|
||
|
"tooltip", "Redo",
|
||
|
"icon-stock-id", GTK_STOCK_REDO,
|
||
|
"closure-signal", "redo",
|
||
|
"condition::sensitive", "can-redo",
|
||
|
NULL);
|
||
|
|
||
|
moo_edit_class_new_action (klass, "Cut",
|
||
|
"name", "Cut",
|
||
|
"stock-id", GTK_STOCK_CUT,
|
||
|
"closure-signal", "cut-clipboard",
|
||
|
"condition::sensitive", "has-selection",
|
||
|
NULL);
|
||
|
|
||
|
moo_edit_class_new_action (klass, "Copy",
|
||
|
"name", "Copy",
|
||
|
"stock-id", GTK_STOCK_COPY,
|
||
|
"closure-signal", "copy-clipboard",
|
||
|
"condition::sensitive", "has-selection",
|
||
|
NULL);
|
||
|
|
||
|
moo_edit_class_new_action (klass, "Paste",
|
||
|
"name", "Paste",
|
||
|
"stock-id", GTK_STOCK_PASTE,
|
||
|
"closure-signal", "paste-clipboard",
|
||
|
NULL);
|
||
|
|
||
|
moo_edit_class_new_action (klass, "SelectAll",
|
||
|
"name", "Select All",
|
||
|
"label", "Select _All",
|
||
|
"tooltip", "Select all",
|
||
|
"icon-stock-id", GTK_STOCK_SELECT_ALL,
|
||
|
"closure-callback", moo_text_view_select_all,
|
||
|
"condition::sensitive", "has-text",
|
||
|
NULL);
|
||
|
}
|
||
|
|
||
|
|
||
|
/****************************************************************************/
|
||
|
/* MooEditAction
|
||
|
*/
|
||
|
|
||
|
G_DEFINE_TYPE (MooEditAction, moo_edit_action, MOO_TYPE_ACTION);
|
||
|
|
||
|
enum {
|
||
|
PROP_0,
|
||
|
PROP_DOC,
|
||
|
PROP_FLAGS,
|
||
|
PROP_LANGS
|
||
|
};
|
||
|
|
||
|
static void
|
||
|
moo_edit_action_finalize (GObject *object)
|
||
|
{
|
||
|
MooEditAction *action = MOO_EDIT_ACTION (object);
|
||
|
g_slist_foreach (action->langs, (GFunc) g_free, NULL);
|
||
|
g_slist_free (action->langs);
|
||
|
G_OBJECT_CLASS(moo_edit_action_parent_class)->finalize (object);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_action_get_property (GObject *object,
|
||
|
guint prop_id,
|
||
|
GValue *value,
|
||
|
GParamSpec *pspec)
|
||
|
{
|
||
|
MooEditAction *action = MOO_EDIT_ACTION (object);
|
||
|
|
||
|
switch (prop_id)
|
||
|
{
|
||
|
case PROP_DOC:
|
||
|
g_value_set_object (value, action->doc);
|
||
|
break;
|
||
|
|
||
|
case PROP_FLAGS:
|
||
|
g_value_set_flags (value, action->flags);
|
||
|
break;
|
||
|
|
||
|
case PROP_LANGS:
|
||
|
g_value_set_pointer (value, action->langs);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
string_slist_free (GSList *list)
|
||
|
{
|
||
|
g_slist_foreach (list, (GFunc) g_free, NULL);
|
||
|
g_slist_free (list);
|
||
|
}
|
||
|
|
||
|
|
||
|
static GSList *
|
||
|
string_slist_copy (GSList *list)
|
||
|
{
|
||
|
GSList *copy = NULL;
|
||
|
|
||
|
while (list)
|
||
|
{
|
||
|
copy = g_slist_prepend (copy, g_strdup (list->data));
|
||
|
list = list->next;
|
||
|
}
|
||
|
|
||
|
return g_slist_reverse (copy);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_action_set_property (GObject *object,
|
||
|
guint prop_id,
|
||
|
const GValue *value,
|
||
|
GParamSpec *pspec)
|
||
|
{
|
||
|
MooEditAction *action = MOO_EDIT_ACTION (object);
|
||
|
|
||
|
switch (prop_id)
|
||
|
{
|
||
|
case PROP_DOC:
|
||
|
action->doc = g_value_get_object (value);
|
||
|
g_object_notify (object, "doc");
|
||
|
break;
|
||
|
|
||
|
case PROP_FLAGS:
|
||
|
action->flags = g_value_get_flags (value);
|
||
|
g_object_notify (object, "flags");
|
||
|
break;
|
||
|
|
||
|
case PROP_LANGS:
|
||
|
string_slist_free (action->langs);
|
||
|
action->langs = string_slist_copy (g_value_get_pointer (value));
|
||
|
g_object_notify (object, "langs");
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_action_class_init (MooEditActionClass *klass)
|
||
|
{
|
||
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
||
|
|
||
|
gobject_class->finalize = moo_edit_action_finalize;
|
||
|
gobject_class->set_property = moo_edit_action_set_property;
|
||
|
gobject_class->get_property = moo_edit_action_get_property;
|
||
|
|
||
|
g_object_class_install_property (gobject_class,
|
||
|
PROP_DOC,
|
||
|
g_param_spec_object ("doc",
|
||
|
"doc",
|
||
|
"doc",
|
||
|
MOO_TYPE_EDIT,
|
||
|
G_PARAM_READWRITE));
|
||
|
|
||
|
g_object_class_install_property (gobject_class,
|
||
|
PROP_FLAGS,
|
||
|
g_param_spec_flags ("flags",
|
||
|
"flags",
|
||
|
"flags",
|
||
|
MOO_TYPE_EDIT_ACTION_FLAGS, 0,
|
||
|
G_PARAM_READWRITE));
|
||
|
|
||
|
g_object_class_install_property (gobject_class,
|
||
|
PROP_LANGS,
|
||
|
g_param_spec_pointer ("langs",
|
||
|
"langs",
|
||
|
"langs",
|
||
|
G_PARAM_READWRITE));
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_action_init (G_GNUC_UNUSED MooEditAction *action)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
moo_edit_action_check (MooEditAction *action)
|
||
|
{
|
||
|
g_return_if_fail (action->doc != NULL);
|
||
|
|
||
|
if (action->flags)
|
||
|
{
|
||
|
gboolean sensitive = (action->flags & MOO_EDIT_ACTION_NEED_FILE) ?
|
||
|
moo_edit_get_filename (action->doc) != NULL : TRUE;
|
||
|
g_object_set (action, "sensitive", sensitive, NULL);
|
||
|
}
|
||
|
|
||
|
if (action->langs)
|
||
|
{
|
||
|
GSList *l;
|
||
|
MooLang *lang = moo_text_view_get_lang (MOO_TEXT_VIEW (action->doc));
|
||
|
const char *lang_id = lang ? lang->id : MOO_LANG_NONE;
|
||
|
gboolean visible = FALSE;
|
||
|
|
||
|
for (l = action->langs; l != NULL; l = l->next)
|
||
|
{
|
||
|
if (l->data && !strcmp (l->data, lang_id))
|
||
|
{
|
||
|
visible = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
g_object_set (action, "visible", visible, NULL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
GType
|
||
|
moo_edit_action_flags_get_type (void)
|
||
|
{
|
||
|
static GType type;
|
||
|
|
||
|
if (!type)
|
||
|
{
|
||
|
static GFlagsValue values[] = {
|
||
|
{ MOO_EDIT_ACTION_NEED_FILE, (char*) "MOO_EDIT_ACTION_NEED_FILE", (char*) "need-file" },
|
||
|
{ 0, NULL, NULL },
|
||
|
};
|
||
|
|
||
|
type = g_flags_register_static ("MooEditActionFlags", values);
|
||
|
}
|
||
|
|
||
|
return type;
|
||
|
}
|