/* * mooeditaction-factory.cpp * * Copyright (C) 2004-2016 by Yevgen Muntyan * * 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. * * You should have received a copy of the GNU Lesser General Public * License along with medit. If not, see . */ #include "mooedit/mooeditaction-factory.h" #include "mooedit/mooeditaction.h" #include "mooedit/mooedit-private.h" #include "mooutils/mooutils-gobject.h" #include "mooutils/mooactionfactory.h" #include "mooutils/mooactionbase.h" #include "mooutils/moomenuaction.h" #include "mooutils/mooi18n.h" #include "mooutils/mootype-macros.h" #include #include typedef GtkAction *(*MooEditActionFunc) (MooEdit *edit, gpointer data); 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, MooEditView *view); #define MOO_EDIT_ACTIONS_QUARK (moo_edit_actions_quark ()) MOO_DEFINE_QUARK_STATIC (moo-edit-actions, moo_edit_actions_quark) typedef struct { MooActionFactory *action; char **doc_conditions; char **view_conditions; } ActionInfo; static ActionInfo* action_info_new (MooActionFactory *action, char **doc_conditions, char **view_conditions) { ActionInfo *info; g_return_val_if_fail (MOO_IS_ACTION_FACTORY (action), NULL); info = g_new0 (ActionInfo, 1); g_object_ref (action); info->action = action; info->doc_conditions = g_strdupv (doc_conditions); info->view_conditions = g_strdupv (view_conditions); return info; } static void action_info_free (ActionInfo *info) { if (info) { g_object_unref (info->action); g_strfreev (info->doc_conditions); g_strfreev (info->view_conditions); g_free (info); } } static void moo_edit_add_action (MooEdit *edit, GtkAction *action) { GtkActionGroup *group; g_return_if_fail (MOO_IS_EDIT (edit)); g_return_if_fail (GTK_IS_ACTION (action)); group = moo_edit_get_actions (edit); gtk_action_group_add_action (group, action); } static void moo_edit_remove_action (MooEdit *edit, const char *action_id) { GtkActionGroup *group; GtkAction *action; g_return_if_fail (MOO_IS_EDIT (edit)); g_return_if_fail (action_id != NULL); group = moo_edit_get_actions (edit); action = gtk_action_group_get_action (group, action_id); gtk_action_group_remove_action (group, action); } static GtkAction* create_action (const char *action_id, ActionInfo *info, MooEdit *edit) { GtkAction *action; char **p; MooEditView *view; g_return_val_if_fail (MOO_IS_EDIT (edit), NULL); g_return_val_if_fail (info != NULL, NULL); g_return_val_if_fail (MOO_IS_ACTION_FACTORY (info->action), NULL); g_return_val_if_fail (action_id && action_id[0], NULL); 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, (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, (char*) 0); else action = moo_action_factory_create_action (info->action, edit, "name", action_id, (char*) 0); g_return_val_if_fail (action != NULL, NULL); if (g_type_is_a (info->action->action_type, MOO_TYPE_EDIT_ACTION)) g_object_set (action, "doc", edit, (char*) 0); g_object_set_data (G_OBJECT (action), "moo-edit", edit); for (p = info->doc_conditions; p && *p; 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); } 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); } 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 GHashTable * get_actions_hash (GType type) { return (GHashTable*) g_type_get_qdata (type, MOO_EDIT_ACTIONS_QUARK); } static void moo_edit_class_install_action (MooEditClass *klass, const char *action_id, MooActionFactory *factory, char **doc_conditions, char **view_conditions) { GHashTable *actions; ActionInfo *info; GType type; g_return_if_fail (MOO_IS_EDIT_CLASS (klass)); g_return_if_fail (MOO_IS_ACTION_FACTORY (factory)); g_return_if_fail (action_id && action_id[0]); type = G_OBJECT_CLASS_TYPE (klass); actions = get_actions_hash (type); 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 (factory, doc_conditions, view_conditions); g_hash_table_insert (actions, g_strdup (action_id), info); for (const auto& doc: Edit::_moo_edit_instances) { if (g_type_is_a (G_OBJECT_TYPE (doc), type)) { GtkAction *action = create_action (action_id, info, doc); if (action) { moo_edit_add_action (doc, action); g_object_unref (action); } } } } static 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 *doc_conditions = NULL; GPtrArray *view_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)); doc_conditions = g_ptr_array_new (); view_conditions = g_ptr_array_new (); name = first_prop_name; while (name) { GParameter param = { 0 }; GParamSpec *pspec; char *err = NULL; /* ignore id property */ if (!strcmp (name, "id") || !strcmp (name, "name")) { g_critical ("id property specified"); 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", err); g_free (err); goto error; } action_type = _moo_value_get_gtype (¶m.value); if (!g_type_is_a (action_type, MOO_TYPE_ACTION_BASE)) { g_warning ("invalid action type %s", g_type_name (action_type)); goto error; } action_class = G_OBJECT_CLASS (g_type_class_ref (action_type)); } else if (g_str_has_prefix (name, "condition::") || g_str_has_prefix (name, "view-condition::")) { GPtrArray *conditions; gboolean is_doc = g_str_has_prefix (name, "condition::"); const char *suffix = strstr (name, "::"); if (!suffix || !suffix[1] || !suffix[2]) { g_warning ("invalid condition name '%s'", name); goto error; } suffix += 2; name = va_arg (var_args, gchar*); if (!name) { g_warning ("unterminated '%s' property", suffix); goto error; } conditions = is_doc ? doc_conditions : view_conditions; g_ptr_array_add (conditions, g_strdup (suffix)); g_ptr_array_add (conditions, g_strdup (name)); } else { if (!action_class) { if (!action_type) action_type = MOO_TYPE_EDIT_ACTION; action_class = G_OBJECT_CLASS (g_type_class_ref (action_type)); } pspec = g_object_class_find_property (action_class, name); if (!pspec) { g_warning ("no property '%s' in class '%s'", 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", 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 { MooActionFactory *action_factory = NULL; action_factory = moo_action_factory_new_a (action_type, (GParameter*) action_params->data, action_params->len); if (!action_factory) { g_warning ("error in moo_object_factory_new_a()"); goto error; } _moo_param_array_free ((GParameter*) action_params->data, action_params->len); g_array_free (action_params, FALSE); action_params = NULL; g_ptr_array_add (doc_conditions, NULL); g_ptr_array_add (view_conditions, NULL); moo_edit_class_install_action (klass, action_id, action_factory, (char**) doc_conditions->pdata, (char**) view_conditions->pdata); 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); 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 (doc_conditions) { g_ptr_array_foreach (doc_conditions, (GFunc) extern_g_free, NULL); g_ptr_array_free (doc_conditions, TRUE); } if (view_conditions) { g_ptr_array_foreach (view_conditions, (GFunc) extern_g_free, NULL); g_ptr_array_free (view_conditions, TRUE); } if (action_class) g_type_class_unref (action_class); } static GtkAction * custom_action_factory_func (MooEdit *edit, MooActionFactory *factory) { const char *action_id; MooEditActionFunc func; gpointer func_data; g_return_val_if_fail (MOO_IS_EDIT (edit), NULL); 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"); 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); return func (edit, func_data); } static void moo_edit_class_new_action_custom (MooEditClass *klass, const char *action_id, MooEditActionFunc func, gpointer data, GDestroyNotify notify) { MooActionFactory *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_action_factory_new_func ((MooActionFactoryFunc) 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", (gpointer) 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, NULL); g_object_unref (action_factory); } static GtkAction * type_action_func (MooEdit *edit, gpointer klass) { GQuark quark = g_quark_from_static_string ("moo-edit-class-action-id"); 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)); } void moo_edit_class_new_action_type (MooEditClass *edit_klass, const char *id, GType type) { gpointer klass; GQuark quark; 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); 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)); moo_edit_class_new_action_custom (edit_klass, id, type_action_func, klass, g_type_class_unref); } void moo_edit_class_remove_action (MooEditClass *klass, const char *action_id) { GHashTable *actions; GType type; g_return_if_fail (MOO_IS_EDIT_CLASS (klass)); type = G_OBJECT_CLASS_TYPE (klass); actions = get_actions_hash (type); if (actions) g_hash_table_remove (actions, action_id); for (const auto& doc: Edit::_moo_edit_instances) { if (g_type_is_a(G_OBJECT_TYPE(doc), type)) moo_edit_remove_action(doc, action_id); } } GtkActionGroup * moo_edit_get_actions (MooEdit *edit) { g_return_val_if_fail (MOO_IS_EDIT (edit), NULL); return moo_action_collection_get_group (edit->priv->actions.gobj(), NULL); } GtkAction * moo_edit_get_action_by_id (MooEdit *edit, const char *action_id) { GtkActionGroup *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 gtk_action_group_get_action (actions, action_id); } static void add_action (const char *id, ActionInfo *info, MooEdit *edit) { GtkAction *action = create_action (id, info, edit); if (action) { moo_edit_add_action (edit, action); g_object_unref (action); } } void Edit::_add_class_actions() { GType type = G_OBJECT_TYPE (gobj()); while (true) { GHashTable *actions; actions = get_actions_hash (type); if (actions) g_hash_table_foreach (actions, (GHFunc) add_action, gobj()); if (type == MOO_TYPE_EDIT) break; type = g_type_parent (type); } } static GtkWidget * create_input_methods_menu_item (GtkAction *action) { MooEditView *view; GtkWidget *item, *menu; gboolean visible = TRUE; view = MOO_EDIT_VIEW (g_object_get_data (G_OBJECT (action), "moo-edit-view")); 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); gtk_im_multicontext_append_menuitems (GTK_IM_MULTICONTEXT (GTK_TEXT_VIEW (view)->im_context), GTK_MENU_SHELL (menu)); g_object_get (gtk_widget_get_settings (GTK_WIDGET (view)), "gtk-show-input-method-menu", &visible, (char*) 0); g_object_set (action, "visible", visible, (char*) 0); return item; } static GtkWidget * create_special_chars_menu_item (GtkAction *action) { MooEditView *view; GtkWidget *item, *menu; gboolean visible = TRUE; view = MOO_EDIT_VIEW (g_object_get_data (G_OBJECT (action), "moo-edit-view")); 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); append_special_char_menuitems (GTK_MENU_SHELL (menu), view); g_object_get (gtk_widget_get_settings (GTK_WIDGET (view)), "gtk-show-unicode-menu", &visible, (char*) 0); g_object_set (action, "visible", visible, (char*) 0); return item; } void Edit::_class_init_actions(MooEditClass *klass) { moo_edit_class_new_action (klass, "Undo", "display-name", GTK_STOCK_UNDO, "label", GTK_STOCK_UNDO, "tooltip", GTK_STOCK_UNDO, "stock-id", GTK_STOCK_UNDO, "closure-signal", "undo", "closure-proxy-func", moo_edit_get_view, "view-condition::sensitive", "can-undo", (char*) 0); moo_edit_class_new_action (klass, "Redo", "display-name", GTK_STOCK_REDO, "label", GTK_STOCK_REDO, "tooltip", GTK_STOCK_REDO, "stock-id", GTK_STOCK_REDO, "closure-signal", "redo", "closure-proxy-func", moo_edit_get_view, "view-condition::sensitive", "can-redo", (char*) 0); moo_edit_class_new_action (klass, "Cut", "display-name", GTK_STOCK_CUT, "stock-id", GTK_STOCK_CUT, "closure-signal", "cut-clipboard", "closure-proxy-func", moo_edit_get_view, "view-condition::sensitive", "has-selection", (char*) 0); moo_edit_class_new_action (klass, "Copy", "display-name", GTK_STOCK_COPY, "stock-id", GTK_STOCK_COPY, "closure-signal", "copy-clipboard", "closure-proxy-func", moo_edit_get_view, "view-condition::sensitive", "has-selection", (char*) 0); moo_edit_class_new_action (klass, "Paste", "display-name", GTK_STOCK_PASTE, "stock-id", GTK_STOCK_PASTE, "closure-signal", "paste-clipboard", "closure-proxy-func", moo_edit_get_view, (char*) 0); moo_edit_class_new_action (klass, "SelectAll", "display-name", GTK_STOCK_SELECT_ALL, "label", GTK_STOCK_SELECT_ALL, "tooltip", GTK_STOCK_SELECT_ALL, "stock-id", GTK_STOCK_SELECT_ALL, "closure-callback", moo_text_view_select_all, "closure-proxy-func", moo_edit_get_view, "view-condition::sensitive", "has-text", (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, (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, (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)); 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, 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); } }