/* * e-mail-reader.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 * * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * */ #ifdef HAVE_CONFIG_H #include #endif #include "e-mail-reader.h" #include #include #ifdef HAVE_XFREE #include #endif #include "shell/e-shell-utils.h" #include "libemail-engine/e-mail-enumtypes.h" #include "libemail-engine/e-mail-utils.h" #include "libemail-engine/mail-mt.h" #include "libemail-engine/mail-ops.h" #include "em-format/e-mail-formatter.h" #include "em-format/e-mail-parser.h" #include "em-format/e-mail-part-utils.h" #include "e-mail-backend.h" #include "e-mail-browser.h" #include "e-mail-reader-utils.h" #include "e-mail-ui-session.h" #include "e-mail-view.h" #include "em-composer-utils.h" #include "em-event.h" #include "em-folder-selector.h" #include "em-folder-tree.h" #include "em-utils.h" #include "mail-autofilter.h" #include "mail-vfolder-ui.h" #include "message-list.h" #define E_MAIL_READER_GET_PRIVATE(obj) \ ((EMailReaderPrivate *) g_object_get_qdata \ (G_OBJECT (obj), quark_private)) #define d(x) typedef struct _EMailReaderClosure EMailReaderClosure; typedef struct _EMailReaderPrivate EMailReaderPrivate; struct _EMailReaderClosure { EMailReader *reader; EActivity *activity; gchar *message_uid; }; struct _EMailReaderPrivate { EMailForwardStyle forward_style; EMailReplyStyle reply_style; /* This timer runs when the user selects a single message. */ guint message_selected_timeout_id; /* This allows message retrieval to be cancelled if another * message is selected before the retrieval has completed. */ GCancellable *retrieving_message; /* These flags work together to prevent message selection * restoration after a folder switch from automatically * marking the message as read. We only want that to * happen when the -user- selects a message. */ guint folder_was_just_selected : 1; guint restoring_message_selection : 1; guint avoid_next_mark_as_seen : 1; guint group_by_threads : 1; }; enum { CHANGED, COMPOSER_CREATED, FOLDER_LOADED, MESSAGE_LOADED, MESSAGE_SEEN, SHOW_SEARCH_BAR, UPDATE_ACTIONS, LAST_SIGNAL }; /* Remembers the previously selected folder when transferring messages. */ static gchar *default_xfer_messages_uri; static GQuark quark_private; static guint signals[LAST_SIGNAL]; G_DEFINE_INTERFACE (EMailReader, e_mail_reader, G_TYPE_INITIALLY_UNOWNED) static void mail_reader_set_display_formatter_for_message (EMailReader *reader, EMailDisplay *display, const gchar *message_uid, CamelMimeMessage *message, CamelFolder *folder); static void mail_reader_closure_free (EMailReaderClosure *closure) { if (closure->reader != NULL) g_object_unref (closure->reader); if (closure->activity != NULL) g_object_unref (closure->activity); g_free (closure->message_uid); g_slice_free (EMailReaderClosure, closure); } static void mail_reader_private_free (EMailReaderPrivate *priv) { if (priv->message_selected_timeout_id > 0) g_source_remove (priv->message_selected_timeout_id); if (priv->retrieving_message != NULL) { g_cancellable_cancel (priv->retrieving_message); g_object_unref (priv->retrieving_message); priv->retrieving_message = 0; } g_slice_free (EMailReaderPrivate, priv); } static void action_mail_add_sender_cb (GtkAction *action, EMailReader *reader) { EShell *shell; EMailBackend *backend; EMailSession *session; EShellBackend *shell_backend; CamelInternetAddress *cia; CamelMessageInfo *info = NULL; CamelFolder *folder; GPtrArray *uids; const gchar *address; const gchar *message_uid; folder = e_mail_reader_ref_folder (reader); backend = e_mail_reader_get_backend (reader); session = e_mail_backend_get_session (backend); uids = e_mail_reader_get_selected_uids (reader); g_return_if_fail (uids != NULL && uids->len == 1); message_uid = g_ptr_array_index (uids, 0); info = camel_folder_get_message_info (folder, message_uid); if (info == NULL) goto exit; address = camel_message_info_from (info); if (address == NULL || *address == '\0') goto exit; /* XXX EBookShellBackend should be listening for this * event. Kind of kludgey, but works for now. */ shell_backend = E_SHELL_BACKEND (backend); shell = e_shell_backend_get_shell (shell_backend); e_shell_event (shell, "contact-quick-add-email", (gpointer) address); /* Remove this address from the photo cache. */ cia = camel_internet_address_new (); if (camel_address_decode (CAMEL_ADDRESS (cia), address) > 0) { EPhotoCache *photo_cache; const gchar *address_only = NULL; photo_cache = e_mail_ui_session_get_photo_cache ( E_MAIL_UI_SESSION (session)); camel_internet_address_get (cia, 0, NULL, &address_only); e_photo_cache_remove_photo (photo_cache, address_only); } g_object_unref (cia); exit: if (info != NULL) camel_message_info_unref (info); g_ptr_array_unref (uids); g_clear_object (&folder); } static void action_add_to_address_book_cb (GtkAction *action, EMailReader *reader) { EShell *shell; EMailBackend *backend; EMailDisplay *display; EMailSession *session; EShellBackend *shell_backend; CamelInternetAddress *cia; EPhotoCache *photo_cache; EWebView *web_view; CamelURL *curl; const gchar *uri; const gchar *address_only = NULL; gchar *email; /* This action is defined in EMailDisplay. */ backend = e_mail_reader_get_backend (reader); session = e_mail_backend_get_session (backend); display = e_mail_reader_get_mail_display (reader); if (display == NULL) return; web_view = E_WEB_VIEW (display); uri = e_web_view_get_selected_uri (web_view); g_return_if_fail (uri != NULL); curl = camel_url_new (uri, NULL); g_return_if_fail (curl != NULL); if (curl->path == NULL || *curl->path == '\0') goto exit; cia = camel_internet_address_new (); if (camel_address_decode (CAMEL_ADDRESS (cia), curl->path) < 0) { g_object_unref (cia); goto exit; } /* XXX EBookShellBackend should be listening for this * event. Kind of kludgey, but works for now. */ shell_backend = E_SHELL_BACKEND (backend); shell = e_shell_backend_get_shell (shell_backend); email = camel_address_format (CAMEL_ADDRESS (cia)); e_shell_event (shell, "contact-quick-add-email", email); g_free (email); /* Remove this address from the photo cache. */ photo_cache = e_mail_ui_session_get_photo_cache ( E_MAIL_UI_SESSION (session)); camel_internet_address_get (cia, 0, NULL, &address_only); e_photo_cache_remove_photo (photo_cache, address_only); g_object_unref (cia); exit: camel_url_free (curl); } static void action_mail_charset_cb (GtkRadioAction *action, GtkRadioAction *current, EMailReader *reader) { EMailDisplay *display; EMailFormatter *formatter; if (action != current) return; display = e_mail_reader_get_mail_display (reader); formatter = e_mail_display_get_formatter (display); if (formatter != NULL) { const gchar *charset; /* Charset for "Default" action will be NULL. */ charset = g_object_get_data (G_OBJECT (action), "charset"); e_mail_formatter_set_charset (formatter, charset); } } static void action_mail_check_for_junk_cb (GtkAction *action, EMailReader *reader) { EMailBackend *backend; EMailSession *session; CamelFolder *folder; GPtrArray *uids; folder = e_mail_reader_ref_folder (reader); backend = e_mail_reader_get_backend (reader); uids = e_mail_reader_get_selected_uids (reader); session = e_mail_backend_get_session (backend); mail_filter_folder ( session, folder, uids, E_FILTER_SOURCE_JUNKTEST, FALSE); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_copy_cb (GtkAction *action, EMailReader *reader) { CamelFolder *folder; EMailBackend *backend; EMailSession *session; EMFolderSelector *selector; EMFolderTree *folder_tree; EMFolderTreeModel *model; GtkWidget *dialog; GtkWindow *window; GPtrArray *uids; const gchar *uri; backend = e_mail_reader_get_backend (reader); session = e_mail_backend_get_session (backend); folder = e_mail_reader_ref_folder (reader); window = e_mail_reader_get_window (reader); uids = e_mail_reader_get_selected_uids (reader); model = em_folder_tree_model_get_default (); dialog = em_folder_selector_new ( window, model, EM_FOLDER_SELECTOR_CAN_CREATE, _("Copy to Folder"), NULL, _("C_opy")); selector = EM_FOLDER_SELECTOR (dialog); folder_tree = em_folder_selector_get_folder_tree (selector); em_folder_tree_set_excluded ( folder_tree, EMFT_EXCLUDE_NOSELECT | EMFT_EXCLUDE_VIRTUAL | EMFT_EXCLUDE_VTRASH); if (default_xfer_messages_uri != NULL) em_folder_tree_set_selected ( folder_tree, default_xfer_messages_uri, FALSE); if (gtk_dialog_run (GTK_DIALOG (dialog)) != GTK_RESPONSE_OK) goto exit; uri = em_folder_selector_get_selected_uri (selector); g_free (default_xfer_messages_uri); default_xfer_messages_uri = g_strdup (uri); if (uri != NULL) mail_transfer_messages ( session, folder, uids, FALSE, uri, 0, NULL, NULL); exit: gtk_widget_destroy (dialog); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_delete_cb (GtkAction *action, EMailReader *reader) { guint32 mask = CAMEL_MESSAGE_SEEN | CAMEL_MESSAGE_DELETED; guint32 set = CAMEL_MESSAGE_SEEN | CAMEL_MESSAGE_DELETED; if (!e_mail_reader_confirm_delete (reader)) return; /* FIXME Verify all selected messages are deletable. * But handle it by disabling this action. */ if (e_mail_reader_mark_selected (reader, mask, set) == 1) e_mail_reader_select_next_message (reader, FALSE); } static void action_mail_filter_on_mailing_list_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_filter_from_selected (reader, AUTO_MLIST); } static void action_mail_filter_on_recipients_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_filter_from_selected (reader, AUTO_TO); } static void action_mail_filter_on_sender_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_filter_from_selected (reader, AUTO_FROM); } static void action_mail_filter_on_subject_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_filter_from_selected (reader, AUTO_SUBJECT); } static void action_mail_filters_apply_cb (GtkAction *action, EMailReader *reader) { EMailBackend *backend; EMailSession *session; CamelFolder *folder; GPtrArray *uids; folder = e_mail_reader_ref_folder (reader); backend = e_mail_reader_get_backend (reader); uids = e_mail_reader_get_selected_uids (reader); session = e_mail_backend_get_session (backend); mail_filter_folder ( session, folder, uids, E_FILTER_SOURCE_DEMAND, FALSE); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_remove_attachments_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_remove_attachments (reader); } static void action_mail_remove_duplicates_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_remove_duplicates (reader); } static void action_mail_find_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_show_search_bar (reader); } static void action_mail_flag_clear_cb (GtkAction *action, EMailReader *reader) { EMailDisplay *display; CamelFolder *folder; GtkWindow *window; GPtrArray *uids; folder = e_mail_reader_ref_folder (reader); display = e_mail_reader_get_mail_display (reader); uids = e_mail_reader_get_selected_uids (reader); window = e_mail_reader_get_window (reader); em_utils_flag_for_followup_clear (window, folder, uids); e_mail_display_reload (display); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_flag_completed_cb (GtkAction *action, EMailReader *reader) { EMailDisplay *display; CamelFolder *folder; GtkWindow *window; GPtrArray *uids; folder = e_mail_reader_ref_folder (reader); display = e_mail_reader_get_mail_display (reader); uids = e_mail_reader_get_selected_uids (reader); window = e_mail_reader_get_window (reader); em_utils_flag_for_followup_completed (window, folder, uids); e_mail_display_reload (display); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_flag_for_followup_cb (GtkAction *action, EMailReader *reader) { CamelFolder *folder; GPtrArray *uids; folder = e_mail_reader_ref_folder (reader); uids = e_mail_reader_get_selected_uids (reader); em_utils_flag_for_followup (reader, folder, uids); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_forward_cb (GtkAction *action, EMailReader *reader) { GtkWindow *window; GPtrArray *uids; window = e_mail_reader_get_window (reader); uids = e_mail_reader_get_selected_uids (reader); g_return_if_fail (uids != NULL); if (em_utils_ask_open_many (window, uids->len)) { CamelFolder *folder; folder = e_mail_reader_ref_folder (reader); e_mail_reader_forward_messages ( reader, folder, uids, e_mail_reader_get_forward_style (reader)); g_clear_object (&folder); } g_ptr_array_unref (uids); } static void action_mail_forward_attached_cb (GtkAction *action, EMailReader *reader) { GtkWindow *window; GPtrArray *uids; window = e_mail_reader_get_window (reader); uids = e_mail_reader_get_selected_uids (reader); g_return_if_fail (uids != NULL); if (em_utils_ask_open_many (window, uids->len)) { CamelFolder *folder; folder = e_mail_reader_ref_folder (reader); e_mail_reader_forward_messages ( reader, folder, uids, E_MAIL_FORWARD_STYLE_ATTACHED); g_clear_object (&folder); } g_ptr_array_unref (uids); } static void action_mail_forward_inline_cb (GtkAction *action, EMailReader *reader) { GtkWindow *window; GPtrArray *uids; window = e_mail_reader_get_window (reader); uids = e_mail_reader_get_selected_uids (reader); g_return_if_fail (uids != NULL); if (em_utils_ask_open_many (window, uids->len)) { CamelFolder *folder; folder = e_mail_reader_ref_folder (reader); e_mail_reader_forward_messages ( reader, folder, uids, E_MAIL_FORWARD_STYLE_INLINE); g_clear_object (&folder); } g_ptr_array_unref (uids); } static void action_mail_forward_quoted_cb (GtkAction *action, EMailReader *reader) { GtkWindow *window; GPtrArray *uids; window = e_mail_reader_get_window (reader); uids = e_mail_reader_get_selected_uids (reader); g_return_if_fail (uids != NULL); if (em_utils_ask_open_many (window, uids->len)) { CamelFolder *folder; folder = e_mail_reader_ref_folder (reader); e_mail_reader_forward_messages ( reader, folder, uids, E_MAIL_FORWARD_STYLE_QUOTED); g_clear_object (&folder); } g_ptr_array_unref (uids); } static void action_mail_load_images_cb (GtkAction *action, EMailReader *reader) { EMailDisplay *display; display = e_mail_reader_get_mail_display (reader); e_mail_display_load_images (display); } static void action_mail_mark_important_cb (GtkAction *action, EMailReader *reader) { guint32 mask = CAMEL_MESSAGE_FLAGGED | CAMEL_MESSAGE_DELETED; guint32 set = CAMEL_MESSAGE_FLAGGED; e_mail_reader_mark_selected (reader, mask, set); } static void action_mail_mark_junk_cb (GtkAction *action, EMailReader *reader) { guint32 mask = CAMEL_MESSAGE_SEEN | CAMEL_MESSAGE_JUNK | CAMEL_MESSAGE_NOTJUNK | CAMEL_MESSAGE_JUNK_LEARN; guint32 set = CAMEL_MESSAGE_SEEN | CAMEL_MESSAGE_JUNK | CAMEL_MESSAGE_JUNK_LEARN; if (e_mail_reader_mark_selected (reader, mask, set) == 1) { CamelFolder *folder; gboolean select_next_message; folder = e_mail_reader_ref_folder (reader); select_next_message = (folder != NULL) && (folder->folder_flags & CAMEL_FOLDER_IS_JUNK); if (select_next_message) e_mail_reader_select_next_message (reader, TRUE); g_clear_object (&folder); } } static void action_mail_mark_notjunk_cb (GtkAction *action, EMailReader *reader) { guint32 mask = CAMEL_MESSAGE_JUNK | CAMEL_MESSAGE_NOTJUNK | CAMEL_MESSAGE_JUNK_LEARN; guint32 set = CAMEL_MESSAGE_NOTJUNK | CAMEL_MESSAGE_JUNK_LEARN; if (e_mail_reader_mark_selected (reader, mask, set) == 1) { CamelFolder *folder; gboolean select_next_message; folder = e_mail_reader_ref_folder (reader); select_next_message = (folder != NULL) && (folder->folder_flags & CAMEL_FOLDER_IS_JUNK); if (select_next_message) e_mail_reader_select_next_message (reader, TRUE); } } static void action_mail_mark_read_cb (GtkAction *action, EMailReader *reader) { guint32 mask = CAMEL_MESSAGE_SEEN; guint32 set = CAMEL_MESSAGE_SEEN; e_mail_reader_mark_selected (reader, mask, set); } static void action_mail_mark_unimportant_cb (GtkAction *action, EMailReader *reader) { guint32 mask = CAMEL_MESSAGE_FLAGGED; guint32 set = 0; e_mail_reader_mark_selected (reader, mask, set); } static void action_mail_mark_unread_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; EMFolderTreeModel *model; CamelFolder *folder; guint32 mask = CAMEL_MESSAGE_SEEN | CAMEL_MESSAGE_DELETED; guint32 set = 0; guint n_marked; message_list = e_mail_reader_get_message_list (reader); n_marked = e_mail_reader_mark_selected (reader, mask, set); if (MESSAGE_LIST (message_list)->seen_id != 0) { g_source_remove (MESSAGE_LIST (message_list)->seen_id); MESSAGE_LIST (message_list)->seen_id = 0; } folder = e_mail_reader_ref_folder (reader); /* Notify the tree model that the user has marked messages as * unread so it doesn't mistake the event as new mail arriving. */ model = em_folder_tree_model_get_default (); em_folder_tree_model_user_marked_unread (model, folder, n_marked); g_clear_object (&folder); } static void action_mail_message_edit_cb (GtkAction *action, EMailReader *reader) { EShell *shell; EMailBackend *backend; ESourceRegistry *registry; CamelFolder *folder; GPtrArray *uids; gboolean replace; uids = e_mail_reader_get_selected_uids (reader); g_return_if_fail (uids != NULL); backend = e_mail_reader_get_backend (reader); shell = e_shell_backend_get_shell (E_SHELL_BACKEND (backend)); registry = e_shell_get_registry (shell); folder = e_mail_reader_ref_folder (reader); replace = em_utils_folder_is_drafts (registry, folder); e_mail_reader_edit_messages (reader, folder, uids, replace, replace); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_message_new_cb (GtkAction *action, EMailReader *reader) { EShell *shell; EMailBackend *backend; EShellBackend *shell_backend; CamelFolder *folder; EMsgComposer *composer; folder = e_mail_reader_ref_folder (reader); backend = e_mail_reader_get_backend (reader); shell_backend = E_SHELL_BACKEND (backend); shell = e_shell_backend_get_shell (shell_backend); composer = em_utils_compose_new_message (shell, folder); e_mail_reader_composer_created (reader, composer, NULL); g_clear_object (&folder); } static void action_mail_message_open_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_open_selected_mail (reader); } static void action_mail_move_cb (GtkAction *action, EMailReader *reader) { CamelFolder *folder; EMailBackend *backend; EMailSession *session; EMFolderSelector *selector; EMFolderTree *folder_tree; EMFolderTreeModel *model; GtkWidget *dialog; GtkWindow *window; GPtrArray *uids; const gchar *uri; backend = e_mail_reader_get_backend (reader); session = e_mail_backend_get_session (backend); folder = e_mail_reader_ref_folder (reader); uids = e_mail_reader_get_selected_uids (reader); window = e_mail_reader_get_window (reader); model = em_folder_tree_model_get_default (); dialog = em_folder_selector_new ( window, model, EM_FOLDER_SELECTOR_CAN_CREATE, _("Move to Folder"), NULL, _("_Move")); selector = EM_FOLDER_SELECTOR (dialog); folder_tree = em_folder_selector_get_folder_tree (selector); em_folder_tree_set_excluded ( folder_tree, EMFT_EXCLUDE_NOSELECT | EMFT_EXCLUDE_VIRTUAL | EMFT_EXCLUDE_VTRASH); if (default_xfer_messages_uri != NULL) em_folder_tree_set_selected ( folder_tree, default_xfer_messages_uri, FALSE); if (gtk_dialog_run (GTK_DIALOG (dialog)) != GTK_RESPONSE_OK) goto exit; uri = em_folder_selector_get_selected_uri (selector); g_free (default_xfer_messages_uri); default_xfer_messages_uri = g_strdup (uri); if (uri != NULL) { mail_transfer_messages ( session, folder, uids, TRUE, uri, 0, NULL, NULL); uids = NULL; } exit: if (uids != NULL) g_ptr_array_unref (uids); gtk_widget_destroy (dialog); g_clear_object (&folder); } static void action_mail_next_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; MessageListSelectDirection direction; guint32 flags, mask; direction = MESSAGE_LIST_SELECT_NEXT; flags = 0; mask = 0; message_list = e_mail_reader_get_message_list (reader); message_list_select ( MESSAGE_LIST (message_list), direction, flags, mask); } static void action_mail_next_important_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; MessageListSelectDirection direction; guint32 flags, mask; direction = MESSAGE_LIST_SELECT_NEXT | MESSAGE_LIST_SELECT_WRAP; flags = CAMEL_MESSAGE_FLAGGED; mask = CAMEL_MESSAGE_FLAGGED; message_list = e_mail_reader_get_message_list (reader); message_list_select ( MESSAGE_LIST (message_list), direction, flags, mask); } static void action_mail_next_thread_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; message_list = e_mail_reader_get_message_list (reader); message_list_select_next_thread (MESSAGE_LIST (message_list)); } static void action_mail_next_unread_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; MessageListSelectDirection direction; guint32 flags, mask; direction = MESSAGE_LIST_SELECT_NEXT | MESSAGE_LIST_SELECT_WRAP; flags = 0; mask = CAMEL_MESSAGE_SEEN; message_list = e_mail_reader_get_message_list (reader); message_list_select ( MESSAGE_LIST (message_list), direction, flags, mask); } static void action_mail_previous_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; MessageListSelectDirection direction; guint32 flags, mask; direction = MESSAGE_LIST_SELECT_PREVIOUS; flags = 0; mask = 0; message_list = e_mail_reader_get_message_list (reader); message_list_select ( MESSAGE_LIST (message_list), direction, flags, mask); } static void action_mail_previous_important_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; MessageListSelectDirection direction; guint32 flags, mask; direction = MESSAGE_LIST_SELECT_PREVIOUS | MESSAGE_LIST_SELECT_WRAP; flags = CAMEL_MESSAGE_FLAGGED; mask = CAMEL_MESSAGE_FLAGGED; message_list = e_mail_reader_get_message_list (reader); message_list_select ( MESSAGE_LIST (message_list), direction, flags, mask); } static void action_mail_previous_thread_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; message_list = e_mail_reader_get_message_list (reader); message_list_select_prev_thread (MESSAGE_LIST (message_list)); } static void action_mail_previous_unread_cb (GtkAction *action, EMailReader *reader) { GtkWidget *message_list; MessageListSelectDirection direction; guint32 flags, mask; direction = MESSAGE_LIST_SELECT_PREVIOUS | MESSAGE_LIST_SELECT_WRAP; flags = 0; mask = CAMEL_MESSAGE_SEEN; message_list = e_mail_reader_get_message_list (reader); message_list_select ( MESSAGE_LIST (message_list), direction, flags, mask); } static void action_mail_print_cb (GtkAction *action, EMailReader *reader) { GtkPrintOperationAction print_action; print_action = GTK_PRINT_OPERATION_ACTION_PRINT_DIALOG; e_mail_reader_print (reader, print_action); } static void action_mail_print_preview_cb (GtkAction *action, EMailReader *reader) { GtkPrintOperationAction print_action; print_action = GTK_PRINT_OPERATION_ACTION_PREVIEW; e_mail_reader_print (reader, print_action); } static void mail_reader_redirect_cb (CamelFolder *folder, GAsyncResult *result, EMailReaderClosure *closure) { EShell *shell; EMailBackend *backend; EAlertSink *alert_sink; CamelMimeMessage *message; EMsgComposer *composer; GError *error = NULL; alert_sink = e_activity_get_alert_sink (closure->activity); message = camel_folder_get_message_finish (folder, result, &error); if (e_activity_handle_cancellation (closure->activity, error)) { g_warn_if_fail (message == NULL); mail_reader_closure_free (closure); g_error_free (error); return; } else if (error != NULL) { g_warn_if_fail (message == NULL); e_alert_submit ( alert_sink, "mail:no-retrieve-message", error->message, NULL); mail_reader_closure_free (closure); g_error_free (error); return; } g_return_if_fail (CAMEL_IS_MIME_MESSAGE (message)); backend = e_mail_reader_get_backend (closure->reader); shell = e_shell_backend_get_shell (E_SHELL_BACKEND (backend)); composer = em_utils_redirect_message (shell, message); e_mail_reader_composer_created (closure->reader, composer, message); g_object_unref (message); mail_reader_closure_free (closure); } static void action_mail_redirect_cb (GtkAction *action, EMailReader *reader) { EActivity *activity; GCancellable *cancellable; EMailReaderClosure *closure; GtkWidget *message_list; CamelFolder *folder; const gchar *message_uid; message_list = e_mail_reader_get_message_list (reader); message_uid = MESSAGE_LIST (message_list)->cursor_uid; g_return_if_fail (message_uid != NULL); /* Open the message asynchronously. */ activity = e_mail_reader_new_activity (reader); cancellable = e_activity_get_cancellable (activity); closure = g_slice_new0 (EMailReaderClosure); closure->activity = activity; closure->reader = g_object_ref (reader); folder = e_mail_reader_ref_folder (reader); camel_folder_get_message ( folder, message_uid, G_PRIORITY_DEFAULT, cancellable, (GAsyncReadyCallback) mail_reader_redirect_cb, closure); g_clear_object (&folder); } static void action_mail_reply_all_check (CamelFolder *folder, GAsyncResult *result, EMailReaderClosure *closure) { EAlertSink *alert_sink; CamelMimeMessage *message; CamelInternetAddress *to, *cc; gint recip_count = 0; EMailReplyType type = E_MAIL_REPLY_TO_ALL; GError *error = NULL; alert_sink = e_activity_get_alert_sink (closure->activity); message = camel_folder_get_message_finish (folder, result, &error); if (e_activity_handle_cancellation (closure->activity, error)) { g_warn_if_fail (message == NULL); mail_reader_closure_free (closure); g_error_free (error); return; } else if (error != NULL) { g_warn_if_fail (message == NULL); e_alert_submit ( alert_sink, "mail:no-retrieve-message", error->message, NULL); mail_reader_closure_free (closure); g_error_free (error); return; } g_return_if_fail (CAMEL_IS_MIME_MESSAGE (message)); to = camel_mime_message_get_recipients ( message, CAMEL_RECIPIENT_TYPE_TO); cc = camel_mime_message_get_recipients ( message, CAMEL_RECIPIENT_TYPE_CC); recip_count = camel_address_length (CAMEL_ADDRESS (to)); recip_count += camel_address_length (CAMEL_ADDRESS (cc)); if (recip_count >= 15) { GtkWidget *dialog; GtkWidget *check; GtkWidget *container; gint response; dialog = e_alert_dialog_new_for_args ( e_mail_reader_get_window (closure->reader), "mail:ask-reply-many-recips", NULL); container = e_alert_dialog_get_content_area ( E_ALERT_DIALOG (dialog)); /* Check buttons */ check = gtk_check_button_new_with_mnemonic ( _("_Do not ask me again.")); gtk_box_pack_start ( GTK_BOX (container), check, FALSE, FALSE, 0); gtk_widget_show (check); response = gtk_dialog_run (GTK_DIALOG (dialog)); if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (check))) { GSettings *settings; const gchar *key; settings = g_settings_new ("org.gnome.evolution.mail"); key = "prompt-on-reply-many-recips"; g_settings_set_boolean (settings, key, FALSE); g_object_unref (settings); } gtk_widget_destroy (dialog); switch (response) { case GTK_RESPONSE_NO: type = E_MAIL_REPLY_TO_SENDER; break; case GTK_RESPONSE_CANCEL: case GTK_RESPONSE_DELETE_EVENT: goto exit; default: break; } } e_mail_reader_reply_to_message (closure->reader, message, type); exit: g_object_unref (message); mail_reader_closure_free (closure); } static void action_mail_reply_all_cb (GtkAction *action, EMailReader *reader) { GSettings *settings; const gchar *key; guint32 state; gboolean ask; state = e_mail_reader_check_state (reader); settings = g_settings_new ("org.gnome.evolution.mail"); key = "prompt-on-reply-many-recips"; ask = g_settings_get_boolean (settings, key); g_object_unref (settings); if (ask && !(state & E_MAIL_READER_SELECTION_IS_MAILING_LIST)) { EActivity *activity; GCancellable *cancellable; EMailReaderClosure *closure; CamelFolder *folder; GtkWidget *message_list; const gchar *message_uid; message_list = e_mail_reader_get_message_list (reader); message_uid = MESSAGE_LIST (message_list)->cursor_uid; g_return_if_fail (message_uid != NULL); activity = e_mail_reader_new_activity (reader); cancellable = e_activity_get_cancellable (activity); closure = g_slice_new0 (EMailReaderClosure); closure->activity = activity; closure->reader = g_object_ref (reader); folder = e_mail_reader_ref_folder (reader); camel_folder_get_message ( folder, message_uid, G_PRIORITY_DEFAULT, cancellable, (GAsyncReadyCallback) action_mail_reply_all_check, closure); g_clear_object (&folder); return; } e_mail_reader_reply_to_message (reader, NULL, E_MAIL_REPLY_TO_ALL); } static void action_mail_reply_group_cb (GtkAction *action, EMailReader *reader) { GSettings *settings; gboolean reply_list; guint32 state; const gchar *key; state = e_mail_reader_check_state (reader); settings = g_settings_new ("org.gnome.evolution.mail"); key = "composer-group-reply-to-list"; reply_list = g_settings_get_boolean (settings, key); g_object_unref (settings); if (reply_list && (state & E_MAIL_READER_SELECTION_IS_MAILING_LIST)) { e_mail_reader_reply_to_message ( reader, NULL, E_MAIL_REPLY_TO_LIST); } else action_mail_reply_all_cb (action, reader); } static void action_mail_reply_list_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_reply_to_message (reader, NULL, E_MAIL_REPLY_TO_LIST); } static gboolean message_is_list_administrative (CamelMimeMessage *message) { const gchar *header; header = camel_medium_get_header ( CAMEL_MEDIUM (message), "X-List-Administrivia"); if (header == NULL) return FALSE; while (*header == ' ' || *header == '\t') header++; return g_ascii_strncasecmp (header, "yes", 3) == 0; } static void action_mail_reply_sender_check (CamelFolder *folder, GAsyncResult *result, EMailReaderClosure *closure) { EAlertSink *alert_sink; CamelMimeMessage *message; EMailReplyType type = E_MAIL_REPLY_TO_SENDER; GSettings *settings; gboolean ask_ignore_list_reply_to; gboolean ask_list_reply_to; gboolean munged_list_message; gboolean active; const gchar *key; GError *local_error = NULL; alert_sink = e_activity_get_alert_sink (closure->activity); message = camel_folder_get_message_finish ( folder, result, &local_error); /* Sanity check. */ g_return_if_fail ( ((message != NULL) && (local_error == NULL)) || ((message == NULL) && (local_error != NULL))); if (e_activity_handle_cancellation (closure->activity, local_error)) { mail_reader_closure_free (closure); g_error_free (local_error); return; } else if (local_error != NULL) { e_alert_submit ( alert_sink, "mail:no-retrieve-message", local_error->message, NULL); mail_reader_closure_free (closure); g_error_free (local_error); return; } settings = g_settings_new ("org.gnome.evolution.mail"); key = "composer-ignore-list-reply-to"; ask_ignore_list_reply_to = g_settings_get_boolean (settings, key); key = "prompt-on-list-reply-to"; ask_list_reply_to = g_settings_get_boolean (settings, key); munged_list_message = em_utils_is_munged_list_message (message); if (message_is_list_administrative (message)) { /* Do not ask for messages which are list administrative, * like list confirmation messages. */ } else if (ask_ignore_list_reply_to || !munged_list_message) { /* Don't do the "Are you sure you want to reply in private?" * pop-up if it's a Reply-To: munged list message... unless * we're ignoring munging. */ GtkWidget *dialog; GtkWidget *check; GtkWidget *container; gint response; dialog = e_alert_dialog_new_for_args ( e_mail_reader_get_window (closure->reader), "mail:ask-list-private-reply", NULL); container = e_alert_dialog_get_content_area ( E_ALERT_DIALOG (dialog)); /* Check buttons */ check = gtk_check_button_new_with_mnemonic ( _("_Do not ask me again.")); gtk_box_pack_start ( GTK_BOX (container), check, FALSE, FALSE, 0); gtk_widget_show (check); response = gtk_dialog_run (GTK_DIALOG (dialog)); active = gtk_toggle_button_get_active ( GTK_TOGGLE_BUTTON (check)); if (active) { key = "prompt-on-private-list-reply"; g_settings_set_boolean (settings, key, FALSE); } gtk_widget_destroy (dialog); if (response == GTK_RESPONSE_YES) type = E_MAIL_REPLY_TO_ALL; else if (response == GTK_RESPONSE_OK) type = E_MAIL_REPLY_TO_LIST; else if (response == GTK_RESPONSE_CANCEL || response == GTK_RESPONSE_DELETE_EVENT) { goto exit; } } else if (ask_list_reply_to) { GtkWidget *dialog; GtkWidget *container; GtkWidget *check_again; GtkWidget *check_always_ignore; gint response; dialog = e_alert_dialog_new_for_args ( e_mail_reader_get_window (closure->reader), "mail:ask-list-honour-reply-to", NULL); container = e_alert_dialog_get_content_area ( E_ALERT_DIALOG (dialog)); check_again = gtk_check_button_new_with_mnemonic ( _("_Do not ask me again.")); gtk_box_pack_start ( GTK_BOX (container), check_again, FALSE, FALSE, 0); gtk_widget_show (check_again); check_always_ignore = gtk_check_button_new_with_mnemonic ( _("_Always ignore Reply-To: for mailing lists.")); gtk_box_pack_start ( GTK_BOX (container), check_always_ignore, FALSE, FALSE, 0); gtk_widget_show (check_always_ignore); response = gtk_dialog_run (GTK_DIALOG (dialog)); active = gtk_toggle_button_get_active ( GTK_TOGGLE_BUTTON (check_again)); if (active) { key = "prompt-on-list-reply-to"; g_settings_set_boolean (settings, key, FALSE); } key = "composer-ignore-list-reply-to"; active = gtk_toggle_button_get_active ( GTK_TOGGLE_BUTTON (check_always_ignore)); g_settings_set_boolean (settings, key, active); gtk_widget_destroy (dialog); switch (response) { case GTK_RESPONSE_NO: type = E_MAIL_REPLY_TO_FROM; break; case GTK_RESPONSE_OK: type = E_MAIL_REPLY_TO_LIST; break; case GTK_RESPONSE_CANCEL: case GTK_RESPONSE_DELETE_EVENT: goto exit; default: break; } } e_mail_reader_reply_to_message (closure->reader, message, type); exit: g_object_unref (settings); g_object_unref (message); mail_reader_closure_free (closure); } static void action_mail_reply_sender_cb (GtkAction *action, EMailReader *reader) { GSettings *settings; gboolean ask_list_reply_to; gboolean ask_private_list_reply; gboolean ask; guint32 state; const gchar *key; state = e_mail_reader_check_state (reader); settings = g_settings_new ("org.gnome.evolution.mail"); key = "prompt-on-list-reply-to"; ask_list_reply_to = g_settings_get_boolean (settings, key); key = "prompt-on-private-list-reply"; ask_private_list_reply = g_settings_get_boolean (settings, key); g_object_unref (settings); ask = (ask_private_list_reply || ask_list_reply_to); if (ask && (state & E_MAIL_READER_SELECTION_IS_MAILING_LIST)) { EActivity *activity; GCancellable *cancellable; EMailReaderClosure *closure; CamelFolder *folder; GtkWidget *message_list; const gchar *message_uid; message_list = e_mail_reader_get_message_list (reader); message_uid = MESSAGE_LIST (message_list)->cursor_uid; g_return_if_fail (message_uid != NULL); activity = e_mail_reader_new_activity (reader); cancellable = e_activity_get_cancellable (activity); closure = g_slice_new0 (EMailReaderClosure); closure->activity = activity; closure->reader = g_object_ref (reader); folder = e_mail_reader_ref_folder (reader); camel_folder_get_message ( folder, message_uid, G_PRIORITY_DEFAULT, cancellable, (GAsyncReadyCallback) action_mail_reply_sender_check, closure); g_clear_object (&folder); return; } e_mail_reader_reply_to_message (reader, NULL, E_MAIL_REPLY_TO_SENDER); } static void action_mail_reply_recipient_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_reply_to_message (reader, NULL, E_MAIL_REPLY_TO_RECIPIENT); } static void action_mail_save_as_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_save_messages (reader); } static void action_mail_search_folder_from_mailing_list_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_vfolder_from_selected (reader, AUTO_MLIST); } static void action_mail_search_folder_from_recipients_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_vfolder_from_selected (reader, AUTO_TO); } static void action_mail_search_folder_from_sender_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_vfolder_from_selected (reader, AUTO_FROM); } static void action_mail_search_folder_from_subject_cb (GtkAction *action, EMailReader *reader) { e_mail_reader_create_vfolder_from_selected (reader, AUTO_SUBJECT); } static void action_mail_show_all_headers_cb (GtkToggleAction *action, EMailReader *reader) { EMailDisplay *display; EMailFormatterMode mode; display = e_mail_reader_get_mail_display (reader); /* Ignore action when viewing message source. */ mode = e_mail_display_get_mode (display); if (mode == E_MAIL_FORMATTER_MODE_SOURCE) return; if (mode == E_MAIL_FORMATTER_MODE_RAW) return; if (gtk_toggle_action_get_active (action)) mode = E_MAIL_FORMATTER_MODE_ALL_HEADERS; else mode = E_MAIL_FORMATTER_MODE_NORMAL; e_mail_display_set_mode (display, mode); } static void mail_source_retrieved (GObject *source_object, GAsyncResult *result, gpointer user_data) { EMailReaderClosure *closure; CamelMimeMessage *message; EMailDisplay *display; GError *error = NULL; closure = (EMailReaderClosure *) user_data; display = e_mail_reader_get_mail_display (closure->reader); message = camel_folder_get_message_finish ( CAMEL_FOLDER (source_object), result, &error); /* Sanity check. */ g_return_if_fail ( ((message != NULL) && (error == NULL)) || ((message == NULL) && (error != NULL))); if (message != NULL) { mail_reader_set_display_formatter_for_message ( closure->reader, display, closure->message_uid, message, CAMEL_FOLDER (source_object)); } else { gchar *status; status = g_strdup_printf ( "%s
%s", _("Failed to retrieve message:"), error->message); e_mail_display_set_status (display, status); g_free (status); g_error_free (error); } e_activity_set_state (closure->activity, E_ACTIVITY_COMPLETED); mail_reader_closure_free (closure); } static void action_mail_show_source_cb (GtkAction *action, EMailReader *reader) { EMailDisplay *display; EMailBackend *backend; GtkWidget *browser; CamelFolder *folder; GPtrArray *uids; const gchar *message_uid; gchar *string; EActivity *activity; GCancellable *cancellable; EMailReaderClosure *closure; backend = e_mail_reader_get_backend (reader); folder = e_mail_reader_ref_folder (reader); uids = e_mail_reader_get_selected_uids (reader); g_return_if_fail (uids != NULL && uids->len == 1); message_uid = g_ptr_array_index (uids, 0); browser = e_mail_browser_new ( backend, NULL, NULL, E_MAIL_FORMATTER_MODE_SOURCE); e_mail_reader_set_folder (E_MAIL_READER (browser), folder); e_mail_reader_set_message (E_MAIL_READER (browser), message_uid); display = e_mail_reader_get_mail_display (E_MAIL_READER (browser)); string = g_strdup_printf (_("Retrieving message '%s'"), message_uid); e_mail_display_set_part_list (display, NULL); e_mail_display_set_status (display, string); gtk_widget_show (browser); activity = e_mail_reader_new_activity (reader); e_activity_set_text (activity, string); cancellable = e_activity_get_cancellable (activity); g_free (string); closure = g_slice_new0 (EMailReaderClosure); closure->reader = g_object_ref (browser); closure->activity = g_object_ref (activity); closure->message_uid = g_strdup (message_uid); camel_folder_get_message ( folder, message_uid, G_PRIORITY_DEFAULT, cancellable, mail_source_retrieved, closure); g_object_unref (activity); g_ptr_array_unref (uids); g_clear_object (&folder); } static void action_mail_toggle_important_cb (GtkAction *action, EMailReader *reader) { CamelFolder *folder; GPtrArray *uids; guint ii; folder = e_mail_reader_ref_folder (reader); uids = e_mail_reader_get_selected_uids (reader); camel_folder_freeze (folder); for (ii = 0; ii < uids->len; ii++) { guint32 flags; flags = camel_folder_get_message_flags ( folder, uids->pdata[ii]); flags ^= CAMEL_MESSAGE_FLAGGED; if (flags & CAMEL_MESSAGE_FLAGGED) flags &= ~CAMEL_MESSAGE_DELETED; camel_folder_set_message_flags ( folder, uids->pdata[ii], CAMEL_MESSAGE_FLAGGED | CAMEL_MESSAGE_DELETED, flags); } camel_folder_thaw (folder); g_clear_object (&folder); g_ptr_array_unref (uids); } static void action_mail_undelete_cb (GtkAction *action, EMailReader *reader) { guint32 mask = CAMEL_MESSAGE_DELETED; guint32 set = 0; e_mail_reader_mark_selected (reader, mask, set); } static void action_mail_zoom_100_cb (GtkAction *action, EMailReader *reader) { EMailDisplay *display; display = e_mail_reader_get_mail_display (reader); webkit_web_view_set_zoom_level (WEBKIT_WEB_VIEW (display), 1.0); } static void action_mail_zoom_in_cb (GtkAction *action, EMailReader *reader) { EMailDisplay *display; display = e_mail_reader_get_mail_display (reader); webkit_web_view_zoom_in (WEBKIT_WEB_VIEW (display)); } static void action_mail_zoom_out_cb (GtkAction *action, EMailReader *reader) { EMailDisplay *display; display = e_mail_reader_get_mail_display (reader); webkit_web_view_zoom_out (WEBKIT_WEB_VIEW (display)); } static void action_search_folder_recipient_cb (GtkAction *action, EMailReader *reader) { EMailBackend *backend; EMailSession *session; EWebView *web_view; CamelURL *curl; const gchar *uri; /* This action is defined in EMailDisplay. */ web_view = E_WEB_VIEW (e_mail_reader_get_mail_display (reader)); uri = e_web_view_get_selected_uri (web_view); g_return_if_fail (uri != NULL); curl = camel_url_new (uri, NULL); g_return_if_fail (curl != NULL); backend = e_mail_reader_get_backend (reader); session = e_mail_backend_get_session (backend); if (curl->path != NULL && *curl->path != '\0') { CamelFolder *folder; CamelInternetAddress *inet_addr; folder = e_mail_reader_ref_folder (reader); inet_addr = camel_internet_address_new (); camel_address_decode (CAMEL_ADDRESS (inet_addr), curl->path); vfolder_gui_add_from_address ( session, inet_addr, AUTO_TO, folder); g_object_unref (inet_addr); g_clear_object (&folder); } camel_url_free (curl); } static void action_search_folder_sender_cb (GtkAction *action, EMailReader *reader) { EMailBackend *backend; EMailSession *session; EWebView *web_view; CamelURL *curl; const gchar *uri; /* This action is defined in EMailDisplay. */ web_view = E_WEB_VIEW (e_mail_reader_get_mail_display (reader)); uri = e_web_view_get_selected_uri (web_view); g_return_if_fail (uri != NULL); curl = camel_url_new (uri, NULL); g_return_if_fail (curl != NULL); backend = e_mail_reader_get_backend (reader); session = e_mail_backend_get_session (backend); if (curl->path != NULL && *curl->path != '\0') { CamelFolder *folder; CamelInternetAddress *inet_addr; folder = e_mail_reader_ref_folder (reader); inet_addr = camel_internet_address_new (); camel_address_decode (CAMEL_ADDRESS (inet_addr), curl->path); vfolder_gui_add_from_address ( session, inet_addr, AUTO_FROM, folder); g_object_unref (inet_addr); g_clear_object (&folder); } camel_url_free (curl); } static GtkActionEntry mail_reader_entries[] = { { "mail-add-sender", NULL, N_("A_dd Sender to Address Book"), NULL, N_("Add sender to address book"), G_CALLBACK (action_mail_add_sender_cb) }, { "mail-check-for-junk", "mail-mark-junk", N_("Check for _Junk"), NULL, N_("Filter the selected messages for junk status"), G_CALLBACK (action_mail_check_for_junk_cb) }, { "mail-copy", "mail-copy", N_("_Copy to Folder..."), "y", N_("Copy selected messages to another folder"), G_CALLBACK (action_mail_copy_cb) }, { "mail-delete", "user-trash", N_("_Delete Message"), "d", N_("Mark the selected messages for deletion"), G_CALLBACK (action_mail_delete_cb) }, { "mail-filter-rule-for-mailing-list", NULL, N_("Create a Filter Rule for Mailing _List..."), NULL, N_("Create a rule to filter messages to this mailing list"), G_CALLBACK (action_mail_filter_on_mailing_list_cb) }, { "mail-filter-rule-for-recipients", NULL, N_("Create a Filter Rule for _Recipients..."), NULL, N_("Create a rule to filter messages to these recipients"), G_CALLBACK (action_mail_filter_on_recipients_cb) }, { "mail-filter-rule-for-sender", NULL, N_("Create a Filter Rule for Se_nder..."), NULL, N_("Create a rule to filter messages from this sender"), G_CALLBACK (action_mail_filter_on_sender_cb) }, { "mail-filter-rule-for-subject", NULL, N_("Create a Filter Rule for _Subject..."), NULL, N_("Create a rule to filter messages with this subject"), G_CALLBACK (action_mail_filter_on_subject_cb) }, { "mail-filters-apply", "stock_mail-filters-apply", N_("A_pply Filters"), "y", N_("Apply filter rules to the selected messages"), G_CALLBACK (action_mail_filters_apply_cb) }, { "mail-find", GTK_STOCK_FIND, N_("_Find in Message..."), "f", N_("Search for text in the body of the displayed message"), G_CALLBACK (action_mail_find_cb) }, { "mail-flag-clear", NULL, N_("_Clear Flag"), NULL, N_("Remove the follow-up flag from the selected messages"), G_CALLBACK (action_mail_flag_clear_cb) }, { "mail-flag-completed", NULL, N_("_Flag Completed"), NULL, N_("Set the follow-up flag to completed on the selected messages"), G_CALLBACK (action_mail_flag_completed_cb) }, { "mail-flag-for-followup", "stock_mail-flag-for-followup", N_("Follow _Up..."), "g", N_("Flag the selected messages for follow-up"), G_CALLBACK (action_mail_flag_for_followup_cb) }, { "mail-forward-attached", NULL, N_("_Attached"), NULL, N_("Forward the selected message to someone as an attachment"), G_CALLBACK (action_mail_forward_attached_cb) }, { "mail-forward-attached-full", NULL, N_("Forward As _Attached"), NULL, N_("Forward the selected message to someone as an attachment"), G_CALLBACK (action_mail_forward_attached_cb) }, { "mail-forward-inline", NULL, N_("_Inline"), NULL, N_("Forward the selected message in the body of a new message"), G_CALLBACK (action_mail_forward_inline_cb) }, { "mail-forward-inline-full", NULL, N_("Forward As _Inline"), NULL, N_("Forward the selected message in the body of a new message"), G_CALLBACK (action_mail_forward_inline_cb) }, { "mail-forward-quoted", NULL, N_("_Quoted"), NULL, N_("Forward the selected message quoted like a reply"), G_CALLBACK (action_mail_forward_quoted_cb) }, { "mail-forward-quoted-full", NULL, N_("Forward As _Quoted"), NULL, N_("Forward the selected message quoted like a reply"), G_CALLBACK (action_mail_forward_quoted_cb) }, { "mail-load-images", "image-x-generic", N_("_Load Images"), "i", N_("Force images in HTML mail to be loaded"), G_CALLBACK (action_mail_load_images_cb) }, { "mail-mark-important", "mail-mark-important", N_("_Important"), NULL, N_("Mark the selected messages as important"), G_CALLBACK (action_mail_mark_important_cb) }, { "mail-mark-junk", "mail-mark-junk", N_("_Junk"), "j", N_("Mark the selected messages as junk"), G_CALLBACK (action_mail_mark_junk_cb) }, { "mail-mark-notjunk", "mail-mark-notjunk", N_("_Not Junk"), "j", N_("Mark the selected messages as not being junk"), G_CALLBACK (action_mail_mark_notjunk_cb) }, { "mail-mark-read", "mail-mark-read", N_("_Read"), "k", N_("Mark the selected messages as having been read"), G_CALLBACK (action_mail_mark_read_cb) }, { "mail-mark-unimportant", NULL, N_("Uni_mportant"), NULL, N_("Mark the selected messages as unimportant"), G_CALLBACK (action_mail_mark_unimportant_cb) }, { "mail-mark-unread", "mail-mark-unread", N_("_Unread"), "k", N_("Mark the selected messages as not having been read"), G_CALLBACK (action_mail_mark_unread_cb) }, { "mail-message-edit", NULL, N_("_Edit as New Message..."), NULL, N_("Open the selected messages in the composer for editing"), G_CALLBACK (action_mail_message_edit_cb) }, { "mail-message-new", "mail-message-new", N_("Compose _New Message"), "m", N_("Open a window for composing a mail message"), G_CALLBACK (action_mail_message_new_cb) }, { "mail-message-open", NULL, N_("_Open in New Window"), "o", N_("Open the selected messages in a new window"), G_CALLBACK (action_mail_message_open_cb) }, { "mail-move", "mail-move", N_("_Move to Folder..."), "v", N_("Move selected messages to another folder"), G_CALLBACK (action_mail_move_cb) }, { "mail-next", GTK_STOCK_GO_FORWARD, N_("_Next Message"), "Page_Down", N_("Display the next message"), G_CALLBACK (action_mail_next_cb) }, { "mail-next-important", NULL, N_("Next _Important Message"), NULL, N_("Display the next important message"), G_CALLBACK (action_mail_next_important_cb) }, { "mail-next-thread", NULL, N_("Next _Thread"), NULL, N_("Display the next thread"), G_CALLBACK (action_mail_next_thread_cb) }, { "mail-next-unread", NULL, N_("Next _Unread Message"), "bracketright", N_("Display the next unread message"), G_CALLBACK (action_mail_next_unread_cb) }, { "mail-previous", GTK_STOCK_GO_BACK, N_("_Previous Message"), "Page_Up", N_("Display the previous message"), G_CALLBACK (action_mail_previous_cb) }, { "mail-previous-important", NULL, N_("Pr_evious Important Message"), NULL, N_("Display the previous important message"), G_CALLBACK (action_mail_previous_important_cb) }, { "mail-previous-thread", NULL, N_("Previous T_hread"), NULL, N_("Display the previous thread"), G_CALLBACK (action_mail_previous_thread_cb) }, { "mail-previous-unread", NULL, N_("P_revious Unread Message"), "bracketleft", N_("Display the previous unread message"), G_CALLBACK (action_mail_previous_unread_cb) }, { "mail-print", GTK_STOCK_PRINT, NULL, "p", N_("Print this message"), G_CALLBACK (action_mail_print_cb) }, { "mail-print-preview", GTK_STOCK_PRINT_PREVIEW, NULL, NULL, N_("Preview the message to be printed"), G_CALLBACK (action_mail_print_preview_cb) }, { "mail-redirect", NULL, N_("Re_direct"), NULL, N_("Redirect (bounce) the selected message to someone"), G_CALLBACK (action_mail_redirect_cb) }, { "mail-remove-attachments", GTK_STOCK_DELETE, N_("Remo_ve Attachments"), NULL, N_("Remove attachments"), G_CALLBACK (action_mail_remove_attachments_cb) }, { "mail-remove-duplicates", NULL, N_("Remove Du_plicate Messages"), NULL, N_("Checks selected messages for duplicates"), G_CALLBACK (action_mail_remove_duplicates_cb) }, { "mail-reply-all", NULL, N_("Reply to _All"), "r", N_("Compose a reply to all the recipients of the selected message"), G_CALLBACK (action_mail_reply_all_cb) }, { "mail-reply-list", NULL, N_("Reply to _List"), "l", N_("Compose a reply to the mailing list of the selected message"), G_CALLBACK (action_mail_reply_list_cb) }, { "mail-reply-sender", "mail-reply-sender", N_("_Reply to Sender"), "r", N_("Compose a reply to the sender of the selected message"), G_CALLBACK (action_mail_reply_sender_cb) }, { "mail-save-as", GTK_STOCK_SAVE_AS, N_("_Save as mbox..."), "s", N_("Save selected messages as an mbox file"), G_CALLBACK (action_mail_save_as_cb) }, { "mail-show-source", NULL, N_("_Message Source"), "u", N_("Show the raw email source of the message"), G_CALLBACK (action_mail_show_source_cb) }, { "mail-toggle-important", NULL, NULL, /* No menu item; key press only */ NULL, NULL, G_CALLBACK (action_mail_toggle_important_cb) }, { "mail-undelete", NULL, N_("_Undelete Message"), "d", N_("Undelete the selected messages"), G_CALLBACK (action_mail_undelete_cb) }, { "mail-zoom-100", GTK_STOCK_ZOOM_100, N_("_Normal Size"), "0", N_("Reset the text to its original size"), G_CALLBACK (action_mail_zoom_100_cb) }, { "mail-zoom-in", GTK_STOCK_ZOOM_IN, N_("_Zoom In"), "plus", N_("Increase the text size"), G_CALLBACK (action_mail_zoom_in_cb) }, { "mail-zoom-out", GTK_STOCK_ZOOM_OUT, N_("Zoom _Out"), "minus", N_("Decrease the text size"), G_CALLBACK (action_mail_zoom_out_cb) }, /*** Menus ***/ { "mail-create-menu", NULL, N_("Cre_ate"), NULL, NULL, NULL }, { "mail-encoding-menu", NULL, N_("Ch_aracter Encoding"), NULL, NULL, NULL }, { "mail-forward-as-menu", NULL, N_("F_orward As"), NULL, NULL, NULL }, { "mail-reply-group-menu", NULL, N_("_Group Reply"), NULL, NULL, NULL }, { "mail-goto-menu", GTK_STOCK_JUMP_TO, N_("_Go To"), NULL, NULL, NULL }, { "mail-mark-as-menu", NULL, N_("Mar_k As"), NULL, NULL, NULL }, { "mail-message-menu", NULL, N_("_Message"), NULL, NULL, NULL }, { "mail-zoom-menu", NULL, N_("_Zoom"), NULL, NULL, NULL } }; static GtkActionEntry mail_reader_search_folder_entries[] = { { "mail-search-folder-from-mailing-list", NULL, N_("Create a Search Folder from Mailing _List..."), NULL, N_("Create a search folder for this mailing list"), G_CALLBACK (action_mail_search_folder_from_mailing_list_cb) }, { "mail-search-folder-from-recipients", NULL, N_("Create a Search Folder from Recipien_ts..."), NULL, N_("Create a search folder for these recipients"), G_CALLBACK (action_mail_search_folder_from_recipients_cb) }, { "mail-search-folder-from-sender", NULL, N_("Create a Search Folder from Sen_der..."), NULL, N_("Create a search folder for this sender"), G_CALLBACK (action_mail_search_folder_from_sender_cb) }, { "mail-search-folder-from-subject", NULL, N_("Create a Search Folder from S_ubject..."), NULL, N_("Create a search folder for this subject"), G_CALLBACK (action_mail_search_folder_from_subject_cb) }, }; static EPopupActionEntry mail_reader_popup_entries[] = { { "mail-popup-copy", NULL, "mail-copy" }, { "mail-popup-delete", NULL, "mail-delete" }, { "mail-popup-flag-clear", NULL, "mail-flag-clear" }, { "mail-popup-flag-completed", NULL, "mail-flag-completed" }, { "mail-popup-flag-for-followup", N_("Mark for Follo_w Up..."), "mail-flag-for-followup" }, { "mail-popup-forward", NULL, "mail-forward" }, { "mail-popup-mark-important", N_("Mark as _Important"), "mail-mark-important" }, { "mail-popup-mark-junk", N_("Mark as _Junk"), "mail-mark-junk" }, { "mail-popup-mark-notjunk", N_("Mark as _Not Junk"), "mail-mark-notjunk" }, { "mail-popup-mark-read", N_("Mar_k as Read"), "mail-mark-read" }, { "mail-popup-mark-unimportant", N_("Mark as Uni_mportant"), "mail-mark-unimportant" }, { "mail-popup-mark-unread", N_("Mark as _Unread"), "mail-mark-unread" }, { "mail-popup-message-edit", NULL, "mail-message-edit" }, { "mail-popup-move", NULL, "mail-move" }, { "mail-popup-print", NULL, "mail-print" }, { "mail-popup-remove-attachments", NULL, "mail-remove-attachments" }, { "mail-popup-remove-duplicates", NULL, "mail-remove-duplicates" }, { "mail-popup-reply-all", NULL, "mail-reply-all" }, { "mail-popup-reply-sender", NULL, "mail-reply-sender" }, { "mail-popup-save-as", NULL, "mail-save-as" }, { "mail-popup-undelete", NULL, "mail-undelete" } }; static GtkToggleActionEntry mail_reader_toggle_entries[] = { { "mail-caret-mode", NULL, N_("_Caret Mode"), "F7", N_("Show a blinking cursor in the body of displayed messages"), NULL, /* No callback required */ FALSE }, { "mail-show-all-headers", NULL, N_("All Message _Headers"), NULL, N_("Show messages with all email headers"), G_CALLBACK (action_mail_show_all_headers_cb), FALSE } }; static void mail_reader_double_click_cb (EMailReader *reader, gint row, ETreePath path, gint col, GdkEvent *event) { GtkAction *action; /* Ignore double clicks on columns that handle their own state. */ if (MESSAGE_LIST_COLUMN_IS_ACTIVE (col)) return; action = e_mail_reader_get_action (reader, "mail-message-open"); gtk_action_activate (action); } static gboolean mail_reader_key_press_event_cb (EMailReader *reader, GdkEventKey *event) { GtkAction *action; const gchar *action_name; if (!gtk_widget_has_focus (GTK_WIDGET (reader))) { WebKitWebFrame *frame; WebKitDOMDocument *dom; WebKitDOMElement *element; EMailDisplay *display; gchar *name = NULL; display = e_mail_reader_get_mail_display (reader); frame = webkit_web_view_get_focused_frame (WEBKIT_WEB_VIEW (display)); if (frame != NULL) { dom = webkit_web_frame_get_dom_document (frame); /* intentionally used "static_cast" */ element = webkit_dom_html_document_get_active_element ((WebKitDOMHTMLDocument *) dom); if (element != NULL) 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 != NULL && (g_ascii_strcasecmp (name, "INPUT") == 0 || g_ascii_strcasecmp (name, "TEXTAREA") == 0)) { g_free (name); return FALSE; } g_free (name); } } if ((event->state & GDK_CONTROL_MASK) != 0) goto ctrl; /* alone */ switch (event->keyval) { case GDK_KEY_Delete: case GDK_KEY_KP_Delete: action_name = "mail-delete"; break; case GDK_KEY_Return: case GDK_KEY_KP_Enter: case GDK_KEY_ISO_Enter: action_name = "mail-message-open"; break; case GDK_KEY_period: case GDK_KEY_bracketright: action_name = "mail-next-unread"; break; case GDK_KEY_comma: case GDK_KEY_bracketleft: action_name = "mail-previous-unread"; break; #ifdef HAVE_XFREE case XF86XK_Reply: action_name = "mail-reply-all"; break; case XF86XK_MailForward: action_name = "mail-forward"; break; #endif case GDK_KEY_exclam: action_name = "mail-toggle-important"; break; default: return FALSE; } goto exit; ctrl: /* Ctrl + */ switch (event->keyval) { case GDK_KEY_period: action_name = "mail-next-unread"; break; case GDK_KEY_comma: action_name = "mail-previous-unread"; break; default: return FALSE; } exit: action = e_mail_reader_get_action (reader, action_name); gtk_action_activate (action); return TRUE; } static gint mail_reader_key_press_cb (EMailReader *reader, gint row, ETreePath path, gint col, GdkEvent *event) { return mail_reader_key_press_event_cb (reader, &event->key); } static gboolean mail_reader_message_seen_cb (gpointer user_data) { EMailReaderClosure *closure = user_data; EMailReader *reader; GtkWidget *message_list; EMailPartList *parts; EMailDisplay *display; CamelMimeMessage *message; const gchar *current_uid; const gchar *message_uid; gboolean uid_is_current = TRUE; reader = closure->reader; message_uid = closure->message_uid; display = e_mail_reader_get_mail_display (reader); parts = e_mail_display_get_part_list (display); message_list = e_mail_reader_get_message_list (reader); if (e_tree_is_dragging (E_TREE (message_list))) return FALSE; current_uid = MESSAGE_LIST (message_list)->cursor_uid; uid_is_current &= (g_strcmp0 (current_uid, message_uid) == 0); if (parts != NULL) message = e_mail_part_list_get_message (parts); else message = NULL; if (uid_is_current && message != NULL) g_signal_emit ( reader, signals[MESSAGE_SEEN], 0, message_uid, message); return FALSE; } static gboolean schedule_timeout_mark_seen (EMailReader *reader) { MessageList *message_list; GSettings *settings; gboolean schedule_timeout; gint timeout_interval; const gchar *message_uid; message_list = MESSAGE_LIST (e_mail_reader_get_message_list (reader)); message_uid = message_list->cursor_uid; if (message_uid == NULL || e_tree_is_dragging (E_TREE (message_list))) return FALSE; settings = g_settings_new ("org.gnome.evolution.mail"); /* FIXME These should be EMailReader properties. */ schedule_timeout = (message_uid != NULL) && g_settings_get_boolean (settings, "mark-seen"); timeout_interval = g_settings_get_int (settings, "mark-seen-timeout"); g_object_unref (settings); if (message_list->seen_id > 0) { g_source_remove (message_list->seen_id); message_list->seen_id = 0; } if (schedule_timeout) { EMailReaderClosure *timeout_closure; timeout_closure = g_slice_new0 (EMailReaderClosure); timeout_closure->reader = g_object_ref (reader); timeout_closure->message_uid = g_strdup (message_uid); MESSAGE_LIST (message_list)->seen_id = e_named_timeout_add_full ( G_PRIORITY_DEFAULT, timeout_interval, mail_reader_message_seen_cb, timeout_closure, (GDestroyNotify) mail_reader_closure_free); } return schedule_timeout; } static gboolean discard_timeout_mark_seen_cb (EMailReader *reader) { MessageList *message_list; g_return_val_if_fail (reader != NULL, FALSE); message_list = MESSAGE_LIST (e_mail_reader_get_message_list (reader)); g_return_val_if_fail (message_list != NULL, FALSE); if (message_list->seen_id > 0) { g_source_remove (message_list->seen_id); message_list->seen_id = 0; } return FALSE; } static void mail_reader_message_loaded_cb (CamelFolder *folder, GAsyncResult *result, EMailReaderClosure *closure) { EMailReader *reader; EMailReaderPrivate *priv; CamelMimeMessage *message = NULL; GtkWidget *message_list; const gchar *message_uid; GError *error = NULL; reader = closure->reader; message_uid = closure->message_uid; priv = E_MAIL_READER_GET_PRIVATE (reader); /* If the private struct is NULL, the EMailReader was destroyed * while we were loading the message and we're likely holding the * last reference. Nothing to do but drop the reference. * FIXME Use a GWeakRef instead of this hack. */ if (priv == NULL) { mail_reader_closure_free (closure); return; } message = camel_folder_get_message_finish (folder, result, &error); /* If the user picked a different message in the time it took * to fetch this message, then don't bother rendering it. */ if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) { g_clear_error (&error); goto exit; } message_list = e_mail_reader_get_message_list (reader); if (message_list == NULL) { /* For cases where message fetching took so long that * user closed the message window before this was called. */ goto exit; } if (message != NULL) g_signal_emit ( reader, signals[MESSAGE_LOADED], 0, message_uid, message); exit: priv->restoring_message_selection = FALSE; if (error != NULL) { EPreviewPane *preview_pane; EWebView *web_view; preview_pane = e_mail_reader_get_preview_pane (reader); web_view = e_preview_pane_get_web_view (preview_pane); e_alert_submit ( E_ALERT_SINK (web_view), "mail:no-retrieve-message", error->message, NULL); } g_clear_error (&error); mail_reader_closure_free (closure); g_clear_object (&message); } static gboolean mail_reader_message_selected_timeout_cb (gpointer user_data) { EMailReader *reader; EMailReaderPrivate *priv; EMailDisplay *display; GtkWidget *message_list; const gchar *cursor_uid; const gchar *format_uid; EMailPartList *parts; reader = E_MAIL_READER (user_data); priv = E_MAIL_READER_GET_PRIVATE (reader); message_list = e_mail_reader_get_message_list (reader); display = e_mail_reader_get_mail_display (reader); parts = e_mail_display_get_part_list (display); cursor_uid = MESSAGE_LIST (message_list)->cursor_uid; if (parts != NULL) format_uid = e_mail_part_list_get_message_uid (parts); else format_uid = NULL; if (MESSAGE_LIST (message_list)->last_sel_single) { GtkWidget *widget; gboolean display_visible; gboolean selected_uid_changed; /* Decide whether to download the full message now. */ widget = GTK_WIDGET (display); display_visible = gtk_widget_get_mapped (widget); selected_uid_changed = (g_strcmp0 (cursor_uid, format_uid) != 0); if (display_visible && selected_uid_changed) { EMailReaderClosure *closure; GCancellable *cancellable; CamelFolder *folder; EActivity *activity; gchar *string; string = g_strdup_printf ( _("Retrieving message '%s'"), cursor_uid); e_mail_display_set_part_list (display, NULL); e_mail_display_set_status (display, string); g_free (string); activity = e_mail_reader_new_activity (reader); e_activity_set_text (activity, _("Retrieving message")); cancellable = e_activity_get_cancellable (activity); closure = g_slice_new0 (EMailReaderClosure); closure->activity = activity; closure->reader = g_object_ref (reader); closure->message_uid = g_strdup (cursor_uid); folder = e_mail_reader_ref_folder (reader); camel_folder_get_message ( folder, cursor_uid, G_PRIORITY_DEFAULT, cancellable, (GAsyncReadyCallback) mail_reader_message_loaded_cb, closure); g_clear_object (&folder); if (priv->retrieving_message != NULL) g_object_unref (priv->retrieving_message); priv->retrieving_message = g_object_ref (cancellable); } } else { e_mail_display_set_part_list (display, NULL); priv->restoring_message_selection = FALSE; } priv->message_selected_timeout_id = 0; return FALSE; } static void mail_reader_message_selected_cb (EMailReader *reader, const gchar *message_uid) { EMailReaderPrivate *priv; MessageList *message_list; priv = E_MAIL_READER_GET_PRIVATE (reader); /* Cancel the previous message retrieval activity. */ g_cancellable_cancel (priv->retrieving_message); /* Cancel the seen timer. */ message_list = MESSAGE_LIST (e_mail_reader_get_message_list (reader)); if (message_list != NULL && message_list->seen_id) { g_source_remove (message_list->seen_id); message_list->seen_id = 0; } /* Cancel the message selected timer. */ if (priv->message_selected_timeout_id > 0) { g_source_remove (priv->message_selected_timeout_id); priv->message_selected_timeout_id = 0; } /* If a folder was just selected then we are now automatically * restoring the previous message selection. We behave slightly * differently than if the user had selected the message. */ priv->restoring_message_selection = priv->folder_was_just_selected; priv->folder_was_just_selected = FALSE; if (message_list_selected_count (message_list) != 1) { EMailDisplay *display; display = e_mail_reader_get_mail_display (reader); e_mail_display_set_part_list (display, NULL); e_web_view_clear (E_WEB_VIEW (display)); } else if (priv->restoring_message_selection) { /* Skip the timeout if we're restoring the previous message * selection. The timeout is there for when we're scrolling * rapidly through the message list. */ mail_reader_message_selected_timeout_cb (reader); } else { priv->message_selected_timeout_id = e_named_timeout_add ( 100, mail_reader_message_selected_timeout_cb, reader); } e_mail_reader_changed (reader); } static void mail_reader_message_cursor_change_cb (EMailReader *reader) { MessageList *message_list; EMailReaderPrivate *priv; g_return_if_fail (reader != NULL); priv = E_MAIL_READER_GET_PRIVATE (reader); g_return_if_fail (priv != NULL); message_list = MESSAGE_LIST (e_mail_reader_get_message_list (reader)); g_return_if_fail (message_list != NULL); if (message_list->seen_id == 0 && E_IS_MAIL_VIEW (reader) && e_mail_view_get_preview_visible (E_MAIL_VIEW (reader)) && !priv->avoid_next_mark_as_seen) schedule_timeout_mark_seen (reader); } static void mail_reader_emit_folder_loaded (EMailReader *reader) { EMailReaderPrivate *priv; MessageList *message_list; priv = E_MAIL_READER_GET_PRIVATE (reader); message_list = MESSAGE_LIST (e_mail_reader_get_message_list (reader)); if (priv && (message_list_count (message_list) <= 0 || message_list_selected_count (message_list) <= 0)) priv->avoid_next_mark_as_seen = FALSE; g_signal_emit (reader, signals[FOLDER_LOADED], 0); } static EAlertSink * mail_reader_get_alert_sink (EMailReader *reader) { EPreviewPane *preview_pane; preview_pane = e_mail_reader_get_preview_pane (reader); return E_ALERT_SINK (preview_pane); } static GPtrArray * mail_reader_get_selected_uids (EMailReader *reader) { GtkWidget *message_list; message_list = e_mail_reader_get_message_list (reader); return message_list_get_selected (MESSAGE_LIST (message_list)); } static CamelFolder * mail_reader_ref_folder (EMailReader *reader) { GtkWidget *message_list; message_list = e_mail_reader_get_message_list (reader); return message_list_ref_folder (MESSAGE_LIST (message_list)); } static void mail_reader_set_folder (EMailReader *reader, CamelFolder *folder) { EMailReaderPrivate *priv; EMailDisplay *display; CamelFolder *previous_folder; GtkWidget *message_list; EMailBackend *backend; EShell *shell; gboolean sync_folder; priv = E_MAIL_READER_GET_PRIVATE (reader); backend = e_mail_reader_get_backend (reader); display = e_mail_reader_get_mail_display (reader); message_list = e_mail_reader_get_message_list (reader); previous_folder = e_mail_reader_ref_folder (reader); backend = e_mail_reader_get_backend (reader); shell = e_shell_backend_get_shell (E_SHELL_BACKEND (backend)); /* Only synchronize the real folder if we're online. */ sync_folder = (previous_folder != NULL) && (CAMEL_IS_VEE_FOLDER (previous_folder) || e_shell_get_online (shell)); if (sync_folder) mail_sync_folder (previous_folder, TRUE, NULL, NULL); /* Skip the rest if we're already viewing the folder. */ if (folder != previous_folder) { e_web_view_clear (E_WEB_VIEW (display)); priv->folder_was_just_selected = (folder != NULL); /* This is to make sure any post-poned changes in Search * Folders will be propagated on folder selection. */ if (CAMEL_IS_VEE_FOLDER (folder)) mail_sync_folder (folder, FALSE, NULL, NULL); message_list_set_folder (MESSAGE_LIST (message_list), folder); mail_reader_emit_folder_loaded (reader); } g_clear_object (&previous_folder); } static void mail_reader_set_message (EMailReader *reader, const gchar *message_uid) { GtkWidget *message_list; message_list = e_mail_reader_get_message_list (reader); message_list_select_uid ( MESSAGE_LIST (message_list), message_uid, FALSE); } static void mail_reader_folder_loaded (EMailReader *reader) { guint32 state; state = e_mail_reader_check_state (reader); e_mail_reader_update_actions (reader, state); } static void set_mail_display_part_list (GObject *object, GAsyncResult *result, gpointer user_data) { EMailPartList *part_list; EMailReader *reader; EMailDisplay *display; reader = E_MAIL_READER (object); display = e_mail_reader_get_mail_display (reader); part_list = e_mail_reader_parse_message_finish (reader, result); e_mail_display_set_part_list (display, part_list); e_mail_display_load (display, NULL); /* Remove the reference added when parts list was * created, so that only owners are EMailDisplays. */ g_object_unref (part_list); } static void mail_reader_set_display_formatter_for_message (EMailReader *reader, EMailDisplay *display, const gchar *message_uid, CamelMimeMessage *message, CamelFolder *folder) { CamelObjectBag *registry; EMailPartList *parts; EMailReaderPrivate *priv; gchar *mail_uri; priv = E_MAIL_READER_GET_PRIVATE (reader); mail_uri = e_mail_part_build_uri (folder, message_uid, NULL, NULL); registry = e_mail_part_list_get_registry (); parts = camel_object_bag_peek (registry, mail_uri); g_free (mail_uri); if (parts == NULL) { e_mail_reader_parse_message ( reader, folder, message_uid, message, priv->retrieving_message, set_mail_display_part_list, NULL); } else { e_mail_display_set_part_list (display, parts); e_mail_display_load (display, NULL); g_object_unref (parts); } } static void mail_reader_message_loaded (EMailReader *reader, const gchar *message_uid, CamelMimeMessage *message) { EMailReaderPrivate *priv; GtkWidget *message_list; EMailBackend *backend; CamelFolder *folder; EMailDisplay *display; EShellBackend *shell_backend; EShell *shell; EMEvent *event; EMEventTargetMessage *target; GError *error = NULL; priv = E_MAIL_READER_GET_PRIVATE (reader); folder = e_mail_reader_ref_folder (reader); backend = e_mail_reader_get_backend (reader); display = e_mail_reader_get_mail_display (reader); message_list = e_mail_reader_get_message_list (reader); shell_backend = E_SHELL_BACKEND (backend); shell = e_shell_backend_get_shell (shell_backend); /** @Event: message.reading * @Title: Viewing a message * @Target: EMEventTargetMessage * * message.reading is emitted whenever a user views a message. */ event = em_event_peek (); target = em_event_target_new_message ( event, folder, message, message_uid, 0, NULL); e_event_emit ( (EEvent *) event, "message.reading", (EEventTarget *) target); mail_reader_set_display_formatter_for_message ( reader, display, message_uid, message, folder); /* Reset the shell view icon. */ e_shell_event (shell, "mail-icon", (gpointer) "evolution-mail"); if (MESSAGE_LIST (message_list)->seen_id > 0) { g_source_remove (MESSAGE_LIST (message_list)->seen_id); MESSAGE_LIST (message_list)->seen_id = 0; } /* Determine whether to mark the message as read. */ if (message != NULL && !priv->restoring_message_selection && !priv->avoid_next_mark_as_seen && schedule_timeout_mark_seen (reader)) { g_clear_error (&error); } else if (error != NULL) { e_alert_submit ( E_ALERT_SINK (display), "mail:no-retrieve-message", error->message, NULL); g_error_free (error); } priv->avoid_next_mark_as_seen = FALSE; g_clear_object (&folder); } static void mail_reader_message_seen (EMailReader *reader, const gchar *message_uid, CamelMimeMessage *message) { CamelFolder *folder; guint32 mask, set; mask = CAMEL_MESSAGE_SEEN; set = CAMEL_MESSAGE_SEEN; folder = e_mail_reader_ref_folder (reader); camel_folder_set_message_flags (folder, message_uid, mask, set); g_clear_object (&folder); } static void mail_reader_show_search_bar (EMailReader *reader) { EPreviewPane *preview_pane; preview_pane = e_mail_reader_get_preview_pane (reader); e_preview_pane_show_search_bar (preview_pane); } static void mail_reader_update_actions (EMailReader *reader, guint32 state) { GtkAction *action; const gchar *action_name; gboolean sensitive; /* Be descriptive. */ gboolean any_messages_selected; gboolean enable_flag_clear; gboolean enable_flag_completed; gboolean enable_flag_for_followup; gboolean have_enabled_account; gboolean multiple_messages_selected; gboolean selection_has_attachment_messages; gboolean selection_has_deleted_messages; gboolean selection_has_important_messages; gboolean selection_has_junk_messages; gboolean selection_has_not_junk_messages; gboolean selection_has_read_messages; gboolean selection_has_undeleted_messages; gboolean selection_has_unimportant_messages; gboolean selection_has_unread_messages; gboolean selection_is_mailing_list; gboolean single_message_selected; gboolean first_message_selected = FALSE; gboolean last_message_selected = FALSE; have_enabled_account = (state & E_MAIL_READER_HAVE_ENABLED_ACCOUNT); single_message_selected = (state & E_MAIL_READER_SELECTION_SINGLE); multiple_messages_selected = (state & E_MAIL_READER_SELECTION_MULTIPLE); /* FIXME Missing CAN_ADD_SENDER */ enable_flag_clear = (state & E_MAIL_READER_SELECTION_FLAG_CLEAR); enable_flag_completed = (state & E_MAIL_READER_SELECTION_FLAG_COMPLETED); enable_flag_for_followup = (state & E_MAIL_READER_SELECTION_FLAG_FOLLOWUP); selection_has_attachment_messages = (state & E_MAIL_READER_SELECTION_HAS_ATTACHMENTS); selection_has_deleted_messages = (state & E_MAIL_READER_SELECTION_HAS_DELETED); selection_has_important_messages = (state & E_MAIL_READER_SELECTION_HAS_IMPORTANT); selection_has_junk_messages = (state & E_MAIL_READER_SELECTION_HAS_JUNK); selection_has_not_junk_messages = (state & E_MAIL_READER_SELECTION_HAS_NOT_JUNK); selection_has_read_messages = (state & E_MAIL_READER_SELECTION_HAS_READ); selection_has_undeleted_messages = (state & E_MAIL_READER_SELECTION_HAS_UNDELETED); selection_has_unimportant_messages = (state & E_MAIL_READER_SELECTION_HAS_UNIMPORTANT); selection_has_unread_messages = (state & E_MAIL_READER_SELECTION_HAS_UNREAD); selection_is_mailing_list = (state & E_MAIL_READER_SELECTION_IS_MAILING_LIST); any_messages_selected = (single_message_selected || multiple_messages_selected); if (any_messages_selected) { MessageList *message_list; gint row = -1, count = -1; ETreeTableAdapter *etta; ETreePath node = NULL; message_list = MESSAGE_LIST ( e_mail_reader_get_message_list (reader)); etta = e_tree_get_table_adapter (E_TREE (message_list)); if (message_list->cursor_uid != NULL) node = g_hash_table_lookup ( message_list->uid_nodemap, message_list->cursor_uid); if (node != NULL) { row = e_tree_table_adapter_row_of_node (etta, node); count = e_table_model_row_count (E_TABLE_MODEL (etta)); } first_message_selected = row <= 0; last_message_selected = row < 0 || row + 1 >= count; } action_name = "mail-add-sender"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-check-for-junk"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-copy"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-create-menu"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); /* If a single message is selected, let the user hit delete to * advance the cursor even if the message is already deleted. */ action_name = "mail-delete"; sensitive = single_message_selected || selection_has_undeleted_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-filters-apply"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-find"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-flag-clear"; sensitive = enable_flag_clear; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-flag-completed"; sensitive = enable_flag_completed; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-flag-for-followup"; sensitive = enable_flag_for_followup; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward"; sensitive = have_enabled_account && any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward-attached"; sensitive = have_enabled_account && any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward-attached-full"; sensitive = have_enabled_account && any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward-as-menu"; sensitive = have_enabled_account && any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward-inline"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward-inline-full"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward-quoted"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-forward-quoted-full"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-goto-menu"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-load-images"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-mark-as-menu"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-mark-important"; sensitive = selection_has_unimportant_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-mark-junk"; sensitive = selection_has_not_junk_messages && !(state & E_MAIL_READER_FOLDER_IS_JUNK); action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-mark-notjunk"; sensitive = selection_has_junk_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-mark-read"; sensitive = selection_has_unread_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-mark-unimportant"; sensitive = selection_has_important_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-mark-unread"; sensitive = selection_has_read_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-message-edit"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-message-new"; sensitive = have_enabled_account; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-message-open"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-move"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-next"; sensitive = any_messages_selected && !last_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-next-important"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-next-thread"; sensitive = single_message_selected && !last_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-next-unread"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-previous"; sensitive = any_messages_selected && !first_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-previous-important"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-previous-unread"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-previous-thread"; sensitive = any_messages_selected && !first_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-print"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-print-preview"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-redirect"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-remove-attachments"; sensitive = any_messages_selected && selection_has_attachment_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-remove-duplicates"; sensitive = multiple_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-reply-all"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-reply-group"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-reply-group-menu"; sensitive = have_enabled_account && any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-reply-list"; sensitive = have_enabled_account && single_message_selected && selection_is_mailing_list; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-reply-sender"; sensitive = have_enabled_account && single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-save-as"; sensitive = any_messages_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-show-source"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-undelete"; sensitive = selection_has_deleted_messages; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-zoom-100"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-zoom-in"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); action_name = "mail-zoom-out"; sensitive = single_message_selected; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_sensitive (action, sensitive); } static void mail_reader_init_charset_actions (EMailReader *reader, GtkActionGroup *action_group) { GtkRadioAction *default_action; GSList *radio_group; radio_group = e_charset_add_radio_actions ( action_group, "mail-charset-", NULL, G_CALLBACK (action_mail_charset_cb), reader); /* XXX Add a tooltip! */ default_action = gtk_radio_action_new ( "mail-charset-default", _("Default"), NULL, NULL, -1); gtk_radio_action_set_group (default_action, radio_group); g_signal_connect ( default_action, "changed", G_CALLBACK (action_mail_charset_cb), reader); gtk_action_group_add_action ( action_group, GTK_ACTION (default_action)); gtk_radio_action_set_current_value (default_action, -1); } static void e_mail_reader_default_init (EMailReaderInterface *interface) { quark_private = g_quark_from_static_string ("e-mail-reader-private"); interface->get_alert_sink = mail_reader_get_alert_sink; interface->get_selected_uids = mail_reader_get_selected_uids; interface->ref_folder = mail_reader_ref_folder; interface->set_folder = mail_reader_set_folder; interface->set_message = mail_reader_set_message; interface->open_selected_mail = e_mail_reader_open_selected; interface->folder_loaded = mail_reader_folder_loaded; interface->message_loaded = mail_reader_message_loaded; interface->message_seen = mail_reader_message_seen; interface->show_search_bar = mail_reader_show_search_bar; interface->update_actions = mail_reader_update_actions; g_object_interface_install_property ( interface, g_param_spec_enum ( "forward-style", "Forward Style", "How to forward messages", E_TYPE_MAIL_FORWARD_STYLE, E_MAIL_FORWARD_STYLE_ATTACHED, G_PARAM_READWRITE)); g_object_interface_install_property ( interface, g_param_spec_boolean ( "group-by-threads", "Group by Threads", "Whether to group messages by threads", FALSE, G_PARAM_READWRITE)); g_object_interface_install_property ( interface, g_param_spec_enum ( "reply-style", "Reply Style", "How to reply to messages", E_TYPE_MAIL_REPLY_STYLE, E_MAIL_REPLY_STYLE_QUOTED, G_PARAM_READWRITE)); signals[CHANGED] = g_signal_new ( "changed", G_OBJECT_CLASS_TYPE (interface), G_SIGNAL_RUN_FIRST, 0, NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[COMPOSER_CREATED] = g_signal_new ( "composer-created", G_OBJECT_CLASS_TYPE (interface), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (EMailReaderInterface, composer_created), NULL, NULL, NULL, G_TYPE_NONE, 2, E_TYPE_MSG_COMPOSER, CAMEL_TYPE_MIME_MESSAGE); signals[FOLDER_LOADED] = g_signal_new ( "folder-loaded", G_OBJECT_CLASS_TYPE (interface), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (EMailReaderInterface, folder_loaded), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[MESSAGE_LOADED] = g_signal_new ( "message-loaded", G_OBJECT_CLASS_TYPE (interface), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EMailReaderInterface, message_loaded), NULL, NULL, e_marshal_VOID__STRING_OBJECT, G_TYPE_NONE, 2, G_TYPE_STRING, CAMEL_TYPE_MIME_MESSAGE); signals[MESSAGE_SEEN] = g_signal_new ( "message-seen", G_OBJECT_CLASS_TYPE (interface), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EMailReaderInterface, message_seen), NULL, NULL, e_marshal_VOID__STRING_OBJECT, G_TYPE_NONE, 2, G_TYPE_STRING, CAMEL_TYPE_MIME_MESSAGE); signals[SHOW_SEARCH_BAR] = g_signal_new ( "show-search-bar", G_OBJECT_CLASS_TYPE (interface), G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (EMailReaderInterface, show_search_bar), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[UPDATE_ACTIONS] = g_signal_new ( "update-actions", G_OBJECT_CLASS_TYPE (interface), G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (EMailReaderInterface, update_actions), NULL, NULL, g_cclosure_marshal_VOID__UINT, G_TYPE_NONE, 1, G_TYPE_UINT); } void e_mail_reader_init (EMailReader *reader, gboolean init_actions, gboolean connect_signals) { EMenuToolAction *menu_tool_action; GtkActionGroup *action_group; GtkWidget *message_list; GtkAction *action; const gchar *action_name; EMailDisplay *display; GSettings *settings; g_return_if_fail (E_IS_MAIL_READER (reader)); message_list = e_mail_reader_get_message_list (reader); display = e_mail_reader_get_mail_display (reader); /* Initialize a private struct. */ g_object_set_qdata_full ( G_OBJECT (reader), quark_private, g_slice_new0 (EMailReaderPrivate), (GDestroyNotify) mail_reader_private_free); g_object_bind_property ( reader, "group-by-threads", message_list, "group-by-threads", G_BINDING_SYNC_CREATE); if (!init_actions) goto connect_signals; /* Add the "standard" EMailReader actions. */ action_group = e_mail_reader_get_action_group ( reader, E_MAIL_READER_ACTION_GROUP_STANDARD); /* The "mail-forward" action is special: it uses a GtkMenuToolButton * for its toolbar item type. So we have to create it separately. */ menu_tool_action = e_menu_tool_action_new ( "mail-forward", _("_Forward"), _("Forward the selected message to someone"), NULL); gtk_action_set_icon_name ( GTK_ACTION (menu_tool_action), "mail-forward"); g_signal_connect ( menu_tool_action, "activate", G_CALLBACK (action_mail_forward_cb), reader); gtk_action_group_add_action_with_accel ( action_group, GTK_ACTION (menu_tool_action), "f"); /* Likewise the "mail-reply-group" action. */ /* For Translators: "Group Reply" will reply either to a mailing list * (if possible and if that configuration option is enabled), or else * it will reply to all. The word "Group" was chosen because it covers * either of those, without too strongly implying one or the other. */ menu_tool_action = e_menu_tool_action_new ( "mail-reply-group", _("Group Reply"), _("Reply to the mailing list, or to all recipients"), NULL); gtk_action_set_icon_name ( GTK_ACTION (menu_tool_action), "mail-reply-all"); g_signal_connect ( menu_tool_action, "activate", G_CALLBACK (action_mail_reply_group_cb), reader); gtk_action_group_add_action_with_accel ( action_group, GTK_ACTION (menu_tool_action), "g"); /* Add the other actions the normal way. */ gtk_action_group_add_actions ( action_group, mail_reader_entries, G_N_ELEMENTS (mail_reader_entries), reader); e_action_group_add_popup_actions ( action_group, mail_reader_popup_entries, G_N_ELEMENTS (mail_reader_popup_entries)); gtk_action_group_add_toggle_actions ( action_group, mail_reader_toggle_entries, G_N_ELEMENTS (mail_reader_toggle_entries), reader); mail_reader_init_charset_actions (reader, action_group); /* Add EMailReader actions for Search Folders. The action group * should be made invisible if Search Folders are disabled. */ action_group = e_mail_reader_get_action_group ( reader, E_MAIL_READER_ACTION_GROUP_SEARCH_FOLDERS); gtk_action_group_add_actions ( action_group, mail_reader_search_folder_entries, G_N_ELEMENTS (mail_reader_search_folder_entries), reader); display = e_mail_reader_get_mail_display (reader); /* Bind GObject properties to GSettings keys. */ settings = g_settings_new ("org.gnome.evolution.mail"); action_name = "mail-caret-mode"; action = e_mail_reader_get_action (reader, action_name); g_settings_bind ( settings, "caret-mode", action, "active", G_SETTINGS_BIND_DEFAULT); action_name = "mail-show-all-headers"; action = e_mail_reader_get_action (reader, action_name); g_settings_bind ( settings, "show-all-headers", action, "active", G_SETTINGS_BIND_DEFAULT); /* Mode change when viewing message source is ignored. */ if (e_mail_display_get_mode (display) == E_MAIL_FORMATTER_MODE_SOURCE || e_mail_display_get_mode (display) == E_MAIL_FORMATTER_MODE_RAW) { gtk_action_set_sensitive (action, FALSE); gtk_action_set_visible (action, FALSE); } g_object_unref (settings); /* Fine tuning. */ action_name = "mail-delete"; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_short_label (action, _("Delete")); action_name = "mail-forward"; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_is_important (action, TRUE); action_name = "mail-reply-group"; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_is_important (action, TRUE); action_name = "mail-next"; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_short_label (action, _("Next")); action_name = "mail-previous"; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_short_label (action, _("Previous")); action_name = "mail-reply-all"; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_is_important (action, TRUE); action_name = "mail-reply-sender"; action = e_mail_reader_get_action (reader, action_name); gtk_action_set_is_important (action, TRUE); gtk_action_set_short_label (action, _("Reply")); action_name = "add-to-address-book"; action = e_mail_display_get_action (display, action_name); g_signal_connect ( action, "activate", G_CALLBACK (action_add_to_address_book_cb), reader); action_name = "send-reply"; action = e_mail_display_get_action (display, action_name); g_signal_connect ( action, "activate", G_CALLBACK (action_mail_reply_recipient_cb), reader); action_name = "search-folder-recipient"; action = e_mail_display_get_action (display, action_name); g_signal_connect ( action, "activate", G_CALLBACK (action_search_folder_recipient_cb), reader); action_name = "search-folder-sender"; action = e_mail_display_get_action (display, action_name); g_signal_connect ( action, "activate", G_CALLBACK (action_search_folder_sender_cb), reader); #ifndef G_OS_WIN32 /* Lockdown integration. */ settings = g_settings_new ("org.gnome.desktop.lockdown"); action_name = "mail-print"; action = e_mail_reader_get_action (reader, action_name); g_settings_bind ( settings, "disable-printing", action, "visible", G_SETTINGS_BIND_GET | G_SETTINGS_BIND_NO_SENSITIVITY | G_SETTINGS_BIND_INVERT_BOOLEAN); action_name = "mail-print-preview"; action = e_mail_reader_get_action (reader, action_name); g_settings_bind ( settings, "disable-printing", action, "visible", G_SETTINGS_BIND_GET | G_SETTINGS_BIND_NO_SENSITIVITY | G_SETTINGS_BIND_INVERT_BOOLEAN); action_name = "mail-save-as"; action = e_mail_reader_get_action (reader, action_name); g_settings_bind ( settings, "disable-save-to-disk", action, "visible", G_SETTINGS_BIND_GET | G_SETTINGS_BIND_NO_SENSITIVITY | G_SETTINGS_BIND_INVERT_BOOLEAN); g_object_unref (settings); #endif /* Bind properties. */ action_name = "mail-caret-mode"; action = e_mail_reader_get_action (reader, action_name); g_object_bind_property ( action, "active", display, "caret-mode", G_BINDING_BIDIRECTIONAL | G_BINDING_SYNC_CREATE); connect_signals: if (!connect_signals) return; /* Connect signals. */ g_signal_connect_swapped ( display, "key-press-event", G_CALLBACK (mail_reader_key_press_event_cb), reader); g_signal_connect_swapped ( message_list, "message-selected", G_CALLBACK (mail_reader_message_selected_cb), reader); /* re-schedule mark-as-seen,... */ g_signal_connect_swapped ( message_list, "cursor-change", G_CALLBACK (mail_reader_message_cursor_change_cb), reader); /* but do not mark-as-seen if... */ g_signal_connect_swapped ( message_list, "tree-drag-begin", G_CALLBACK (discard_timeout_mark_seen_cb), reader); g_signal_connect_swapped ( message_list, "tree-drag-end", G_CALLBACK (discard_timeout_mark_seen_cb), reader); g_signal_connect_swapped ( message_list, "right-click", G_CALLBACK (discard_timeout_mark_seen_cb), reader); g_signal_connect_swapped ( message_list, "message-list-built", G_CALLBACK (mail_reader_emit_folder_loaded), reader); g_signal_connect_swapped ( message_list, "double-click", G_CALLBACK (mail_reader_double_click_cb), reader); g_signal_connect_swapped ( message_list, "key-press", G_CALLBACK (mail_reader_key_press_cb), reader); g_signal_connect_swapped ( message_list, "selection-change", G_CALLBACK (e_mail_reader_changed), reader); } void e_mail_reader_changed (EMailReader *reader) { g_return_if_fail (E_IS_MAIL_READER (reader)); g_signal_emit (reader, signals[CHANGED], 0); } guint32 e_mail_reader_check_state (EMailReader *reader) { EShell *shell; GPtrArray *uids; CamelFolder *folder; CamelStore *store = NULL; EMailBackend *backend; ESourceRegistry *registry; EMailSession *mail_session; EMailAccountStore *account_store; const gchar *tag; gboolean can_clear_flags = FALSE; gboolean can_flag_completed = FALSE; gboolean can_flag_for_followup = FALSE; gboolean has_attachments = FALSE; gboolean has_deleted = FALSE; gboolean has_important = FALSE; gboolean has_junk = FALSE; gboolean has_not_junk = FALSE; gboolean has_read = FALSE; gboolean has_undeleted = FALSE; gboolean has_unimportant = FALSE; gboolean has_unread = FALSE; gboolean have_enabled_account = FALSE; gboolean drafts_or_outbox = FALSE; gboolean store_supports_vjunk = FALSE; gboolean is_mailing_list; gboolean is_junk_folder = FALSE; guint32 state = 0; guint ii; g_return_val_if_fail (E_IS_MAIL_READER (reader), 0); backend = e_mail_reader_get_backend (reader); shell = e_shell_backend_get_shell (E_SHELL_BACKEND (backend)); registry = e_shell_get_registry (shell); mail_session = e_mail_backend_get_session (backend); account_store = e_mail_ui_session_get_account_store ( E_MAIL_UI_SESSION (mail_session)); folder = e_mail_reader_ref_folder (reader); uids = e_mail_reader_get_selected_uids (reader); if (folder != NULL) { store = camel_folder_get_parent_store (folder); store_supports_vjunk = (store->flags & CAMEL_STORE_VJUNK); is_junk_folder = (folder->folder_flags & CAMEL_FOLDER_IS_JUNK) != 0; drafts_or_outbox = em_utils_folder_is_drafts (registry, folder) || em_utils_folder_is_outbox (registry, folder); } /* Initialize this flag based on whether there are any * messages selected. We will update it in the loop. */ is_mailing_list = (uids->len > 0); for (ii = 0; ii < uids->len; ii++) { CamelMessageInfo *info; const gchar *string; guint32 flags; info = camel_folder_get_message_info ( folder, uids->pdata[ii]); if (info == NULL) continue; flags = camel_message_info_flags (info); if (flags & CAMEL_MESSAGE_SEEN) has_read = TRUE; else has_unread = TRUE; if (flags & CAMEL_MESSAGE_ATTACHMENTS) has_attachments = TRUE; if (drafts_or_outbox) { has_junk = FALSE; has_not_junk = FALSE; } else if (store_supports_vjunk) { guint32 bitmask; /* XXX Strictly speaking, this logic is correct. * Problem is there's nothing in the message * list that indicates whether a message is * already marked "Not Junk". So the user may * think the "Not Junk" button is enabling and * disabling itself randomly as he reads mail. */ if (flags & CAMEL_MESSAGE_JUNK) has_junk = TRUE; if (flags & CAMEL_MESSAGE_NOTJUNK) has_not_junk = TRUE; bitmask = CAMEL_MESSAGE_JUNK | CAMEL_MESSAGE_NOTJUNK; /* If neither junk flag is set, the * message can be marked either way. */ if ((flags & bitmask) == 0) { has_junk = TRUE; has_not_junk = TRUE; } } else { has_junk = TRUE; has_not_junk = TRUE; } if (flags & CAMEL_MESSAGE_DELETED) has_deleted = TRUE; else has_undeleted = TRUE; if (flags & CAMEL_MESSAGE_FLAGGED) has_important = TRUE; else has_unimportant = TRUE; tag = camel_message_info_user_tag (info, "follow-up"); if (tag != NULL && *tag != '\0') { can_clear_flags = TRUE; tag = camel_message_info_user_tag ( info, "completed-on"); if (tag == NULL || *tag == '\0') can_flag_completed = TRUE; } else can_flag_for_followup = TRUE; string = camel_message_info_mlist (info); is_mailing_list &= (string != NULL && *string != '\0'); camel_message_info_unref (info); } have_enabled_account = e_mail_account_store_have_enabled_service ( account_store, CAMEL_TYPE_STORE); if (have_enabled_account) state |= E_MAIL_READER_HAVE_ENABLED_ACCOUNT; if (uids->len == 1) state |= E_MAIL_READER_SELECTION_SINGLE; if (uids->len > 1) state |= E_MAIL_READER_SELECTION_MULTIPLE; if (!drafts_or_outbox && uids->len == 1) state |= E_MAIL_READER_SELECTION_CAN_ADD_SENDER; if (can_clear_flags) state |= E_MAIL_READER_SELECTION_FLAG_CLEAR; if (can_flag_completed) state |= E_MAIL_READER_SELECTION_FLAG_COMPLETED; if (can_flag_for_followup) state |= E_MAIL_READER_SELECTION_FLAG_FOLLOWUP; if (has_attachments) state |= E_MAIL_READER_SELECTION_HAS_ATTACHMENTS; if (has_deleted) state |= E_MAIL_READER_SELECTION_HAS_DELETED; if (has_important) state |= E_MAIL_READER_SELECTION_HAS_IMPORTANT; if (has_junk) state |= E_MAIL_READER_SELECTION_HAS_JUNK; if (has_not_junk) state |= E_MAIL_READER_SELECTION_HAS_NOT_JUNK; if (has_read) state |= E_MAIL_READER_SELECTION_HAS_READ; if (has_undeleted) state |= E_MAIL_READER_SELECTION_HAS_UNDELETED; if (has_unimportant) state |= E_MAIL_READER_SELECTION_HAS_UNIMPORTANT; if (has_unread) state |= E_MAIL_READER_SELECTION_HAS_UNREAD; if (is_mailing_list) state |= E_MAIL_READER_SELECTION_IS_MAILING_LIST; if (is_junk_folder) state |= E_MAIL_READER_FOLDER_IS_JUNK; g_clear_object (&folder); g_ptr_array_unref (uids); return state; } EActivity * e_mail_reader_new_activity (EMailReader *reader) { EActivity *activity; EMailBackend *backend; EAlertSink *alert_sink; GCancellable *cancellable; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); activity = e_activity_new (); alert_sink = e_mail_reader_get_alert_sink (reader); e_activity_set_alert_sink (activity, alert_sink); cancellable = camel_operation_new (); e_activity_set_cancellable (activity, cancellable); g_object_unref (cancellable); backend = e_mail_reader_get_backend (reader); e_shell_backend_add_activity (E_SHELL_BACKEND (backend), activity); return activity; } void e_mail_reader_update_actions (EMailReader *reader, guint32 state) { g_return_if_fail (E_IS_MAIL_READER (reader)); g_signal_emit (reader, signals[UPDATE_ACTIONS], 0, state); } GtkAction * e_mail_reader_get_action (EMailReader *reader, const gchar *action_name) { GtkAction *action = NULL; gint ii; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); g_return_val_if_fail (action_name != NULL, NULL); for (ii = 0; ii < E_MAIL_READER_NUM_ACTION_GROUPS; ii++) { GtkActionGroup *group; group = e_mail_reader_get_action_group (reader, ii); action = gtk_action_group_get_action (group, action_name); if (action != NULL) break; } if (action == NULL) g_critical ( "%s: action '%s' not found", G_STRFUNC, action_name); return action; } GtkActionGroup * e_mail_reader_get_action_group (EMailReader *reader, EMailReaderActionGroup group) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_action_group != NULL, NULL); return interface->get_action_group (reader, group); } EAlertSink * e_mail_reader_get_alert_sink (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_alert_sink != NULL, NULL); return interface->get_alert_sink (reader); } EMailBackend * e_mail_reader_get_backend (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_backend != NULL, NULL); return interface->get_backend (reader); } EMailDisplay * e_mail_reader_get_mail_display (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_mail_display != NULL, NULL); return interface->get_mail_display (reader); } gboolean e_mail_reader_get_hide_deleted (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), FALSE); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_hide_deleted != NULL, FALSE); return interface->get_hide_deleted (reader); } GtkWidget * e_mail_reader_get_message_list (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_message_list != NULL, NULL); return interface->get_message_list (reader); } GtkMenu * e_mail_reader_get_popup_menu (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_popup_menu != NULL, NULL); return interface->get_popup_menu (reader); } EPreviewPane * e_mail_reader_get_preview_pane (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_preview_pane != NULL, NULL); return interface->get_preview_pane (reader); } GPtrArray * e_mail_reader_get_selected_uids (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_selected_uids != NULL, NULL); return interface->get_selected_uids (reader); } GtkWindow * e_mail_reader_get_window (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->get_window != NULL, NULL); return interface->get_window (reader); } CamelFolder * e_mail_reader_ref_folder (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), NULL); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->ref_folder != NULL, NULL); return interface->ref_folder (reader); } void e_mail_reader_set_folder (EMailReader *reader, CamelFolder *folder) { EMailReaderInterface *interface; g_return_if_fail (E_IS_MAIL_READER (reader)); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_if_fail (interface->set_folder != NULL); interface->set_folder (reader, folder); } void e_mail_reader_set_message (EMailReader *reader, const gchar *message_uid) { EMailReaderInterface *interface; g_return_if_fail (E_IS_MAIL_READER (reader)); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_if_fail (interface->set_message != NULL); interface->set_message (reader, message_uid); } guint e_mail_reader_open_selected_mail (EMailReader *reader) { EMailReaderInterface *interface; g_return_val_if_fail (E_IS_MAIL_READER (reader), 0); interface = E_MAIL_READER_GET_INTERFACE (reader); g_return_val_if_fail (interface->open_selected_mail != NULL, 0); return interface->open_selected_mail (reader); } EMailForwardStyle e_mail_reader_get_forward_style (EMailReader *reader) { EMailReaderPrivate *priv; g_return_val_if_fail (E_IS_MAIL_READER (reader), 0); priv = E_MAIL_READER_GET_PRIVATE (reader); return priv->forward_style; } void e_mail_reader_set_forward_style (EMailReader *reader, EMailForwardStyle style) { EMailReaderPrivate *priv; g_return_if_fail (E_IS_MAIL_READER (reader)); priv = E_MAIL_READER_GET_PRIVATE (reader); if (priv->forward_style == style) return; priv->forward_style = style; g_object_notify (G_OBJECT (reader), "forward-style"); } gboolean e_mail_reader_get_group_by_threads (EMailReader *reader) { EMailReaderPrivate *priv; g_return_val_if_fail (E_IS_MAIL_READER (reader), FALSE); priv = E_MAIL_READER_GET_PRIVATE (reader); return priv->group_by_threads; } void e_mail_reader_set_group_by_threads (EMailReader *reader, gboolean group_by_threads) { EMailReaderPrivate *priv; g_return_if_fail (E_IS_MAIL_READER (reader)); priv = E_MAIL_READER_GET_PRIVATE (reader); if (priv->group_by_threads == group_by_threads) return; priv->group_by_threads = group_by_threads; g_object_notify (G_OBJECT (reader), "group-by-threads"); } EMailReplyStyle e_mail_reader_get_reply_style (EMailReader *reader) { EMailReaderPrivate *priv; g_return_val_if_fail (E_IS_MAIL_READER (reader), 0); priv = E_MAIL_READER_GET_PRIVATE (reader); return priv->reply_style; } void e_mail_reader_set_reply_style (EMailReader *reader, EMailReplyStyle style) { EMailReaderPrivate *priv; g_return_if_fail (E_IS_MAIL_READER (reader)); priv = E_MAIL_READER_GET_PRIVATE (reader); if (priv->reply_style == style) return; priv->reply_style = style; g_object_notify (G_OBJECT (reader), "reply-style"); } void e_mail_reader_create_charset_menu (EMailReader *reader, GtkUIManager *ui_manager, guint merge_id) { GtkAction *action; const gchar *action_name; const gchar *path; GSList *list; g_return_if_fail (E_IS_MAIL_READER (reader)); g_return_if_fail (GTK_IS_UI_MANAGER (ui_manager)); action_name = "mail-charset-default"; action = e_mail_reader_get_action (reader, action_name); g_return_if_fail (action != NULL); list = gtk_radio_action_get_group (GTK_RADIO_ACTION (action)); list = g_slist_copy (list); list = g_slist_remove (list, action); list = g_slist_sort (list, (GCompareFunc) e_action_compare_by_label); path = "/main-menu/view-menu/mail-message-view-actions/mail-encoding-menu"; while (list != NULL) { action = list->data; gtk_ui_manager_add_ui ( ui_manager, merge_id, path, gtk_action_get_name (action), gtk_action_get_name (action), GTK_UI_MANAGER_AUTO, FALSE); list = g_slist_delete_link (list, list); } gtk_ui_manager_ensure_update (ui_manager); } void e_mail_reader_show_search_bar (EMailReader *reader) { g_return_if_fail (E_IS_MAIL_READER (reader)); g_signal_emit (reader, signals[SHOW_SEARCH_BAR], 0); } void e_mail_reader_avoid_next_mark_as_seen (EMailReader *reader) { EMailReaderPrivate *priv; MessageList *message_list; g_return_if_fail (reader != NULL); priv = E_MAIL_READER_GET_PRIVATE (reader); g_return_if_fail (priv != NULL); message_list = MESSAGE_LIST (e_mail_reader_get_message_list (reader)); g_return_if_fail (message_list != NULL); priv->avoid_next_mark_as_seen = TRUE; } /** * e_mail_reader_composer_created: * @reader: an #EMailReader * @composer: an #EMsgComposer * @message: the source #CamelMimeMessage, or %NULL * * Emits a #EMailReader::composer-created signal to indicate the @composer * window was created in response to a user action on @reader. Examples of * such actions include replying, forwarding, and composing a new message. * If applicable, the source @message (i.e. the message being replied to or * forwarded) should be included. **/ void e_mail_reader_composer_created (EMailReader *reader, EMsgComposer *composer, CamelMimeMessage *message) { g_return_if_fail (E_IS_MAIL_READER (reader)); g_return_if_fail (E_IS_MSG_COMPOSER (composer)); if (message != NULL) g_return_if_fail (CAMEL_IS_MIME_MESSAGE (message)); g_signal_emit ( reader, signals[COMPOSER_CREATED], 0, composer, message); }