/* * This program 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) version 3. * * 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with the program; if not, see * * * Authors: * Diego Escalante Urrelo * Bharath Acharya * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * Copyright (C) 2008 - Diego Escalante Urrelo * */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define GCONF_KEY_TEMPLATE_PLACEHOLDERS "/apps/evolution/mail/template_placeholders" typedef struct { GConfClient *gconf; GtkWidget *treeview; GtkWidget *clue_add; GtkWidget *clue_edit; GtkWidget *clue_remove; GtkListStore *store; } UIData; enum { CLUE_KEYWORD_COLUMN, CLUE_VALUE_COLUMN, CLUE_N_COLUMNS }; GtkWidget * e_plugin_lib_get_configure_widget (EPlugin *plugin); gboolean init_composer_actions (GtkUIManager *ui_manager, EMsgComposer *composer); gboolean init_shell_actions (GtkUIManager *ui_manager, EShellWindow *shell_window); gint e_plugin_lib_enable (EPlugin *plugin, gboolean enabled); /* Thanks to attachment reminder plugin for this*/ static void commit_changes (UIData *ui); static void key_cell_edited_callback (GtkCellRendererText *cell, gchar *path_string, gchar *new_text,UIData *ui); static void value_cell_edited_callback (GtkCellRendererText *cell, gchar *path_string, gchar *new_text,UIData *ui); static gboolean clue_foreach_check_isempty (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, UIData *ui); static gboolean plugin_enabled; static void selection_changed (GtkTreeSelection *selection, UIData *ui) { GtkTreeModel *model; GtkTreeIter iter; if (gtk_tree_selection_get_selected (selection, &model, &iter)) { gtk_widget_set_sensitive (ui->clue_edit, TRUE); gtk_widget_set_sensitive (ui->clue_remove, TRUE); } else { gtk_widget_set_sensitive (ui->clue_edit, FALSE); gtk_widget_set_sensitive (ui->clue_remove, FALSE); } } static void destroy_ui_data (gpointer data) { UIData *ui = (UIData *) data; if (!ui) return; g_object_unref (ui->gconf); g_free (ui); } static void commit_changes (UIData *ui) { GtkTreeModel *model = NULL; GSList *clue_list = NULL; GtkTreeIter iter; gboolean valid; model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview)); valid = gtk_tree_model_get_iter_first (model, &iter); while (valid) { gchar *keyword, *value; gchar *key; gtk_tree_model_get (model, &iter, CLUE_KEYWORD_COLUMN, &keyword, -1); gtk_tree_model_get (model, &iter, CLUE_VALUE_COLUMN, &value, -1); /* Check if the keyword and value are not empty */ if ((keyword) && (value) && (g_utf8_strlen(g_strstrip(keyword), -1) > 0) && (g_utf8_strlen(g_strstrip(value), -1) > 0)) { key = g_strdup_printf("%s=%s", keyword, value); clue_list = g_slist_append (clue_list, key); } valid = gtk_tree_model_iter_next (model, &iter); } gconf_client_set_list (ui->gconf, GCONF_KEY_TEMPLATE_PLACEHOLDERS, GCONF_VALUE_STRING, clue_list, NULL); g_slist_foreach (clue_list, (GFunc) g_free, NULL); g_slist_free (clue_list); } static void clue_check_isempty (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, UIData *ui) { GtkTreeSelection *selection; gchar *keyword = NULL; gboolean valid; selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview)); /* move to the previous node */ valid = gtk_tree_path_prev (path); gtk_tree_model_get (model, iter, CLUE_KEYWORD_COLUMN, &keyword, -1); if ((keyword) && !(g_utf8_strlen (g_strstrip (keyword), -1) > 0)) gtk_list_store_remove (ui->store, iter); /* Check if we have a valid row to select. If not, then select * the previous row */ if (gtk_list_store_iter_is_valid (GTK_LIST_STORE (model), iter)) { gtk_tree_selection_select_iter (selection, iter); } else { if (path && valid) { gtk_tree_model_get_iter (model, iter, path); gtk_tree_selection_select_iter (selection, iter); } } gtk_widget_grab_focus (ui->treeview); g_free (keyword); } static gboolean clue_foreach_check_isempty (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, UIData *ui) { gboolean valid; valid = gtk_tree_model_get_iter_first (model, iter); while (valid && gtk_list_store_iter_is_valid (ui->store, iter)) { gchar *keyword = NULL; gtk_tree_model_get (model, iter, CLUE_KEYWORD_COLUMN, &keyword, -1); /* Check if the keyword is not empty and then emit the row-changed signal (if we delete the row, then the iter gets corrupted) */ if ((keyword) && !(g_utf8_strlen (g_strstrip (keyword), -1) > 0)) gtk_tree_model_row_changed (model, path, iter); g_free (keyword); valid = gtk_tree_model_iter_next (model, iter); } return FALSE; } static void key_cell_edited_callback (GtkCellRendererText *cell, gchar *path_string, gchar *new_text, UIData *ui) { GtkTreeModel *model; GtkTreeIter iter; gchar *value; model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview)); gtk_tree_model_get_iter_from_string (model, &iter, path_string); gtk_tree_model_get (model, &iter, CLUE_VALUE_COLUMN, &value, -1); gtk_list_store_set (GTK_LIST_STORE (model), &iter, CLUE_KEYWORD_COLUMN, new_text, CLUE_VALUE_COLUMN, value, -1); commit_changes (ui); } static void value_cell_edited_callback (GtkCellRendererText *cell, gchar *path_string, gchar *new_text, UIData *ui) { GtkTreeModel *model; GtkTreeIter iter; gchar *keyword; model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview)); gtk_tree_model_get_iter_from_string (model, &iter, path_string); gtk_tree_model_get (model, &iter, CLUE_KEYWORD_COLUMN, &keyword, -1); gtk_list_store_set (GTK_LIST_STORE (model), &iter, CLUE_KEYWORD_COLUMN, keyword, CLUE_VALUE_COLUMN, new_text, -1); commit_changes (ui); } static void clue_add_clicked (GtkButton *button, UIData *ui) { GtkTreeModel *model; GtkTreeIter iter; gchar *new_clue = NULL; GtkTreeViewColumn *focus_col; GtkTreePath *path; model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview)); gtk_tree_model_foreach (model, (GtkTreeModelForeachFunc) clue_foreach_check_isempty, ui); /* Disconnect from signal so that we can create an empty row */ g_signal_handlers_disconnect_matched(G_OBJECT(model), G_SIGNAL_MATCH_FUNC, 0, 0, NULL, clue_check_isempty, ui); /* TODO : Trim and check for blank strings */ new_clue = g_strdup (""); gtk_list_store_append (GTK_LIST_STORE (model), &iter); gtk_list_store_set (GTK_LIST_STORE (model), &iter, CLUE_KEYWORD_COLUMN, new_clue, CLUE_VALUE_COLUMN, new_clue, -1); focus_col = gtk_tree_view_get_column (GTK_TREE_VIEW (ui->treeview), CLUE_KEYWORD_COLUMN); path = gtk_tree_model_get_path (model, &iter); if (path) { gtk_tree_view_set_cursor (GTK_TREE_VIEW (ui->treeview), path, focus_col, TRUE); gtk_tree_view_row_activated(GTK_TREE_VIEW(ui->treeview), path, focus_col); gtk_tree_path_free (path); } /* We have done our job, connect back to the signal */ g_signal_connect(G_OBJECT(model), "row-changed", G_CALLBACK(clue_check_isempty), ui); } static void clue_remove_clicked (GtkButton *button, UIData *ui) { GtkTreeSelection *selection; GtkTreeModel *model; GtkTreeIter iter; GtkTreePath *path; gboolean valid; gint len; valid = FALSE; selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview)); if (!gtk_tree_selection_get_selected (selection, &model, &iter)) return; /* Get the path and move to the previous node :) */ path = gtk_tree_model_get_path (model, &iter); if (path) valid = gtk_tree_path_prev(path); gtk_list_store_remove (GTK_LIST_STORE (model), &iter); len = gtk_tree_model_iter_n_children (model, NULL); if (len > 0) { if (gtk_list_store_iter_is_valid (GTK_LIST_STORE(model), &iter)) { gtk_tree_selection_select_iter (selection, &iter); } else { if (path && valid) { gtk_tree_model_get_iter(model, &iter, path); gtk_tree_selection_select_iter (selection, &iter); } } } else { gtk_widget_set_sensitive (ui->clue_edit, FALSE); gtk_widget_set_sensitive (ui->clue_remove, FALSE); } gtk_widget_grab_focus(ui->treeview); gtk_tree_path_free (path); commit_changes (ui); } static void clue_edit_clicked (GtkButton *button, UIData *ui) { GtkTreeSelection *selection; GtkTreeModel *model; GtkTreePath *path; GtkTreeIter iter; GtkTreeViewColumn *focus_col; selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview)); if (!gtk_tree_selection_get_selected (selection, &model, &iter)) return; focus_col = gtk_tree_view_get_column (GTK_TREE_VIEW (ui->treeview), CLUE_KEYWORD_COLUMN); path = gtk_tree_model_get_path (model, &iter); if (path) { gtk_tree_view_set_cursor (GTK_TREE_VIEW (ui->treeview), path, focus_col, TRUE); gtk_tree_path_free (path); } } GtkWidget * e_plugin_lib_get_configure_widget (EPlugin *epl) { GtkCellRenderer *renderer_key, *renderer_value; GtkTreeSelection *selection; GtkTreeIter iter; GConfClient *gconf = gconf_client_get_default(); GtkWidget *hbox; GSList *clue_list = NULL, *list; GtkTreeModel *model; GtkWidget *templates_configuration_box; GtkWidget *clue_container; GtkWidget *scrolledwindow1; GtkWidget *clue_treeview; GtkWidget *vbuttonbox2; GtkWidget *clue_add; GtkWidget *clue_edit; GtkWidget *clue_remove; UIData *ui = g_new0 (UIData, 1); templates_configuration_box = gtk_vbox_new (FALSE, 5); gtk_widget_show (templates_configuration_box); gtk_widget_set_size_request (templates_configuration_box, 385, 189); clue_container = gtk_hbox_new (FALSE, 6); gtk_widget_show (clue_container); gtk_box_pack_start (GTK_BOX (templates_configuration_box), clue_container, TRUE, TRUE, 0); scrolledwindow1 = gtk_scrolled_window_new (NULL, NULL); gtk_widget_show (scrolledwindow1); gtk_box_pack_start (GTK_BOX (clue_container), scrolledwindow1, TRUE, TRUE, 0); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolledwindow1), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); clue_treeview = gtk_tree_view_new (); gtk_widget_show (clue_treeview); gtk_container_add (GTK_CONTAINER (scrolledwindow1), clue_treeview); gtk_container_set_border_width (GTK_CONTAINER (clue_treeview), 1); vbuttonbox2 = gtk_vbutton_box_new (); gtk_widget_show (vbuttonbox2); gtk_box_pack_start (GTK_BOX (clue_container), vbuttonbox2, FALSE, TRUE, 0); gtk_button_box_set_layout (GTK_BUTTON_BOX (vbuttonbox2), GTK_BUTTONBOX_START); gtk_box_set_spacing (GTK_BOX (vbuttonbox2), 6); clue_add = gtk_button_new_from_stock ("gtk-add"); gtk_widget_show (clue_add); gtk_container_add (GTK_CONTAINER (vbuttonbox2), clue_add); gtk_widget_set_can_default (clue_add, TRUE); clue_edit = gtk_button_new_from_stock ("gtk-edit"); gtk_widget_show (clue_edit); gtk_container_add (GTK_CONTAINER (vbuttonbox2), clue_edit); gtk_widget_set_can_default (clue_edit, TRUE); clue_remove = gtk_button_new_from_stock ("gtk-remove"); gtk_widget_show (clue_remove); gtk_container_add (GTK_CONTAINER (vbuttonbox2), clue_remove); gtk_widget_set_can_default (clue_remove, TRUE); ui->gconf = gconf_client_get_default (); ui->treeview = clue_treeview; ui->store = gtk_list_store_new (CLUE_N_COLUMNS, G_TYPE_STRING, G_TYPE_STRING); gtk_tree_view_set_model (GTK_TREE_VIEW (ui->treeview), GTK_TREE_MODEL (ui->store)); renderer_key = gtk_cell_renderer_text_new (); gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (ui->treeview), -1, _("Keywords"), renderer_key, "text", CLUE_KEYWORD_COLUMN, NULL); g_object_set (G_OBJECT (renderer_key), "editable", TRUE, NULL); g_signal_connect(renderer_key, "edited", (GCallback) key_cell_edited_callback, ui); renderer_value = gtk_cell_renderer_text_new (); gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (ui->treeview), -1, _("Values"), renderer_value, "text", CLUE_VALUE_COLUMN, NULL); g_object_set (G_OBJECT (renderer_value), "editable", TRUE, NULL); g_signal_connect(renderer_value, "edited", (GCallback) value_cell_edited_callback, ui); selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (ui->treeview)); gtk_tree_selection_set_mode (selection, GTK_SELECTION_SINGLE); g_signal_connect (G_OBJECT (selection), "changed", G_CALLBACK (selection_changed), ui); gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (ui->treeview), TRUE); ui->clue_add = clue_add; g_signal_connect (G_OBJECT (ui->clue_add), "clicked", G_CALLBACK (clue_add_clicked), ui); ui->clue_remove = clue_remove; g_signal_connect (G_OBJECT (ui->clue_remove), "clicked", G_CALLBACK (clue_remove_clicked), ui); gtk_widget_set_sensitive (ui->clue_remove, FALSE); ui->clue_edit = clue_edit; g_signal_connect (G_OBJECT (ui->clue_edit), "clicked", G_CALLBACK (clue_edit_clicked), ui); gtk_widget_set_sensitive (ui->clue_edit, FALSE); model = gtk_tree_view_get_model (GTK_TREE_VIEW (ui->treeview)); g_signal_connect(G_OBJECT(model), "row-changed", G_CALLBACK(clue_check_isempty), ui); /* Populate tree view with values from gconf */ clue_list = gconf_client_get_list ( gconf, GCONF_KEY_TEMPLATE_PLACEHOLDERS, GCONF_VALUE_STRING, NULL ); for (list = clue_list; list; list = g_slist_next (list)) { gchar **temp = g_strsplit (list->data, "=", 2); gtk_list_store_append (ui->store, &iter); gtk_list_store_set (ui->store, &iter, CLUE_KEYWORD_COLUMN, temp[0], CLUE_VALUE_COLUMN, temp[1], -1); g_strfreev(temp); } if (clue_list) { g_slist_foreach (clue_list, (GFunc) g_free, NULL); g_slist_free (clue_list); } /* Add the list here */ hbox = gtk_vbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), templates_configuration_box, TRUE, TRUE, 0); /* to let free data properly on destroy of configuration widget */ g_object_set_data_full (G_OBJECT (hbox), "myui-data", ui, destroy_ui_data); return hbox; } static void create_new_message (CamelFolder *folder, const gchar *uid, CamelMimeMessage *message, gpointer data) { GtkAction *action = data; CamelMimeMessage *new, *template; struct _camel_header_raw *header; CamelStream *mem; g_return_if_fail (data != NULL); g_return_if_fail (message != NULL); folder = e_mail_local_get_folder (E_MAIL_FOLDER_TEMPLATES); template = g_object_get_data (G_OBJECT (action), "template"); /* The new message we are creating */ new = camel_mime_message_new(); /* make the exact copy of the template message, with all its attachments and message structure */ mem = camel_stream_mem_new (); camel_data_wrapper_write_to_stream ( CAMEL_DATA_WRAPPER (template), mem, NULL); camel_stream_reset (mem, NULL); camel_data_wrapper_construct_from_stream (CAMEL_DATA_WRAPPER (new), mem, NULL); g_object_unref (mem); /* Add the headers from the message we are replying to, so CC and that * stuff is preserved. */ header = ((CamelMimePart *)message)->headers; while (header) { if (g_ascii_strncasecmp (header->name, "content-", 8) != 0) { camel_medium_add_header((CamelMedium *) new, header->name, header->value); } header = header->next; } /* Set the To: field to the same To: field of the message we are replying to. */ camel_mime_message_set_recipients (new, CAMEL_RECIPIENT_TYPE_TO, camel_mime_message_get_from (message)); /* Copy the CC and BCC from the template.*/ camel_mime_message_set_recipients (new, CAMEL_RECIPIENT_TYPE_CC, camel_mime_message_get_recipients (template, CAMEL_RECIPIENT_TYPE_CC)); camel_mime_message_set_recipients (new, CAMEL_RECIPIENT_TYPE_BCC, camel_mime_message_get_recipients (template, CAMEL_RECIPIENT_TYPE_BCC)); /* Create the composer */ em_utils_edit_message (new, folder); g_object_unref (new); } static void action_reply_with_template_cb (GtkAction *action, const gchar *message_uid) { CamelFolder *folder; g_return_if_fail (message_uid != NULL); folder = CAMEL_FOLDER (g_object_get_data (G_OBJECT (action), "message_folder")); g_return_if_fail (folder != NULL); g_object_ref (G_OBJECT (action)); mail_get_message (folder, message_uid, create_new_message, action, mail_msg_unordered_push); g_object_unref (G_OBJECT (action)); } static void build_template_menus_recurse (GtkUIManager *ui_manager, GtkActionGroup *action_group, const gchar *menu_path, guint *action_count, guint merge_id, CamelFolderInfo *folder_info, CamelFolder *message_folder, const gchar *message_uid) { CamelStore *store; store = e_mail_local_get_store (); while (folder_info != NULL) { CamelFolder *folder; GPtrArray *uids; GtkAction *action; const gchar *action_label; const gchar *folder_name; gchar *action_name; gchar *path; guint ii; folder = camel_store_get_folder ( store, folder_info->full_name, 0, NULL); folder_name = camel_folder_get_name (folder); action_name = g_strdup_printf ( "templates-menu-%d", *action_count); *action_count = *action_count + 1; /* To avoid having a Templates dir, we ignore the top level */ if (g_str_has_suffix (folder_name, "Templates")) action_label = _("Templates"); else action_label = folder_name; action = gtk_action_new ( action_name, action_label, NULL, NULL); gtk_action_group_add_action (action_group, action); gtk_ui_manager_add_ui ( ui_manager, merge_id, menu_path, action_name, action_name, GTK_UI_MANAGER_MENU, FALSE); path = g_strdup_printf ("%s/%s", menu_path, action_name); g_object_unref (action); g_free (action_name); /* Add submenus, if any. */ if (folder_info->child != NULL) build_template_menus_recurse ( ui_manager, action_group, path, action_count, merge_id, folder_info->child, message_folder, message_uid); /* Get the UIDs for this folder and add them to the menu. */ uids = camel_folder_get_uids (folder); for (ii = 0; ii < uids->len; ii++) { CamelMimeMessage *template; const gchar *uid = uids->pdata[ii], *muid; guint32 flags; /* If the UIDs is marked for deletion, skip it. */ flags = camel_folder_get_message_flags (folder, uid); if (flags & CAMEL_MESSAGE_DELETED) continue; template = camel_folder_get_message (folder, uid, NULL); g_object_ref (template); action_label = camel_mime_message_get_subject (template); if (action_label == NULL || *action_label == '\0') action_label = _("No Title"); action_name = g_strdup_printf ( "templates-item-%d", *action_count); *action_count = *action_count + 1; action = gtk_action_new ( action_name, action_label, NULL, NULL); muid = camel_pstring_strdup (message_uid); g_object_ref (message_folder); g_object_set_data_full ( G_OBJECT (action), "message_uid", (gpointer) muid, (GDestroyNotify) camel_pstring_free); g_object_set_data_full ( G_OBJECT (action), "message_folder", message_folder, (GDestroyNotify) g_object_unref); g_object_set_data_full ( G_OBJECT (action), "template", template, (GDestroyNotify) g_object_unref); g_signal_connect ( action, "activate", G_CALLBACK (action_reply_with_template_cb), (gpointer) muid); gtk_action_group_add_action (action_group, action); gtk_ui_manager_add_ui ( ui_manager, merge_id, path, action_name, action_name, GTK_UI_MANAGER_MENUITEM, FALSE); g_object_unref (action); g_free (action_name); } camel_folder_free_uids (folder, uids); g_free (path); folder_info = folder_info->next; } } static void action_template_cb (GtkAction *action, EMsgComposer *composer) { CamelMessageInfo *info; CamelMimeMessage *msg; CamelFolder *folder; /* Get the templates folder and all UIDs of the messages there. */ folder = e_mail_local_get_folder (E_MAIL_FOLDER_TEMPLATES); msg = e_msg_composer_get_message_draft (composer); info = camel_message_info_new (NULL); /* FIXME: what's the ~0 for? :) */ camel_message_info_set_flags ( info, CAMEL_MESSAGE_SEEN | CAMEL_MESSAGE_DRAFT, ~0); mail_append_mail (folder, msg, info, NULL, composer); } static GtkActionEntry composer_entries[] = { { "template", GTK_STOCK_SAVE, N_("Save as _Template"), "t", N_("Save as Template"), G_CALLBACK (action_template_cb) } }; static void update_actions_cb (EShellView *shell_view) { EShellContent *shell_content; EShellWindow *shell_window; GtkActionGroup *action_group; GtkUIManager *ui_manager; CamelFolderInfo *folder_info; CamelFolder *templates_folder; CamelFolder *folder; CamelStore *store; EMailReader *reader; GPtrArray *uids; const gchar *full_name; guint action_count = 0; guint merge_id; gpointer data; shell_content = e_shell_view_get_shell_content (shell_view); shell_window = e_shell_view_get_shell_window (shell_view); ui_manager = e_shell_window_get_ui_manager (shell_window); action_group = e_lookup_action_group (ui_manager, "templates"); data = g_object_get_data (G_OBJECT (action_group), "merge-id"); merge_id = GPOINTER_TO_UINT (data); g_return_if_fail (merge_id > 0); gtk_ui_manager_remove_ui (ui_manager, merge_id); e_action_group_remove_all_actions (action_group); if (!plugin_enabled) return; reader = E_MAIL_READER (shell_content); folder = e_mail_reader_get_folder (reader); uids = e_mail_reader_get_selected_uids (reader); if (uids->len != 1) goto exit; /* Now recursively build template submenus in the pop-up menu. */ store = e_mail_local_get_store (); templates_folder = e_mail_local_get_folder (E_MAIL_FOLDER_TEMPLATES); full_name = camel_folder_get_full_name (templates_folder); folder_info = camel_store_get_folder_info ( store, full_name, CAMEL_STORE_FOLDER_INFO_RECURSIVE | CAMEL_STORE_FOLDER_INFO_FAST, NULL); build_template_menus_recurse ( ui_manager, action_group, "/mail-message-popup/mail-message-templates", &action_count, merge_id, folder_info, folder, uids->pdata[0]); camel_store_free_folder_info (store, folder_info); exit: em_utils_uids_free (uids); } gboolean init_composer_actions (GtkUIManager *ui_manager, EMsgComposer *composer) { GtkhtmlEditor *editor; editor = GTKHTML_EDITOR (composer); /* Add actions to the "composer" action group. */ gtk_action_group_add_actions ( gtkhtml_editor_get_action_group (editor, "composer"), composer_entries, G_N_ELEMENTS (composer_entries), composer); return TRUE; } static void mail_shell_view_created_cb (EShellWindow *shell_window, EShellView *shell_view) { g_signal_connect ( shell_view, "update-actions", G_CALLBACK (update_actions_cb), NULL); } gboolean init_shell_actions (GtkUIManager *ui_manager, EShellWindow *shell_window) { EShellView *shell_view; GtkActionGroup *action_group; guint merge_id; /* This is where we keep dynamically-built menu items. */ e_shell_window_add_action_group (shell_window, "templates"); action_group = e_lookup_action_group (ui_manager, "templates"); merge_id = gtk_ui_manager_new_merge_id (ui_manager); g_object_set_data ( G_OBJECT (action_group), "merge-id", GUINT_TO_POINTER (merge_id)); /* Be careful not to instantiate the mail view ourselves. */ shell_view = e_shell_window_peek_shell_view (shell_window, "mail"); if (shell_view != NULL) mail_shell_view_created_cb (shell_window, shell_view); else g_signal_connect ( shell_window, "shell-view-created::mail", G_CALLBACK (mail_shell_view_created_cb), NULL); return TRUE; } gint e_plugin_lib_enable (EPlugin *plugin, gboolean enabled) { plugin_enabled = enabled; return 0; }