medit/moo/mooedit/mooeditaction-factory.cpp

793 lines
26 KiB
C++
Raw Normal View History

2006-04-20 03:57:05 -05:00
/*
2016-01-02 13:51:52 -08:00
* mooeditaction-factory.cpp
2006-04-20 03:57:05 -05:00
*
2016-01-02 13:51:52 -08:00
* Copyright (C) 2004-2016 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
2006-04-20 03:57:05 -05:00
*
* This file is part of medit. medit 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.1 of the License,
* or (at your option) any later version.
2006-04-20 03:57:05 -05:00
*
* You should have received a copy of the GNU Lesser General Public
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
2006-04-20 03:57:05 -05:00
*/
#include "mooedit/mooeditaction-factory.h"
#include "mooedit/mooeditaction.h"
2006-04-20 03:57:05 -05:00
#include "mooedit/mooedit-private.h"
#include "mooutils/mooutils-gobject.h"
2006-05-08 23:57:16 -05:00
#include "mooutils/mooactionfactory.h"
#include "mooutils/mooactionbase.h"
#include "mooutils/moomenuaction.h"
#include "mooutils/mooi18n.h"
2008-09-07 00:15:07 -05:00
#include "mooutils/mootype-macros.h"
2006-04-20 03:57:05 -05:00
#include <string.h>
#include <gobject/gvaluecollector.h>
typedef GtkAction *(*MooEditActionFunc) (MooEdit *edit,
gpointer data);
2006-08-16 00:18:16 -05:00
static void moo_edit_add_action (MooEdit *edit,
GtkAction *action);
static void moo_edit_remove_action (MooEdit *edit,
const char *action_id);
static void moo_edit_class_new_actionv (MooEditClass *klass,
const char *id,
const char *first_prop_name,
va_list props);
static void moo_edit_class_new_action_custom (MooEditClass *klass,
const char *id,
MooEditActionFunc func,
gpointer data,
GDestroyNotify notify);
static void append_special_char_menuitems (GtkMenuShell *menu,
2011-01-17 03:01:11 -08:00
MooEditView *view);
2006-04-20 03:57:05 -05:00
2008-09-07 00:15:07 -05:00
#define MOO_EDIT_ACTIONS_QUARK (moo_edit_actions_quark ())
MOO_DEFINE_QUARK_STATIC (moo-edit-actions, moo_edit_actions_quark)
2006-04-20 03:57:05 -05:00
typedef struct {
2006-05-08 23:57:16 -05:00
MooActionFactory *action;
2010-12-18 23:58:18 -08:00
char **doc_conditions;
char **view_conditions;
2006-04-20 03:57:05 -05:00
} ActionInfo;
static ActionInfo*
2006-05-08 23:57:16 -05:00
action_info_new (MooActionFactory *action,
2010-12-18 23:58:18 -08:00
char **doc_conditions,
char **view_conditions)
2006-04-20 03:57:05 -05:00
{
ActionInfo *info;
2006-05-08 23:57:16 -05:00
g_return_val_if_fail (MOO_IS_ACTION_FACTORY (action), NULL);
2006-04-20 03:57:05 -05:00
info = g_new0 (ActionInfo, 1);
2010-02-01 22:44:03 -08:00
g_object_ref (action);
info->action = action;
2010-12-18 23:58:18 -08:00
info->doc_conditions = g_strdupv (doc_conditions);
info->view_conditions = g_strdupv (view_conditions);
2006-04-20 03:57:05 -05:00
return info;
}
static void
action_info_free (ActionInfo *info)
{
if (info)
{
g_object_unref (info->action);
2010-12-18 23:58:18 -08:00
g_strfreev (info->doc_conditions);
g_strfreev (info->view_conditions);
2006-04-20 03:57:05 -05:00
g_free (info);
}
}
static void
moo_edit_add_action (MooEdit *edit,
2006-05-08 23:57:16 -05:00
GtkAction *action)
2006-04-20 03:57:05 -05:00
{
2006-05-08 23:57:16 -05:00
GtkActionGroup *group;
2006-04-20 03:57:05 -05:00
g_return_if_fail (MOO_IS_EDIT (edit));
2006-05-08 23:57:16 -05:00
g_return_if_fail (GTK_IS_ACTION (action));
2006-04-20 03:57:05 -05:00
group = moo_edit_get_actions (edit);
2006-05-08 23:57:16 -05:00
gtk_action_group_add_action (group, action);
2006-04-20 03:57:05 -05:00
}
static void
moo_edit_remove_action (MooEdit *edit,
const char *action_id)
{
2006-05-08 23:57:16 -05:00
GtkActionGroup *group;
GtkAction *action;
2006-04-20 03:57:05 -05:00
g_return_if_fail (MOO_IS_EDIT (edit));
g_return_if_fail (action_id != NULL);
group = moo_edit_get_actions (edit);
2006-05-08 23:57:16 -05:00
action = gtk_action_group_get_action (group, action_id);
gtk_action_group_remove_action (group, action);
2006-04-20 03:57:05 -05:00
}
2006-05-08 23:57:16 -05:00
static GtkAction*
2006-04-20 03:57:05 -05:00
create_action (const char *action_id,
ActionInfo *info,
MooEdit *edit)
{
2006-05-08 23:57:16 -05:00
GtkAction *action;
2010-12-18 23:58:18 -08:00
char **p;
MooEditView *view;
2006-04-20 03:57:05 -05:00
2010-12-18 23:58:18 -08:00
g_return_val_if_fail (MOO_IS_EDIT (edit), NULL);
2006-04-20 03:57:05 -05:00
g_return_val_if_fail (info != NULL, NULL);
2006-05-08 23:57:16 -05:00
g_return_val_if_fail (MOO_IS_ACTION_FACTORY (info->action), NULL);
2006-04-20 03:57:05 -05:00
g_return_val_if_fail (action_id && action_id[0], NULL);
2010-12-18 23:58:18 -08:00
view = moo_edit_get_view (edit);
if (g_type_is_a (info->action->action_type, MOO_TYPE_ACTION))
action = moo_action_factory_create_action (info->action, edit,
"closure-object", edit,
"name", action_id,
2010-02-01 22:44:03 -08:00
(char*) 0);
else if (g_type_is_a (info->action->action_type, MOO_TYPE_TOGGLE_ACTION))
action = moo_action_factory_create_action (info->action, edit,
"toggled-object", edit,
"name", action_id,
2010-02-01 22:44:03 -08:00
(char*) 0);
else
action = moo_action_factory_create_action (info->action, edit,
"name", action_id,
2010-02-01 22:44:03 -08:00
(char*) 0);
2006-04-20 03:57:05 -05:00
g_return_val_if_fail (action != NULL, NULL);
2006-05-08 23:57:16 -05:00
if (g_type_is_a (info->action->action_type, MOO_TYPE_EDIT_ACTION))
2010-02-01 22:44:03 -08:00
g_object_set (action, "doc", edit, (char*) 0);
g_object_set_data (G_OBJECT (action), "moo-edit", edit);
2006-04-20 03:57:05 -05:00
2010-12-18 23:58:18 -08:00
for (p = info->doc_conditions; p && *p; p += 2)
2006-04-20 03:57:05 -05:00
{
2010-12-18 23:58:18 -08:00
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);
}
2006-04-20 03:57:05 -05:00
2010-12-18 23:58:18 -08:00
for (p = info->view_conditions; p && *p; p += 2)
{
if (p[1][0] == '!')
moo_bind_bool_property (action, p[0], view, p[1] + 1, TRUE);
else
moo_bind_bool_property (action, p[0], view, p[1], FALSE);
2006-04-20 03:57:05 -05:00
}
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);
}
2010-02-01 22:44:03 -08:00
static GHashTable *
get_actions_hash (GType type)
{
return (GHashTable*) g_type_get_qdata (type, MOO_EDIT_ACTIONS_QUARK);
}
2006-04-20 03:57:05 -05:00
static void
2006-12-04 04:54:12 -06:00
moo_edit_class_install_action (MooEditClass *klass,
const char *action_id,
MooActionFactory *factory,
2010-12-18 23:58:18 -08:00
char **doc_conditions,
char **view_conditions)
2006-04-20 03:57:05 -05:00
{
GHashTable *actions;
ActionInfo *info;
GType type;
2010-02-01 22:44:03 -08:00
MooEditList *l;
2006-04-20 03:57:05 -05:00
g_return_if_fail (MOO_IS_EDIT_CLASS (klass));
2006-12-04 04:54:12 -06:00
g_return_if_fail (MOO_IS_ACTION_FACTORY (factory));
2006-04-20 03:57:05 -05:00
g_return_if_fail (action_id && action_id[0]);
type = G_OBJECT_CLASS_TYPE (klass);
2010-02-01 22:44:03 -08:00
actions = get_actions_hash (type);
2006-04-20 03:57:05 -05:00
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);
2010-12-18 23:58:18 -08:00
info = action_info_new (factory, doc_conditions, view_conditions);
2006-04-20 03:57:05 -05:00
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))
{
2006-05-08 23:57:16 -05:00
GtkAction *action = create_action (action_id, info, l->data);
2006-04-20 03:57:05 -05:00
if (action)
2007-08-06 13:17:25 -05:00
{
2006-04-20 03:57:05 -05:00
moo_edit_add_action (l->data, action);
2007-08-06 13:17:25 -05:00
g_object_unref (action);
}
2006-04-20 03:57:05 -05:00
}
}
}
2006-08-16 00:18:16 -05:00
static void
2006-04-20 03:57:05 -05:00
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;
2010-12-18 23:58:18 -08:00
GPtrArray *doc_conditions = NULL;
GPtrArray *view_conditions = NULL;
2006-04-20 03:57:05 -05:00
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));
2010-12-18 23:58:18 -08:00
doc_conditions = g_ptr_array_new ();
view_conditions = g_ptr_array_new ();
2006-04-20 03:57:05 -05:00
name = first_prop_name;
while (name)
{
2010-02-01 22:44:03 -08:00
GParameter param = { 0 };
2006-04-20 03:57:05 -05:00
GParamSpec *pspec;
char *err = NULL;
/* ignore id property */
2006-05-08 23:57:16 -05:00
if (!strcmp (name, "id") || !strcmp (name, "name"))
2006-04-20 03:57:05 -05:00
{
g_critical ("id property specified");
2006-04-20 03:57:05 -05:00
goto error;
}
if (!strcmp (name, "action-type::") || !strcmp (name, "action_type::"))
{
g_value_init (&param.value, MOO_TYPE_GTYPE);
G_VALUE_COLLECT (&param.value, var_args, 0, &err);
if (err)
{
g_warning ("%s", err);
2006-04-20 03:57:05 -05:00
g_free (err);
goto error;
}
2006-08-15 03:21:44 -05:00
action_type = _moo_value_get_gtype (&param.value);
2006-04-20 03:57:05 -05:00
if (!g_type_is_a (action_type, MOO_TYPE_ACTION_BASE))
2006-04-20 03:57:05 -05:00
{
g_warning ("invalid action type %s", g_type_name (action_type));
2006-04-20 03:57:05 -05:00
goto error;
}
2010-02-01 22:44:03 -08:00
action_class = G_OBJECT_CLASS (g_type_class_ref (action_type));
2006-04-20 03:57:05 -05:00
}
2010-12-18 23:58:18 -08:00
else if (g_str_has_prefix (name, "condition::") ||
g_str_has_prefix (name, "view-condition::"))
2006-04-20 03:57:05 -05:00
{
2010-12-18 23:58:18 -08:00
GPtrArray *conditions;
gboolean is_doc = g_str_has_prefix (name, "condition::");
2006-04-20 03:57:05 -05:00
const char *suffix = strstr (name, "::");
if (!suffix || !suffix[1] || !suffix[2])
{
g_warning ("invalid condition name '%s'", name);
2006-04-20 03:57:05 -05:00
goto error;
}
2010-12-18 23:58:18 -08:00
suffix += 2;
2006-04-20 03:57:05 -05:00
name = va_arg (var_args, gchar*);
if (!name)
{
g_warning ("unterminated '%s' property", suffix);
2006-04-20 03:57:05 -05:00
goto error;
}
2010-12-18 23:58:18 -08:00
conditions = is_doc ? doc_conditions : view_conditions;
g_ptr_array_add (conditions, g_strdup (suffix));
2010-12-08 01:11:51 -08:00
g_ptr_array_add (conditions, g_strdup (name));
2006-04-20 03:57:05 -05:00
}
else
{
if (!action_class)
{
if (!action_type)
action_type = MOO_TYPE_EDIT_ACTION;
2010-02-01 22:44:03 -08:00
action_class = G_OBJECT_CLASS (g_type_class_ref (action_type));
2006-04-20 03:57:05 -05:00
}
pspec = g_object_class_find_property (action_class, name);
2006-04-20 03:57:05 -05:00
if (!pspec)
{
g_warning ("no property '%s' in class '%s'",
name, g_type_name (action_type));
2006-04-20 03:57:05 -05:00
goto error;
}
g_value_init (&param.value, G_PARAM_SPEC_VALUE_TYPE (pspec));
G_VALUE_COLLECT (&param.value, var_args, 0, &err);
if (err)
{
g_warning ("%s", err);
2006-04-20 03:57:05 -05:00
g_free (err);
g_value_unset (&param.value);
goto error;
}
param.name = g_strdup (name);
g_array_append_val (action_params, param);
}
name = va_arg (var_args, gchar*);
}
G_STMT_START
{
2006-05-08 23:57:16 -05:00
MooActionFactory *action_factory = NULL;
2006-04-20 03:57:05 -05:00
2006-05-08 23:57:16 -05:00
action_factory = moo_action_factory_new_a (action_type,
2006-04-20 03:57:05 -05:00
(GParameter*) action_params->data,
action_params->len);
if (!action_factory)
{
g_warning ("error in moo_object_factory_new_a()");
2006-04-20 03:57:05 -05:00
goto error;
}
2006-08-15 03:21:44 -05:00
_moo_param_array_free ((GParameter*) action_params->data, action_params->len);
2006-04-20 03:57:05 -05:00
g_array_free (action_params, FALSE);
action_params = NULL;
2010-12-18 23:58:18 -08:00
g_ptr_array_add (doc_conditions, NULL);
g_ptr_array_add (view_conditions, NULL);
2006-04-20 03:57:05 -05:00
moo_edit_class_install_action (klass,
action_id,
action_factory,
2010-12-18 23:58:18 -08:00
(char**) doc_conditions->pdata,
(char**) view_conditions->pdata);
2006-04-20 03:57:05 -05:00
2010-12-18 23:58:18 -08:00
g_strfreev ((char**) doc_conditions->pdata);
g_strfreev ((char**) view_conditions->pdata);
g_ptr_array_free (doc_conditions, FALSE);
g_ptr_array_free (view_conditions, FALSE);
2006-04-20 03:57:05 -05:00
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 (&params[i].value);
g_free ((char*) params[i].name);
}
g_array_free (action_params, TRUE);
}
2010-12-18 23:58:18 -08:00
if (doc_conditions)
2006-04-20 03:57:05 -05:00
{
2015-12-31 16:41:12 -08:00
g_ptr_array_foreach (doc_conditions, (GFunc) g_free, NULL);
2010-12-18 23:58:18 -08:00
g_ptr_array_free (doc_conditions, TRUE);
}
if (view_conditions)
{
2015-12-31 16:41:12 -08:00
g_ptr_array_foreach (view_conditions, (GFunc) g_free, NULL);
2010-12-18 23:58:18 -08:00
g_ptr_array_free (view_conditions, TRUE);
2006-04-20 03:57:05 -05:00
}
if (action_class)
g_type_class_unref (action_class);
}
2006-05-08 23:57:16 -05:00
static GtkAction *
2006-04-20 03:57:05 -05:00
custom_action_factory_func (MooEdit *edit,
2006-05-08 23:57:16 -05:00
MooActionFactory *factory)
2006-04-20 03:57:05 -05:00
{
const char *action_id;
MooEditActionFunc func;
gpointer func_data;
g_return_val_if_fail (MOO_IS_EDIT (edit), NULL);
2010-02-01 22:44:03 -08:00
action_id = (const char *) g_object_get_data (G_OBJECT (factory), "moo-edit-class-action-id");
func = (MooEditActionFunc) g_object_get_data (G_OBJECT (factory), "moo-edit-class-action-func");
2006-04-20 03:57:05 -05:00
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);
2006-05-08 23:57:16 -05:00
return func (edit, func_data);
2006-04-20 03:57:05 -05:00
}
2006-08-16 00:18:16 -05:00
static void
2006-04-20 03:57:05 -05:00
moo_edit_class_new_action_custom (MooEditClass *klass,
const char *action_id,
MooEditActionFunc func,
gpointer data,
GDestroyNotify notify)
{
2006-05-08 23:57:16 -05:00
MooActionFactory *action_factory;
2006-04-20 03:57:05 -05:00
g_return_if_fail (MOO_IS_EDIT_CLASS (klass));
g_return_if_fail (action_id && action_id[0]);
g_return_if_fail (func != NULL);
2006-05-08 23:57:16 -05:00
action_factory = moo_action_factory_new_func ((MooActionFactoryFunc) custom_action_factory_func, NULL);
2006-04-20 03:57:05 -05:00
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);
2010-02-01 22:44:03 -08:00
g_object_set_data (G_OBJECT (action_factory), "moo-edit-class-action-func", (gpointer) func);
2006-04-20 03:57:05 -05:00
g_object_set_data_full (G_OBJECT (action_factory), "moo-edit-class-action-func-data",
data, notify);
2010-12-18 23:58:18 -08:00
moo_edit_class_install_action (klass, action_id, action_factory, NULL, NULL);
2006-04-20 03:57:05 -05:00
g_object_unref (action_factory);
}
2006-05-08 23:57:16 -05:00
static GtkAction *
2006-04-29 04:39:25 -05:00
type_action_func (MooEdit *edit,
gpointer klass)
{
2006-05-08 23:57:16 -05:00
GQuark quark = g_quark_from_static_string ("moo-edit-class-action-id");
2010-02-01 22:44:03 -08:00
const char *id = (const char*) g_type_get_qdata (G_TYPE_FROM_CLASS (klass), quark);
return GTK_ACTION (g_object_new (G_TYPE_FROM_CLASS (klass),
"doc", edit, "name", id, (char*) 0));
2006-04-29 04:39:25 -05:00
}
void
moo_edit_class_new_action_type (MooEditClass *edit_klass,
const char *id,
GType type)
{
gpointer klass;
2006-05-08 23:57:16 -05:00
GQuark quark;
2006-04-29 04:39:25 -05:00
g_return_if_fail (g_type_is_a (type, MOO_TYPE_EDIT_ACTION));
klass = g_type_class_ref (type);
g_return_if_fail (klass != NULL);
2006-05-08 23:57:16 -05:00
quark = g_quark_from_static_string ("moo-edit-class-action-id");
g_free (g_type_get_qdata (type, quark));
g_type_set_qdata (type, quark, g_strdup (id));
2006-04-29 04:39:25 -05:00
moo_edit_class_new_action_custom (edit_klass, id, type_action_func,
klass, g_type_class_unref);
}
2006-04-20 03:57:05 -05:00
void
moo_edit_class_remove_action (MooEditClass *klass,
const char *action_id)
{
GHashTable *actions;
GType type;
2010-02-01 22:44:03 -08:00
MooEditList *l;
2006-04-20 03:57:05 -05:00
g_return_if_fail (MOO_IS_EDIT_CLASS (klass));
type = G_OBJECT_CLASS_TYPE (klass);
2010-02-01 22:44:03 -08:00
actions = get_actions_hash (type);
2006-04-20 03:57:05 -05:00
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);
}
2006-05-08 23:57:16 -05:00
GtkActionGroup *
2006-04-20 03:57:05 -05:00
moo_edit_get_actions (MooEdit *edit)
{
g_return_val_if_fail (MOO_IS_EDIT (edit), NULL);
2016-01-03 06:13:38 -08:00
return moo_action_collection_get_group (edit->priv->actions.get(), NULL);
2006-04-20 03:57:05 -05:00
}
2006-05-08 23:57:16 -05:00
GtkAction *
2006-04-20 03:57:05 -05:00
moo_edit_get_action_by_id (MooEdit *edit,
const char *action_id)
{
2006-05-08 23:57:16 -05:00
GtkActionGroup *actions;
2006-04-20 03:57:05 -05:00
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);
2006-05-08 23:57:16 -05:00
return gtk_action_group_get_action (actions, action_id);
2006-04-20 03:57:05 -05:00
}
static void
add_action (const char *id,
ActionInfo *info,
MooEdit *edit)
{
2006-05-08 23:57:16 -05:00
GtkAction *action = create_action (id, info, edit);
2006-04-20 03:57:05 -05:00
if (action)
2007-08-06 13:17:25 -05:00
{
2006-04-20 03:57:05 -05:00
moo_edit_add_action (edit, action);
2007-08-06 13:17:25 -05:00
g_object_unref (action);
}
2006-04-20 03:57:05 -05:00
}
2016-01-04 09:56:33 -08:00
void Edit::_add_class_actions()
2006-04-20 03:57:05 -05:00
{
2016-01-04 03:56:42 -08:00
GType type = G_OBJECT_TYPE (g());
2006-04-20 03:57:05 -05:00
2016-01-04 03:56:42 -08:00
while (true)
2006-04-20 03:57:05 -05:00
{
GHashTable *actions;
2010-02-01 22:44:03 -08:00
actions = get_actions_hash (type);
2006-04-20 03:57:05 -05:00
if (actions)
2016-01-04 03:56:42 -08:00
g_hash_table_foreach (actions, (GHFunc) add_action, g());
2006-04-20 03:57:05 -05:00
if (type == MOO_TYPE_EDIT)
break;
type = g_type_parent (type);
}
}
static GtkWidget *
create_input_methods_menu_item (GtkAction *action)
{
2011-01-17 03:01:11 -08:00
MooEditView *view;
GtkWidget *item, *menu;
gboolean visible = TRUE;
2015-12-31 04:53:46 -08:00
view = MOO_EDIT_VIEW (g_object_get_data (G_OBJECT (action), "moo-edit-view"));
2011-01-17 03:01:11 -08:00
g_return_val_if_fail (MOO_IS_EDIT_VIEW (view), NULL);
item = gtk_menu_item_new ();
menu = gtk_menu_new ();
gtk_widget_show (menu);
gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), menu);
2011-01-17 03:01:11 -08:00
gtk_im_multicontext_append_menuitems (GTK_IM_MULTICONTEXT (GTK_TEXT_VIEW (view)->im_context),
GTK_MENU_SHELL (menu));
2010-12-18 23:58:18 -08:00
g_object_get (gtk_widget_get_settings (GTK_WIDGET (view)),
2008-01-05 02:13:05 -06:00
"gtk-show-input-method-menu", &visible,
2010-02-01 22:44:03 -08:00
(char*) 0);
g_object_set (action, "visible", visible, (char*) 0);
return item;
}
static GtkWidget *
create_special_chars_menu_item (GtkAction *action)
{
2011-01-17 03:01:11 -08:00
MooEditView *view;
GtkWidget *item, *menu;
gboolean visible = TRUE;
2015-12-31 04:53:46 -08:00
view = MOO_EDIT_VIEW (g_object_get_data (G_OBJECT (action), "moo-edit-view"));
2011-01-17 03:01:11 -08:00
g_return_val_if_fail (MOO_IS_EDIT_VIEW (view), NULL);
item = gtk_menu_item_new ();
menu = gtk_menu_new ();
gtk_widget_show (menu);
gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), menu);
2010-12-18 23:58:18 -08:00
append_special_char_menuitems (GTK_MENU_SHELL (menu), view);
2010-12-18 23:58:18 -08:00
g_object_get (gtk_widget_get_settings (GTK_WIDGET (view)),
2008-01-05 02:13:05 -06:00
"gtk-show-unicode-menu", &visible,
2010-02-01 22:44:03 -08:00
(char*) 0);
g_object_set (action, "visible", visible, (char*) 0);
return item;
}
2016-01-04 09:56:33 -08:00
void Edit::_class_init_actions(MooEditClass *klass)
2006-04-20 03:57:05 -05:00
{
moo_edit_class_new_action (klass, "Undo",
2006-12-08 03:24:03 -06:00
"display-name", GTK_STOCK_UNDO,
"label", GTK_STOCK_UNDO,
"tooltip", GTK_STOCK_UNDO,
2006-05-08 23:57:16 -05:00
"stock-id", GTK_STOCK_UNDO,
2006-04-20 03:57:05 -05:00
"closure-signal", "undo",
2010-12-18 23:58:18 -08:00
"closure-proxy-func", moo_edit_get_view,
"view-condition::sensitive", "can-undo",
2010-02-01 22:44:03 -08:00
(char*) 0);
2006-04-20 03:57:05 -05:00
moo_edit_class_new_action (klass, "Redo",
2006-12-08 03:24:03 -06:00
"display-name", GTK_STOCK_REDO,
"label", GTK_STOCK_REDO,
"tooltip", GTK_STOCK_REDO,
2006-05-08 23:57:16 -05:00
"stock-id", GTK_STOCK_REDO,
2006-04-20 03:57:05 -05:00
"closure-signal", "redo",
2010-12-18 23:58:18 -08:00
"closure-proxy-func", moo_edit_get_view,
"view-condition::sensitive", "can-redo",
2010-02-01 22:44:03 -08:00
(char*) 0);
2006-04-20 03:57:05 -05:00
moo_edit_class_new_action (klass, "Cut",
2006-12-08 03:24:03 -06:00
"display-name", GTK_STOCK_CUT,
2006-04-20 03:57:05 -05:00
"stock-id", GTK_STOCK_CUT,
"closure-signal", "cut-clipboard",
2010-12-18 23:58:18 -08:00
"closure-proxy-func", moo_edit_get_view,
"view-condition::sensitive", "has-selection",
2010-02-01 22:44:03 -08:00
(char*) 0);
2006-04-20 03:57:05 -05:00
moo_edit_class_new_action (klass, "Copy",
2006-12-08 03:24:03 -06:00
"display-name", GTK_STOCK_COPY,
2006-04-20 03:57:05 -05:00
"stock-id", GTK_STOCK_COPY,
"closure-signal", "copy-clipboard",
2010-12-18 23:58:18 -08:00
"closure-proxy-func", moo_edit_get_view,
"view-condition::sensitive", "has-selection",
2010-02-01 22:44:03 -08:00
(char*) 0);
2006-04-20 03:57:05 -05:00
moo_edit_class_new_action (klass, "Paste",
2006-12-08 03:24:03 -06:00
"display-name", GTK_STOCK_PASTE,
2006-04-20 03:57:05 -05:00
"stock-id", GTK_STOCK_PASTE,
"closure-signal", "paste-clipboard",
2010-12-18 23:58:18 -08:00
"closure-proxy-func", moo_edit_get_view,
2010-02-01 22:44:03 -08:00
(char*) 0);
2006-04-20 03:57:05 -05:00
moo_edit_class_new_action (klass, "SelectAll",
2006-12-08 03:24:03 -06:00
"display-name", GTK_STOCK_SELECT_ALL,
"label", GTK_STOCK_SELECT_ALL,
"tooltip", GTK_STOCK_SELECT_ALL,
2006-05-08 23:57:16 -05:00
"stock-id", GTK_STOCK_SELECT_ALL,
2006-04-20 03:57:05 -05:00
"closure-callback", moo_text_view_select_all,
2010-12-18 23:58:18 -08:00
"closure-proxy-func", moo_edit_get_view,
"view-condition::sensitive", "has-text",
2010-02-01 22:44:03 -08:00
(char*) 0);
moo_edit_class_new_action (klass, "InputMethods",
"action-type::", MOO_TYPE_MENU_ACTION,
"label", D_("Input _Methods", "gtk20"),
"menu-func", create_input_methods_menu_item,
2010-02-01 22:44:03 -08:00
(char*) 0);
moo_edit_class_new_action (klass, "SpecialChars",
"action-type::", MOO_TYPE_MENU_ACTION,
"label", D_("_Insert Unicode Control Character", "gtk20"),
"menu-func", create_special_chars_menu_item,
2010-02-01 22:44:03 -08:00
(char*) 0);
}
/*
* _gtk_text_util_append_special_char_menuitems from gtk/gtktextutil.c
*/
static const struct BidiMenuEntry {
const char *label;
gunichar ch;
} bidi_menu_entries[] = {
{ "LRM _Left-to-right mark", 0x200E },
{ "RLM _Right-to-left mark", 0x200F },
{ "LRE Left-to-right _embedding", 0x202A },
{ "RLE Right-to-left e_mbedding", 0x202B },
{ "LRO Left-to-right _override", 0x202D },
{ "RLO Right-to-left o_verride", 0x202E },
{ "PDF _Pop directional formatting", 0x202C },
{ "ZWS _Zero width space", 0x200B },
{ "ZWJ Zero width _joiner", 0x200D },
{ "ZWNJ Zero width _non-joiner", 0x200C }
};
static void
bidi_menu_item_activate (GtkWidget *menuitem,
GtkTextView *view)
{
GtkTextBuffer *buffer;
struct BidiMenuEntry *entry;
gboolean had_selection;
char string[7];
g_return_if_fail (GTK_IS_TEXT_VIEW (view));
2010-02-01 22:44:03 -08:00
entry = (struct BidiMenuEntry*) g_object_get_data (G_OBJECT (menuitem), "unicode-menu-entry");
g_return_if_fail (entry != NULL);
string[g_unichar_to_utf8 (entry->ch, string)] = 0;
buffer = gtk_text_view_get_buffer (view);
gtk_text_buffer_begin_user_action (buffer);
had_selection = gtk_text_buffer_get_selection_bounds (buffer, NULL, NULL);
gtk_text_buffer_delete_selection (buffer, TRUE, view->editable);
if (!had_selection && view->overwrite_mode)
{
GtkTextIter insert, end;
gtk_text_buffer_get_iter_at_mark (buffer, &insert, gtk_text_buffer_get_insert (buffer));
if (!gtk_text_iter_ends_line (&insert))
{
gtk_text_iter_forward_cursor_positions (&end, 1);
gtk_text_buffer_delete_interactive (buffer, &insert, &end, view->editable);
}
}
gtk_text_buffer_insert_interactive_at_cursor (buffer, string, -1, view->editable);
gtk_text_buffer_end_user_action (buffer);
gtk_text_view_scroll_mark_onscreen (view, gtk_text_buffer_get_insert (buffer));
}
static void
append_special_char_menuitems (GtkMenuShell *menu,
2011-01-17 03:01:11 -08:00
MooEditView *view)
{
guint i;
for (i = 0; i < G_N_ELEMENTS (bidi_menu_entries); i++)
{
GtkWidget *menuitem;
menuitem = gtk_menu_item_new_with_mnemonic (D_(bidi_menu_entries[i].label, "gtk20"));
g_object_set_data (G_OBJECT (menuitem), "unicode-menu-entry",
(gpointer) &bidi_menu_entries[i]);
g_signal_connect (menuitem, "activate", G_CALLBACK (bidi_menu_item_activate), view);
gtk_widget_show (menuitem);
gtk_menu_shell_append (menu, menuitem);
}
2006-04-20 03:57:05 -05:00
}