aboutsummaryrefslogtreecommitdiffstats
path: root/modules/mail/e-mail-shell-view-private.c
diff options
context:
space:
mode:
Diffstat (limited to 'modules/mail/e-mail-shell-view-private.c')
-rw-r--r--modules/mail/e-mail-shell-view-private.c1548
1 files changed, 1548 insertions, 0 deletions
diff --git a/modules/mail/e-mail-shell-view-private.c b/modules/mail/e-mail-shell-view-private.c
new file mode 100644
index 0000000000..f33c6fe07e
--- /dev/null
+++ b/modules/mail/e-mail-shell-view-private.c
@@ -0,0 +1,1548 @@
+/*
+ * e-mail-shell-view-private.c
+ *
+ * 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 <http://www.gnu.org/licenses/>
+ *
+ *
+ * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include "e-mail-shell-view-private.h"
+
+#include "e-util/e-util-private.h"
+
+typedef struct _AsyncContext AsyncContext;
+
+struct _AsyncContext {
+ EActivity *activity;
+ EMailReader *reader;
+ EShellView *shell_view;
+};
+
+static void
+async_context_free (AsyncContext *context)
+{
+ if (context->activity != NULL)
+ g_object_unref (context->activity);
+
+ if (context->reader != NULL)
+ g_object_unref (context->reader);
+
+ if (context->shell_view != NULL)
+ g_object_unref (context->shell_view);
+
+ g_slice_free (AsyncContext, context);
+}
+
+static void
+mail_shell_view_got_folder_cb (CamelStore *store,
+ GAsyncResult *result,
+ AsyncContext *context)
+{
+ EAlertSink *alert_sink;
+ CamelFolder *folder;
+ GError *error = NULL;
+
+ alert_sink = e_activity_get_alert_sink (context->activity);
+
+ folder = camel_store_get_folder_finish (store, result, &error);
+
+ if (e_activity_handle_cancellation (context->activity, error)) {
+ g_warn_if_fail (folder == NULL);
+ async_context_free (context);
+ g_error_free (error);
+ return;
+
+ } else if (error != NULL) {
+ g_warn_if_fail (folder == NULL);
+ e_alert_submit (
+ alert_sink, "mail:folder-open",
+ error->message, NULL);
+ async_context_free (context);
+ g_error_free (error);
+ return;
+ }
+
+ e_mail_reader_set_folder (context->reader, folder);
+ e_shell_view_update_actions (context->shell_view);
+
+ g_object_unref (folder);
+
+ async_context_free (context);
+}
+
+static void
+mail_shell_view_folder_tree_selected_cb (EMailShellView *mail_shell_view,
+ CamelStore *store,
+ const gchar *folder_name,
+ CamelFolderInfoFlags flags,
+ EMFolderTree *folder_tree)
+{
+ EMailShellContent *mail_shell_content;
+ EShellView *shell_view;
+ EMailReader *reader;
+ EMailView *mail_view;
+ GCancellable *cancellable;
+ AsyncContext *context;
+ EActivity *activity;
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ reader = E_MAIL_READER (mail_view);
+
+ /* Cancel any unfinished open folder operations. */
+ if (mail_shell_view->priv->opening_folder != NULL) {
+ g_cancellable_cancel (mail_shell_view->priv->opening_folder);
+ g_object_unref (mail_shell_view->priv->opening_folder);
+ mail_shell_view->priv->opening_folder = NULL;
+ }
+
+ /* If we are to clear the message list, do so immediately. */
+ if ((flags & CAMEL_FOLDER_NOSELECT) || folder_name == NULL) {
+ e_mail_reader_set_folder (reader, NULL);
+ e_shell_view_update_actions (shell_view);
+ return;
+ }
+
+ g_warn_if_fail (CAMEL_IS_STORE (store));
+
+ /* Open the selected folder asynchronously. */
+
+ activity = e_mail_reader_new_activity (reader);
+ cancellable = e_activity_get_cancellable (activity);
+ mail_shell_view->priv->opening_folder = g_object_ref (cancellable);
+
+ context = g_slice_new0 (AsyncContext);
+ context->activity = activity;
+ context->reader = g_object_ref (reader);
+ context->shell_view = g_object_ref (shell_view);
+
+ camel_store_get_folder (
+ store, folder_name, 0, G_PRIORITY_DEFAULT, cancellable,
+ (GAsyncReadyCallback) mail_shell_view_got_folder_cb, context);
+}
+
+static gboolean
+mail_shell_view_folder_tree_key_press_event_cb (EMailShellView *mail_shell_view,
+ GdkEventKey *event)
+{
+ EMailShellContent *mail_shell_content;
+ EMailView *mail_view;
+ gboolean handled = FALSE;
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ if ((event->state & GDK_CONTROL_MASK) != 0)
+ goto ctrl;
+
+ /* <keyval> alone */
+ switch (event->keyval) {
+ case GDK_KEY_period:
+ case GDK_KEY_comma:
+ case GDK_KEY_bracketleft:
+ case GDK_KEY_bracketright:
+ goto emit;
+
+ default:
+ goto exit;
+ }
+
+ctrl:
+ /* Ctrl + <keyval> */
+ switch (event->keyval) {
+ case GDK_KEY_period:
+ case GDK_KEY_comma:
+ goto emit;
+
+ default:
+ goto exit;
+ }
+
+ /* All branches jump past this. */
+ g_return_val_if_reached (FALSE);
+
+emit:
+ /* Forward the key press to the EMailReader interface. */
+ g_signal_emit_by_name (mail_view, "key-press-event", event, &handled);
+
+exit:
+ return handled;
+}
+
+static void
+mail_shell_view_folder_tree_selection_done_cb (EMailShellView *mail_shell_view,
+ GtkWidget *menu)
+{
+ EMailShellContent *mail_shell_content;
+ EMailShellSidebar *mail_shell_sidebar;
+ EMFolderTree *folder_tree;
+ GtkWidget *message_list;
+ EMailReader *reader;
+ EMailView *mail_view;
+ CamelFolder *folder;
+ gchar *list_uri = NULL;
+ gchar *tree_uri;
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ mail_shell_sidebar = mail_shell_view->priv->mail_shell_sidebar;
+ folder_tree = e_mail_shell_sidebar_get_folder_tree (mail_shell_sidebar);
+
+ reader = E_MAIL_READER (mail_view);
+ message_list = e_mail_reader_get_message_list (reader);
+
+ /* Don't use e_mail_reader_ref_folder() here. The fact that the
+ * method gets the folder from the message list is supposed to be
+ * a hidden implementation detail, and we want to explicitly get
+ * the folder URI from the message list here. */
+ folder = message_list_ref_folder (MESSAGE_LIST (message_list));
+ if (folder != NULL) {
+ list_uri = e_mail_folder_uri_from_folder (folder);
+ g_object_unref (folder);
+ }
+
+ tree_uri = em_folder_tree_get_selected_uri (folder_tree);
+
+ /* If the folder tree and message list disagree on the current
+ * folder, reset the folder tree to match the message list. */
+ if (list_uri != NULL && g_strcmp0 (tree_uri, list_uri) != 0)
+ em_folder_tree_set_selected (folder_tree, list_uri, FALSE);
+
+ g_free (list_uri);
+ g_free (tree_uri);
+
+ /* Disconnect from the "selection-done" signal. */
+ g_signal_handlers_disconnect_by_func (
+ menu, mail_shell_view_folder_tree_selection_done_cb,
+ mail_shell_view);
+}
+
+static void
+mail_shell_view_folder_tree_popup_event_cb (EShellView *shell_view,
+ GdkEvent *button_event)
+{
+ GtkWidget *menu;
+
+ menu = e_shell_view_show_popup_menu (
+ shell_view, "/mail-folder-popup", button_event);
+
+ g_signal_connect_object (
+ menu, "selection-done",
+ G_CALLBACK (mail_shell_view_folder_tree_selection_done_cb),
+ shell_view, G_CONNECT_SWAPPED);
+}
+
+static gboolean
+mail_shell_view_mail_display_needs_key (EMailDisplay *mail_display,
+ gboolean with_input)
+{
+ gboolean needs_key = FALSE;
+
+ if (gtk_widget_has_focus (GTK_WIDGET (mail_display))) {
+ WebKitWebFrame *frame;
+ WebKitDOMDocument *dom;
+ WebKitDOMElement *element;
+ gchar *name = NULL;
+
+ frame = webkit_web_view_get_focused_frame (WEBKIT_WEB_VIEW (mail_display));
+ if (!frame)
+ return FALSE;
+ dom = webkit_web_frame_get_dom_document (frame);
+ /* intentionally used "static_cast" */
+ element = webkit_dom_html_document_get_active_element ((WebKitDOMHTMLDocument *) dom);
+
+ if (element)
+ name = webkit_dom_node_get_node_name (WEBKIT_DOM_NODE (element));
+
+ /* if INPUT or TEXTAREA has focus, then any key press should go there */
+ if (name && ((with_input && g_ascii_strcasecmp (name, "INPUT") == 0) || g_ascii_strcasecmp (name, "TEXTAREA") == 0)) {
+ needs_key = TRUE;
+ }
+
+ g_free (name);
+ }
+
+ return needs_key;
+}
+
+static gboolean
+mail_shell_view_key_press_event_cb (EMailShellView *mail_shell_view,
+ GdkEventKey *event)
+{
+ EShellView *shell_view;
+ EShellWindow *shell_window;
+ EShellContent *shell_content;
+ EMailView *mail_view;
+ EMailReader *reader;
+ EMailDisplay *mail_display;
+ GtkAction *action;
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ shell_window = e_shell_view_get_shell_window (shell_view);
+
+ if ((event->state & (GDK_CONTROL_MASK | GDK_SHIFT_MASK | GDK_MOD1_MASK)) != 0)
+ return FALSE;
+
+ shell_content = e_shell_view_get_shell_content (shell_view);
+ mail_view = e_mail_shell_content_get_mail_view (E_MAIL_SHELL_CONTENT (shell_content));
+ reader = E_MAIL_READER (mail_view);
+ mail_display = e_mail_reader_get_mail_display (reader);
+
+ switch (event->keyval) {
+ case GDK_KEY_space:
+ action = ACTION (MAIL_SMART_FORWARD);
+ break;
+
+ case GDK_KEY_BackSpace:
+ action = ACTION (MAIL_SMART_BACKWARD);
+ break;
+
+ case GDK_KEY_Home:
+ case GDK_KEY_Left:
+ case GDK_KEY_Up:
+ case GDK_KEY_Right:
+ case GDK_KEY_Down:
+ case GDK_KEY_Prior:
+ case GDK_KEY_Next:
+ case GDK_KEY_End:
+ case GDK_KEY_Begin:
+ if (!mail_shell_view_mail_display_needs_key (mail_display, FALSE) &&
+ webkit_web_view_get_main_frame (WEBKIT_WEB_VIEW (mail_display)) !=
+ webkit_web_view_get_focused_frame (WEBKIT_WEB_VIEW (mail_display))) {
+ WebKitDOMDocument *document;
+ WebKitDOMDOMWindow *window;
+
+ document = webkit_web_view_get_dom_document (WEBKIT_WEB_VIEW (mail_display));
+ window = webkit_dom_document_get_default_view (document);
+
+ /* Workaround WebKit bug for key navigation, when inner IFRAME is focused.
+ * EMailView's inner IFRAMEs have disabled scrolling, but WebKit doesn't post
+ * key navigation events to parent's frame, thus the view doesn't scroll.
+ * This is a poor workaround for this issue, the main frame is focused,
+ * which has scrolling enabled.
+ */
+ webkit_dom_dom_window_focus (window);
+ }
+
+ return FALSE;
+ default:
+ return FALSE;
+ }
+
+ if (mail_shell_view_mail_display_needs_key (mail_display, TRUE))
+ return FALSE;
+
+ gtk_action_activate (action);
+
+ return TRUE;
+}
+
+static gboolean
+mail_shell_view_message_list_key_press_cb (EMailShellView *mail_shell_view,
+ gint row,
+ ETreePath path,
+ gint col,
+ GdkEvent *event)
+{
+ return mail_shell_view_key_press_event_cb (
+ mail_shell_view, &event->key);
+}
+
+static gboolean
+mail_shell_view_message_list_popup_menu_cb (EShellView *shell_view)
+{
+ const gchar *widget_path;
+
+ widget_path = "/mail-message-popup";
+ e_shell_view_show_popup_menu (shell_view, widget_path, NULL);
+
+ return TRUE;
+}
+
+static gboolean
+mail_shell_view_message_list_right_click_cb (EShellView *shell_view,
+ gint row,
+ ETreePath path,
+ gint col,
+ GdkEvent *button_event)
+{
+ const gchar *widget_path;
+
+ widget_path = "/mail-message-popup";
+ e_shell_view_show_popup_menu (shell_view, widget_path, button_event);
+
+ return TRUE;
+}
+
+static gboolean
+mail_shell_view_popup_event_cb (EMailShellView *mail_shell_view,
+ const gchar *uri)
+{
+ EMailShellContent *mail_shell_content;
+ EMailDisplay *display;
+ EShellView *shell_view;
+ EMailReader *reader;
+ EMailView *mail_view;
+ GtkMenu *menu;
+
+ if (uri != NULL)
+ return FALSE;
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ reader = E_MAIL_READER (mail_view);
+ display = e_mail_reader_get_mail_display (reader);
+
+ if (e_web_view_get_cursor_image_src (E_WEB_VIEW (display)) != NULL)
+ return FALSE;
+
+ menu = e_mail_reader_get_popup_menu (reader);
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ e_shell_view_update_actions (shell_view);
+
+ gtk_menu_popup (
+ menu, NULL, NULL, NULL, NULL,
+ 0, gtk_get_current_event_time ());
+
+ return TRUE;
+}
+
+static void
+mail_shell_view_reader_changed_cb (EMailShellView *mail_shell_view,
+ EMailReader *reader)
+{
+ GtkWidget *message_list;
+ EMailDisplay *display;
+ EShellView *shell_view;
+ EShellTaskbar *shell_taskbar;
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ shell_taskbar = e_shell_view_get_shell_taskbar (shell_view);
+
+ display = e_mail_reader_get_mail_display (reader);
+ message_list = e_mail_reader_get_message_list (reader);
+
+ e_shell_view_update_actions (E_SHELL_VIEW (mail_shell_view));
+ e_mail_shell_view_update_sidebar (mail_shell_view);
+
+ /* Connect if its not connected already */
+ if (g_signal_handler_find (
+ message_list, G_SIGNAL_MATCH_FUNC, 0, 0, NULL,
+ mail_shell_view_message_list_key_press_cb, NULL))
+ return;
+
+ g_signal_connect_object (
+ message_list, "key-press",
+ G_CALLBACK (mail_shell_view_message_list_key_press_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ message_list, "popup-menu",
+ G_CALLBACK (mail_shell_view_message_list_popup_menu_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ message_list, "right-click",
+ G_CALLBACK (mail_shell_view_message_list_right_click_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ display, "key-press-event",
+ G_CALLBACK (mail_shell_view_key_press_event_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ display, "popup-event",
+ G_CALLBACK (mail_shell_view_popup_event_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ display, "status-message",
+ G_CALLBACK (e_shell_taskbar_set_message),
+ shell_taskbar, G_CONNECT_SWAPPED);
+}
+
+static void
+mail_shell_view_reader_update_actions_cb (EMailReader *reader,
+ guint32 state,
+ EMailShellView *mail_shell_view)
+{
+ EMailShellContent *mail_shell_content;
+
+ g_return_if_fail (mail_shell_view != NULL);
+ g_return_if_fail (mail_shell_view->priv != NULL);
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ e_mail_reader_update_actions (E_MAIL_READER (mail_shell_content), state);
+}
+
+static void
+mail_shell_view_prepare_for_quit_done_cb (CamelFolder *folder,
+ gpointer user_data)
+{
+ g_object_unref (E_ACTIVITY (user_data));
+}
+
+static void
+mail_shell_view_prepare_for_quit_cb (EMailShellView *mail_shell_view,
+ EActivity *activity)
+{
+ EMailShellContent *mail_shell_content;
+ CamelFolder *folder;
+ EMailReader *reader;
+ EMailView *mail_view;
+ GtkWidget *message_list;
+
+ /* If we got here, it means the application is shutting down
+ * and this is the last EMailShellView instance. Synchronize
+ * the currently selected folder before we terminate. */
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ reader = E_MAIL_READER (mail_view);
+ folder = e_mail_reader_ref_folder (reader);
+ message_list = e_mail_reader_get_message_list (reader);
+
+ message_list_save_state (MESSAGE_LIST (message_list));
+
+ if (folder != NULL) {
+ mail_sync_folder (
+ folder, TRUE,
+ mail_shell_view_prepare_for_quit_done_cb,
+ g_object_ref (activity));
+ g_object_unref (folder);
+ }
+}
+
+static void
+mail_shell_view_notify_view_id_cb (EMailShellView *mail_shell_view)
+{
+ EMailShellContent *mail_shell_content;
+ GalViewInstance *view_instance;
+ EMailView *mail_view;
+ const gchar *view_id;
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ view_instance = e_mail_view_get_view_instance (mail_view);
+ view_id = e_shell_view_get_view_id (E_SHELL_VIEW (mail_shell_view));
+
+ /* A NULL view ID implies we're in a custom view. But you can
+ * only get to a custom view via the "Define Views" dialog, which
+ * would have already modified the view instance appropriately.
+ * Furthermore, there's no way to refer to a custom view by ID
+ * anyway, since custom views have no IDs. */
+ if (view_id == NULL)
+ return;
+
+ gal_view_instance_set_current_view_id (view_instance, view_id);
+}
+
+static void
+mail_shell_view_search_filter_changed_cb (EMailShellView *mail_shell_view)
+{
+ EMailShellContent *mail_shell_content;
+ EMailView *mail_view;
+
+ g_return_if_fail (mail_shell_view != NULL);
+ g_return_if_fail (mail_shell_view->priv != NULL);
+
+ if (e_shell_view_is_execute_search_blocked (E_SHELL_VIEW (mail_shell_view)))
+ return;
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ e_mail_reader_avoid_next_mark_as_seen (E_MAIL_READER (mail_view));
+}
+
+void
+e_mail_shell_view_private_init (EMailShellView *mail_shell_view)
+{
+ g_signal_connect (
+ mail_shell_view, "notify::view-id",
+ G_CALLBACK (mail_shell_view_notify_view_id_cb), NULL);
+}
+
+void
+e_mail_shell_view_private_constructed (EMailShellView *mail_shell_view)
+{
+ EMailShellViewPrivate *priv = mail_shell_view->priv;
+ EMailShellContent *mail_shell_content;
+ EMailShellSidebar *mail_shell_sidebar;
+ EShell *shell;
+ EShellView *shell_view;
+ EShellBackend *shell_backend;
+ EShellContent *shell_content;
+ EShellSidebar *shell_sidebar;
+ EShellTaskbar *shell_taskbar;
+ EShellWindow *shell_window;
+ EShellSearchbar *searchbar;
+ EMFolderTree *folder_tree;
+ EActionComboBox *combo_box;
+ ERuleContext *context;
+ EFilterRule *rule = NULL;
+ GtkTreeSelection *selection;
+ GtkUIManager *ui_manager;
+ GtkWidget *message_list;
+ EMailLabelListStore *label_store;
+ EMailBackend *backend;
+ EMailSession *session;
+ EMailReader *reader;
+ EMailView *mail_view;
+ EMailDisplay *display;
+ const gchar *source;
+ guint merge_id;
+ gint ii = 0;
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ shell_backend = e_shell_view_get_shell_backend (shell_view);
+ shell_content = e_shell_view_get_shell_content (shell_view);
+ shell_sidebar = e_shell_view_get_shell_sidebar (shell_view);
+ shell_taskbar = e_shell_view_get_shell_taskbar (shell_view);
+ shell_window = e_shell_view_get_shell_window (shell_view);
+
+ ui_manager = e_shell_window_get_ui_manager (shell_window);
+
+ shell = e_shell_window_get_shell (shell_window);
+
+ backend = E_MAIL_BACKEND (shell_backend);
+ session = e_mail_backend_get_session (backend);
+ label_store = e_mail_ui_session_get_label_store (
+ E_MAIL_UI_SESSION (session));
+
+ e_shell_window_add_action_group (shell_window, "mail");
+ e_shell_window_add_action_group (shell_window, "mail-filter");
+ e_shell_window_add_action_group (shell_window, "mail-label");
+ e_shell_window_add_action_group (shell_window, "search-folders");
+
+ merge_id = gtk_ui_manager_new_merge_id (ui_manager);
+ priv->label_merge_id = merge_id;
+
+ /* Cache these to avoid lots of awkward casting. */
+ priv->mail_shell_backend = g_object_ref (shell_backend);
+ priv->mail_shell_content = g_object_ref (shell_content);
+ priv->mail_shell_sidebar = g_object_ref (shell_sidebar);
+
+ mail_shell_sidebar = E_MAIL_SHELL_SIDEBAR (shell_sidebar);
+ folder_tree = e_mail_shell_sidebar_get_folder_tree (mail_shell_sidebar);
+ selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_tree));
+
+ mail_shell_content = E_MAIL_SHELL_CONTENT (shell_content);
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+ searchbar = e_mail_shell_content_get_searchbar (mail_shell_content);
+ combo_box = e_shell_searchbar_get_scope_combo_box (searchbar);
+
+ reader = E_MAIL_READER (shell_content);
+ display = e_mail_reader_get_mail_display (reader);
+ message_list = e_mail_reader_get_message_list (reader);
+
+ em_folder_tree_set_selectable_widget (folder_tree, message_list);
+
+ /* The folder tree and scope combo box are both insensitive
+ * when searching beyond the currently selected folder. */
+ g_object_bind_property (
+ folder_tree, "sensitive",
+ combo_box, "sensitive",
+ G_BINDING_BIDIRECTIONAL |
+ G_BINDING_SYNC_CREATE);
+
+ combo_box = e_shell_searchbar_get_filter_combo_box (searchbar);
+ g_signal_connect_object (
+ combo_box, "changed",
+ G_CALLBACK (mail_shell_view_search_filter_changed_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ folder_tree, "folder-selected",
+ G_CALLBACK (mail_shell_view_folder_tree_selected_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ folder_tree, "key-press-event",
+ G_CALLBACK (mail_shell_view_folder_tree_key_press_event_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ folder_tree, "popup-event",
+ G_CALLBACK (mail_shell_view_folder_tree_popup_event_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ message_list, "key-press",
+ G_CALLBACK (mail_shell_view_message_list_key_press_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ message_list, "popup-menu",
+ G_CALLBACK (mail_shell_view_message_list_popup_menu_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ message_list, "right-click",
+ G_CALLBACK (mail_shell_view_message_list_right_click_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ reader, "changed",
+ G_CALLBACK (mail_shell_view_reader_changed_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ mail_view, "update-actions",
+ G_CALLBACK (mail_shell_view_reader_update_actions_cb),
+ mail_shell_view, 0);
+
+ g_signal_connect_object (
+ reader, "folder-loaded",
+ G_CALLBACK (e_mail_view_update_view_instance),
+ mail_view, G_CONNECT_SWAPPED);
+
+ /* Use the same callback as "changed". */
+ g_signal_connect_object (
+ reader, "folder-loaded",
+ G_CALLBACK (mail_shell_view_reader_changed_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ reader, "folder-loaded",
+ G_CALLBACK (e_mail_shell_view_restore_state),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ label_store, "row-changed",
+ G_CALLBACK (e_mail_shell_view_update_search_filter),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ label_store, "row-deleted",
+ G_CALLBACK (e_mail_shell_view_update_search_filter),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ label_store, "row-inserted",
+ G_CALLBACK (e_mail_shell_view_update_search_filter),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ display, "key-press-event",
+ G_CALLBACK (mail_shell_view_key_press_event_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ display, "popup-event",
+ G_CALLBACK (mail_shell_view_popup_event_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ display, "status-message",
+ G_CALLBACK (e_shell_taskbar_set_message),
+ shell_taskbar, G_CONNECT_SWAPPED);
+
+ g_signal_connect_object (
+ mail_shell_view, "toggled",
+ G_CALLBACK (e_mail_shell_view_update_send_receive_menus),
+ mail_shell_view, G_CONNECT_AFTER | G_CONNECT_SWAPPED);
+
+ /* Need to keep the handler ID so we can disconnect it in
+ * dispose(). The shell outlives us and we don't want it
+ * invoking callbacks on finalized shell views. */
+ priv->prepare_for_quit_handler_id =
+ g_signal_connect_object (
+ shell, "prepare-for-quit",
+ G_CALLBACK (mail_shell_view_prepare_for_quit_cb),
+ mail_shell_view, G_CONNECT_SWAPPED);
+
+ e_mail_reader_init (reader, TRUE, FALSE);
+ e_mail_shell_view_actions_init (mail_shell_view);
+ e_mail_shell_view_update_search_filter (mail_shell_view);
+
+ /* This binding must come after e_mail_reader_init(). */
+ g_object_bind_property (
+ shell_content, "group-by-threads",
+ mail_view, "group-by-threads",
+ G_BINDING_BIDIRECTIONAL |
+ G_BINDING_SYNC_CREATE);
+
+ /* Populate built-in rules for search entry popup menu.
+ * Keep the assertions, please. If the conditions aren't
+ * met we're going to crash anyway, just more mysteriously. */
+ context = E_SHELL_VIEW_GET_CLASS (shell_view)->search_context;
+ source = E_FILTER_SOURCE_DEMAND;
+ while ((rule = e_rule_context_next_rule (context, rule, source))) {
+ if (!rule->system)
+ continue;
+ g_assert (ii < MAIL_NUM_SEARCH_RULES);
+ priv->search_rules[ii++] = g_object_ref (rule);
+ }
+ g_assert (ii == MAIL_NUM_SEARCH_RULES);
+
+ /* Now that we're all set up, simulate selecting a folder. */
+ g_signal_emit_by_name (selection, "changed");
+}
+
+void
+e_mail_shell_view_private_dispose (EMailShellView *mail_shell_view)
+{
+ EMailShellViewPrivate *priv = mail_shell_view->priv;
+ gint ii;
+
+ /* XXX It's a little awkward to have to dig up the
+ * shell this late in the game. Should we just
+ * keep a direct reference to it? Not sure. */
+ if (priv->prepare_for_quit_handler_id > 0) {
+ EShellBackend *shell_backend;
+ EShell *shell;
+
+ shell_backend = E_SHELL_BACKEND (priv->mail_shell_backend);
+ shell = e_shell_backend_get_shell (shell_backend);
+
+ g_signal_handler_disconnect (
+ shell, priv->prepare_for_quit_handler_id);
+ priv->prepare_for_quit_handler_id = 0;
+ }
+
+ g_clear_object (&priv->mail_shell_backend);
+ g_clear_object (&priv->mail_shell_content);
+ g_clear_object (&priv->mail_shell_sidebar);
+
+ for (ii = 0; ii < MAIL_NUM_SEARCH_RULES; ii++)
+ g_clear_object (&priv->search_rules[ii]);
+
+ if (priv->opening_folder != NULL) {
+ g_cancellable_cancel (priv->opening_folder);
+ g_clear_object (&priv->opening_folder);
+ }
+
+ if (priv->search_account_all != NULL) {
+ g_object_unref (priv->search_account_all);
+ priv->search_account_all = NULL;
+ }
+
+ if (priv->search_account_current != NULL) {
+ g_object_unref (priv->search_account_current);
+ priv->search_account_current = NULL;
+ }
+
+ if (priv->search_account_cancel != NULL) {
+ g_object_unref (priv->search_account_cancel);
+ priv->search_account_cancel = NULL;
+ }
+}
+
+void
+e_mail_shell_view_private_finalize (EMailShellView *mail_shell_view)
+{
+ /* XXX Nothing to do? */
+}
+
+void
+e_mail_shell_view_restore_state (EMailShellView *mail_shell_view)
+{
+ EMailShellContent *mail_shell_content;
+ EShellSearchbar *searchbar;
+ EMailReader *reader;
+ EMailView *mail_view;
+ CamelFolder *folder;
+ CamelVeeFolder *vee_folder;
+ const gchar *old_state_group;
+ gchar *folder_uri;
+ gchar *new_state_group;
+
+ /* XXX Move this to EMailShellContent. */
+
+ g_return_if_fail (E_IS_MAIL_SHELL_VIEW (mail_shell_view));
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+ searchbar = e_mail_shell_content_get_searchbar (mail_shell_content);
+
+ reader = E_MAIL_READER (mail_view);
+ folder = e_mail_reader_ref_folder (reader);
+
+ if (folder == NULL) {
+ if (e_shell_searchbar_get_state_group (searchbar)) {
+ e_shell_searchbar_set_state_group (searchbar, NULL);
+ e_shell_searchbar_load_state (searchbar);
+ }
+ return;
+ }
+
+ /* Do not restore state if we're running a "Current Account"
+ * or "All Accounts" search, since we don't want the search
+ * criteria to be destroyed in those cases. */
+
+ vee_folder = mail_shell_view->priv->search_account_all;
+ if (vee_folder != NULL && folder == CAMEL_FOLDER (vee_folder))
+ goto exit;
+
+ vee_folder = mail_shell_view->priv->search_account_current;
+ if (vee_folder != NULL && folder == CAMEL_FOLDER (vee_folder))
+ goto exit;
+
+ folder_uri = e_mail_folder_uri_from_folder (folder);
+ new_state_group = g_strdup_printf ("Folder %s", folder_uri);
+ old_state_group = e_shell_searchbar_get_state_group (searchbar);
+ g_free (folder_uri);
+
+ /* Avoid loading search state unnecessarily. */
+ if (g_strcmp0 (new_state_group, old_state_group) != 0) {
+ e_shell_searchbar_set_state_group (searchbar, new_state_group);
+ e_shell_searchbar_load_state (searchbar);
+ }
+
+ g_free (new_state_group);
+
+exit:
+ g_clear_object (&folder);
+}
+
+void
+e_mail_shell_view_update_sidebar (EMailShellView *mail_shell_view)
+{
+ EMailShellContent *mail_shell_content;
+ EShellBackend *shell_backend;
+ EShellSidebar *shell_sidebar;
+ EShellView *shell_view;
+ EShell *shell;
+ EMailReader *reader;
+ EMailView *mail_view;
+ ESourceRegistry *registry;
+ CamelStore *parent_store;
+ CamelFolder *folder;
+ GPtrArray *uids;
+ GString *buffer;
+ gboolean store_is_local;
+ const gchar *display_name;
+ const gchar *folder_name;
+ const gchar *uid;
+ gchar *title;
+ guint32 num_deleted;
+ guint32 num_junked;
+ guint32 num_junked_not_deleted;
+ guint32 num_unread;
+ guint32 num_visible;
+
+ g_return_if_fail (E_IS_MAIL_SHELL_VIEW (mail_shell_view));
+
+ mail_shell_content = mail_shell_view->priv->mail_shell_content;
+ mail_view = e_mail_shell_content_get_mail_view (mail_shell_content);
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ shell_backend = e_shell_view_get_shell_backend (shell_view);
+ shell_sidebar = e_shell_view_get_shell_sidebar (shell_view);
+
+ shell = e_shell_backend_get_shell (shell_backend);
+ registry = e_shell_get_registry (shell);
+
+ reader = E_MAIL_READER (mail_view);
+ folder = e_mail_reader_ref_folder (reader);
+
+ /* If no folder is selected, reset the sidebar banners
+ * to their default values and stop. */
+ if (folder == NULL) {
+ GtkAction *action;
+ gchar *label;
+
+ action = e_shell_view_get_action (shell_view);
+
+ g_object_get (action, "label", &label, NULL);
+ e_shell_sidebar_set_secondary_text (shell_sidebar, NULL);
+ e_shell_view_set_title (shell_view, label);
+ g_free (label);
+
+ return;
+ }
+
+ folder_name = camel_folder_get_display_name (folder);
+ parent_store = camel_folder_get_parent_store (folder);
+
+ num_deleted = camel_folder_summary_get_deleted_count (folder->summary);
+ num_junked = camel_folder_summary_get_junk_count (folder->summary);
+ num_junked_not_deleted =
+ camel_folder_summary_get_junk_not_deleted_count (folder->summary);
+ num_unread = camel_folder_summary_get_unread_count (folder->summary);
+ num_visible = camel_folder_summary_get_visible_count (folder->summary);
+
+ buffer = g_string_sized_new (256);
+ uids = e_mail_reader_get_selected_uids (reader);
+
+ if (uids->len > 1)
+ g_string_append_printf (
+ buffer, ngettext ("%d selected, ", "%d selected, ",
+ uids->len), uids->len);
+
+ /* "Trash" folder (virtual or real) */
+ if (folder->folder_flags & CAMEL_FOLDER_IS_TRASH) {
+ if (CAMEL_IS_VTRASH_FOLDER (folder)) {
+ /* For a virtual Trash folder, count
+ * the messages marked for deletion. */
+ g_string_append_printf (
+ buffer, ngettext ("%d deleted",
+ "%d deleted", num_deleted), num_deleted);
+ } else {
+ /* For a regular Trash folder, just
+ * count the visible messages.
+ *
+ * XXX Open question: what to do about messages
+ * marked for deletion in Trash? Probably
+ * this is the wrong question to be asking
+ * anyway. Deleting a message in a real
+ * Trash should permanently expunge the
+ * message (if the server supports that),
+ * which would eliminate this corner case. */
+ if (!e_mail_reader_get_hide_deleted (reader))
+ num_visible += num_deleted;
+
+ g_string_append_printf (
+ buffer, ngettext ("%d deleted",
+ "%d deleted", num_visible), num_visible);
+ }
+
+ /* "Junk" folder (virtual or real) */
+ } else if (folder->folder_flags & CAMEL_FOLDER_IS_JUNK) {
+ if (e_mail_reader_get_hide_deleted (reader)) {
+ /* Junk folder with deleted messages hidden. */
+ g_string_append_printf (
+ buffer, ngettext ("%d junk",
+ "%d junk", num_junked_not_deleted),
+ num_junked_not_deleted);
+ } else {
+ /* Junk folder with deleted messages visible. */
+ g_string_append_printf (
+ buffer, ngettext ("%d junk", "%d junk",
+ num_junked), num_junked);
+ }
+
+ /* "Drafts" folder */
+ } else if (em_utils_folder_is_drafts (registry, folder)) {
+ g_string_append_printf (
+ buffer, ngettext ("%d draft", "%d drafts",
+ num_visible), num_visible);
+
+ /* "Outbox" folder */
+ } else if (em_utils_folder_is_outbox (registry, folder)) {
+ g_string_append_printf (
+ buffer, ngettext ("%d unsent", "%d unsent",
+ num_visible), num_visible);
+
+ /* "Sent" folder */
+ } else if (em_utils_folder_is_sent (registry, folder)) {
+ g_string_append_printf (
+ buffer, ngettext ("%d sent", "%d sent",
+ num_visible), num_visible);
+
+ /* Normal folder */
+ } else {
+ if (!e_mail_reader_get_hide_deleted (reader))
+ num_visible +=
+ num_deleted - num_junked +
+ num_junked_not_deleted;
+
+ if (num_unread > 0 && uids->len <= 1)
+ g_string_append_printf (
+ buffer, ngettext ("%d unread, ",
+ "%d unread, ", num_unread), num_unread);
+ g_string_append_printf (
+ buffer, ngettext ("%d total", "%d total",
+ num_visible), num_visible);
+ }
+
+ g_ptr_array_unref (uids);
+
+ uid = camel_service_get_uid (CAMEL_SERVICE (parent_store));
+ store_is_local = (g_strcmp0 (uid, E_MAIL_SESSION_LOCAL_UID) == 0);
+
+ /* Choose a suitable folder name for displaying. */
+ display_name = folder_name;
+ if (store_is_local) {
+ if (strcmp (folder_name, "Drafts") == 0)
+ display_name = _("Drafts");
+ else if (strcmp (folder_name, "Inbox") == 0)
+ display_name = _("Inbox");
+ else if (strcmp (folder_name, "Outbox") == 0)
+ display_name = _("Outbox");
+ else if (strcmp (folder_name, "Sent") == 0)
+ display_name = _("Sent");
+ else if (strcmp (folder_name, "Templates") == 0)
+ display_name = _("Templates");
+ else if (strcmp (folder_name, "Trash") == 0)
+ display_name = _("Trash");
+ }
+ if (strcmp (folder_name, "INBOX") == 0)
+ display_name = _("Inbox");
+
+ title = g_strdup_printf ("%s (%s)", display_name, buffer->str);
+ e_shell_sidebar_set_secondary_text (shell_sidebar, buffer->str);
+ e_shell_view_set_title (shell_view, title);
+ g_free (title);
+
+ g_string_free (buffer, TRUE);
+
+ g_clear_object (&folder);
+}
+
+typedef struct {
+ GtkMenuShell *menu;
+ CamelSession *session;
+ EMailAccountStore *account_store;
+
+ /* GtkMenuItem -> CamelService */
+ GHashTable *menu_items;
+
+ /* Signal handlers */
+ gulong service_added_id;
+ gulong service_removed_id;
+ gulong service_enabled_id;
+ gulong service_disabled_id;
+} SendReceiveData;
+
+static gboolean
+send_receive_can_use_service (EMailAccountStore *account_store,
+ CamelService *service,
+ GtkTreeIter *piter)
+{
+ GtkTreeModel *model;
+ GtkTreeIter iter;
+ gboolean found = FALSE, enabled = FALSE, builtin = TRUE;
+
+ if (!CAMEL_IS_STORE (service))
+ return FALSE;
+
+ model = GTK_TREE_MODEL (account_store);
+
+ if (piter) {
+ found = TRUE;
+ iter = *piter;
+ } else if (gtk_tree_model_get_iter_first (model, &iter)) {
+ CamelService *adept;
+
+ do {
+ adept = NULL;
+
+ gtk_tree_model_get (
+ model, &iter,
+ E_MAIL_ACCOUNT_STORE_COLUMN_SERVICE, &adept,
+ -1);
+
+ if (service == adept) {
+ found = TRUE;
+ g_object_unref (adept);
+ break;
+ }
+
+ if (adept)
+ g_object_unref (adept);
+ } while (gtk_tree_model_iter_next (model, &iter));
+ }
+
+ if (!found)
+ return FALSE;
+
+ gtk_tree_model_get (
+ model, &iter,
+ E_MAIL_ACCOUNT_STORE_COLUMN_ENABLED, &enabled,
+ E_MAIL_ACCOUNT_STORE_COLUMN_BUILTIN, &builtin,
+ -1);
+
+ return enabled && !builtin;
+}
+
+static GtkMenuItem *
+send_receive_find_menu_item (SendReceiveData *data,
+ gpointer service)
+{
+ GHashTableIter iter;
+ gpointer menu_item;
+ gpointer candidate;
+
+ g_hash_table_iter_init (&iter, data->menu_items);
+
+ while (g_hash_table_iter_next (&iter, &menu_item, &candidate))
+ if (service == candidate)
+ return GTK_MENU_ITEM (menu_item);
+
+ return NULL;
+}
+
+static void
+send_receive_account_item_activate_cb (GtkMenuItem *menu_item,
+ SendReceiveData *data)
+{
+ CamelService *service;
+
+ service = g_hash_table_lookup (data->menu_items, menu_item);
+ g_return_if_fail (CAMEL_IS_SERVICE (service));
+
+ mail_receive_service (service);
+}
+
+static void
+send_receive_add_to_menu (SendReceiveData *data,
+ CamelService *service,
+ gint position)
+{
+ GtkWidget *menu_item;
+ CamelProvider *provider;
+
+ if (send_receive_find_menu_item (data, service) != NULL)
+ return;
+
+ provider = camel_service_get_provider (service);
+
+ menu_item = gtk_menu_item_new ();
+ gtk_widget_show (menu_item);
+
+ g_object_bind_property (
+ service, "display-name",
+ menu_item, "label",
+ G_BINDING_SYNC_CREATE);
+
+ if (provider && (provider->flags & CAMEL_PROVIDER_IS_REMOTE) != 0) {
+ gpointer object;
+
+ if (CAMEL_IS_OFFLINE_STORE (service) ||
+ CAMEL_IS_DISCO_STORE (service))
+ object = g_object_ref (service);
+ else
+ object = camel_service_ref_session (service);
+
+ g_object_bind_property (
+ object, "online",
+ menu_item, "sensitive",
+ G_BINDING_SYNC_CREATE);
+
+ g_object_unref (object);
+ }
+
+ g_hash_table_insert (
+ data->menu_items, menu_item,
+ g_object_ref (service));
+
+ g_signal_connect (
+ menu_item, "activate",
+ G_CALLBACK (send_receive_account_item_activate_cb), data);
+
+ /* Position is with respect to the sorted list of CamelService-s,
+ * not menu item position. */
+ if (position < 0)
+ gtk_menu_shell_append (data->menu, menu_item);
+ else
+ gtk_menu_shell_insert (data->menu, menu_item, position + 4);
+}
+
+static void
+send_receive_gather_services (gpointer menu_item,
+ gpointer service,
+ gpointer queue)
+{
+ g_queue_push_head (queue, service);
+}
+
+static gint
+sort_services_cb (gconstpointer service1,
+ gconstpointer service2,
+ gpointer account_store)
+{
+ return e_mail_account_store_compare_services (account_store, CAMEL_SERVICE (service1), CAMEL_SERVICE (service2));
+}
+
+static void
+send_receive_menu_service_added_cb (EMailAccountStore *account_store,
+ CamelService *service,
+ SendReceiveData *data)
+{
+ GQueue *services;
+
+ if (!send_receive_can_use_service (account_store, service, NULL))
+ return;
+
+ services = g_queue_new ();
+
+ g_queue_push_head (services, service);
+ g_hash_table_foreach (data->menu_items, send_receive_gather_services, services);
+ g_queue_sort (services, sort_services_cb, account_store);
+
+ send_receive_add_to_menu (data, service, g_queue_index (services, service));
+
+ g_queue_free (services);
+}
+
+static void
+send_receive_menu_service_removed_cb (EMailAccountStore *account_store,
+ CamelService *service,
+ SendReceiveData *data)
+{
+ GtkMenuItem *menu_item;
+
+ menu_item = send_receive_find_menu_item (data, service);
+ if (menu_item == NULL)
+ return;
+
+ g_hash_table_remove (data->menu_items, menu_item);
+
+ gtk_container_remove (
+ GTK_CONTAINER (data->menu),
+ GTK_WIDGET (menu_item));
+}
+
+static void
+send_receive_data_free (SendReceiveData *data)
+{
+ g_signal_handler_disconnect (data->account_store, data->service_added_id);
+ g_signal_handler_disconnect (data->account_store, data->service_removed_id);
+ g_signal_handler_disconnect (data->account_store, data->service_enabled_id);
+ g_signal_handler_disconnect (data->account_store, data->service_disabled_id);
+
+ g_object_unref (data->session);
+ g_object_unref (data->account_store);
+
+ g_hash_table_destroy (data->menu_items);
+
+ g_slice_free (SendReceiveData, data);
+}
+
+static SendReceiveData *
+send_receive_data_new (EMailShellView *mail_shell_view,
+ GtkWidget *menu)
+{
+ SendReceiveData *data;
+ EShellView *shell_view;
+ EShellBackend *shell_backend;
+ EMailAccountStore *account_store;
+ EMailBackend *backend;
+ EMailSession *session;
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ shell_backend = e_shell_view_get_shell_backend (shell_view);
+
+ backend = E_MAIL_BACKEND (shell_backend);
+ session = e_mail_backend_get_session (backend);
+ account_store = e_mail_ui_session_get_account_store (
+ E_MAIL_UI_SESSION (session));
+
+ data = g_slice_new0 (SendReceiveData);
+ data->menu = GTK_MENU_SHELL (menu); /* do not reference */
+ data->session = g_object_ref (session);
+ data->account_store = g_object_ref (account_store);
+
+ data->menu_items = g_hash_table_new_full (
+ (GHashFunc) g_direct_hash,
+ (GEqualFunc) g_direct_equal,
+ (GDestroyNotify) NULL,
+ (GDestroyNotify) g_object_unref);
+
+ data->service_added_id = g_signal_connect (
+ account_store, "service-added",
+ G_CALLBACK (send_receive_menu_service_added_cb), data);
+ data->service_removed_id = g_signal_connect (
+ account_store, "service-removed",
+ G_CALLBACK (send_receive_menu_service_removed_cb), data);
+ data->service_enabled_id = g_signal_connect (
+ account_store, "service-enabled",
+ G_CALLBACK (send_receive_menu_service_added_cb), data);
+ data->service_disabled_id = g_signal_connect (
+ account_store, "service-disabled",
+ G_CALLBACK (send_receive_menu_service_removed_cb), data);
+
+ g_object_weak_ref (
+ G_OBJECT (menu), (GWeakNotify)
+ send_receive_data_free, data);
+
+ return data;
+}
+
+static GtkWidget *
+create_send_receive_submenu (EMailShellView *mail_shell_view)
+{
+ EShellView *shell_view;
+ EShellWindow *shell_window;
+ EShellBackend *shell_backend;
+ EMailAccountStore *account_store;
+ EMailBackend *backend;
+ EMailSession *session;
+ GtkWidget *menu;
+ GtkAccelGroup *accel_group;
+ GtkUIManager *ui_manager;
+ GtkAction *action;
+ GtkTreeModel *model;
+ GtkTreeIter iter;
+ SendReceiveData *data;
+
+ g_return_val_if_fail (mail_shell_view != NULL, NULL);
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ shell_window = e_shell_view_get_shell_window (shell_view);
+ shell_backend = e_shell_view_get_shell_backend (shell_view);
+
+ backend = E_MAIL_BACKEND (shell_backend);
+ session = e_mail_backend_get_session (backend);
+ account_store = e_mail_ui_session_get_account_store (E_MAIL_UI_SESSION (session));
+
+ menu = gtk_menu_new ();
+ ui_manager = e_shell_window_get_ui_manager (shell_window);
+ accel_group = gtk_ui_manager_get_accel_group (ui_manager);
+
+ action = e_shell_window_get_action (shell_window, "mail-send-receive");
+ gtk_action_set_accel_group (action, accel_group);
+ gtk_menu_shell_append (
+ GTK_MENU_SHELL (menu),
+ gtk_action_create_menu_item (action));
+
+ action = e_shell_window_get_action (
+ shell_window, "mail-send-receive-receive-all");
+ gtk_action_set_accel_group (action, accel_group);
+ gtk_menu_shell_append (
+ GTK_MENU_SHELL (menu),
+ gtk_action_create_menu_item (action));
+
+ action = e_shell_window_get_action (
+ shell_window, "mail-send-receive-send-all");
+ gtk_action_set_accel_group (action, accel_group);
+ gtk_menu_shell_append (
+ GTK_MENU_SHELL (menu),
+ gtk_action_create_menu_item (action));
+
+ gtk_menu_shell_append (
+ GTK_MENU_SHELL (menu),
+ gtk_separator_menu_item_new ());
+
+ data = send_receive_data_new (mail_shell_view, menu);
+
+ model = GTK_TREE_MODEL (account_store);
+ if (gtk_tree_model_get_iter_first (model, &iter)) {
+ CamelService *service;
+
+ do {
+ service = NULL;
+
+ gtk_tree_model_get (
+ model, &iter,
+ E_MAIL_ACCOUNT_STORE_COLUMN_SERVICE, &service,
+ -1);
+
+ if (send_receive_can_use_service (account_store, service, &iter))
+ send_receive_add_to_menu (data, service, -1);
+
+ if (service)
+ g_object_unref (service);
+ } while (gtk_tree_model_iter_next (model, &iter));
+ }
+
+ gtk_widget_show_all (menu);
+
+ return menu;
+}
+
+void
+e_mail_shell_view_update_send_receive_menus (EMailShellView *mail_shell_view)
+{
+ EMailShellViewPrivate *priv;
+ EShellWindow *shell_window;
+ EShellView *shell_view;
+ GtkWidget *widget;
+ const gchar *widget_path;
+
+ g_return_if_fail (E_IS_MAIL_SHELL_VIEW (mail_shell_view));
+
+ priv = E_MAIL_SHELL_VIEW_GET_PRIVATE (mail_shell_view);
+
+ shell_view = E_SHELL_VIEW (mail_shell_view);
+ shell_window = e_shell_view_get_shell_window (shell_view);
+
+ if (!e_shell_view_is_active (shell_view)) {
+ if (priv->send_receive_tool_item) {
+ GtkWidget *toolbar;
+
+ toolbar = e_shell_window_get_managed_widget (
+ shell_window, "/main-toolbar");
+ g_return_if_fail (toolbar != NULL);
+
+ gtk_container_remove (
+ GTK_CONTAINER (toolbar),
+ GTK_WIDGET (priv->send_receive_tool_item));
+ gtk_container_remove (
+ GTK_CONTAINER (toolbar),
+ GTK_WIDGET (priv->send_receive_tool_separator));
+
+ priv->send_receive_tool_item = NULL;
+ priv->send_receive_tool_separator = NULL;
+ }
+
+ return;
+ }
+
+ widget_path =
+ "/main-menu/file-menu"
+ "/mail-send-receiver/mail-send-receive-submenu";
+ widget = e_shell_window_get_managed_widget (shell_window, widget_path);
+ if (widget != NULL)
+ gtk_menu_item_set_submenu (
+ GTK_MENU_ITEM (widget),
+ create_send_receive_submenu (mail_shell_view));
+
+ if (!priv->send_receive_tool_item) {
+ GtkWidget *toolbar;
+ GtkToolItem *tool_item;
+ gint index;
+
+ toolbar = e_shell_window_get_managed_widget (
+ shell_window, "/main-toolbar");
+ g_return_if_fail (toolbar != NULL);
+
+ widget_path =
+ "/main-toolbar/toolbar-actions/mail-send-receiver";
+ widget = e_shell_window_get_managed_widget (
+ shell_window, widget_path);
+ g_return_if_fail (widget != NULL);
+
+ index = gtk_toolbar_get_item_index (
+ GTK_TOOLBAR (toolbar), GTK_TOOL_ITEM (widget));
+
+ tool_item = gtk_separator_tool_item_new ();
+ gtk_toolbar_insert (GTK_TOOLBAR (toolbar), tool_item, index);
+ gtk_widget_show (GTK_WIDGET (tool_item));
+ priv->send_receive_tool_separator = tool_item;
+
+ tool_item = GTK_TOOL_ITEM (
+ e_menu_tool_button_new (_("Send / Receive")));
+ gtk_tool_item_set_is_important (tool_item, TRUE);
+ gtk_toolbar_insert (GTK_TOOLBAR (toolbar), tool_item, index);
+ gtk_widget_show (GTK_WIDGET (tool_item));
+ priv->send_receive_tool_item = tool_item;
+
+ g_object_bind_property (
+ ACTION (MAIL_SEND_RECEIVE), "sensitive",
+ tool_item, "sensitive",
+ G_BINDING_SYNC_CREATE);
+ }
+
+ if (priv->send_receive_tool_item)
+ gtk_menu_tool_button_set_menu (
+ GTK_MENU_TOOL_BUTTON (priv->send_receive_tool_item),
+ create_send_receive_submenu (mail_shell_view));
+}