1 /* Copyright (c) 2006, Nokia Corporation
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * * Neither the name of the Nokia Corporation nor the names of its
14 * contributors may be used to endorse or promote products derived from
15 * this software without specific prior written permission.
17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
18 * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
20 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
21 * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
22 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
23 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
24 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
25 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
26 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 #endif /*HAVE_CONFIG_H*/
34 #include <glib/gi18n.h>
35 #include <glib/gprintf.h>
37 #include <modest-runtime.h>
38 #include <modest-defs.h>
39 #include <modest-tny-folder.h>
40 #include <modest-tny-msg.h>
41 #include <modest-tny-account.h>
42 #include <modest-address-book.h>
43 #include "modest-error.h"
44 #include "modest-ui-actions.h"
45 #include "modest-tny-platform-factory.h"
46 #include "modest-platform.h"
47 #include "modest-debug.h"
48 #include <tny-mime-part.h>
49 #include <tny-camel-folder.h>
50 #include <tny-camel-imap-folder.h>
51 #include <tny-camel-pop-folder.h>
52 #ifdef MODEST_TOOLKIT_HILDON2
53 #include <modest-accounts-window.h>
54 #include <hildon/hildon-pannable-area.h>
55 #include <hildon/hildon-gtk.h>
56 #include <modest-header-window.h>
57 #include <modest-folder-window.h>
58 #include <modest-maemo-utils.h>
61 #ifdef MODEST_PLATFORM_MAEMO
62 #include "maemo/modest-osso-state-saving.h"
63 #endif /* MODEST_PLATFORM_MAEMO */
64 #ifndef MODEST_TOOLKIT_GTK
65 #include "maemo/modest-hildon-includes.h"
66 #include "maemo/modest-connection-specific-smtp-window.h"
67 #endif /* !MODEST_TOOLKIT_GTK */
68 #include <modest-utils.h>
70 #include "widgets/modest-ui-constants.h"
71 #include <widgets/modest-main-window.h>
72 #include <widgets/modest-msg-view-window.h>
73 #include <widgets/modest-account-view-window.h>
74 #include <widgets/modest-details-dialog.h>
75 #include <widgets/modest-attachments-view.h>
76 #include "widgets/modest-folder-view.h"
77 #include "widgets/modest-global-settings-dialog.h"
78 #include "modest-account-mgr-helpers.h"
79 #include "modest-mail-operation.h"
80 #include "modest-text-utils.h"
81 #include <modest-widget-memory.h>
82 #include <tny-error.h>
83 #include <tny-simple-list.h>
84 #include <tny-msg-view.h>
85 #include <tny-device.h>
86 #include <tny-merge-folder.h>
88 #include <gtkhtml/gtkhtml.h>
90 #define MIN_FREE_SPACE 5 * 1024 * 1024
91 #define MODEST_MOVE_TO_DIALOG_FOLDER_VIEW "move-to-dialog-folder-view"
93 typedef struct _GetMsgAsyncHelper {
95 ModestMailOperation *mail_op;
102 typedef enum _ReplyForwardAction {
106 } ReplyForwardAction;
108 typedef struct _ReplyForwardHelper {
109 guint reply_forward_type;
110 ReplyForwardAction action;
113 GtkWidget *parent_window;
115 } ReplyForwardHelper;
117 typedef struct _MoveToHelper {
118 GtkTreeRowReference *reference;
122 typedef struct _PasteAsAttachmentHelper {
123 ModestMsgEditWindow *window;
125 } PasteAsAttachmentHelper;
133 * The do_headers_action uses this kind of functions to perform some
134 * action to each member of a list of headers
136 typedef void (*HeadersFunc) (TnyHeader *header, ModestWindow *win, gpointer user_data);
138 static void do_headers_action (ModestWindow *win,
142 static void open_msg_cb (ModestMailOperation *mail_op,
149 static void reply_forward_cb (ModestMailOperation *mail_op,
156 static void reply_forward (ReplyForwardAction action, ModestWindow *win);
158 static void folder_refreshed_cb (ModestMailOperation *mail_op,
162 static void on_send_receive_finished (ModestMailOperation *mail_op,
165 static gint header_list_count_uncached_msgs (TnyList *header_list);
167 static gboolean connect_to_get_msg (ModestWindow *win,
168 gint num_of_uncached_msgs,
169 TnyAccount *account);
171 static gboolean remote_folder_has_leave_on_server (TnyFolderStore *folder);
173 static void do_create_folder (GtkWindow *window,
174 TnyFolderStore *parent_folder,
175 const gchar *suggested_name);
177 static TnyAccount *get_account_from_folder_store (TnyFolderStore *folder_store);
179 static void modest_ui_actions_on_main_window_move_to (GtkAction *action,
180 GtkWidget *folder_view,
181 TnyFolderStore *dst_folder,
182 ModestMainWindow *win);
183 #ifdef MODEST_TOOLKIT_HILDON2
184 static void modest_ui_actions_on_folder_window_move_to (GtkWidget *folder_view,
185 TnyFolderStore *dst_folder,
190 static void modest_ui_actions_on_window_move_to (GtkAction *action,
191 TnyList *list_to_move,
192 TnyFolderStore *dst_folder,
196 * This function checks whether a TnyFolderStore is a pop account
199 remote_folder_has_leave_on_server (TnyFolderStore *folder)
204 g_return_val_if_fail (TNY_IS_FOLDER_STORE (folder), FALSE);
206 account = get_account_from_folder_store (folder);
207 result = (modest_protocol_registry_protocol_type_has_leave_on_server (modest_runtime_get_protocol_registry (),
208 modest_tny_account_get_protocol_type (account)));
209 g_object_unref (account);
214 /* FIXME: this should be merged with the similar code in modest-account-view-window */
215 /* Show the account creation wizard dialog.
216 * returns: TRUE if an account was created. FALSE if the user cancelled.
219 modest_ui_actions_run_account_setup_wizard (ModestWindow *win)
221 gboolean result = FALSE;
223 gint dialog_response;
225 /* there is no such wizard yet */
226 wizard = GTK_WINDOW (modest_platform_get_account_settings_wizard ());
227 modest_window_mgr_set_modal (modest_runtime_get_window_mgr(), GTK_WINDOW (wizard), (GtkWindow *) win);
229 #ifndef MODEST_TOOLKIT_HILDON2
230 /* always present a main window in the background
231 * we do it here, so we cannot end up with two wizards (as this
232 * function might be called in modest_window_mgr_get_main_window as well */
234 win = modest_window_mgr_get_main_window (modest_runtime_get_window_mgr(),
235 TRUE); /* create if not existent */
239 ModestWindowMgr *mgr;
241 mgr = modest_runtime_get_window_mgr ();
243 window_list = modest_window_mgr_get_window_list (mgr);
244 if (window_list == NULL) {
245 ModestWindow *old_win;
246 win = MODEST_WINDOW (modest_accounts_window_new ());
247 if (modest_window_mgr_register_window (mgr, win, NULL)) {
248 gtk_widget_show_all (GTK_WIDGET (win));
250 gtk_widget_destroy (GTK_WIDGET (win));
255 win = MODEST_WINDOW (modest_folder_window_new (NULL));
256 if (modest_window_mgr_register_window (mgr, win, NULL)) {
257 gtk_widget_show_all (GTK_WIDGET (win));
259 gtk_widget_destroy (GTK_WIDGET (win));
263 g_list_free (window_list);
269 gtk_window_set_transient_for (GTK_WINDOW (wizard), GTK_WINDOW (win));
271 /* make sure the mainwindow is visible. We need to present the
272 wizard again to give it the focus back. show_all are needed
273 in order to get the widgets properly drawn (MainWindow main
274 paned won't be in its right position and the dialog will be
276 #ifndef MODEST_TOOLKIT_HILDON2
277 gtk_widget_show_all (GTK_WIDGET (win));
278 gtk_widget_show_all (GTK_WIDGET (wizard));
279 gtk_window_present (GTK_WINDOW (win));
280 gtk_window_present (GTK_WINDOW (wizard));
283 dialog_response = gtk_dialog_run (GTK_DIALOG (wizard));
284 gtk_widget_destroy (GTK_WIDGET (wizard));
285 if (gtk_events_pending ())
286 gtk_main_iteration ();
288 if (dialog_response == GTK_RESPONSE_CANCEL) {
291 /* Check whether an account was created: */
292 result = modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
299 modest_ui_actions_on_about (GtkAction *action, ModestWindow *win)
302 const gchar *authors[] = {
303 "Dirk-Jan C. Binnema <dirk-jan.binnema@nokia.com>",
306 about = gtk_about_dialog_new ();
307 gtk_about_dialog_set_name (GTK_ABOUT_DIALOG(about), PACKAGE_NAME);
308 gtk_about_dialog_set_version (GTK_ABOUT_DIALOG(about),PACKAGE_VERSION);
309 gtk_about_dialog_set_copyright (GTK_ABOUT_DIALOG(about),
310 _("Copyright (c) 2006, Nokia Corporation\n"
311 "All rights reserved."));
312 gtk_about_dialog_set_comments (GTK_ABOUT_DIALOG(about),
313 _("a modest e-mail client\n\n"
314 "design and implementation: Dirk-Jan C. Binnema\n"
315 "contributions from the fine people at KC and Ig\n"
316 "uses the tinymail email framework written by Philip van Hoof"));
317 gtk_about_dialog_set_authors (GTK_ABOUT_DIALOG(about), authors);
318 gtk_about_dialog_set_website (GTK_ABOUT_DIALOG(about), "http://modest.garage.maemo.org");
319 gtk_window_set_transient_for (GTK_WINDOW (about), GTK_WINDOW (win));
320 gtk_window_set_modal (GTK_WINDOW (about), TRUE);
322 gtk_dialog_run (GTK_DIALOG (about));
323 gtk_widget_destroy(about);
327 * Gets the list of currently selected messages. If the win is the
328 * main window, then it returns a newly allocated list of the headers
329 * selected in the header view. If win is the msg view window, then
330 * the value returned is a list with just a single header.
332 * The caller of this funcion must free the list.
335 get_selected_headers (ModestWindow *win)
337 if (MODEST_IS_MAIN_WINDOW(win)) {
338 GtkWidget *header_view;
340 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
341 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
342 return modest_header_view_get_selected_headers (MODEST_HEADER_VIEW(header_view));
344 } else if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
345 /* for MsgViewWindows, we simply return a list with one element */
347 TnyList *list = NULL;
349 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
350 if (header != NULL) {
351 list = tny_simple_list_new ();
352 tny_list_prepend (list, G_OBJECT(header));
353 g_object_unref (G_OBJECT(header));
358 #ifdef MODEST_TOOLKIT_HILDON2
359 } else if (MODEST_IS_HEADER_WINDOW (win)) {
360 GtkWidget *header_view;
362 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
363 return modest_header_view_get_selected_headers (MODEST_HEADER_VIEW(header_view));
369 static GtkTreeRowReference *
370 get_next_after_selected_headers (ModestHeaderView *header_view)
372 GtkTreeSelection *sel;
373 GList *selected_rows, *node;
375 GtkTreeRowReference *result;
378 model = gtk_tree_view_get_model (GTK_TREE_VIEW (header_view));
379 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (header_view));
380 selected_rows = gtk_tree_selection_get_selected_rows (sel, NULL);
382 if (selected_rows == NULL)
385 node = g_list_last (selected_rows);
386 path = gtk_tree_path_copy ((GtkTreePath *) node->data);
387 gtk_tree_path_next (path);
389 result = gtk_tree_row_reference_new (model, path);
391 gtk_tree_path_free (path);
392 g_list_foreach (selected_rows, (GFunc) gtk_tree_path_free, NULL);
393 g_list_free (selected_rows);
399 headers_action_mark_as_read (TnyHeader *header,
403 TnyHeaderFlags flags;
405 g_return_if_fail (TNY_IS_HEADER(header));
407 flags = tny_header_get_flags (header);
408 if (flags & TNY_HEADER_FLAG_SEEN) return;
409 tny_header_set_flag (header, TNY_HEADER_FLAG_SEEN);
413 headers_action_mark_as_unread (TnyHeader *header,
417 TnyHeaderFlags flags;
419 g_return_if_fail (TNY_IS_HEADER(header));
421 flags = tny_header_get_flags (header);
422 if (flags & TNY_HEADER_FLAG_SEEN) {
423 tny_header_unset_flag (header, TNY_HEADER_FLAG_SEEN);
427 /** After deleing a message that is currently visible in a window,
428 * show the next message from the list, or close the window if there are no more messages.
431 modest_ui_actions_refresh_message_window_after_delete (ModestMsgViewWindow* win)
433 /* Close msg view window or select next */
434 if (!modest_msg_view_window_select_next_message (win) &&
435 !modest_msg_view_window_select_previous_message (win)) {
437 g_signal_emit_by_name (G_OBJECT (win), "delete-event", NULL, &ret_value);
443 modest_ui_actions_on_delete_message (GtkAction *action, ModestWindow *win)
445 modest_ui_actions_on_edit_mode_delete_message (win);
449 modest_ui_actions_on_edit_mode_delete_message (ModestWindow *win)
451 TnyList *header_list = NULL;
452 TnyIterator *iter = NULL;
453 TnyHeader *header = NULL;
454 gchar *message = NULL;
457 ModestWindowMgr *mgr;
458 GtkWidget *header_view = NULL;
459 gboolean retval = TRUE;
461 g_return_val_if_fail (MODEST_IS_WINDOW(win), FALSE);
463 /* Check first if the header view has the focus */
464 if (MODEST_IS_MAIN_WINDOW (win)) {
466 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
467 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
468 if (!gtk_widget_is_focus (header_view))
472 /* Get the headers, either from the header view (if win is the main window),
473 * or from the message view window: */
474 header_list = get_selected_headers (win);
475 if (!header_list) return FALSE;
477 /* Check if any of the headers are already opened, or in the process of being opened */
478 if (MODEST_IS_MAIN_WINDOW (win)) {
479 gint opened_headers = 0;
481 iter = tny_list_create_iterator (header_list);
482 mgr = modest_runtime_get_window_mgr ();
483 while (!tny_iterator_is_done (iter)) {
484 header = TNY_HEADER (tny_iterator_get_current (iter));
486 if (modest_window_mgr_find_registered_header (mgr, header, NULL))
488 g_object_unref (header);
490 tny_iterator_next (iter);
492 g_object_unref (iter);
494 if (opened_headers > 0) {
497 msg = g_strdup_printf (_("mcen_nc_unable_to_delete_n_messages"),
500 modest_platform_run_information_dialog (GTK_WINDOW (win), (const gchar *) msg, FALSE);
503 g_object_unref (header_list);
509 if (tny_list_get_length(header_list) == 1) {
510 iter = tny_list_create_iterator (header_list);
511 header = TNY_HEADER (tny_iterator_get_current (iter));
514 subject = tny_header_dup_subject (header);
516 subject = g_strdup (_("mail_va_no_subject"));
517 desc = g_strdup_printf ("%s", subject);
519 g_object_unref (header);
522 g_object_unref (iter);
524 message = g_strdup_printf(ngettext("emev_nc_delete_message", "emev_nc_delete_messages",
525 tny_list_get_length(header_list)), desc);
527 /* Confirmation dialog */
528 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
532 if (response == GTK_RESPONSE_OK) {
533 ModestWindow *main_window = NULL;
534 ModestWindowMgr *mgr = NULL;
535 GtkTreeModel *model = NULL;
536 GtkTreeSelection *sel = NULL;
537 GList *sel_list = NULL, *tmp = NULL;
538 GtkTreeRowReference *next_row_reference = NULL;
539 GtkTreeRowReference *prev_row_reference = NULL;
540 GtkTreePath *next_path = NULL;
541 GtkTreePath *prev_path = NULL;
542 ModestMailOperation *mail_op = NULL;
544 /* Find last selected row */
545 if (MODEST_IS_MAIN_WINDOW (win)) {
546 model = gtk_tree_view_get_model (GTK_TREE_VIEW (header_view));
547 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (header_view));
548 sel_list = gtk_tree_selection_get_selected_rows (sel, &model);
549 for (tmp=sel_list; tmp; tmp=tmp->next) {
550 if (tmp->next == NULL) {
551 prev_path = gtk_tree_path_copy((GtkTreePath *) tmp->data);
552 next_path = gtk_tree_path_copy((GtkTreePath *) tmp->data);
554 gtk_tree_path_prev (prev_path);
555 gtk_tree_path_next (next_path);
557 prev_row_reference = gtk_tree_row_reference_new (model, prev_path);
558 next_row_reference = gtk_tree_row_reference_new (model, next_path);
563 /* Disable window dimming management */
564 modest_window_disable_dimming (MODEST_WINDOW(win));
566 /* Remove each header. If it's a view window header_view == NULL */
567 mail_op = modest_mail_operation_new ((GObject *) win);
568 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
570 modest_mail_operation_remove_msgs (mail_op, header_list, FALSE);
571 g_object_unref (mail_op);
573 /* Enable window dimming management */
575 gtk_tree_selection_unselect_all (sel);
577 modest_window_enable_dimming (MODEST_WINDOW(win));
579 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
580 modest_ui_actions_refresh_message_window_after_delete (MODEST_MSG_VIEW_WINDOW (win));
582 /* Get main window */
583 mgr = modest_runtime_get_window_mgr ();
584 main_window = modest_window_mgr_get_main_window (mgr, FALSE); /* don't create */
585 } else if (MODEST_IS_MAIN_WINDOW (win)) {
586 /* Move cursor to next row */
589 /* Select next or previous row */
590 if (gtk_tree_row_reference_valid (next_row_reference)) {
591 gtk_tree_selection_select_path (sel, next_path);
593 else if (gtk_tree_row_reference_valid (prev_row_reference)) {
594 gtk_tree_selection_select_path (sel, prev_path);
598 if (gtk_tree_row_reference_valid (next_row_reference))
599 gtk_tree_row_reference_free (next_row_reference);
600 if (next_path != NULL)
601 gtk_tree_path_free (next_path);
602 if (gtk_tree_row_reference_valid (prev_row_reference))
603 gtk_tree_row_reference_free (prev_row_reference);
604 if (prev_path != NULL)
605 gtk_tree_path_free (prev_path);
608 /* Update toolbar dimming state */
610 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
611 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
615 g_list_foreach (sel_list, (GFunc) gtk_tree_path_free, NULL);
616 g_list_free (sel_list);
625 g_object_unref (header_list);
633 /* delete either message or folder, based on where we are */
635 modest_ui_actions_on_delete_message_or_folder (GtkAction *action, ModestWindow *win)
637 g_return_if_fail (MODEST_IS_WINDOW(win));
639 /* Check first if the header view has the focus */
640 if (MODEST_IS_MAIN_WINDOW (win)) {
642 w = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
643 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
644 if (gtk_widget_is_focus (w)) {
645 modest_ui_actions_on_delete_folder (action, MODEST_WINDOW(win));
649 modest_ui_actions_on_delete_message (action, win);
653 modest_ui_actions_on_quit (GtkAction *action, ModestWindow *win)
655 ModestWindowMgr *mgr = NULL;
657 #ifdef MODEST_PLATFORM_MAEMO
658 modest_osso_save_state();
659 #endif /* MODEST_PLATFORM_MAEMO */
661 g_debug ("closing down, clearing %d item(s) from operation queue",
662 modest_mail_operation_queue_num_elements
663 (modest_runtime_get_mail_operation_queue()));
665 /* cancel all outstanding operations */
666 modest_mail_operation_queue_cancel_all
667 (modest_runtime_get_mail_operation_queue());
669 g_debug ("queue has been cleared");
672 /* Check if there are opened editing windows */
673 mgr = modest_runtime_get_window_mgr ();
674 modest_window_mgr_close_all_windows (mgr);
676 /* note: when modest-tny-account-store is finalized,
677 it will automatically set all network connections
680 /* gtk_main_quit (); */
684 modest_ui_actions_on_close_window (GtkAction *action, ModestWindow *win)
688 g_signal_emit_by_name (G_OBJECT (win), "delete-event", NULL, &ret_value);
690 /* if (MODEST_IS_MSG_VIEW_WINDOW (win)) { */
691 /* gtk_widget_destroy (GTK_WIDGET (win)); */
692 /* } else if (MODEST_IS_MSG_EDIT_WINDOW (win)) { */
693 /* gboolean ret_value; */
694 /* g_signal_emit_by_name (G_OBJECT (win), "delete-event", NULL, &ret_value); */
695 /* } else if (MODEST_IS_WINDOW (win)) { */
696 /* gtk_widget_destroy (GTK_WIDGET (win)); */
698 /* g_return_if_reached (); */
703 modest_ui_actions_add_to_contacts (GtkAction *action, ModestWindow *win)
705 g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (win));
707 modest_msg_view_window_add_to_contacts (MODEST_MSG_VIEW_WINDOW (win));
711 modest_ui_actions_on_add_to_contacts (GtkAction *action, ModestWindow *win)
713 GtkClipboard *clipboard = NULL;
714 gchar *selection = NULL;
716 clipboard = gtk_clipboard_get (GDK_SELECTION_PRIMARY);
717 selection = gtk_clipboard_wait_for_text (clipboard);
719 /* Question: why is the clipboard being used here?
720 * It doesn't really make a lot of sense. */
724 modest_address_book_add_address (selection);
730 modest_ui_actions_on_new_account (GtkAction *action,
731 ModestWindow *window)
733 if (!modest_ui_actions_run_account_setup_wizard (window)) {
734 g_debug ("%s: wizard was already running", __FUNCTION__);
739 modest_ui_actions_on_accounts (GtkAction *action,
742 /* This is currently only implemented for Maemo */
743 if (!modest_account_mgr_has_accounts (modest_runtime_get_account_mgr(), TRUE)) {
744 if (!modest_ui_actions_run_account_setup_wizard (win))
745 g_debug ("%s: wizard was already running", __FUNCTION__);
749 /* Show the list of accounts */
750 GtkWindow *account_win = GTK_WINDOW (modest_account_view_window_new ());
752 /* The accounts dialog must be modal */
753 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (), GTK_WINDOW (account_win), (GtkWindow *) win);
754 modest_utils_show_dialog_and_forget (GTK_WINDOW (win), GTK_DIALOG (account_win));
759 modest_ui_actions_on_smtp_servers (GtkAction *action, ModestWindow *win)
761 /* This is currently only implemented for Maemo,
762 * because it requires an API (libconic) to detect different connection
765 #ifndef MODEST_TOOLKIT_GTK /* Defined in config.h */
767 /* Create the window if necessary: */
768 GtkWidget *specific_window = GTK_WIDGET (modest_connection_specific_smtp_window_new ());
769 modest_connection_specific_smtp_window_fill_with_connections (
770 MODEST_CONNECTION_SPECIFIC_SMTP_WINDOW (specific_window),
771 modest_runtime_get_account_mgr());
773 /* Show the window: */
774 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (),
775 GTK_WINDOW (specific_window), (GtkWindow *) win);
776 gtk_widget_show (specific_window);
777 #endif /* !MODEST_TOOLKIT_GTK */
781 modest_ui_actions_compose_msg(ModestWindow *win,
784 const gchar *bcc_str,
785 const gchar *subject_str,
786 const gchar *body_str,
788 gboolean set_as_modified)
790 gchar *account_name = NULL;
791 const gchar *mailbox;
793 TnyAccount *account = NULL;
794 TnyFolder *folder = NULL;
795 gchar *from_str = NULL, *signature = NULL, *body = NULL;
796 gboolean use_signature = FALSE;
797 ModestWindow *msg_win = NULL;
798 ModestAccountMgr *mgr = modest_runtime_get_account_mgr();
799 ModestTnyAccountStore *store = modest_runtime_get_account_store();
800 GnomeVFSFileSize total_size, allowed_size;
802 /* we check for low-mem */
803 if (modest_platform_check_memory_low (win, TRUE))
806 #ifdef MODEST_TOOLKIT_HILDON2
807 account_name = g_strdup (modest_window_get_active_account(win));
810 account_name = modest_account_mgr_get_default_account(mgr);
813 g_printerr ("modest: no account found\n");
817 mailbox = modest_window_get_active_mailbox (win);
818 account = modest_tny_account_store_get_server_account (store, account_name, TNY_ACCOUNT_TYPE_STORE);
820 g_printerr ("modest: failed to get tnyaccount for '%s'\n", account_name);
823 folder = modest_tny_account_get_special_folder (account, TNY_FOLDER_TYPE_DRAFTS);
825 g_printerr ("modest: failed to find Drafts folder\n");
828 from_str = modest_account_mgr_get_from_string (mgr, account_name, mailbox);
830 g_printerr ("modest: failed get from string for '%s'\n", account_name);
834 signature = modest_account_mgr_get_signature (mgr, account_name, &use_signature);
835 if (body_str != NULL) {
836 body = use_signature ? g_strconcat(body_str, "\n--\n", signature, NULL) : g_strdup(body_str);
838 body = use_signature ? g_strconcat("\n--\n", signature, NULL) : g_strdup("");
841 msg = modest_tny_msg_new (to_str, from_str, cc_str, bcc_str, subject_str, NULL, NULL, body, NULL, NULL, NULL);
843 g_printerr ("modest: failed to create new msg\n");
847 /* Create and register edit window */
848 /* This is destroyed by TODO. */
850 allowed_size = MODEST_MAX_ATTACHMENT_SIZE;
851 msg_win = modest_msg_edit_window_new (msg, account_name, mailbox, FALSE);
853 if (!modest_window_mgr_register_window (modest_runtime_get_window_mgr(), msg_win, win)) {
854 gtk_widget_destroy (GTK_WIDGET (msg_win));
857 modest_msg_edit_window_set_modified (MODEST_MSG_EDIT_WINDOW (msg_win), set_as_modified);
858 gtk_widget_show_all (GTK_WIDGET (msg_win));
860 while (attachments) {
862 modest_msg_edit_window_attach_file_one((ModestMsgEditWindow *)msg_win,
863 attachments->data, allowed_size);
865 if (total_size > allowed_size) {
866 g_warning ("%s: total size: %u",
867 __FUNCTION__, (unsigned int)total_size);
870 allowed_size -= total_size;
872 attachments = g_slist_next(attachments);
879 g_free (account_name);
881 g_object_unref (G_OBJECT(account));
883 g_object_unref (G_OBJECT(folder));
885 g_object_unref (G_OBJECT(msg));
889 modest_ui_actions_on_new_msg (GtkAction *action, ModestWindow *win)
891 /* if there are no accounts yet, just show the wizard */
892 if (!modest_account_mgr_has_accounts (modest_runtime_get_account_mgr(), TRUE))
893 if (!modest_ui_actions_run_account_setup_wizard (win))
896 modest_ui_actions_compose_msg(win, NULL, NULL, NULL, NULL, NULL, NULL, FALSE);
901 modest_ui_actions_msg_retrieval_check (ModestMailOperation *mail_op,
905 ModestMailOperationStatus status;
907 /* If there is no message or the operation was not successful */
908 status = modest_mail_operation_get_status (mail_op);
909 if (!msg || status != MODEST_MAIL_OPERATION_STATUS_SUCCESS) {
912 /* If it's a memory low issue, then show a banner */
913 error = modest_mail_operation_get_error (mail_op);
914 if (error && error->domain == MODEST_MAIL_OPERATION_ERROR &&
915 error->code == MODEST_MAIL_OPERATION_ERROR_LOW_MEMORY) {
916 GObject *source = modest_mail_operation_get_source (mail_op);
917 modest_platform_run_information_dialog (GTK_IS_WINDOW (source) ? GTK_WINDOW (source) : NULL,
918 _KR("memr_ib_operation_disabled"),
920 g_object_unref (source);
923 if (error && ((error->code == TNY_SERVICE_ERROR_NO_SUCH_MESSAGE) ||
924 error->code == TNY_SERVICE_ERROR_MESSAGE_NOT_AVAILABLE)) {
925 gchar *subject, *msg, *format = NULL;
927 subject = tny_header_dup_subject (header);
929 subject = g_strdup (_("mail_va_no_subject"));
931 account = modest_mail_operation_get_account (mail_op);
933 ModestProtocol *protocol;
934 ModestProtocolType proto;
935 proto = modest_tny_account_get_protocol_type (account);
936 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (), proto);
938 format = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
939 g_object_unref (account);
943 format = g_strdup (_("emev_ni_ui_imap_message_not_available_in_server"));
945 msg = g_strdup_printf (format, subject);
946 modest_platform_run_information_dialog (NULL, msg, FALSE);
952 /* Remove the header from the preregistered uids */
953 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
971 OpenMsgBannerInfo *banner_info;
972 GtkTreeRowReference *rowref;
976 open_msg_banner_idle (gpointer userdata)
978 OpenMsgBannerInfo *banner_info = (OpenMsgBannerInfo *) userdata;
980 gdk_threads_enter ();
981 banner_info->idle_handler = 0;
982 banner_info->banner = modest_platform_animation_banner (NULL, NULL, banner_info->message);
983 if (banner_info->banner)
984 g_object_ref (banner_info->banner);
986 gdk_threads_leave ();
992 get_header_view_from_window (ModestWindow *window)
994 GtkWidget *header_view;
996 if (MODEST_IS_MAIN_WINDOW (window)) {
997 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
998 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
999 #ifdef MODEST_TOOLKIT_HILDON2
1000 } else if (MODEST_IS_HEADER_WINDOW (window)){
1001 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (window)));
1011 get_info_from_header (TnyHeader *header, gboolean *is_draft, gboolean *can_open)
1014 gchar *account = NULL;
1015 TnyFolderType folder_type = TNY_FOLDER_TYPE_UNKNOWN;
1020 folder = tny_header_get_folder (header);
1021 /* Gets folder type (OUTBOX headers will be opened in edit window */
1022 if (modest_tny_folder_is_local_folder (folder)) {
1023 folder_type = modest_tny_folder_get_local_or_mmc_folder_type (folder);
1024 if (folder_type == TNY_FOLDER_TYPE_INVALID)
1025 g_warning ("%s: BUG: TNY_FOLDER_TYPE_INVALID", __FUNCTION__);
1028 if (folder_type == TNY_FOLDER_TYPE_OUTBOX) {
1029 TnyTransportAccount *traccount = NULL;
1030 ModestTnyAccountStore *accstore = modest_runtime_get_account_store();
1031 traccount = modest_tny_account_store_get_transport_account_from_outbox_header(accstore, header);
1033 ModestTnySendQueue *send_queue = NULL;
1034 ModestTnySendQueueStatus status;
1036 account = g_strdup(modest_tny_account_get_parent_modest_account_name_for_server_account(
1037 TNY_ACCOUNT(traccount)));
1038 send_queue = modest_runtime_get_send_queue(traccount, TRUE);
1039 if (TNY_IS_SEND_QUEUE (send_queue)) {
1040 msg_id = modest_tny_send_queue_get_msg_id (header);
1041 status = modest_tny_send_queue_get_msg_status(send_queue, msg_id);
1043 /* Only open messages in outbox with the editor if they are in Failed state */
1044 if (status == MODEST_TNY_SEND_QUEUE_FAILED) {
1047 #ifdef MODEST_TOOLKIT_HILDON2
1049 /* In Fremantle we can not
1050 open any message from
1051 outbox which is not in
1057 g_object_unref(traccount);
1059 g_warning("Cannot get transport account for message in outbox!!");
1061 } else if (folder_type == TNY_FOLDER_TYPE_DRAFTS) {
1062 *is_draft = TRUE; /* Open in editor if the message is in the Drafts folder */
1066 TnyAccount *acc = tny_folder_get_account (folder);
1069 g_strdup (modest_tny_account_get_parent_modest_account_name_for_server_account (acc));
1070 g_object_unref (acc);
1074 g_object_unref (folder);
1080 open_msg_cb (ModestMailOperation *mail_op,
1087 ModestWindowMgr *mgr = NULL;
1088 ModestWindow *parent_win = NULL;
1089 ModestWindow *win = NULL;
1090 gchar *account = NULL;
1091 gboolean open_in_editor = FALSE;
1093 OpenMsgHelper *helper = (OpenMsgHelper *) user_data;
1095 /* Do nothing if there was any problem with the mail
1096 operation. The error will be shown by the error_handler of
1097 the mail operation */
1098 if (!modest_ui_actions_msg_retrieval_check (mail_op, header, msg))
1101 parent_win = (ModestWindow *) modest_mail_operation_get_source (mail_op);
1103 /* Mark header as read */
1104 headers_action_mark_as_read (header, MODEST_WINDOW(parent_win), NULL);
1106 account = get_info_from_header (header, &open_in_editor, &can_open);
1110 account = g_strdup (modest_window_get_active_account (MODEST_WINDOW (parent_win)));
1112 account = modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
1114 if (open_in_editor) {
1115 ModestAccountMgr *mgr = modest_runtime_get_account_mgr ();
1116 gchar *from_header = NULL, *acc_name;
1117 gchar *mailbox = NULL;
1119 from_header = tny_header_dup_from (header);
1121 /* we cannot edit without a valid account... */
1122 if (!modest_account_mgr_has_accounts(mgr, TRUE)) {
1123 if (!modest_ui_actions_run_account_setup_wizard(parent_win)) {
1124 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1126 g_free (from_header);
1131 acc_name = modest_utils_get_account_name_from_recipient (from_header, &mailbox);
1132 g_free (from_header);
1138 win = modest_msg_edit_window_new (msg, account, mailbox, TRUE);
1142 gchar *uid = modest_tny_folder_get_header_unique_id (header);
1143 const gchar *mailbox = NULL;
1145 if (parent_win && MODEST_IS_WINDOW (parent_win))
1146 mailbox = modest_window_get_active_mailbox (MODEST_WINDOW (parent_win));
1148 if (helper->rowref && helper->model) {
1149 win = modest_msg_view_window_new_with_header_model (msg, account, mailbox, (const gchar*) uid,
1150 helper->model, helper->rowref);
1152 win = modest_msg_view_window_new_for_attachment (msg, account, mailbox, (const gchar*) uid);
1157 /* Register and show new window */
1159 mgr = modest_runtime_get_window_mgr ();
1160 if (!modest_window_mgr_register_window (mgr, win, NULL)) {
1161 gtk_widget_destroy (GTK_WIDGET (win));
1164 gtk_widget_show_all (GTK_WIDGET(win));
1167 /* Update toolbar dimming state */
1168 if (MODEST_IS_MAIN_WINDOW (parent_win)) {
1169 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (parent_win));
1175 g_object_unref (parent_win);
1179 is_memory_full_error (GError *error, ModestMailOperation *mail_op)
1181 gboolean enough_free_space = TRUE;
1182 GnomeVFSURI *cache_dir_uri;
1183 const gchar *cache_dir = NULL;
1184 GnomeVFSFileSize free_space;
1185 TnyAccountStore *acc_store;
1187 acc_store = TNY_ACCOUNT_STORE (modest_runtime_get_account_store ());
1189 /* Cache dir is different in case we're using an external storage (like MMC account) */
1191 TnyAccount *account = modest_mail_operation_get_account (mail_op);
1193 if (modest_tny_account_is_memory_card_account (account)) {
1194 cache_dir = g_getenv (MODEST_MMC1_VOLUMEPATH_ENV);
1196 g_object_unref (account);
1200 /* Get the default local cache dir */
1202 cache_dir = tny_account_store_get_cache_dir (acc_store);
1204 cache_dir_uri = gnome_vfs_uri_new (cache_dir);
1205 if (cache_dir_uri) {
1206 if (gnome_vfs_get_volume_free_space (cache_dir_uri, &free_space) == GNOME_VFS_OK) {
1207 if (free_space < MIN_FREE_SPACE)
1208 enough_free_space = FALSE;
1210 gnome_vfs_uri_unref (cache_dir_uri);
1213 if ((error->code == TNY_SYSTEM_ERROR_MEMORY ||
1214 /* When asking for a mail and no space left on device
1215 tinymail returns this error */
1216 error->code == TNY_SERVICE_ERROR_MESSAGE_NOT_AVAILABLE ||
1217 /* When the folder summary could not be read or
1219 error->code == TNY_IO_ERROR_WRITE ||
1220 error->code == TNY_IO_ERROR_READ) &&
1221 !enough_free_space) {
1229 check_memory_full_error (GtkWidget *parent_window, GError *err)
1234 if (is_memory_full_error (err, NULL))
1235 modest_platform_information_banner (parent_window,
1236 NULL, _KR("cerm_device_memory_full"));
1237 else if (err->code == TNY_SYSTEM_ERROR_MEMORY)
1238 /* If the account was created in memory full
1239 conditions then tinymail won't be able to
1240 connect so it'll return this error code */
1241 modest_platform_information_banner (parent_window,
1242 NULL, _("emev_ui_imap_inbox_select_error"));
1250 modest_ui_actions_disk_operations_error_handler (ModestMailOperation *mail_op,
1253 const GError *error;
1254 GObject *win = NULL;
1255 ModestMailOperationStatus status;
1257 win = modest_mail_operation_get_source (mail_op);
1258 error = modest_mail_operation_get_error (mail_op);
1259 status = modest_mail_operation_get_status (mail_op);
1261 /* If the mail op has been cancelled then it's not an error:
1262 don't show any message */
1263 if (status != MODEST_MAIL_OPERATION_STATUS_CANCELED) {
1264 if (is_memory_full_error ((GError *) error, mail_op)) {
1265 modest_platform_information_banner ((GtkWidget *) win,
1266 NULL, _KR("cerm_device_memory_full"));
1267 } else if (error->code == TNY_SYSTEM_ERROR_MEMORY) {
1268 modest_platform_information_banner ((GtkWidget *) win,
1269 NULL, _("emev_ui_imap_inbox_select_error"));
1270 } else if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
1271 error->code == MODEST_MAIL_OPERATION_ERROR_FILE_IO) {
1272 modest_platform_information_banner ((GtkWidget *) win,
1273 NULL, _CS ("sfil_ni_unable_to_open_file_not_found"));
1274 } else if (user_data) {
1275 modest_platform_information_banner ((GtkWidget *) win,
1281 g_object_unref (win);
1285 * Returns the account a list of headers belongs to. It returns a
1286 * *new* reference so don't forget to unref it
1289 get_account_from_header_list (TnyList *headers)
1291 TnyAccount *account = NULL;
1293 if (tny_list_get_length (headers) > 0) {
1294 TnyIterator *iter = tny_list_create_iterator (headers);
1295 TnyHeader *header = TNY_HEADER (tny_iterator_get_current (iter));
1296 TnyFolder *folder = tny_header_get_folder (header);
1299 g_object_unref (header);
1301 while (!tny_iterator_is_done (iter)) {
1302 header = TNY_HEADER (tny_iterator_get_current (iter));
1303 folder = tny_header_get_folder (header);
1306 g_object_unref (header);
1308 tny_iterator_next (iter);
1313 account = tny_folder_get_account (folder);
1314 g_object_unref (folder);
1318 g_object_unref (header);
1320 g_object_unref (iter);
1326 get_account_from_header (TnyHeader *header)
1328 TnyAccount *account = NULL;
1331 folder = tny_header_get_folder (header);
1334 account = tny_folder_get_account (folder);
1335 g_object_unref (folder);
1341 open_msg_helper_destroyer (gpointer user_data)
1343 OpenMsgHelper *helper = (OpenMsgHelper *) user_data;
1345 if (helper->banner_info) {
1346 g_free (helper->banner_info->message);
1347 if (helper->banner_info->idle_handler > 0) {
1348 g_source_remove (helper->banner_info->idle_handler);
1349 helper->banner_info->idle_handler = 0;
1351 if (helper->banner_info->banner != NULL) {
1352 gtk_widget_destroy (helper->banner_info->banner);
1353 g_object_unref (helper->banner_info->banner);
1354 helper->banner_info->banner = NULL;
1356 g_slice_free (OpenMsgBannerInfo, helper->banner_info);
1357 helper->banner_info = NULL;
1359 g_object_unref (helper->model);
1360 g_object_unref (helper->header);
1361 gtk_tree_row_reference_free (helper->rowref);
1362 g_slice_free (OpenMsgHelper, helper);
1366 open_msg_performer(gboolean canceled,
1368 GtkWindow *parent_window,
1369 TnyAccount *account,
1372 ModestMailOperation *mail_op = NULL;
1373 gchar *error_msg = NULL;
1374 ModestProtocolType proto;
1375 TnyConnectionStatus status;
1376 OpenMsgHelper *helper = NULL;
1377 ModestProtocol *protocol;
1378 ModestProtocolRegistry *protocol_registry;
1381 helper = (OpenMsgHelper *) user_data;
1383 status = tny_account_get_connection_status (account);
1384 if (err || canceled) {
1385 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (), helper->header);
1386 /* Free the helper */
1387 open_msg_helper_destroyer (helper);
1389 /* In memory full conditions we could get this error here */
1390 check_memory_full_error ((GtkWidget *) parent_window, err);
1395 /* Get the error message depending on the protocol */
1396 proto = modest_tny_account_get_protocol_type (account);
1397 if (proto == MODEST_PROTOCOL_REGISTRY_TYPE_INVALID) {
1398 proto = MODEST_PROTOCOLS_STORE_MAILDIR;
1401 protocol_registry = modest_runtime_get_protocol_registry ();
1402 subject = tny_header_dup_subject (helper->header);
1404 protocol = modest_protocol_registry_get_protocol_by_type (protocol_registry, proto);
1405 error_msg = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
1409 if (error_msg == NULL) {
1410 error_msg = g_strdup (_("mail_ni_ui_folder_get_msg_folder_error"));
1413 #ifndef MODEST_TOOLKIT_HILDON2
1414 gboolean show_open_draft = FALSE;
1415 if (modest_protocol_registry_protocol_type_has_tag (protocol_registry,
1417 MODEST_PROTOCOL_REGISTRY_LOCAL_STORE_PROTOCOLS)) {
1419 TnyFolderType folder_type;
1421 folder = tny_header_get_folder (helper->header);
1422 folder_type = modest_tny_folder_get_local_or_mmc_folder_type (folder);
1423 show_open_draft = (folder_type == TNY_FOLDER_TYPE_DRAFTS);
1424 g_object_unref (folder);
1428 #ifdef MODEST_TOOLKIT_HILDON2
1431 gchar *account_name = get_info_from_header (helper->header, &is_draft, &can_open);
1434 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (), helper->header);
1435 g_free (account_name);
1436 open_msg_helper_destroyer (helper);
1441 ModestWindow *window;
1442 GtkWidget *header_view;
1445 header_view = get_header_view_from_window (MODEST_WINDOW (parent_window));
1446 uid = modest_tny_folder_get_header_unique_id (helper->header);
1448 const gchar *mailbox = NULL;
1449 mailbox = modest_window_get_active_mailbox (MODEST_WINDOW (parent_window));
1450 window = modest_msg_view_window_new_from_header_view
1451 (MODEST_HEADER_VIEW (header_view), account_name, mailbox, uid, helper->rowref);
1452 if (window != NULL) {
1453 if (!modest_window_mgr_register_window (modest_runtime_get_window_mgr (),
1455 gtk_widget_destroy (GTK_WIDGET (window));
1457 gtk_widget_show_all (GTK_WIDGET(window));
1461 g_free (account_name);
1463 open_msg_helper_destroyer (helper);
1466 g_free (account_name);
1468 /* Create the mail operation */
1470 modest_mail_operation_new_with_error_handling ((GObject *) parent_window,
1471 modest_ui_actions_disk_operations_error_handler,
1472 g_strdup (error_msg), g_free);
1473 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
1477 #ifndef MODEST_TOOLKIT_HILDON2
1478 if (show_open_draft) {
1479 helper->banner_info = g_slice_new (OpenMsgBannerInfo);
1480 helper->banner_info->message = g_strdup (_("mail_ib_opening_draft_message"));
1481 helper->banner_info->banner = NULL;
1482 helper->banner_info->idle_handler = g_timeout_add (500, open_msg_banner_idle,
1483 helper->banner_info);
1489 headers = TNY_LIST (tny_simple_list_new ());
1490 tny_list_prepend (headers, G_OBJECT (helper->header));
1491 modest_mail_operation_get_msgs_full (mail_op,
1495 open_msg_helper_destroyer);
1496 g_object_unref (headers);
1503 g_object_unref (mail_op);
1504 g_object_unref (account);
1508 * This function is used by both modest_ui_actions_on_open and
1509 * modest_ui_actions_on_header_activated. This way we always do the
1510 * same when trying to open messages.
1513 open_msg_from_header (TnyHeader *header, GtkTreeRowReference *rowref, ModestWindow *win)
1515 ModestWindowMgr *mgr = NULL;
1516 TnyAccount *account;
1517 gboolean cached = FALSE;
1519 GtkWidget *header_view = NULL;
1520 OpenMsgHelper *helper;
1521 ModestWindow *window;
1523 g_return_if_fail (header != NULL && rowref != NULL);
1525 mgr = modest_runtime_get_window_mgr ();
1528 header_view = get_header_view_from_window (MODEST_WINDOW (win));
1529 if (header_view == NULL)
1532 /* Get the account */
1533 account = get_account_from_header (header);
1538 found = modest_window_mgr_find_registered_header (mgr, header, &window);
1540 /* Do not open again the message and present the
1541 window to the user */
1544 #ifndef MODEST_TOOLKIT_HILDON2
1545 gtk_window_present (GTK_WINDOW (window));
1548 /* the header has been registered already, we don't do
1549 * anything but wait for the window to come up*/
1550 g_debug ("header %p already registered, waiting for window", header);
1555 /* Open each message */
1556 cached = tny_header_get_flags (header) & TNY_HEADER_FLAG_CACHED;
1558 /* Allways download if we are online. */
1559 if (!tny_device_is_online (modest_runtime_get_device ())) {
1562 /* If ask for user permission to download the messages */
1563 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1564 _("mcen_nc_get_msg"));
1566 /* End if the user does not want to continue */
1567 if (response == GTK_RESPONSE_CANCEL) {
1573 /* We register the window for opening */
1574 modest_window_mgr_register_header (mgr, header, NULL);
1576 /* Create the helper. We need to get a reference to the model
1577 here because it could change while the message is readed
1578 (the user could switch between folders) */
1579 helper = g_slice_new (OpenMsgHelper);
1580 helper->model = g_object_ref (gtk_tree_view_get_model (GTK_TREE_VIEW (header_view)));
1581 helper->header = g_object_ref (header);
1582 helper->rowref = gtk_tree_row_reference_copy (rowref);
1583 helper->banner_info = NULL;
1585 /* Connect to the account and perform */
1587 modest_platform_connect_and_perform ((GtkWindow *) win, TRUE, g_object_ref (account),
1588 open_msg_performer, helper);
1590 /* Call directly the performer, do not need to connect */
1591 open_msg_performer (FALSE, NULL, (GtkWindow *) win,
1592 g_object_ref (account), helper);
1597 g_object_unref (account);
1601 modest_ui_actions_on_open (GtkAction *action, ModestWindow *win)
1608 /* we check for low-mem; in that case, show a warning, and don't allow
1611 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
1615 headers = get_selected_headers (win);
1619 headers_count = tny_list_get_length (headers);
1620 if (headers_count != 1) {
1621 if (headers_count > 1) {
1622 /* Don't allow activation if there are more than one message selected */
1623 modest_platform_information_banner (NULL, NULL, _("mcen_ib_select_one_message"));
1626 g_object_unref (headers);
1630 iter = tny_list_create_iterator (headers);
1631 header = TNY_HEADER (tny_iterator_get_current (iter));
1632 g_object_unref (iter);
1636 open_msg_from_header (header, NULL, win);
1637 g_object_unref (header);
1640 g_object_unref(headers);
1644 rf_helper_window_closed (gpointer data,
1647 ReplyForwardHelper *helper = (ReplyForwardHelper *) data;
1649 helper->parent_window = NULL;
1652 static ReplyForwardHelper*
1653 create_reply_forward_helper (ReplyForwardAction action,
1655 guint reply_forward_type,
1658 ReplyForwardHelper *rf_helper = NULL;
1659 const gchar *active_acc = modest_window_get_active_account (win);
1660 const gchar *active_mailbox = modest_window_get_active_mailbox (win);
1662 rf_helper = g_slice_new0 (ReplyForwardHelper);
1663 rf_helper->reply_forward_type = reply_forward_type;
1664 rf_helper->action = action;
1665 rf_helper->parent_window = (MODEST_IS_WINDOW (win)) ? GTK_WIDGET (win) : NULL;
1666 rf_helper->header = (header) ? g_object_ref (header) : NULL;
1667 rf_helper->account_name = (active_acc) ?
1668 g_strdup (active_acc) :
1669 modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
1670 rf_helper->mailbox = g_strdup (active_mailbox);
1672 /* Note that window could be destroyed just AFTER calling
1673 register_window so we must ensure that this pointer does
1674 not hold invalid references */
1675 if (rf_helper->parent_window)
1676 g_object_weak_ref (G_OBJECT (rf_helper->parent_window),
1677 rf_helper_window_closed, rf_helper);
1683 free_reply_forward_helper (gpointer data)
1685 ReplyForwardHelper *helper;
1687 helper = (ReplyForwardHelper *) data;
1688 g_free (helper->account_name);
1689 g_free (helper->mailbox);
1691 g_object_unref (helper->header);
1692 if (helper->parent_window)
1693 g_object_weak_unref (G_OBJECT (helper->parent_window),
1694 rf_helper_window_closed, helper);
1695 g_slice_free (ReplyForwardHelper, helper);
1699 reply_forward_cb (ModestMailOperation *mail_op,
1706 TnyMsg *new_msg = NULL;
1707 ReplyForwardHelper *rf_helper;
1708 ModestWindow *msg_win = NULL;
1709 ModestEditType edit_type;
1711 TnyAccount *account = NULL;
1712 ModestWindowMgr *mgr = NULL;
1713 gchar *signature = NULL;
1714 gboolean use_signature;
1716 /* If there was any error. The mail operation could be NULL,
1717 this means that we already have the message downloaded and
1718 that we didn't do a mail operation to retrieve it */
1719 rf_helper = (ReplyForwardHelper *) user_data;
1720 if (mail_op && !modest_ui_actions_msg_retrieval_check (mail_op, header, msg))
1723 from = modest_account_mgr_get_from_string (modest_runtime_get_account_mgr(),
1724 rf_helper->account_name, rf_helper->mailbox);
1725 signature = modest_account_mgr_get_signature (modest_runtime_get_account_mgr(),
1726 rf_helper->account_name,
1729 /* Create reply mail */
1730 switch (rf_helper->action) {
1733 modest_tny_msg_create_reply_msg (msg, header, from,
1734 (use_signature) ? signature : NULL,
1735 rf_helper->reply_forward_type,
1736 MODEST_TNY_MSG_REPLY_MODE_SENDER);
1738 case ACTION_REPLY_TO_ALL:
1740 modest_tny_msg_create_reply_msg (msg, header, from,
1741 (use_signature) ? signature : NULL,
1742 rf_helper->reply_forward_type,
1743 MODEST_TNY_MSG_REPLY_MODE_ALL);
1744 edit_type = MODEST_EDIT_TYPE_REPLY;
1746 case ACTION_FORWARD:
1748 modest_tny_msg_create_forward_msg (msg, from, (use_signature) ? signature : NULL,
1749 rf_helper->reply_forward_type);
1750 edit_type = MODEST_EDIT_TYPE_FORWARD;
1753 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1755 g_return_if_reached ();
1763 g_warning ("%s: failed to create message\n", __FUNCTION__);
1767 account = modest_tny_account_store_get_server_account (modest_runtime_get_account_store(),
1768 rf_helper->account_name,
1769 TNY_ACCOUNT_TYPE_STORE);
1771 g_warning ("%s: failed to get tnyaccount for '%s'\n", __FUNCTION__, rf_helper->account_name);
1775 /* Create and register the windows */
1776 msg_win = modest_msg_edit_window_new (new_msg, rf_helper->account_name, rf_helper->mailbox, FALSE);
1777 mgr = modest_runtime_get_window_mgr ();
1778 modest_window_mgr_register_window (mgr, msg_win, (ModestWindow *) rf_helper->parent_window);
1780 /* Note that register_window could have deleted the account */
1781 if (MODEST_IS_WINDOW (rf_helper->parent_window)) {
1782 gdouble parent_zoom;
1784 parent_zoom = modest_window_get_zoom (MODEST_WINDOW (rf_helper->parent_window));
1785 modest_window_set_zoom (msg_win, parent_zoom);
1788 /* Show edit window */
1789 gtk_widget_show_all (GTK_WIDGET (msg_win));
1792 /* We always unregister the header because the message is
1793 forwarded or replied so the original one is no longer
1795 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1798 g_object_unref (G_OBJECT (new_msg));
1800 g_object_unref (G_OBJECT (account));
1801 free_reply_forward_helper (rf_helper);
1804 /* Checks a list of headers. If any of them are not currently
1805 * downloaded (CACHED) then returns TRUE else returns FALSE.
1808 header_list_count_uncached_msgs (TnyList *header_list)
1811 gint uncached_messages = 0;
1813 iter = tny_list_create_iterator (header_list);
1814 while (!tny_iterator_is_done (iter)) {
1817 header = TNY_HEADER (tny_iterator_get_current (iter));
1819 if (!(tny_header_get_flags (header) & TNY_HEADER_FLAG_CACHED))
1820 uncached_messages ++;
1821 g_object_unref (header);
1824 tny_iterator_next (iter);
1826 g_object_unref (iter);
1828 return uncached_messages;
1831 /* Returns FALSE if the user does not want to download the
1832 * messages. Returns TRUE if the user allowed the download.
1835 connect_to_get_msg (ModestWindow *win,
1836 gint num_of_uncached_msgs,
1837 TnyAccount *account)
1839 GtkResponseType response;
1841 /* Allways download if we are online. */
1842 if (tny_device_is_online (modest_runtime_get_device ()))
1845 /* If offline, then ask for user permission to download the messages */
1846 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1847 ngettext("mcen_nc_get_msg",
1849 num_of_uncached_msgs));
1851 if (response == GTK_RESPONSE_CANCEL)
1854 return modest_platform_connect_and_wait((GtkWindow *) win, account);
1858 reply_forward_performer (gboolean canceled,
1860 GtkWindow *parent_window,
1861 TnyAccount *account,
1864 ReplyForwardHelper *rf_helper = NULL;
1865 ModestMailOperation *mail_op;
1867 rf_helper = (ReplyForwardHelper *) user_data;
1869 if (canceled || err) {
1870 free_reply_forward_helper (rf_helper);
1874 /* Retrieve the message */
1875 modest_window_mgr_register_header (modest_runtime_get_window_mgr (), rf_helper->header, NULL);
1876 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT (parent_window),
1877 modest_ui_actions_disk_operations_error_handler,
1879 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
1880 modest_mail_operation_get_msg (mail_op, rf_helper->header, TRUE, reply_forward_cb, rf_helper);
1883 g_object_unref(mail_op);
1887 * Common code for the reply and forward actions
1890 reply_forward (ReplyForwardAction action, ModestWindow *win)
1892 ReplyForwardHelper *rf_helper = NULL;
1893 guint reply_forward_type;
1895 g_return_if_fail (MODEST_IS_WINDOW(win));
1897 /* we check for low-mem; in that case, show a warning, and don't allow
1898 * reply/forward (because it could potentially require a lot of memory */
1899 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
1903 /* we need an account when editing */
1904 if (!modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE)) {
1905 if (!modest_ui_actions_run_account_setup_wizard (win))
1909 reply_forward_type =
1910 modest_conf_get_int (modest_runtime_get_conf (),
1911 (action == ACTION_FORWARD) ?
1912 MODEST_CONF_FORWARD_TYPE :
1913 MODEST_CONF_REPLY_TYPE,
1916 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
1918 TnyHeader *header = NULL;
1919 /* Get header and message. Do not free them here, the
1920 reply_forward_cb must do it */
1921 msg = modest_msg_view_window_get_message (MODEST_MSG_VIEW_WINDOW(win));
1922 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
1924 if (msg && header) {
1926 rf_helper = create_reply_forward_helper (action, win,
1927 reply_forward_type, header);
1928 reply_forward_cb (NULL, header, FALSE, msg, NULL, rf_helper);
1930 g_warning("%s: no message or header found in viewer\n", __FUNCTION__);
1934 g_object_unref (msg);
1936 g_object_unref (header);
1938 TnyHeader *header = NULL;
1940 gboolean do_retrieve = TRUE;
1941 TnyList *header_list = NULL;
1943 header_list = get_selected_headers (win);
1946 /* Check that only one message is selected for replying */
1947 if (tny_list_get_length (header_list) != 1) {
1948 modest_platform_information_banner ((win) ? GTK_WIDGET (win) : NULL,
1949 NULL, _("mcen_ib_select_one_message"));
1950 g_object_unref (header_list);
1954 /* Only reply/forward to one message */
1955 iter = tny_list_create_iterator (header_list);
1956 header = TNY_HEADER (tny_iterator_get_current (iter));
1957 g_object_unref (iter);
1959 /* Retrieve messages */
1960 do_retrieve = (action == ACTION_FORWARD) ||
1961 (reply_forward_type != MODEST_TNY_MSG_REPLY_TYPE_CITE);
1964 TnyAccount *account = NULL;
1965 TnyFolder *folder = NULL;
1966 gdouble download = TRUE;
1967 guint uncached_msgs = 0;
1969 folder = tny_header_get_folder (header);
1971 goto do_retrieve_frees;
1972 account = tny_folder_get_account (folder);
1974 goto do_retrieve_frees;
1976 uncached_msgs = header_list_count_uncached_msgs (header_list);
1978 if (uncached_msgs > 0) {
1979 /* Allways download if we are online. */
1980 if (!tny_device_is_online (modest_runtime_get_device ())) {
1983 /* If ask for user permission to download the messages */
1984 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1985 ngettext("mcen_nc_get_msg",
1989 /* End if the user does not want to continue */
1990 if (response == GTK_RESPONSE_CANCEL)
1997 rf_helper = create_reply_forward_helper (action, win,
1998 reply_forward_type, header);
1999 if (uncached_msgs > 0) {
2000 modest_platform_connect_and_perform (GTK_WINDOW (win),
2002 reply_forward_performer,
2005 reply_forward_performer (FALSE, NULL, GTK_WINDOW (win),
2006 account, rf_helper);
2011 g_object_unref (account);
2013 g_object_unref (folder);
2015 reply_forward_cb (NULL, header, FALSE, NULL, NULL, rf_helper);
2018 g_object_unref (header_list);
2019 g_object_unref (header);
2024 modest_ui_actions_on_reply (GtkAction *action, ModestWindow *win)
2026 g_return_if_fail (MODEST_IS_WINDOW(win));
2028 reply_forward (ACTION_REPLY, win);
2032 modest_ui_actions_on_forward (GtkAction *action, ModestWindow *win)
2034 g_return_if_fail (MODEST_IS_WINDOW(win));
2036 reply_forward (ACTION_FORWARD, win);
2040 modest_ui_actions_on_reply_all (GtkAction *action, ModestWindow *win)
2042 g_return_if_fail (MODEST_IS_WINDOW(win));
2044 reply_forward (ACTION_REPLY_TO_ALL, win);
2048 modest_ui_actions_on_next (GtkAction *action,
2049 ModestWindow *window)
2051 if (MODEST_IS_MAIN_WINDOW (window)) {
2052 GtkWidget *header_view;
2054 header_view = modest_main_window_get_child_widget (
2055 MODEST_MAIN_WINDOW(window),
2056 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2060 modest_header_view_select_next (
2061 MODEST_HEADER_VIEW(header_view));
2062 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
2063 modest_msg_view_window_select_next_message (
2064 MODEST_MSG_VIEW_WINDOW (window));
2066 g_return_if_reached ();
2071 modest_ui_actions_on_prev (GtkAction *action,
2072 ModestWindow *window)
2074 g_return_if_fail (MODEST_IS_WINDOW(window));
2076 if (MODEST_IS_MAIN_WINDOW (window)) {
2077 GtkWidget *header_view;
2078 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(window),
2079 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2083 modest_header_view_select_prev (MODEST_HEADER_VIEW(header_view));
2084 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
2085 modest_msg_view_window_select_previous_message (MODEST_MSG_VIEW_WINDOW (window));
2087 g_return_if_reached ();
2092 modest_ui_actions_on_sort (GtkAction *action,
2093 ModestWindow *window)
2095 GtkWidget *header_view = NULL;
2097 g_return_if_fail (MODEST_IS_WINDOW(window));
2099 if (MODEST_IS_MAIN_WINDOW (window)) {
2100 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(window),
2101 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2102 #ifdef MODEST_TOOLKIT_HILDON2
2103 } else if (MODEST_IS_HEADER_WINDOW (window)) {
2104 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (window)));
2109 modest_platform_information_banner (NULL, NULL, _CS("ckdg_ib_nothing_to_sort"));
2114 /* Show sorting dialog */
2115 modest_utils_run_sort_dialog (GTK_WINDOW (window), MODEST_SORT_HEADERS);
2119 new_messages_arrived (ModestMailOperation *self,
2120 TnyList *new_headers,
2124 gboolean show_visual_notifications;
2126 source = modest_mail_operation_get_source (self);
2127 show_visual_notifications = (source) ? FALSE : TRUE;
2129 g_object_unref (source);
2131 /* Notify new messages have been downloaded. If the
2132 send&receive was invoked by the user then do not show any
2133 visual notification, only play a sound and activate the LED
2134 (for the Maemo version) */
2135 if (TNY_IS_LIST(new_headers) && (tny_list_get_length (new_headers)) > 0)
2136 modest_platform_on_new_headers_received (new_headers,
2137 show_visual_notifications);
2142 retrieve_all_messages_cb (GObject *source,
2144 guint retrieve_limit)
2150 window = GTK_WINDOW (source);
2151 msg = g_strdup_printf (_("mail_nc_msg_count_limit_exceeded"),
2152 num_msgs, retrieve_limit);
2154 /* Ask the user if they want to retrieve all the messages */
2156 modest_platform_run_confirmation_dialog_with_buttons (window, msg,
2157 _("mcen_bd_get_all"),
2158 _("mcen_bd_newest_only"));
2159 /* Free and return */
2161 return (response == GTK_RESPONSE_ACCEPT) ? TRUE : FALSE;
2165 TnyAccount *account;
2167 gchar *account_name;
2168 gboolean poke_status;
2169 gboolean interactive;
2170 ModestMailOperation *mail_op;
2174 do_send_receive_performer (gboolean canceled,
2176 GtkWindow *parent_window,
2177 TnyAccount *account,
2180 SendReceiveInfo *info;
2182 info = (SendReceiveInfo *) user_data;
2184 if (err || canceled) {
2185 /* In memory full conditions we could get this error here */
2186 check_memory_full_error ((GtkWidget *) parent_window, err);
2188 if (info->mail_op) {
2189 modest_mail_operation_queue_remove (modest_runtime_get_mail_operation_queue (),
2195 /* Set send/receive operation in progress */
2196 if (info->win && MODEST_IS_MAIN_WINDOW (info->win)) {
2197 modest_main_window_notify_send_receive_initied (MODEST_MAIN_WINDOW (info->win));
2200 if (info->win && MODEST_IS_MAIN_WINDOW (info->win))
2201 g_signal_connect (G_OBJECT (info->mail_op), "operation-finished",
2202 G_CALLBACK (on_send_receive_finished),
2205 /* Send & receive. */
2206 modest_mail_operation_update_account (info->mail_op, info->account_name, info->poke_status, info->interactive,
2207 (info->win) ? retrieve_all_messages_cb : NULL,
2208 new_messages_arrived, info->win);
2213 g_object_unref (G_OBJECT (info->mail_op));
2214 if (info->account_name)
2215 g_free (info->account_name);
2217 g_object_unref (info->win);
2219 g_object_unref (info->account);
2220 g_slice_free (SendReceiveInfo, info);
2224 * This function performs the send & receive required actions. The
2225 * window is used to create the mail operation. Typically it should
2226 * always be the main window, but we pass it as argument in order to
2230 modest_ui_actions_do_send_receive (const gchar *account_name,
2231 gboolean force_connection,
2232 gboolean poke_status,
2233 gboolean interactive,
2236 gchar *acc_name = NULL;
2237 SendReceiveInfo *info;
2238 ModestTnyAccountStore *acc_store;
2240 /* If no account name was provided then get the current account, and if
2241 there is no current account then pick the default one: */
2242 if (!account_name) {
2244 acc_name = g_strdup (modest_window_get_active_account (win));
2246 acc_name = modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
2248 g_printerr ("modest: cannot get default account\n");
2252 acc_name = g_strdup (account_name);
2255 acc_store = modest_runtime_get_account_store ();
2257 /* Create the info for the connect and perform */
2258 info = g_slice_new (SendReceiveInfo);
2259 info->account_name = acc_name;
2260 info->win = (win) ? g_object_ref (win) : NULL;
2261 info->poke_status = poke_status;
2262 info->interactive = interactive;
2263 info->account = modest_tny_account_store_get_server_account (acc_store, acc_name,
2264 TNY_ACCOUNT_TYPE_STORE);
2265 /* We need to create the operation here, because otherwise it
2266 could happen that the queue emits the queue-empty signal
2267 while we're trying to connect the account */
2268 info->mail_op = modest_mail_operation_new_with_error_handling ((info->win) ? G_OBJECT (info->win) : NULL,
2269 modest_ui_actions_disk_operations_error_handler,
2271 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), info->mail_op);
2273 /* Invoke the connect and perform */
2274 modest_platform_connect_and_perform ((win) ? GTK_WINDOW (win) : NULL,
2275 force_connection, info->account,
2276 do_send_receive_performer, info);
2281 modest_ui_actions_do_cancel_send (const gchar *account_name,
2284 TnyTransportAccount *transport_account;
2285 TnySendQueue *send_queue = NULL;
2286 GError *error = NULL;
2288 /* Get transport account */
2290 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
2291 (modest_runtime_get_account_store(),
2293 TNY_ACCOUNT_TYPE_TRANSPORT));
2294 if (!transport_account) {
2295 g_printerr ("modest: no transport account found for '%s'\n", account_name);
2300 send_queue = TNY_SEND_QUEUE (modest_runtime_get_send_queue (transport_account, TRUE));
2301 if (!TNY_IS_SEND_QUEUE(send_queue)) {
2302 g_set_error (&error, MODEST_MAIL_OPERATION_ERROR,
2303 MODEST_MAIL_OPERATION_ERROR_ITEM_NOT_FOUND,
2304 "modest: could not find send queue for account\n");
2306 /* Cancel the current send */
2307 tny_account_cancel (TNY_ACCOUNT (transport_account));
2309 /* Suspend all pending messages */
2310 tny_send_queue_cancel (send_queue, TNY_SEND_QUEUE_CANCEL_ACTION_SUSPEND, &error);
2314 if (transport_account != NULL)
2315 g_object_unref (G_OBJECT (transport_account));
2319 modest_ui_actions_cancel_send_all (ModestWindow *win)
2321 GSList *account_names, *iter;
2323 account_names = modest_account_mgr_account_names (modest_runtime_get_account_mgr(),
2326 iter = account_names;
2328 modest_ui_actions_do_cancel_send ((const char*) iter->data, win);
2329 iter = g_slist_next (iter);
2332 modest_account_mgr_free_account_names (account_names);
2333 account_names = NULL;
2337 modest_ui_actions_cancel_send (GtkAction *action, ModestWindow *win)
2340 /* Check if accounts exist */
2341 gboolean accounts_exist =
2342 modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
2344 /* If not, allow the user to create an account before trying to send/receive. */
2345 if (!accounts_exist)
2346 modest_ui_actions_on_accounts (NULL, win);
2348 /* Cancel all sending operaitons */
2349 modest_ui_actions_cancel_send_all (win);
2353 * Refreshes all accounts. This function will be used by automatic
2357 modest_ui_actions_do_send_receive_all (ModestWindow *win,
2358 gboolean force_connection,
2359 gboolean poke_status,
2360 gboolean interactive)
2362 GSList *account_names, *iter;
2364 account_names = modest_account_mgr_account_names (modest_runtime_get_account_mgr(),
2367 iter = account_names;
2369 modest_ui_actions_do_send_receive ((const char*) iter->data,
2371 poke_status, interactive, win);
2372 iter = g_slist_next (iter);
2375 modest_account_mgr_free_account_names (account_names);
2376 account_names = NULL;
2380 * Handler of the click on Send&Receive button in the main toolbar
2383 modest_ui_actions_on_send_receive (GtkAction *action, ModestWindow *win)
2385 /* Check if accounts exist */
2386 gboolean accounts_exist;
2389 modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
2391 /* If not, allow the user to create an account before trying to send/receive. */
2392 if (!accounts_exist)
2393 modest_ui_actions_on_accounts (NULL, win);
2395 /* Refresh the current folder. The if is always TRUE it's just an extra check */
2396 if (MODEST_IS_MAIN_WINDOW (win)) {
2397 GtkWidget *folder_view;
2398 TnyFolderStore *folder_store;
2401 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
2402 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2406 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
2409 g_object_unref (folder_store);
2410 /* Refresh the active account. Force the connection if needed
2411 and poke the status of all folders */
2412 modest_ui_actions_do_send_receive (NULL, TRUE, TRUE, TRUE, win);
2413 #ifdef MODEST_TOOLKIT_HILDON2
2414 } else if (MODEST_IS_ACCOUNTS_WINDOW (win)) {
2415 modest_ui_actions_do_send_receive_all (win, TRUE, TRUE, TRUE);
2418 const gchar *active_account;
2419 active_account = modest_window_get_active_account (MODEST_WINDOW (win));
2421 modest_ui_actions_do_send_receive (active_account, TRUE, TRUE, TRUE, win);
2428 modest_ui_actions_toggle_header_list_view (GtkAction *action, ModestMainWindow *main_window)
2431 GtkWidget *header_view;
2433 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2435 header_view = modest_main_window_get_child_widget (main_window,
2436 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2440 conf = modest_runtime_get_conf ();
2442 /* what is saved/restored is depending on the style; thus; we save with
2443 * old style, then update the style, and restore for this new style
2445 modest_widget_memory_save (conf, G_OBJECT(header_view), MODEST_CONF_HEADER_VIEW_KEY);
2447 if (modest_header_view_get_style
2448 (MODEST_HEADER_VIEW(header_view)) == MODEST_HEADER_VIEW_STYLE_DETAILS)
2449 modest_header_view_set_style (MODEST_HEADER_VIEW(header_view),
2450 MODEST_HEADER_VIEW_STYLE_TWOLINES);
2452 modest_header_view_set_style (MODEST_HEADER_VIEW(header_view),
2453 MODEST_HEADER_VIEW_STYLE_DETAILS);
2455 modest_widget_memory_restore (conf, G_OBJECT(header_view),
2456 MODEST_CONF_HEADER_VIEW_KEY);
2461 modest_ui_actions_on_header_selected (ModestHeaderView *header_view,
2463 ModestMainWindow *main_window)
2465 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2466 g_return_if_fail (MODEST_IS_HEADER_VIEW (header_view));
2468 /* in the case the folder is empty, show the empty folder message and focus
2470 if (!header && gtk_widget_is_focus (GTK_WIDGET (header_view))) {
2471 if (modest_header_view_is_empty (header_view)) {
2472 TnyFolder *folder = modest_header_view_get_folder (header_view);
2473 GtkWidget *folder_view =
2474 modest_main_window_get_child_widget (main_window,
2475 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2476 if (folder != NULL) {
2477 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view), folder, FALSE);
2478 g_object_unref (folder);
2480 gtk_widget_grab_focus (GTK_WIDGET (folder_view));
2484 /* If no header has been selected then exit */
2489 if (!gtk_widget_is_focus (GTK_WIDGET(header_view)))
2490 gtk_widget_grab_focus (GTK_WIDGET(header_view));
2492 /* Update toolbar dimming state */
2493 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
2494 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
2498 modest_ui_actions_on_header_activated (ModestHeaderView *header_view,
2501 ModestWindow *window)
2503 GtkWidget *open_widget;
2504 GtkTreeRowReference *rowref;
2506 g_return_if_fail (MODEST_IS_WINDOW(window));
2507 g_return_if_fail (MODEST_IS_HEADER_VIEW (header_view));
2508 g_return_if_fail (TNY_IS_HEADER (header));
2510 if (modest_header_view_count_selected_headers (header_view) > 1) {
2511 /* Don't allow activation if there are more than one message selected */
2512 modest_platform_information_banner (NULL, NULL, _("mcen_ib_select_one_message"));
2516 /* we check for low-mem; in that case, show a warning, and don't allow
2517 * activating headers
2519 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
2522 if (MODEST_IS_MAIN_WINDOW (window)) {
2523 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (window));
2524 open_widget = modest_window_get_action_widget (MODEST_WINDOW (window), "/MenuBar/EmailMenu/EmailOpenMenu");
2525 if (!GTK_WIDGET_IS_SENSITIVE (open_widget))
2529 rowref = gtk_tree_row_reference_new (gtk_tree_view_get_model (GTK_TREE_VIEW (header_view)), path);
2530 open_msg_from_header (header, rowref, MODEST_WINDOW (window));
2531 gtk_tree_row_reference_free (rowref);
2535 set_active_account_from_tny_account (TnyAccount *account,
2536 ModestWindow *window)
2538 const gchar *server_acc_name = tny_account_get_id (account);
2540 /* We need the TnyAccount provided by the
2541 account store because that is the one that
2542 knows the name of the Modest account */
2543 TnyAccount *modest_server_account =
2544 modest_tny_account_store_get_tny_account_by (modest_runtime_get_account_store (),
2545 MODEST_TNY_ACCOUNT_STORE_QUERY_ID,
2547 if (!modest_server_account) {
2548 g_warning ("%s: could not get tny account\n", __FUNCTION__);
2552 /* Update active account, but only if it's not a pseudo-account */
2553 if ((!modest_tny_account_is_virtual_local_folders(modest_server_account)) &&
2554 (!modest_tny_account_is_memory_card_account(modest_server_account))) {
2555 const gchar *modest_acc_name =
2556 modest_tny_account_get_parent_modest_account_name_for_server_account (modest_server_account);
2557 if (modest_acc_name)
2558 modest_window_set_active_account (window, modest_acc_name);
2561 g_object_unref (modest_server_account);
2566 folder_refreshed_cb (ModestMailOperation *mail_op,
2570 ModestMainWindow *win = NULL;
2571 GtkWidget *folder_view;
2572 const GError *error;
2574 g_return_if_fail (TNY_IS_FOLDER (folder));
2576 win = MODEST_MAIN_WINDOW (user_data);
2578 /* Check if the operation failed due to memory low conditions */
2579 error = modest_mail_operation_get_error (mail_op);
2580 if (error && error->domain == MODEST_MAIL_OPERATION_ERROR &&
2581 error->code == MODEST_MAIL_OPERATION_ERROR_LOW_MEMORY) {
2582 modest_platform_run_information_dialog (GTK_WINDOW (win),
2583 _KR("memr_ib_operation_disabled"),
2589 modest_main_window_get_child_widget(win, MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2592 TnyFolderStore *current_folder;
2594 current_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
2595 if (current_folder) {
2596 gboolean different = ((TnyFolderStore *) folder != current_folder);
2597 g_object_unref (current_folder);
2603 /* Check if folder is empty and set headers view contents style */
2604 if (tny_folder_get_all_count (folder) == 0)
2605 modest_main_window_set_contents_style (win,
2606 MODEST_MAIN_WINDOW_CONTENTS_STYLE_EMPTY);
2611 modest_ui_actions_on_folder_selection_changed (ModestFolderView *folder_view,
2612 TnyFolderStore *folder_store,
2614 ModestMainWindow *main_window)
2617 GtkWidget *header_view;
2619 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2621 header_view = modest_main_window_get_child_widget(main_window,
2622 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2626 conf = modest_runtime_get_conf ();
2628 if (TNY_IS_ACCOUNT (folder_store)) {
2630 set_active_account_from_tny_account (TNY_ACCOUNT (folder_store), MODEST_WINDOW (main_window));
2632 /* Show account details */
2633 modest_main_window_set_contents_style (main_window, MODEST_MAIN_WINDOW_CONTENTS_STYLE_DETAILS);
2636 if (TNY_IS_FOLDER (folder_store) && selected) {
2637 TnyAccount *account;
2638 const gchar *account_name = NULL;
2640 /* Update the active account */
2641 account = modest_tny_folder_get_account (TNY_FOLDER (folder_store));
2643 set_active_account_from_tny_account (account, MODEST_WINDOW (main_window));
2645 modest_tny_account_get_parent_modest_account_name_for_server_account (account);
2646 g_object_unref (account);
2650 /* Set the header style by default, it could
2651 be changed later by the refresh callback to
2653 modest_main_window_set_contents_style (main_window,
2654 MODEST_MAIN_WINDOW_CONTENTS_STYLE_HEADERS);
2656 /* Set folder on header view. This function
2657 will call tny_folder_refresh_async so we
2658 pass a callback that will be called when
2659 finished. We use that callback to set the
2660 empty view if there are no messages */
2661 modest_header_view_set_folder (MODEST_HEADER_VIEW(header_view),
2662 TNY_FOLDER (folder_store),
2664 MODEST_WINDOW (main_window),
2665 folder_refreshed_cb,
2668 /* Restore configuration. We need to do this
2669 *after* the set_folder because the widget
2670 memory asks the header view about its
2672 modest_widget_memory_restore (modest_runtime_get_conf (),
2673 G_OBJECT(header_view),
2674 MODEST_CONF_HEADER_VIEW_KEY);
2676 /* No need to save the header view
2677 configuration for Maemo because it only
2678 saves the sorting stuff and that it's
2679 already being done by the sort
2680 dialog. Remove it when the GNOME version
2681 has the same behaviour */
2682 #ifdef MODEST_TOOLKIT_GTK
2683 if (modest_main_window_get_contents_style (main_window) ==
2684 MODEST_MAIN_WINDOW_CONTENTS_STYLE_HEADERS)
2685 modest_widget_memory_save (conf, G_OBJECT (header_view),
2686 MODEST_CONF_HEADER_VIEW_KEY);
2688 modest_header_view_clear (MODEST_HEADER_VIEW(header_view));
2692 /* Update dimming state */
2693 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
2694 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
2698 modest_ui_actions_on_item_not_found (ModestHeaderView *header_view,ModestItemType type,
2705 item = (type == MODEST_ITEM_TYPE_FOLDER) ? "folder" : "message";
2707 online = tny_device_is_online (modest_runtime_get_device());
2710 /* already online -- the item is simply not there... */
2711 dialog = gtk_message_dialog_new (GTK_WINDOW (win),
2713 GTK_MESSAGE_WARNING,
2715 _("The %s you selected cannot be found"),
2717 gtk_dialog_add_button (GTK_DIALOG (dialog),_("mcen_bd_dialog_ok"), GTK_RESPONSE_ACCEPT);
2718 gtk_dialog_run (GTK_DIALOG(dialog));
2720 dialog = gtk_dialog_new_with_buttons (_("Connection requested"),
2723 _("mcen_bd_dialog_cancel"),
2724 GTK_RESPONSE_REJECT,
2725 _("mcen_bd_dialog_ok"),
2726 GTK_RESPONSE_ACCEPT,
2728 txt = g_strdup_printf (_("This %s is not available in offline mode.\n"
2729 "Do you want to get online?"), item);
2730 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox),
2731 gtk_label_new (txt), FALSE, FALSE, 0);
2732 gtk_widget_show_all (GTK_WIDGET(GTK_DIALOG(dialog)->vbox));
2735 gtk_window_set_default_size (GTK_WINDOW(dialog), 300, 300);
2736 if (gtk_dialog_run (GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
2737 /* TODO: Comment about why is this commented out: */
2738 /* modest_platform_connect_and_wait (); */
2741 gtk_widget_destroy (dialog);
2745 modest_ui_actions_on_msg_link_hover (ModestMsgView *msgview, const gchar* link,
2748 /* g_message ("%s %s", __FUNCTION__, link); */
2753 modest_ui_actions_on_msg_link_clicked (ModestMsgView *msgview, const gchar* link,
2756 modest_platform_activate_uri (link);
2760 modest_ui_actions_on_msg_link_contextual (ModestMsgView *msgview, const gchar* link,
2763 modest_platform_show_uri_popup (link);
2767 modest_ui_actions_on_msg_attachment_clicked (ModestMsgView *msgview, TnyMimePart *mime_part,
2770 /* we check for low-mem; in that case, show a warning, and don't allow
2771 * viewing attachments
2773 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
2776 modest_msg_view_window_view_attachment (MODEST_MSG_VIEW_WINDOW (win), mime_part);
2780 modest_ui_actions_on_msg_recpt_activated (ModestMsgView *msgview,
2781 const gchar *address,
2784 /* g_message ("%s %s", __FUNCTION__, address); */
2788 on_save_to_drafts_cb (ModestMailOperation *mail_op,
2789 TnyMsg *saved_draft,
2792 ModestMsgEditWindow *edit_window;
2794 /* TODO: in hildon 2 we have to dim and undim the header views while we're saving */
2795 #ifndef MODEST_TOOLKIT_HILDON2
2796 ModestMainWindow *win;
2798 /* FIXME. Make the header view sensitive again. This is a
2799 * temporary hack. See modest_ui_actions_on_save_to_drafts()
2801 win = MODEST_MAIN_WINDOW(modest_window_mgr_get_main_window(
2802 modest_runtime_get_window_mgr(), FALSE));
2804 GtkWidget *hdrview = modest_main_window_get_child_widget(
2805 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2806 if (hdrview) gtk_widget_set_sensitive(hdrview, TRUE);
2810 edit_window = MODEST_MSG_EDIT_WINDOW (user_data);
2812 /* Set draft is there was no error */
2813 if (!modest_mail_operation_get_error (mail_op))
2814 modest_msg_edit_window_set_draft (edit_window, saved_draft);
2816 g_object_unref(edit_window);
2820 enough_space_for_message (ModestMsgEditWindow *edit_window,
2823 TnyAccountStore *acc_store;
2824 guint64 available_disk, expected_size;
2829 acc_store = TNY_ACCOUNT_STORE (modest_runtime_get_account_store());
2830 available_disk = modest_utils_get_available_space (NULL);
2831 modest_msg_edit_window_get_parts_size (edit_window, &parts_count, &parts_size);
2832 expected_size = modest_tny_msg_estimate_size (data->plain_body,
2837 /* Double check: memory full condition or message too big */
2838 if (available_disk < MIN_FREE_SPACE ||
2839 expected_size > available_disk) {
2841 modest_platform_information_banner (NULL, NULL,
2842 _KR("cerm_device_memory_full"));
2847 * djcb: if we're in low-memory state, we only allow for
2848 * saving messages smaller than
2849 * MODEST_MAX_LOW_MEMORY_MESSAGE_SIZE (see modest-defs.h) this
2850 * should still allow for sending anything critical...
2852 if ((expected_size > MODEST_MAX_LOW_MEMORY_MESSAGE_SIZE) &&
2853 modest_platform_check_memory_low (MODEST_WINDOW(edit_window), TRUE))
2857 * djcb: we also make sure that the attachments are smaller than the max size
2858 * this is for the case where we'd try to forward a message with attachments
2859 * bigger than our max allowed size, or sending an message from drafts which
2860 * somehow got past our checks when attaching.
2862 if (expected_size > MODEST_MAX_ATTACHMENT_SIZE) {
2863 modest_platform_run_information_dialog (
2864 GTK_WINDOW(edit_window),
2865 _KR("memr_ib_operation_disabled"),
2874 modest_ui_actions_on_save_to_drafts (GtkWidget *widget, ModestMsgEditWindow *edit_window)
2876 TnyTransportAccount *transport_account;
2877 ModestMailOperation *mail_operation;
2879 gchar *account_name;
2880 ModestAccountMgr *account_mgr;
2881 gboolean had_error = FALSE;
2882 ModestMainWindow *win = NULL;
2884 g_return_val_if_fail (MODEST_IS_MSG_EDIT_WINDOW(edit_window), FALSE);
2886 data = modest_msg_edit_window_get_msg_data (edit_window);
2889 if (!enough_space_for_message (edit_window, data)) {
2890 modest_msg_edit_window_free_msg_data (edit_window, data);
2894 account_name = g_strdup (data->account_name);
2895 account_mgr = modest_runtime_get_account_mgr();
2897 account_name = g_strdup(modest_window_get_active_account (MODEST_WINDOW(edit_window)));
2899 account_name = modest_account_mgr_get_default_account (account_mgr);
2900 if (!account_name) {
2901 g_printerr ("modest: no account found\n");
2902 modest_msg_edit_window_free_msg_data (edit_window, data);
2906 if (!strcmp (account_name, MODEST_LOCAL_FOLDERS_ACCOUNT_ID)) {
2907 account_name = g_strdup (data->account_name);
2911 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
2912 (modest_runtime_get_account_store (),
2914 TNY_ACCOUNT_TYPE_TRANSPORT));
2915 if (!transport_account) {
2916 g_printerr ("modest: no transport account found for '%s'\n", account_name);
2917 g_free (account_name);
2918 modest_msg_edit_window_free_msg_data (edit_window, data);
2922 /* Create the mail operation */
2923 mail_operation = modest_mail_operation_new_with_error_handling (NULL, modest_ui_actions_disk_operations_error_handler,
2925 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_operation);
2927 modest_mail_operation_save_to_drafts (mail_operation,
2939 data->priority_flags,
2942 on_save_to_drafts_cb,
2943 g_object_ref(edit_window));
2945 #ifdef MODEST_TOOLKIT_HILDON2
2946 /* In hildon2 we always show the information banner on saving to drafts.
2947 * It will be a system information banner in this case.
2949 gchar *text = g_strdup_printf (_("mail_va_saved_to_drafts"), _("mcen_me_folder_drafts"));
2950 modest_platform_information_banner (NULL, NULL, text);
2953 /* Use the main window as the parent of the banner, if the
2954 main window does not exist it won't be shown, if the parent
2955 window exists then it's properly shown. We don't use the
2956 editor window because it could be closed (save to drafts
2957 could happen after closing the window */
2958 win = (ModestMainWindow *)
2959 modest_window_mgr_get_main_window( modest_runtime_get_window_mgr(), FALSE);
2961 gchar *text = g_strdup_printf (_("mail_va_saved_to_drafts"), _("mcen_me_folder_drafts"));
2962 modest_platform_information_banner (GTK_WIDGET (win), NULL, text);
2966 modest_msg_edit_window_set_modified (edit_window, FALSE);
2969 g_free (account_name);
2970 g_object_unref (G_OBJECT (transport_account));
2971 g_object_unref (G_OBJECT (mail_operation));
2973 modest_msg_edit_window_free_msg_data (edit_window, data);
2976 * If the drafts folder is selected then make the header view
2977 * insensitive while the message is being saved to drafts
2978 * (it'll be sensitive again in on_save_to_drafts_cb()). This
2979 * is not very clean but it avoids letting the drafts folder
2980 * in an inconsistent state: the user could edit the message
2981 * being saved and undesirable things would happen.
2982 * In the average case the user won't notice anything at
2983 * all. In the worst case (the user is editing a really big
2984 * file from Drafts) the header view will be insensitive
2985 * during the saving process (10 or 20 seconds, depending on
2986 * the message). Anyway this is just a quick workaround: once
2987 * we find a better solution it should be removed
2988 * See NB#65125 (commend #18) for details.
2990 if (!had_error && win != NULL) {
2991 ModestFolderView *view = MODEST_FOLDER_VIEW(modest_main_window_get_child_widget(
2992 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW));
2994 TnyFolder *folder = TNY_FOLDER(modest_folder_view_get_selected(view));
2996 if (modest_tny_folder_is_local_folder(folder)) {
2997 TnyFolderType folder_type;
2998 folder_type = modest_tny_folder_get_local_or_mmc_folder_type(folder);
2999 if (folder_type == TNY_FOLDER_TYPE_DRAFTS) {
3000 GtkWidget *hdrview = modest_main_window_get_child_widget(
3001 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
3002 if (hdrview) gtk_widget_set_sensitive(hdrview, FALSE);
3006 if (folder != NULL) g_object_unref(folder);
3013 /* For instance, when clicking the Send toolbar button when editing a message: */
3015 modest_ui_actions_on_send (GtkWidget *widget, ModestMsgEditWindow *edit_window)
3017 TnyTransportAccount *transport_account = NULL;
3018 gboolean had_error = FALSE;
3020 ModestAccountMgr *account_mgr;
3021 gchar *account_name;
3022 ModestMailOperation *mail_operation;
3024 g_return_val_if_fail (MODEST_IS_MSG_EDIT_WINDOW(edit_window), TRUE);
3026 if (!modest_msg_edit_window_check_names (edit_window, TRUE))
3029 data = modest_msg_edit_window_get_msg_data (edit_window);
3032 if (!enough_space_for_message (edit_window, data)) {
3033 modest_msg_edit_window_free_msg_data (edit_window, data);
3037 account_mgr = modest_runtime_get_account_mgr();
3038 account_name = g_strdup (data->account_name);
3040 account_name = g_strdup(modest_window_get_active_account (MODEST_WINDOW(edit_window)));
3043 account_name = modest_account_mgr_get_default_account (account_mgr);
3045 if (!account_name) {
3046 modest_msg_edit_window_free_msg_data (edit_window, data);
3047 /* Run account setup wizard */
3048 if (!modest_ui_actions_run_account_setup_wizard (MODEST_WINDOW(edit_window))) {
3053 /* Get the currently-active transport account for this modest account: */
3054 if (strcmp (account_name, MODEST_LOCAL_FOLDERS_ACCOUNT_ID) != 0) {
3056 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
3057 (modest_runtime_get_account_store (),
3058 account_name, TNY_ACCOUNT_TYPE_TRANSPORT));
3061 if (!transport_account) {
3062 modest_msg_edit_window_free_msg_data (edit_window, data);
3063 /* Run account setup wizard */
3064 if (!modest_ui_actions_run_account_setup_wizard(MODEST_WINDOW(edit_window)))
3069 /* Create the mail operation */
3070 mail_operation = modest_mail_operation_new_with_error_handling (NULL, modest_ui_actions_disk_operations_error_handler, NULL, NULL);
3071 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_operation);
3073 modest_mail_operation_send_new_mail (mail_operation,
3087 data->priority_flags);
3089 if (modest_mail_operation_get_status (mail_operation) == MODEST_MAIL_OPERATION_STATUS_IN_PROGRESS)
3090 modest_platform_information_banner (NULL, NULL, _("mcen_ib_outbox_waiting_to_be_sent"));
3092 if (modest_mail_operation_get_error (mail_operation) != NULL) {
3093 const GError *error = modest_mail_operation_get_error (mail_operation);
3094 if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
3095 error->code == MODEST_MAIL_OPERATION_ERROR_INSTANCE_CREATION_FAILED) {
3096 g_warning ("%s failed: %s\n", __FUNCTION__, (modest_mail_operation_get_error (mail_operation))->message);
3097 modest_platform_information_banner (NULL, NULL, _CS("sfil_ni_not_enough_memory"));
3103 g_free (account_name);
3104 g_object_unref (G_OBJECT (transport_account));
3105 g_object_unref (G_OBJECT (mail_operation));
3107 modest_msg_edit_window_free_msg_data (edit_window, data);
3110 modest_msg_edit_window_set_sent (edit_window, TRUE);
3112 /* Save settings and close the window: */
3113 modest_ui_actions_on_close_window (NULL, MODEST_WINDOW (edit_window));
3120 modest_ui_actions_on_toggle_bold (GtkToggleAction *action,
3121 ModestMsgEditWindow *window)
3123 ModestMsgEditFormatState *format_state = NULL;
3125 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3126 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3128 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW (window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3131 format_state = modest_msg_edit_window_get_format_state (window);
3132 g_return_if_fail (format_state != NULL);
3134 format_state->bold = gtk_toggle_action_get_active (action);
3135 modest_msg_edit_window_set_format_state (window, format_state);
3136 g_free (format_state);
3141 modest_ui_actions_on_toggle_italics (GtkToggleAction *action,
3142 ModestMsgEditWindow *window)
3144 ModestMsgEditFormatState *format_state = NULL;
3146 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3147 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3149 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3152 format_state = modest_msg_edit_window_get_format_state (window);
3153 g_return_if_fail (format_state != NULL);
3155 format_state->italics = gtk_toggle_action_get_active (action);
3156 modest_msg_edit_window_set_format_state (window, format_state);
3157 g_free (format_state);
3162 modest_ui_actions_on_toggle_bullets (GtkToggleAction *action,
3163 ModestMsgEditWindow *window)
3165 ModestMsgEditFormatState *format_state = NULL;
3167 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3168 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3170 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW (window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3173 format_state = modest_msg_edit_window_get_format_state (window);
3174 g_return_if_fail (format_state != NULL);
3176 format_state->bullet = gtk_toggle_action_get_active (action);
3177 modest_msg_edit_window_set_format_state (window, format_state);
3178 g_free (format_state);
3183 modest_ui_actions_on_change_justify (GtkRadioAction *action,
3184 GtkRadioAction *selected,
3185 ModestMsgEditWindow *window)
3187 ModestMsgEditFormatState *format_state = NULL;
3188 GtkJustification value;
3190 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3192 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3195 value = gtk_radio_action_get_current_value (selected);
3197 format_state = modest_msg_edit_window_get_format_state (window);
3198 g_return_if_fail (format_state != NULL);
3200 format_state->justification = value;
3201 modest_msg_edit_window_set_format_state (window, format_state);
3202 g_free (format_state);
3206 modest_ui_actions_on_select_editor_color (GtkAction *action,
3207 ModestMsgEditWindow *window)
3209 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3210 g_return_if_fail (GTK_IS_ACTION (action));
3212 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3215 modest_msg_edit_window_select_color (window);
3219 modest_ui_actions_on_select_editor_background_color (GtkAction *action,
3220 ModestMsgEditWindow *window)
3222 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3223 g_return_if_fail (GTK_IS_ACTION (action));
3225 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3228 modest_msg_edit_window_select_background_color (window);
3232 modest_ui_actions_on_insert_image (GObject *object,
3233 ModestMsgEditWindow *window)
3235 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3238 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
3241 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3244 modest_msg_edit_window_insert_image (window);
3248 modest_ui_actions_on_attach_file (GtkAction *action,
3249 ModestMsgEditWindow *window)
3251 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3252 g_return_if_fail (GTK_IS_ACTION (action));
3254 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
3257 modest_msg_edit_window_offer_attach_file (window);
3261 modest_ui_actions_on_remove_attachments (GtkAction *action,
3262 ModestMsgEditWindow *window)
3264 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3266 modest_msg_edit_window_remove_attachments (window, NULL);
3270 do_create_folder_cb (ModestMailOperation *mail_op,
3271 TnyFolderStore *parent_folder,
3272 TnyFolder *new_folder,
3275 gchar *suggested_name = (gchar *) user_data;
3276 GtkWindow *source_win = (GtkWindow *) modest_mail_operation_get_source (mail_op);
3277 const GError *error;
3279 error = modest_mail_operation_get_error (mail_op);
3282 /* Show an error. If there was some problem writing to
3283 disk, show it, otherwise show the generic folder
3284 create error. We do it here and not in an error
3285 handler because the call to do_create_folder will
3286 stop the main loop in a gtk_dialog_run and then,
3287 the message won't be shown until that dialog is
3289 modest_ui_actions_disk_operations_error_handler (mail_op,
3290 _("mail_in_ui_folder_create_error"));
3292 if (!is_memory_full_error ((GError *) error, mail_op)) {
3293 /* Try again if there is no full memory condition */
3294 do_create_folder (source_win, parent_folder, (const gchar *) suggested_name);
3297 /* the 'source_win' is either the ModestMainWindow, or the 'Move to folder'-dialog
3298 * FIXME: any other? */
3299 GtkWidget *folder_view;
3301 if (MODEST_IS_MAIN_WINDOW(source_win))
3303 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (source_win),
3304 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3306 folder_view = GTK_WIDGET(g_object_get_data (G_OBJECT (source_win),
3307 MODEST_MOVE_TO_DIALOG_FOLDER_VIEW));
3309 /* Select the newly created folder. It could happen
3310 that the widget is no longer there (i.e. the window
3311 has been destroyed, so we need to check this */
3313 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view),
3315 g_object_unref (new_folder);
3317 /* Free. Note that the first time it'll be NULL so noop */
3318 g_free (suggested_name);
3319 g_object_unref (source_win);
3324 TnyFolderStore *parent;
3325 } CreateFolderConnect;
3328 do_create_folder_performer (gboolean canceled,
3330 GtkWindow *parent_window,
3331 TnyAccount *account,
3334 CreateFolderConnect *helper = (CreateFolderConnect *) user_data;
3335 ModestMailOperation *mail_op;
3337 if (canceled || err) {
3338 /* In memory full conditions we could get this error here */
3339 check_memory_full_error ((GtkWidget *) parent_window, err);
3343 mail_op = modest_mail_operation_new ((GObject *) parent_window);
3344 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3346 modest_mail_operation_create_folder (mail_op,
3348 (const gchar *) helper->folder_name,
3349 do_create_folder_cb,
3350 g_strdup (helper->folder_name));
3351 g_object_unref (mail_op);
3355 g_object_unref (helper->parent);
3356 if (helper->folder_name)
3357 g_free (helper->folder_name);
3358 g_slice_free (CreateFolderConnect, helper);
3363 do_create_folder (GtkWindow *parent_window,
3364 TnyFolderStore *suggested_parent,
3365 const gchar *suggested_name)
3368 gchar *folder_name = NULL;
3369 TnyFolderStore *parent_folder = NULL;
3371 result = modest_platform_run_new_folder_dialog (GTK_WINDOW (parent_window),
3373 (gchar *) suggested_name,
3377 if (result == GTK_RESPONSE_ACCEPT && parent_folder) {
3378 CreateFolderConnect *helper = (CreateFolderConnect *) g_slice_new0 (CreateFolderConnect);
3379 helper->folder_name = g_strdup (folder_name);
3380 helper->parent = g_object_ref (parent_folder);
3382 modest_platform_connect_if_remote_and_perform (GTK_WINDOW (parent_window),
3385 do_create_folder_performer,
3390 g_free (folder_name);
3392 g_object_unref (parent_folder);
3396 modest_ui_actions_create_folder(GtkWidget *parent_window,
3397 GtkWidget *folder_view)
3399 TnyFolderStore *parent_folder;
3401 #ifdef MODEST_TOOLKIT_HILDON2
3402 ModestTnyAccountStore *acc_store;
3404 acc_store = modest_runtime_get_account_store ();
3406 parent_folder = (TnyFolderStore *)
3407 modest_tny_account_store_get_local_folders_account (acc_store);
3409 parent_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW(folder_view));
3412 if (parent_folder) {
3413 do_create_folder (GTK_WINDOW (parent_window), parent_folder, NULL);
3414 g_object_unref (parent_folder);
3419 modest_ui_actions_on_new_folder (GtkAction *action, ModestWindow *window)
3422 g_return_if_fail (MODEST_IS_WINDOW(window));
3424 if (MODEST_IS_MAIN_WINDOW (window)) {
3425 GtkWidget *folder_view;
3427 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3428 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3432 modest_ui_actions_create_folder (GTK_WIDGET (window), folder_view);
3433 #ifdef MODEST_TOOLKIT_HILDON2
3434 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3435 GtkWidget *folder_view;
3437 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3438 modest_ui_actions_create_folder (GTK_WIDGET (window), folder_view);
3441 g_assert_not_reached ();
3446 modest_ui_actions_rename_folder_error_handler (ModestMailOperation *mail_op,
3449 const GError *error = NULL;
3450 const gchar *message = NULL;
3452 /* Get error message */
3453 error = modest_mail_operation_get_error (mail_op);
3455 g_return_if_reached ();
3457 if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
3458 error->code == MODEST_MAIL_OPERATION_ERROR_FOLDER_EXISTS) {
3459 message = _CS("ckdg_ib_folder_already_exists");
3460 } else if (error->domain == TNY_ERROR_DOMAIN &&
3461 error->code == TNY_SERVICE_ERROR_STATE) {
3462 /* This means that the folder is already in use (a
3463 message is opened for example */
3464 message = _("emev_ni_internal_error");
3466 message = _CS("ckdg_ib_unable_to_rename");
3469 /* We don't set a parent for the dialog because the dialog
3470 will be destroyed so the banner won't appear */
3471 modest_platform_information_banner (NULL, NULL, message);
3475 TnyFolderStore *folder;
3480 on_rename_folder_cb (ModestMailOperation *mail_op,
3481 TnyFolder *new_folder,
3484 ModestFolderView *folder_view;
3486 /* If the window was closed when renaming a folder, or if
3487 * it's not a main window this will happen */
3488 if (!MODEST_IS_FOLDER_VIEW (user_data))
3491 folder_view = MODEST_FOLDER_VIEW (user_data);
3492 /* Note that if the rename fails new_folder will be NULL */
3494 modest_folder_view_select_folder (folder_view, new_folder, FALSE);
3496 modest_folder_view_select_first_inbox_or_local (folder_view);
3498 gtk_widget_grab_focus (GTK_WIDGET (folder_view));
3502 on_rename_folder_performer (gboolean canceled,
3504 GtkWindow *parent_window,
3505 TnyAccount *account,
3508 ModestMailOperation *mail_op = NULL;
3509 GtkTreeSelection *sel = NULL;
3510 GtkWidget *folder_view = NULL;
3511 RenameFolderInfo *data = (RenameFolderInfo*)user_data;
3513 if (canceled || err) {
3514 /* In memory full conditions we could get this error here */
3515 check_memory_full_error ((GtkWidget *) parent_window, err);
3519 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
3520 modest_ui_actions_rename_folder_error_handler,
3521 parent_window, NULL);
3523 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3526 if (MODEST_IS_MAIN_WINDOW(parent_window)) {
3528 folder_view = modest_main_window_get_child_widget (
3529 MODEST_MAIN_WINDOW (parent_window),
3530 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3532 #ifdef MODEST_TOOLKIT_HILDON2
3533 else if (MODEST_IS_FOLDER_WINDOW (parent_window)) {
3534 ModestFolderWindow *folder_window = (ModestFolderWindow *) parent_window;
3535 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (folder_window));
3539 /* Clear the folders view */
3540 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_view));
3541 gtk_tree_selection_unselect_all (sel);
3543 /* Actually rename the folder */
3544 modest_mail_operation_rename_folder (mail_op,
3545 TNY_FOLDER (data->folder),
3546 (const gchar *) (data->new_name),
3547 on_rename_folder_cb,
3549 g_object_unref (mail_op);
3552 g_object_unref (data->folder);
3553 g_free (data->new_name);
3558 modest_ui_actions_on_rename_folder (GtkAction *action,
3559 ModestWindow *window)
3561 modest_ui_actions_on_edit_mode_rename_folder (window);
3565 modest_ui_actions_on_edit_mode_rename_folder (ModestWindow *window)
3567 TnyFolderStore *folder;
3568 GtkWidget *folder_view;
3569 gboolean do_rename = TRUE;
3571 g_return_val_if_fail (MODEST_IS_WINDOW(window), FALSE);
3573 if (MODEST_IS_MAIN_WINDOW (window)) {
3574 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3575 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3579 #ifdef MODEST_TOOLKIT_HILDON2
3580 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3581 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3587 folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW(folder_view));
3592 if (TNY_IS_FOLDER (folder)) {
3593 gchar *folder_name = NULL;
3595 const gchar *current_name;
3596 TnyFolderStore *parent;
3598 current_name = tny_folder_get_name (TNY_FOLDER (folder));
3599 parent = tny_folder_get_folder_store (TNY_FOLDER (folder));
3600 response = modest_platform_run_rename_folder_dialog (GTK_WINDOW (window),
3601 parent, current_name,
3603 g_object_unref (parent);
3605 if (response != GTK_RESPONSE_ACCEPT || strlen (folder_name) == 0) {
3608 RenameFolderInfo *rename_folder_data = g_new0 (RenameFolderInfo, 1);
3609 rename_folder_data->folder = g_object_ref (folder);
3610 rename_folder_data->new_name = folder_name;
3611 modest_platform_connect_if_remote_and_perform (GTK_WINDOW(window), TRUE,
3612 folder, on_rename_folder_performer, rename_folder_data);
3615 g_object_unref (folder);
3620 modest_ui_actions_delete_folder_error_handler (ModestMailOperation *mail_op,
3623 GObject *win = modest_mail_operation_get_source (mail_op);
3625 modest_platform_run_information_dialog ((win) ? GTK_WINDOW (win) : NULL,
3626 _("mail_in_ui_folder_delete_error"),
3628 g_object_unref (win);
3632 TnyFolderStore *folder;
3633 gboolean move_to_trash;
3637 on_delete_folder_cb (gboolean canceled,
3639 GtkWindow *parent_window,
3640 TnyAccount *account,
3643 DeleteFolderInfo *info = (DeleteFolderInfo*) user_data;
3644 GtkWidget *folder_view;
3645 ModestMailOperation *mail_op;
3646 GtkTreeSelection *sel;
3648 if (!MODEST_IS_WINDOW(parent_window) || canceled || (err!=NULL)) {
3649 g_object_unref (G_OBJECT (info->folder));
3654 if (MODEST_IS_MAIN_WINDOW (parent_window)) {
3655 folder_view = modest_main_window_get_child_widget (
3656 MODEST_MAIN_WINDOW (parent_window),
3657 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3658 #ifdef MODEST_TOOLKIT_HILDON2
3659 } else if (MODEST_IS_FOLDER_WINDOW (parent_window)) {
3660 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (parent_window)));
3663 g_object_unref (G_OBJECT (info->folder));
3668 /* Unselect the folder before deleting it to free the headers */
3669 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_view));
3670 gtk_tree_selection_unselect_all (sel);
3672 /* Create the mail operation */
3674 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
3675 modest_ui_actions_delete_folder_error_handler,
3678 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3680 modest_mail_operation_remove_folder (mail_op, TNY_FOLDER (info->folder), info->move_to_trash);
3682 modest_folder_view_select_first_inbox_or_local (MODEST_FOLDER_VIEW (folder_view));
3684 g_object_unref (G_OBJECT (mail_op));
3685 g_object_unref (G_OBJECT (info->folder));
3690 delete_folder (ModestWindow *window, gboolean move_to_trash)
3692 TnyFolderStore *folder;
3693 GtkWidget *folder_view;
3697 g_return_val_if_fail (MODEST_IS_WINDOW(window), FALSE);
3699 if (MODEST_IS_MAIN_WINDOW (window)) {
3701 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3702 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3703 #ifdef MODEST_TOOLKIT_HILDON2
3704 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3705 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3713 folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
3718 /* Show an error if it's an account */
3719 if (!TNY_IS_FOLDER (folder)) {
3720 modest_platform_run_information_dialog (GTK_WINDOW (window),
3721 _("mail_in_ui_folder_delete_error"),
3723 g_object_unref (G_OBJECT (folder));
3728 message = g_strdup_printf (_("mcen_nc_delete_folder_text"),
3729 tny_folder_get_name (TNY_FOLDER (folder)));
3730 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (window),
3731 (const gchar *) message);
3734 if (response == GTK_RESPONSE_OK) {
3735 DeleteFolderInfo *info;
3736 info = g_new0(DeleteFolderInfo, 1);
3737 info->folder = folder;
3738 info->move_to_trash = move_to_trash;
3739 g_object_ref (G_OBJECT (info->folder));
3740 TnyAccount *account = tny_folder_get_account (TNY_FOLDER (folder));
3741 modest_platform_connect_if_remote_and_perform (GTK_WINDOW (window),
3743 TNY_FOLDER_STORE (account),
3744 on_delete_folder_cb, info);
3745 g_object_unref (account);
3750 g_object_unref (G_OBJECT (folder));
3754 modest_ui_actions_on_delete_folder (GtkAction *action,
3755 ModestWindow *window)
3757 modest_ui_actions_on_edit_mode_delete_folder (window);
3761 modest_ui_actions_on_edit_mode_delete_folder (ModestWindow *window)
3763 g_return_val_if_fail (MODEST_IS_WINDOW(window), TRUE);
3765 return delete_folder (window, FALSE);
3769 modest_ui_actions_on_move_folder_to_trash_folder (GtkAction *action, ModestMainWindow *main_window)
3771 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
3773 delete_folder (MODEST_WINDOW (main_window), TRUE);
3777 typedef struct _PasswordDialogFields {
3778 GtkWidget *username;
3779 GtkWidget *password;
3781 } PasswordDialogFields;
3784 password_dialog_check_field (GtkEditable *editable,
3785 PasswordDialogFields *fields)
3788 gboolean any_value_empty = FALSE;
3790 #ifdef MODEST_TOOLKIT_HILDON2
3791 value = hildon_entry_get_text (HILDON_ENTRY (fields->username));
3793 value = gtk_entry_get_text (GTK_ENTRY (fields->username));
3795 if ((value == NULL) || value[0] == '\0') {
3796 any_value_empty = TRUE;
3798 #ifdef MODEST_TOOLKIT_HILDON2
3799 value = hildon_entry_get_text (HILDON_ENTRY (fields->password));
3801 value = gtk_entry_get_text (GTK_ENTRY (fields->password));
3803 if ((value == NULL) || value[0] == '\0') {
3804 any_value_empty = TRUE;
3806 gtk_dialog_set_response_sensitive (GTK_DIALOG (fields->dialog), GTK_RESPONSE_ACCEPT, !any_value_empty);
3810 modest_ui_actions_on_password_requested (TnyAccountStore *account_store,
3811 const gchar* server_account_name,
3816 ModestMainWindow *main_window)
3818 g_return_if_fail(server_account_name);
3819 gboolean completed = FALSE;
3820 PasswordDialogFields *fields = NULL;
3822 /* Initalize output parameters: */
3829 #ifndef MODEST_TOOLKIT_GTK
3830 /* Maemo uses a different (awkward) button order,
3831 * It should probably just use gtk_alternative_dialog_button_order ().
3833 #ifdef MODEST_TOOLKIT_HILDON2
3835 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3838 _HL("wdgt_bd_done"),
3839 GTK_RESPONSE_ACCEPT,
3841 gtk_container_set_border_width (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox),
3842 HILDON_MARGIN_DOUBLE);
3845 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3848 _("mcen_bd_dialog_ok"),
3849 GTK_RESPONSE_ACCEPT,
3850 _("mcen_bd_dialog_cancel"),
3851 GTK_RESPONSE_REJECT,
3853 #endif /* MODEST_TOOLKIT_HILDON2 */
3856 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3860 GTK_RESPONSE_REJECT,
3862 GTK_RESPONSE_ACCEPT,
3864 #endif /* MODEST_TOOLKIT_GTK */
3866 modest_window_mgr_set_modal (modest_runtime_get_window_mgr(), GTK_WINDOW (dialog), NULL);
3868 gchar *server_name = modest_account_mgr_get_server_account_hostname (
3869 modest_runtime_get_account_mgr(), server_account_name);
3870 if (!server_name) {/* This happened once, though I don't know why. murrayc. */
3871 g_warning("%s: Could not get server name for server account '%s'", __FUNCTION__, server_account_name);
3874 gtk_widget_destroy (dialog);
3878 gchar *txt = g_strdup_printf (_("mail_ia_password_info"), server_name);
3879 GtkWidget *label = gtk_label_new (txt);
3880 gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
3882 g_free (server_name);
3883 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), label,
3888 gchar *initial_username = modest_account_mgr_get_server_account_username (
3889 modest_runtime_get_account_mgr(), server_account_name);
3891 #ifdef MODEST_TOOLKIT_HILDON2
3892 GtkWidget *entry_username = hildon_entry_new (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH);
3893 if (initial_username)
3894 hildon_entry_set_text (HILDON_ENTRY (entry_username), initial_username);
3896 GtkWidget *entry_username = gtk_entry_new ();
3897 if (initial_username)
3898 gtk_entry_set_text (GTK_ENTRY (entry_username), initial_username);
3900 /* Dim this if a connection has ever succeeded with this username,
3901 * as per the UI spec: */
3902 /* const gboolean username_known = */
3903 /* modest_account_mgr_get_server_account_username_has_succeeded( */
3904 /* modest_runtime_get_account_mgr(), server_account_name); */
3905 /* gtk_widget_set_sensitive (entry_username, !username_known); */
3907 /* We drop the username sensitive code and disallow changing it here
3908 * as tinymail does not support really changing the username in the callback
3910 gtk_widget_set_sensitive (entry_username, FALSE);
3912 #ifndef MODEST_TOOLKIT_GTK
3913 /* Auto-capitalization is the default, so let's turn it off: */
3914 hildon_gtk_entry_set_input_mode (GTK_ENTRY (entry_username), HILDON_GTK_INPUT_MODE_FULL);
3916 /* Create a size group to be used by all captions.
3917 * Note that HildonCaption does not create a default size group if we do not specify one.
3918 * We use GTK_SIZE_GROUP_HORIZONTAL, so that the widths are the same. */
3919 GtkSizeGroup *sizegroup = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
3921 #ifdef MODEST_TOOLKIT_HILDON2
3922 GtkWidget *caption = modest_maemo_utils_create_captioned (sizegroup, NULL,
3923 _("mail_fi_username"), FALSE,
3926 GtkWidget *caption = hildon_caption_new (sizegroup,
3927 _("mail_fi_username"), entry_username, NULL, HILDON_CAPTION_MANDATORY);
3929 gtk_widget_show (entry_username);
3930 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), caption,
3931 FALSE, FALSE, MODEST_MARGIN_HALF);
3932 gtk_widget_show (caption);
3934 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), entry_username,
3936 #endif /* !MODEST_TOOLKIT_GTK */
3939 #ifdef MODEST_TOOLKIT_HILDON2
3940 GtkWidget *entry_password = hildon_entry_new (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH);
3942 GtkWidget *entry_password = gtk_entry_new ();
3944 gtk_entry_set_visibility (GTK_ENTRY(entry_password), FALSE);
3945 /* gtk_entry_set_invisible_char (GTK_ENTRY(entry_password), "*"); */
3947 #ifndef MODEST_TOOLKIT_GTK
3948 /* Auto-capitalization is the default, so let's turn it off: */
3949 hildon_gtk_entry_set_input_mode (GTK_ENTRY (entry_password),
3950 HILDON_GTK_INPUT_MODE_FULL | HILDON_GTK_INPUT_MODE_INVISIBLE);
3952 #ifdef MODEST_TOOLKIT_HILDON2
3953 caption = modest_maemo_utils_create_captioned (sizegroup, NULL,
3954 _("mail_fi_password"), FALSE,
3957 caption = hildon_caption_new (sizegroup,
3958 _("mail_fi_password"), entry_password, NULL, HILDON_CAPTION_MANDATORY);
3960 gtk_widget_show (entry_password);
3961 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), caption,
3962 FALSE, FALSE, MODEST_MARGIN_HALF);
3963 gtk_widget_show (caption);
3964 g_object_unref (sizegroup);
3966 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), entry_password,
3968 #endif /* !MODEST_TOOLKIT_GTK */
3970 if (initial_username != NULL)
3971 gtk_widget_grab_focus (GTK_WIDGET (entry_password));
3973 /* This is not in the Maemo UI spec:
3974 remember_pass_check = gtk_check_button_new_with_label (_("Remember password"));
3975 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), remember_pass_check,
3979 fields = g_slice_new0 (PasswordDialogFields);
3980 fields->username = entry_username;
3981 fields->password = entry_password;
3982 fields->dialog = dialog;
3984 g_signal_connect (entry_username, "changed", G_CALLBACK (password_dialog_check_field), fields);
3985 g_signal_connect (entry_password, "changed", G_CALLBACK (password_dialog_check_field), fields);
3986 password_dialog_check_field (NULL, fields);
3988 gtk_widget_show_all (GTK_WIDGET(GTK_DIALOG(dialog)->vbox));
3990 while (!completed) {
3992 if (gtk_dialog_run (GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
3994 #ifdef MODEST_TOOLKIT_HILDON2
3995 *username = g_strdup (hildon_entry_get_text (HILDON_ENTRY(entry_username)));
3997 *username = g_strdup (gtk_entry_get_text (GTK_ENTRY(entry_username)));
4000 /* Note that an empty field becomes the "" string */
4001 if (*username && strlen (*username) > 0) {
4002 modest_account_mgr_set_server_account_username (modest_runtime_get_account_mgr(),
4003 server_account_name,
4007 const gboolean username_was_changed =
4008 (strcmp (*username, initial_username) != 0);
4009 if (username_was_changed) {
4010 g_warning ("%s: tinymail does not yet support changing the "
4011 "username in the get_password() callback.\n", __FUNCTION__);
4017 modest_platform_information_banner (GTK_WIDGET (dialog), NULL,
4018 _("mcen_ib_username_pw_incorrect"));
4024 #ifdef MODEST_TOOLKIT_HILDON2
4025 *password = g_strdup (hildon_entry_get_text (HILDON_ENTRY(entry_password)));
4027 *password = g_strdup (gtk_entry_get_text (GTK_ENTRY(entry_password)));
4030 /* We do not save the password in the configuration,
4031 * because this function is only called for passwords that should
4032 * not be remembered:
4033 modest_server_account_set_password (
4034 modest_runtime_get_account_mgr(), server_account_name,
4041 #ifndef MODEST_TOOLKIT_HILDON2
4042 /* Set parent to NULL or the banner will disappear with its parent dialog */
4043 modest_platform_information_banner(NULL, NULL, _("mail_ib_login_cancelled"));
4055 /* This is not in the Maemo UI spec:
4056 if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (remember_pass_check)))
4062 g_free (initial_username);
4063 gtk_widget_destroy (dialog);
4064 g_slice_free (PasswordDialogFields, fields);
4066 /* printf ("DEBUG: %s: cancel=%d\n", __FUNCTION__, *cancel); */
4070 modest_ui_actions_on_cut (GtkAction *action,
4071 ModestWindow *window)
4073 GtkWidget *focused_widget;
4074 GtkClipboard *clipboard;
4076 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4077 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4078 if (GTK_IS_EDITABLE (focused_widget)) {
4079 gtk_editable_cut_clipboard (GTK_EDITABLE(focused_widget));
4080 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4081 gtk_clipboard_store (clipboard);
4082 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4083 GtkTextBuffer *buffer;
4085 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4086 if (modest_text_utils_buffer_selection_is_valid (buffer)) {
4087 gtk_text_buffer_cut_clipboard (buffer, clipboard, TRUE);
4088 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4089 gtk_clipboard_store (clipboard);
4091 } else if (MODEST_IS_HEADER_VIEW (focused_widget)) {
4092 TnyList *header_list = modest_header_view_get_selected_headers (
4093 MODEST_HEADER_VIEW (focused_widget));
4094 gboolean continue_download = FALSE;
4095 gint num_of_unc_msgs;
4097 num_of_unc_msgs = header_list_count_uncached_msgs(header_list);
4099 if (num_of_unc_msgs) {
4100 TnyAccount *account = get_account_from_header_list (header_list);
4102 continue_download = connect_to_get_msg (window, num_of_unc_msgs, account);
4103 g_object_unref (account);
4107 if (num_of_unc_msgs == 0 || continue_download) {
4108 /* modest_platform_information_banner (
4109 NULL, NULL, _CS("mcen_ib_getting_items"));*/
4110 modest_header_view_cut_selection (
4111 MODEST_HEADER_VIEW (focused_widget));
4114 g_object_unref (header_list);
4115 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4116 modest_folder_view_cut_selection (MODEST_FOLDER_VIEW (focused_widget));
4121 modest_ui_actions_on_copy (GtkAction *action,
4122 ModestWindow *window)
4124 GtkClipboard *clipboard;
4125 GtkWidget *focused_widget;
4126 gboolean copied = TRUE;
4128 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4129 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4131 if (GTK_IS_LABEL (focused_widget)) {
4133 selection = modest_text_utils_label_get_selection (GTK_LABEL (focused_widget));
4134 gtk_clipboard_set_text (clipboard, selection, -1);
4136 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4137 gtk_clipboard_store (clipboard);
4138 } else if (GTK_IS_EDITABLE (focused_widget)) {
4139 gtk_editable_copy_clipboard (GTK_EDITABLE(focused_widget));
4140 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4141 gtk_clipboard_store (clipboard);
4142 } else if (GTK_IS_HTML (focused_widget)) {
4145 sel = gtk_html_get_selection_html (GTK_HTML (focused_widget), &len);
4146 if ((sel == NULL) || (sel[0] == '\0')) {
4149 gtk_html_copy (GTK_HTML (focused_widget));
4150 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4151 gtk_clipboard_store (clipboard);
4153 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4154 GtkTextBuffer *buffer;
4155 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4156 if (modest_text_utils_buffer_selection_is_valid (buffer)) {
4157 gtk_text_buffer_copy_clipboard (buffer, clipboard);
4158 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4159 gtk_clipboard_store (clipboard);
4161 } else if (MODEST_IS_HEADER_VIEW (focused_widget)) {
4162 TnyList *header_list = modest_header_view_get_selected_headers (
4163 MODEST_HEADER_VIEW (focused_widget));
4164 gboolean continue_download = FALSE;
4165 gint num_of_unc_msgs;
4167 num_of_unc_msgs = header_list_count_uncached_msgs(header_list);
4169 if (num_of_unc_msgs) {
4170 TnyAccount *account = get_account_from_header_list (header_list);
4172 continue_download = connect_to_get_msg (window, num_of_unc_msgs, account);
4173 g_object_unref (account);
4177 if (num_of_unc_msgs == 0 || continue_download) {
4178 modest_platform_information_banner (
4179 NULL, NULL, _CS("mcen_ib_getting_items"));
4180 modest_header_view_copy_selection (
4181 MODEST_HEADER_VIEW (focused_widget));
4185 g_object_unref (header_list);
4187 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4188 modest_folder_view_copy_selection (MODEST_FOLDER_VIEW (focused_widget));
4191 /* Show information banner if there was a copy to clipboard */
4193 modest_platform_information_banner (
4194 NULL, NULL, _CS("ecoc_ib_edwin_copied"));
4198 modest_ui_actions_on_undo (GtkAction *action,
4199 ModestWindow *window)
4201 ModestEmailClipboard *clipboard = NULL;
4203 if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4204 modest_msg_edit_window_undo (MODEST_MSG_EDIT_WINDOW (window));
4205 } else if (MODEST_IS_MAIN_WINDOW (window)) {
4206 /* Clear clipboard source */
4207 clipboard = modest_runtime_get_email_clipboard ();
4208 modest_email_clipboard_clear (clipboard);
4211 g_return_if_reached ();
4216 modest_ui_actions_on_redo (GtkAction *action,
4217 ModestWindow *window)
4219 if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4220 modest_msg_edit_window_redo (MODEST_MSG_EDIT_WINDOW (window));
4223 g_return_if_reached ();
4229 destroy_information_note (ModestMailOperation *mail_op,
4232 /* destroy information note */
4233 gtk_widget_destroy (GTK_WIDGET(user_data));
4237 destroy_folder_information_note (ModestMailOperation *mail_op,
4238 TnyFolder *new_folder,
4241 /* destroy information note */
4242 gtk_widget_destroy (GTK_WIDGET(user_data));
4247 paste_as_attachment_free (gpointer data)
4249 PasteAsAttachmentHelper *helper = (PasteAsAttachmentHelper *) data;
4251 if (helper->banner) {
4252 gtk_widget_destroy (helper->banner);
4253 g_object_unref (helper->banner);
4259 paste_msg_as_attachment_cb (ModestMailOperation *mail_op,
4264 PasteAsAttachmentHelper *helper = (PasteAsAttachmentHelper *) userdata;
4265 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (helper->window));
4270 modest_msg_edit_window_add_part (MODEST_MSG_EDIT_WINDOW (helper->window), TNY_MIME_PART (msg));
4275 modest_ui_actions_on_paste (GtkAction *action,
4276 ModestWindow *window)
4278 GtkWidget *focused_widget = NULL;
4279 GtkWidget *inf_note = NULL;
4280 ModestMailOperation *mail_op = NULL;
4282 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4283 if (GTK_IS_EDITABLE (focused_widget)) {
4284 gtk_editable_paste_clipboard (GTK_EDITABLE(focused_widget));
4285 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4286 ModestEmailClipboard *e_clipboard = NULL;
4287 e_clipboard = modest_runtime_get_email_clipboard ();
4288 if (modest_email_clipboard_cleared (e_clipboard)) {
4289 GtkTextBuffer *buffer;
4290 GtkClipboard *clipboard;
4292 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4293 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4294 gtk_text_buffer_paste_clipboard (buffer, clipboard, NULL, TRUE);
4295 } else if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4296 ModestMailOperation *mail_op;
4297 TnyFolder *src_folder = NULL;
4298 TnyList *data = NULL;
4300 PasteAsAttachmentHelper *helper = g_new0 (PasteAsAttachmentHelper, 1);
4301 helper->window = MODEST_MSG_EDIT_WINDOW (window);
4302 helper->banner = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4303 _CS("ckct_nw_pasting"));
4304 modest_email_clipboard_get_data (e_clipboard, &src_folder, &data, &delete);
4305 mail_op = modest_mail_operation_new (G_OBJECT (window));
4306 if (helper->banner != NULL) {
4307 g_object_ref (G_OBJECT (helper->banner));
4308 gtk_widget_show (GTK_WIDGET (helper->banner));
4312 modest_mail_operation_get_msgs_full (mail_op,
4314 (GetMsgAsyncUserCallback) paste_msg_as_attachment_cb,
4316 paste_as_attachment_free);
4320 g_object_unref (data);
4322 g_object_unref (src_folder);
4325 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4326 ModestEmailClipboard *clipboard = NULL;
4327 TnyFolder *src_folder = NULL;
4328 TnyFolderStore *folder_store = NULL;
4329 TnyList *data = NULL;
4330 gboolean delete = FALSE;
4332 /* Check clipboard source */
4333 clipboard = modest_runtime_get_email_clipboard ();
4334 if (modest_email_clipboard_cleared (clipboard))
4337 /* Get elements to paste */
4338 modest_email_clipboard_get_data (clipboard, &src_folder, &data, &delete);
4340 /* Create a new mail operation */
4341 mail_op = modest_mail_operation_new (G_OBJECT(window));
4343 /* Get destination folder */
4344 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (focused_widget));
4346 /* transfer messages */
4350 /* Ask for user confirmation */
4352 modest_ui_actions_msgs_move_to_confirmation (window,
4353 TNY_FOLDER (folder_store),
4357 if (response == GTK_RESPONSE_OK) {
4358 /* Launch notification */
4359 inf_note = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4360 _CS("ckct_nw_pasting"));
4361 if (inf_note != NULL) {
4362 gtk_window_set_modal (GTK_WINDOW(inf_note), FALSE);
4363 gtk_widget_show (GTK_WIDGET(inf_note));
4366 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
4367 modest_mail_operation_xfer_msgs (mail_op,
4369 TNY_FOLDER (folder_store),
4371 destroy_information_note,
4374 g_object_unref (mail_op);
4377 } else if (src_folder != NULL) {
4378 /* Launch notification */
4379 inf_note = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4380 _CS("ckct_nw_pasting"));
4381 if (inf_note != NULL) {
4382 gtk_window_set_modal (GTK_WINDOW(inf_note), FALSE);
4383 gtk_widget_show (GTK_WIDGET(inf_note));
4386 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
4387 modest_mail_operation_xfer_folder (mail_op,
4391 destroy_folder_information_note,
4397 g_object_unref (data);
4398 if (src_folder != NULL)
4399 g_object_unref (src_folder);
4400 if (folder_store != NULL)
4401 g_object_unref (folder_store);
4407 modest_ui_actions_on_select_all (GtkAction *action,
4408 ModestWindow *window)
4410 GtkWidget *focused_widget;
4412 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4413 if (MODEST_IS_ATTACHMENTS_VIEW (focused_widget)) {
4414 modest_attachments_view_select_all (MODEST_ATTACHMENTS_VIEW (focused_widget));
4415 } else if (GTK_IS_LABEL (focused_widget)) {
4416 gtk_label_select_region (GTK_LABEL (focused_widget), 0, -1);
4417 } else if (GTK_IS_EDITABLE (focused_widget)) {
4418 gtk_editable_select_region (GTK_EDITABLE(focused_widget), 0, -1);
4419 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4420 GtkTextBuffer *buffer;
4421 GtkTextIter start, end;
4423 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4424 gtk_text_buffer_get_start_iter (buffer, &start);
4425 gtk_text_buffer_get_end_iter (buffer, &end);
4426 gtk_text_buffer_select_range (buffer, &start, &end);
4427 } else if (GTK_IS_HTML (focused_widget)) {
4428 gtk_html_select_all (GTK_HTML (focused_widget));
4429 } else if (MODEST_IS_MAIN_WINDOW (window)) {
4430 GtkWidget *header_view = focused_widget;
4431 GtkTreeSelection *selection = NULL;
4433 if (!(MODEST_IS_HEADER_VIEW (focused_widget))) {
4434 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
4435 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
4438 /* Disable window dimming management */
4439 modest_window_disable_dimming (MODEST_WINDOW(window));
4441 /* Select all messages */
4442 selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(header_view));
4443 gtk_tree_selection_select_all (selection);
4445 /* Set focuse on header view */
4446 gtk_widget_grab_focus (header_view);
4448 /* Enable window dimming management */
4449 modest_window_enable_dimming (MODEST_WINDOW(window));
4450 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (window));
4451 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (window));
4457 modest_ui_actions_on_mark_as_read (GtkAction *action,
4458 ModestWindow *window)
4460 g_return_if_fail (MODEST_IS_WINDOW(window));
4462 /* Mark each header as read */
4463 do_headers_action (window, headers_action_mark_as_read, NULL);
4467 modest_ui_actions_on_mark_as_unread (GtkAction *action,
4468 ModestWindow *window)
4470 g_return_if_fail (MODEST_IS_WINDOW(window));
4472 /* Mark each header as read */
4473 do_headers_action (window, headers_action_mark_as_unread, NULL);
4477 modest_ui_actions_on_change_zoom (GtkRadioAction *action,
4478 GtkRadioAction *selected,
4479 ModestWindow *window)
4483 value = gtk_radio_action_get_current_value (selected);
4484 if (MODEST_IS_WINDOW (window)) {
4485 modest_window_set_zoom (MODEST_WINDOW (window), ((gdouble)value)/100);
4490 modest_ui_actions_msg_edit_on_change_priority (GtkRadioAction *action,
4491 GtkRadioAction *selected,
4492 ModestWindow *window)
4494 TnyHeaderFlags flags;
4495 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4497 flags = gtk_radio_action_get_current_value (selected);
4498 modest_msg_edit_window_set_priority_flags (MODEST_MSG_EDIT_WINDOW (window), flags);
4502 modest_ui_actions_msg_edit_on_change_file_format (GtkRadioAction *action,
4503 GtkRadioAction *selected,
4504 ModestWindow *window)
4508 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4510 file_format = gtk_radio_action_get_current_value (selected);
4511 modest_msg_edit_window_set_file_format (MODEST_MSG_EDIT_WINDOW (window), file_format);
4516 modest_ui_actions_on_zoom_plus (GtkAction *action,
4517 ModestWindow *window)
4519 g_return_if_fail (MODEST_IS_WINDOW (window));
4521 modest_window_zoom_plus (MODEST_WINDOW (window));
4525 modest_ui_actions_on_zoom_minus (GtkAction *action,
4526 ModestWindow *window)
4528 g_return_if_fail (MODEST_IS_WINDOW (window));
4530 modest_window_zoom_minus (MODEST_WINDOW (window));
4534 modest_ui_actions_on_toggle_fullscreen (GtkToggleAction *toggle,
4535 ModestWindow *window)
4537 ModestWindowMgr *mgr;
4538 gboolean fullscreen, active;
4539 g_return_if_fail (MODEST_IS_WINDOW (window));
4541 mgr = modest_runtime_get_window_mgr ();
4543 active = (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (toggle)))?1:0;
4544 fullscreen = modest_window_mgr_get_fullscreen_mode (mgr);
4546 if (active != fullscreen) {
4547 modest_window_mgr_set_fullscreen_mode (mgr, active);
4548 #ifndef MODEST_TOOLKIT_HILDON2
4549 gtk_window_present (GTK_WINDOW (window));
4555 modest_ui_actions_on_change_fullscreen (GtkAction *action,
4556 ModestWindow *window)
4558 ModestWindowMgr *mgr;
4559 gboolean fullscreen;
4561 g_return_if_fail (MODEST_IS_WINDOW (window));
4563 mgr = modest_runtime_get_window_mgr ();
4564 fullscreen = modest_window_mgr_get_fullscreen_mode (mgr);
4565 modest_window_mgr_set_fullscreen_mode (mgr, !fullscreen);
4567 #ifndef MODEST_TOOLKIT_HILDON2
4568 gtk_window_present (GTK_WINDOW (window));
4573 * Used by modest_ui_actions_on_details to call do_headers_action
4576 headers_action_show_details (TnyHeader *header,
4577 ModestWindow *window,
4581 modest_platform_run_header_details_dialog (GTK_WINDOW (window), header);
4585 * Show the header details in a ModestDetailsDialog widget
4588 modest_ui_actions_on_details (GtkAction *action,
4591 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
4595 msg = modest_msg_view_window_get_message (MODEST_MSG_VIEW_WINDOW (win));
4599 header = tny_msg_get_header (msg);
4601 headers_action_show_details (header, win, NULL);
4602 g_object_unref (header);
4604 g_object_unref (msg);
4606 } else if (MODEST_IS_MAIN_WINDOW (win)) {
4607 GtkWidget *folder_view, *header_view;
4609 /* Check which widget has the focus */
4610 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
4611 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
4612 if (gtk_widget_is_focus (folder_view)) {
4613 TnyFolderStore *folder_store
4614 = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
4615 if (!folder_store) {
4616 g_warning ("%s: No item was selected.\n", __FUNCTION__);
4619 /* Show only when it's a folder */
4620 /* This function should not be called for account items,
4621 * because we dim the menu item for them. */
4622 if (TNY_IS_FOLDER (folder_store)) {
4623 modest_platform_run_folder_details_dialog (GTK_WINDOW (win),
4624 TNY_FOLDER (folder_store));
4627 g_object_unref (folder_store);
4630 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
4631 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
4632 /* Show details of each header */
4633 do_headers_action (win, headers_action_show_details, header_view);
4635 #ifdef MODEST_TOOLKIT_HILDON2
4636 } else if (MODEST_IS_HEADER_WINDOW (win)) {
4638 GtkWidget *header_view;
4640 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
4641 folder = modest_header_view_get_folder (MODEST_HEADER_VIEW (header_view));
4643 modest_platform_run_folder_details_dialog (GTK_WINDOW (win),
4645 g_object_unref (folder);
4652 modest_ui_actions_on_toggle_show_cc (GtkToggleAction *toggle,
4653 ModestMsgEditWindow *window)
4655 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4657 modest_msg_edit_window_show_cc (window, gtk_toggle_action_get_active (toggle));
4661 modest_ui_actions_on_toggle_show_bcc (GtkToggleAction *toggle,
4662 ModestMsgEditWindow *window)
4664 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4666 modest_msg_edit_window_show_bcc (window, gtk_toggle_action_get_active (toggle));
4670 modest_ui_actions_toggle_folders_view (GtkAction *action,
4671 ModestMainWindow *main_window)
4673 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
4675 if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)))
4676 modest_main_window_set_style (main_window, MODEST_MAIN_WINDOW_STYLE_SPLIT);
4678 modest_main_window_set_style (main_window, MODEST_MAIN_WINDOW_STYLE_SIMPLE);
4682 modest_ui_actions_on_toggle_toolbar (GtkToggleAction *toggle,
4683 ModestWindow *window)
4685 gboolean active, fullscreen = FALSE;
4686 ModestWindowMgr *mgr;
4688 active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (toggle));
4690 /* Check if we want to toggle the toolbar view in fullscreen
4692 if (!strcmp (gtk_action_get_name (GTK_ACTION (toggle)),
4693 "ViewShowToolbarFullScreen")) {
4697 /* Toggle toolbar */
4698 mgr = modest_runtime_get_window_mgr ();
4699 modest_window_mgr_show_toolbars (mgr, G_TYPE_FROM_INSTANCE (window), active, fullscreen);
4703 modest_ui_actions_msg_edit_on_select_font (GtkAction *action,
4704 ModestMsgEditWindow *window)
4706 modest_msg_edit_window_select_font (window);
4711 modest_ui_actions_on_folder_display_name_changed (ModestFolderView *folder_view,
4712 const gchar *display_name,
4715 /* don't update the display name if it was already set;
4716 * updating the display name apparently is expensive */
4717 const gchar* old_name = gtk_window_get_title (window);
4719 if (display_name == NULL)
4722 if (old_name && display_name && strcmp (old_name, display_name) == 0)
4723 return; /* don't do anything */
4725 /* This is usually used to change the title of the main window, which
4726 * is the one that holds the folder view. Note that this change can
4727 * happen even when the widget doesn't have the focus. */
4728 gtk_window_set_title (window, display_name);
4733 modest_ui_actions_on_select_contacts (GtkAction *action, ModestMsgEditWindow *window)
4735 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4736 modest_msg_edit_window_select_contacts (window);
4740 modest_ui_actions_on_check_names (GtkAction *action, ModestMsgEditWindow *window)
4742 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4743 modest_msg_edit_window_check_names (window, FALSE);
4746 #ifndef MODEST_TOOLKIT_HILDON2
4748 * This function is used to track changes in the selection of the
4749 * folder view that is inside the "move to" dialog to enable/disable
4750 * the OK button because we do not want the user to select a disallowed
4751 * destination for a folder.
4752 * The user also not desired to be able to use NEW button on items where
4753 * folder creation is not possibel.
4756 on_move_to_dialog_folder_selection_changed (ModestFolderView* self,
4757 TnyFolderStore *folder_store,
4761 GtkWidget *dialog = NULL;
4762 gboolean ok_sensitive = TRUE, new_sensitive = TRUE;
4763 gboolean moving_folder = FALSE;
4764 gboolean is_local_account = TRUE;
4765 GtkWidget *folder_view = NULL;
4766 ModestTnyFolderRules rules;
4768 g_return_if_fail (MODEST_IS_FOLDER_VIEW(self));
4773 dialog = gtk_widget_get_ancestor (GTK_WIDGET (self), GTK_TYPE_DIALOG);
4777 /* check if folder_store is an remote account */
4778 if (TNY_IS_ACCOUNT (folder_store)) {
4779 TnyAccount *local_account = NULL;
4780 TnyAccount *mmc_account = NULL;
4781 ModestTnyAccountStore *account_store = NULL;
4783 account_store = modest_runtime_get_account_store ();
4784 local_account = modest_tny_account_store_get_local_folders_account (account_store);
4785 mmc_account = modest_tny_account_store_get_mmc_folders_account (account_store);
4787 if ((gpointer) local_account != (gpointer) folder_store &&
4788 (gpointer) mmc_account != (gpointer) folder_store) {
4789 ModestProtocolType proto;
4790 proto = modest_tny_account_get_protocol_type (TNY_ACCOUNT (folder_store));
4791 if (proto == MODEST_PROTOCOL_REGISTRY_TYPE_INVALID) {
4792 proto = MODEST_PROTOCOLS_STORE_MAILDIR;
4794 is_local_account = FALSE;
4795 /* New button should be dimmed on remote
4797 new_sensitive = (modest_protocol_registry_protocol_type_has_tag (modest_runtime_get_protocol_registry (),
4799 MODEST_PROTOCOL_REGISTRY_STORE_HAS_FOLDERS));
4801 g_object_unref (local_account);
4803 /* It could not exist */
4805 g_object_unref (mmc_account);
4808 /* Check the target folder rules */
4809 if (TNY_IS_FOLDER (folder_store)) {
4810 rules = modest_tny_folder_get_rules (TNY_FOLDER (folder_store));
4811 if (rules & MODEST_FOLDER_RULES_FOLDER_NON_WRITEABLE) {
4812 ok_sensitive = FALSE;
4813 new_sensitive = FALSE;
4818 /* Check if we're moving a folder */
4819 if (MODEST_IS_MAIN_WINDOW (user_data)) {
4820 /* Get the widgets */
4821 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (user_data),
4822 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
4823 if (gtk_widget_is_focus (folder_view))
4824 moving_folder = TRUE;
4827 if (moving_folder) {
4828 TnyFolderStore *moved_folder = NULL, *parent = NULL;
4830 /* Get the folder to move */
4831 moved_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
4833 /* Check that we're not moving to the same folder */
4834 if (TNY_IS_FOLDER (moved_folder)) {
4835 parent = tny_folder_get_folder_store (TNY_FOLDER (moved_folder));
4836 if (parent == folder_store)
4837 ok_sensitive = FALSE;
4838 g_object_unref (parent);
4841 if (ok_sensitive && TNY_IS_ACCOUNT (folder_store)) {
4842 /* Do not allow to move to an account unless it's the
4843 local folders account */
4844 if (!is_local_account)
4845 ok_sensitive = FALSE;
4848 if (ok_sensitive && (moved_folder == folder_store)) {
4849 /* Do not allow to move to itself */
4850 ok_sensitive = FALSE;
4852 g_object_unref (moved_folder);
4854 TnyFolder *src_folder = NULL;
4856 /* Moving a message */
4857 if (MODEST_IS_MSG_VIEW_WINDOW (user_data)) {
4859 TnyHeader *header = NULL;
4860 header = modest_msg_view_window_get_header
4861 (MODEST_MSG_VIEW_WINDOW (user_data));
4862 if (!TNY_IS_HEADER(header))
4863 g_warning ("%s: could not get source header", __FUNCTION__);
4865 src_folder = tny_header_get_folder (header);
4868 g_object_unref (header);
4871 TNY_FOLDER (modest_folder_view_get_selected
4872 (MODEST_FOLDER_VIEW (folder_view)));
4875 if (TNY_IS_FOLDER(src_folder)) {
4876 /* Do not allow to move the msg to the same folder */
4877 /* Do not allow to move the msg to an account */
4878 if ((gpointer) src_folder == (gpointer) folder_store ||
4879 TNY_IS_ACCOUNT (folder_store))
4880 ok_sensitive = FALSE;
4881 g_object_unref (src_folder);
4883 g_warning ("%s: could not get source folder", __FUNCTION__);
4887 /* Set sensitivity of the OK and NEW button */
4888 gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), GTK_RESPONSE_OK, ok_sensitive);
4889 gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), MODEST_GTK_RESPONSE_NEW_FOLDER, new_sensitive);
4894 on_move_to_dialog_response (GtkDialog *dialog,
4898 GtkWidget *parent_win;
4899 MoveToInfo *helper = NULL;
4900 ModestFolderView *folder_view;
4902 helper = (MoveToInfo *) user_data;
4904 parent_win = (GtkWidget *) helper->win;
4905 folder_view = MODEST_FOLDER_VIEW (g_object_get_data (G_OBJECT (dialog),
4906 MODEST_MOVE_TO_DIALOG_FOLDER_VIEW));
4909 TnyFolderStore *dst_folder;
4911 case MODEST_GTK_RESPONSE_NEW_FOLDER:
4912 modest_ui_actions_create_folder (GTK_WIDGET (dialog), GTK_WIDGET (folder_view));
4914 case GTK_RESPONSE_NONE:
4915 case GTK_RESPONSE_CANCEL:
4916 case GTK_RESPONSE_DELETE_EVENT:
4918 case GTK_RESPONSE_OK:
4919 dst_folder = modest_folder_view_get_selected (folder_view);
4921 if (MODEST_IS_MAIN_WINDOW (parent_win)) {
4922 /* Clean list to move used for filtering */
4923 modest_folder_view_set_list_to_move (folder_view, NULL);
4925 modest_ui_actions_on_main_window_move_to (NULL,
4926 GTK_WIDGET (folder_view),
4928 MODEST_MAIN_WINDOW (parent_win));
4929 #ifdef MODEST_TOOLKIT_HILDON2
4930 } else if (MODEST_IS_FOLDER_WINDOW (parent_win)) {
4931 /* Clean list to move used for filtering */
4932 modest_folder_view_set_list_to_move (folder_view, NULL);
4934 modest_ui_actions_on_folder_window_move_to (GTK_WIDGET (folder_view),
4937 GTK_WINDOW (parent_win));
4940 /* if the user selected a root folder
4941 (account) then do not perform any action */
4942 if (TNY_IS_ACCOUNT (dst_folder)) {
4943 g_signal_stop_emission_by_name (dialog, "response");
4947 /* Clean list to move used for filtering */
4948 modest_folder_view_set_list_to_move (folder_view, NULL);
4950 /* Moving from headers window in edit mode */
4951 modest_ui_actions_on_window_move_to (NULL, helper->list,
4953 MODEST_WINDOW (parent_win));
4957 g_object_unref (dst_folder);
4961 g_warning ("%s unexpected response id %d", __FUNCTION__, response);
4964 /* Free the helper and exit */
4966 g_object_unref (helper->list);
4967 g_slice_free (MoveToInfo, helper);
4968 gtk_widget_destroy (GTK_WIDGET (dialog));
4972 create_move_to_dialog (GtkWindow *win,
4973 GtkWidget *folder_view,
4974 TnyList *list_to_move)
4976 GtkWidget *dialog, *tree_view = NULL;
4978 dialog = modest_platform_create_move_to_dialog (win, &tree_view);
4980 #ifndef MODEST_TOOLKIT_HILDON2
4981 /* Track changes in the selection to
4982 * disable the OK button whenever "Move to" is not possible
4983 * disbale NEW button whenever New is not possible */
4984 g_signal_connect (tree_view,
4985 "folder_selection_changed",
4986 G_CALLBACK (on_move_to_dialog_folder_selection_changed),
4990 /* It could happen that we're trying to move a message from a
4991 window (msg window for example) after the main window was
4992 closed, so we can not just get the model of the folder
4994 if (MODEST_IS_FOLDER_VIEW (folder_view)) {
4995 const gchar *visible_id = NULL;
4997 modest_folder_view_set_style (MODEST_FOLDER_VIEW (tree_view),
4998 MODEST_FOLDER_VIEW_STYLE_SHOW_ALL);
4999 modest_folder_view_copy_model (MODEST_FOLDER_VIEW(folder_view),
5000 MODEST_FOLDER_VIEW(tree_view));
5003 modest_folder_view_get_account_id_of_visible_server_account (MODEST_FOLDER_VIEW(folder_view));
5005 /* Show the same account than the one that is shown in the main window */
5006 modest_folder_view_set_account_id_of_visible_server_account (MODEST_FOLDER_VIEW(tree_view),
5009 const gchar *active_account_name = NULL;
5010 ModestAccountMgr *mgr = NULL;
5011 ModestAccountSettings *settings = NULL;
5012 ModestServerAccountSettings *store_settings = NULL;
5014 modest_folder_view_set_style (MODEST_FOLDER_VIEW (tree_view),
5015 MODEST_FOLDER_VIEW_STYLE_SHOW_ALL);
5016 modest_folder_view_update_model (MODEST_FOLDER_VIEW (tree_view),
5017 TNY_ACCOUNT_STORE (modest_runtime_get_account_store ()));
5019 active_account_name = modest_window_get_active_account (MODEST_WINDOW (win));
5020 mgr = modest_runtime_get_account_mgr ();
5021 settings = modest_account_mgr_load_account_settings (mgr, active_account_name);
5024 const gchar *store_account_name;
5025 store_settings = modest_account_settings_get_store_settings (settings);
5026 store_account_name = modest_server_account_settings_get_account_name (store_settings);
5028 modest_folder_view_set_account_id_of_visible_server_account (MODEST_FOLDER_VIEW (tree_view),
5029 store_account_name);
5030 g_object_unref (store_settings);
5031 g_object_unref (settings);
5035 /* we keep a pointer to the embedded folder view, so we can
5036 * retrieve it with get_folder_view_from_move_to_dialog (see
5037 * above) later (needed for focus handling)
5039 g_object_set_data (G_OBJECT(dialog), MODEST_MOVE_TO_DIALOG_FOLDER_VIEW, tree_view);
5041 /* Hide special folders */
5042 modest_folder_view_show_non_move_folders (MODEST_FOLDER_VIEW (tree_view), FALSE);
5044 modest_folder_view_set_list_to_move (MODEST_FOLDER_VIEW (tree_view), list_to_move);
5045 #ifndef MODEST_TOOLKIT_HILDON2
5046 modest_folder_view_select_first_inbox_or_local (MODEST_FOLDER_VIEW (tree_view));
5049 gtk_widget_show (GTK_WIDGET (tree_view));
5055 * Shows a confirmation dialog to the user when we're moving messages
5056 * from a remote server to the local storage. Returns the dialog
5057 * response. If it's other kind of movement then it always returns
5060 * This one is used by the next functions:
5061 * modest_ui_actions_on_paste - commented out
5062 * drag_and_drop_from_header_view (for d&d in modest_folder_view.c)
5065 modest_ui_actions_msgs_move_to_confirmation (ModestWindow *win,
5066 TnyFolder *dest_folder,
5070 gint response = GTK_RESPONSE_OK;
5071 TnyAccount *account = NULL;
5072 TnyFolder *src_folder = NULL;
5073 TnyIterator *iter = NULL;
5074 TnyHeader *header = NULL;
5076 /* return with OK if the destination is a remote folder */
5077 if (modest_tny_folder_is_remote_folder (dest_folder))
5078 return GTK_RESPONSE_OK;
5080 /* Get source folder */
5081 iter = tny_list_create_iterator (headers);
5082 header = TNY_HEADER (tny_iterator_get_current (iter));
5084 src_folder = tny_header_get_folder (header);
5085 g_object_unref (header);
5087 g_object_unref (iter);
5089 /* if no src_folder, message may be an attahcment */
5090 if (src_folder == NULL)
5091 return GTK_RESPONSE_CANCEL;
5093 /* If the source is a local or MMC folder */
5094 if (!modest_tny_folder_is_remote_folder (src_folder)) {
5095 g_object_unref (src_folder);
5096 return GTK_RESPONSE_OK;
5099 /* Get the account */
5100 account = tny_folder_get_account (src_folder);
5102 /* now if offline we ask the user */
5103 if(connect_to_get_msg (win, tny_list_get_length (headers), account))
5104 response = GTK_RESPONSE_OK;
5106 response = GTK_RESPONSE_CANCEL;
5109 g_object_unref (src_folder);
5110 g_object_unref (account);
5116 move_to_helper_destroyer (gpointer user_data)
5118 MoveToHelper *helper = (MoveToHelper *) user_data;
5120 /* Close the "Pasting" information banner */
5121 if (helper->banner) {
5122 gtk_widget_destroy (GTK_WIDGET (helper->banner));
5123 g_object_unref (helper->banner);
5125 if (gtk_tree_row_reference_valid (helper->reference)) {
5126 gtk_tree_row_reference_free (helper->reference);
5127 helper->reference = NULL;
5133 move_to_cb (ModestMailOperation *mail_op,
5136 MoveToHelper *helper = (MoveToHelper *) user_data;
5137 GObject *object = modest_mail_operation_get_source (mail_op);
5139 /* Note that the operation could have failed, in that case do
5141 if (modest_mail_operation_get_status (mail_op) !=
5142 MODEST_MAIL_OPERATION_STATUS_SUCCESS)
5145 if (MODEST_IS_MSG_VIEW_WINDOW (object)) {
5146 ModestMsgViewWindow *self = MODEST_MSG_VIEW_WINDOW (object);
5148 if (!modest_msg_view_window_select_next_message (self) &&
5149 !modest_msg_view_window_select_previous_message (self)) {
5150 /* No more messages to view, so close this window */
5151 modest_ui_actions_on_close_window (NULL, MODEST_WINDOW(self));
5153 } else if (MODEST_IS_MAIN_WINDOW (object) &&
5154 gtk_tree_row_reference_valid (helper->reference)) {
5155 GtkWidget *header_view;
5157 GtkTreeSelection *sel;
5159 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(object),
5160 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5161 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (header_view));
5162 path = gtk_tree_row_reference_get_path (helper->reference);
5163 /* We need to unselect the previous one
5164 because we could be copying instead of
5166 gtk_tree_selection_unselect_all (sel);
5167 gtk_tree_selection_select_path (sel, path);
5168 gtk_tree_path_free (path);
5170 g_object_unref (object);
5173 /* Destroy the helper */
5174 move_to_helper_destroyer (helper);
5178 folder_move_to_cb (ModestMailOperation *mail_op,
5179 TnyFolder *new_folder,
5182 GtkWidget *folder_view;
5185 object = modest_mail_operation_get_source (mail_op);
5186 if (MODEST_IS_MAIN_WINDOW (object)) {
5187 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(object),
5188 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
5189 g_object_ref (folder_view);
5190 g_object_unref (object);
5191 move_to_cb (mail_op, user_data);
5192 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view), new_folder, FALSE);
5193 g_object_unref (folder_view);
5195 move_to_cb (mail_op, user_data);
5200 msgs_move_to_cb (ModestMailOperation *mail_op,
5203 move_to_cb (mail_op, user_data);
5207 modest_ui_actions_move_folder_error_handler (ModestMailOperation *mail_op,
5210 GObject *win = NULL;
5212 #ifndef MODEST_TOOLKIT_HILDON2
5213 ModestWindow *main_window = NULL;
5215 /* Disable next automatic folder selection */
5216 main_window = modest_window_mgr_get_main_window (modest_runtime_get_window_mgr (),
5217 FALSE); /* don't create */
5219 GtkWidget *folder_view = NULL;
5221 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (main_window),
5222 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
5223 modest_folder_view_disable_next_folder_selection (MODEST_FOLDER_VIEW(folder_view));
5225 if (user_data && TNY_IS_FOLDER (user_data)) {
5226 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view),
5227 TNY_FOLDER (user_data), FALSE);
5231 /* Show notification dialog only if the main window exists */
5232 win = modest_mail_operation_get_source (mail_op);
5233 modest_platform_run_information_dialog ((GtkWindow *) win,
5234 _("mail_in_ui_folder_move_target_error"),
5237 g_object_unref (win);
5241 open_msg_for_purge_cb (ModestMailOperation *mail_op,
5250 gint pending_purges = 0;
5251 gboolean some_purged = FALSE;
5252 ModestWindow *win = MODEST_WINDOW (user_data);
5253 ModestWindowMgr *mgr = modest_runtime_get_window_mgr ();
5255 /* If there was any error */
5256 if (!modest_ui_actions_msg_retrieval_check (mail_op, header, msg)) {
5257 modest_window_mgr_unregister_header (mgr, header);
5261 /* Once the message has been retrieved for purging, we check if
5262 * it's all ok for purging */
5264 parts = tny_simple_list_new ();
5265 tny_mime_part_get_parts (TNY_MIME_PART (msg), parts);
5266 iter = tny_list_create_iterator (parts);
5268 while (!tny_iterator_is_done (iter)) {
5270 part = TNY_MIME_PART (tny_iterator_get_current (iter));
5271 if (part && (tny_mime_part_is_attachment (part) || TNY_IS_MSG (part))) {
5272 if (tny_mime_part_is_purged (part))
5279 g_object_unref (part);
5281 tny_iterator_next (iter);
5283 g_object_unref (iter);
5286 if (pending_purges>0) {
5288 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),_("mcen_nc_purge_file_text_inbox"));
5290 if (response == GTK_RESPONSE_OK) {
5293 modest_platform_animation_banner (GTK_WIDGET (win), NULL, _("mcen_me_inbox_remove_attachments"));
5294 iter = tny_list_create_iterator (parts);
5295 while (!tny_iterator_is_done (iter)) {
5298 part = TNY_MIME_PART (tny_iterator_get_current (iter));
5299 if (part && (tny_mime_part_is_attachment (part) || TNY_IS_MSG (part)))
5300 tny_mime_part_set_purged (part);
5303 g_object_unref (part);
5305 tny_iterator_next (iter);
5307 g_object_unref (iter);
5309 tny_msg_rewrite_cache (msg);
5311 gtk_widget_destroy (info);
5315 modest_window_mgr_unregister_header (mgr, header);
5317 g_object_unref (parts);
5321 modest_ui_actions_on_main_window_remove_attachments (GtkAction *action,
5322 ModestMainWindow *win)
5324 GtkWidget *header_view;
5325 TnyList *header_list;
5327 TnyHeaderFlags flags;
5328 ModestWindow *msg_view_window = NULL;
5331 g_return_if_fail (MODEST_IS_MAIN_WINDOW (win));
5333 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
5334 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5336 header_list = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
5338 g_warning ("%s: no header selected", __FUNCTION__);
5342 if (tny_list_get_length (header_list) == 1) {
5343 TnyIterator *iter = tny_list_create_iterator (header_list);
5344 header = TNY_HEADER (tny_iterator_get_current (iter));
5345 g_object_unref (iter);
5349 if (!header || !TNY_IS_HEADER(header)) {
5350 g_warning ("%s: header is not valid", __FUNCTION__);
5354 found = modest_window_mgr_find_registered_header (modest_runtime_get_window_mgr (),
5355 header, &msg_view_window);
5356 flags = tny_header_get_flags (header);
5357 if (!(flags & TNY_HEADER_FLAG_CACHED))
5360 if (msg_view_window != NULL)
5361 modest_msg_view_window_remove_attachments (MODEST_MSG_VIEW_WINDOW (msg_view_window), TRUE);
5363 /* do nothing; uid was registered before, so window is probably on it's way */
5364 g_warning ("debug: header %p has already been registered", header);
5367 ModestMailOperation *mail_op = NULL;
5368 modest_window_mgr_register_header (modest_runtime_get_window_mgr (), header, NULL);
5369 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT (win),
5370 modest_ui_actions_disk_operations_error_handler,
5372 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
5373 modest_mail_operation_get_msg (mail_op, header, FALSE, open_msg_for_purge_cb, win);
5375 g_object_unref (mail_op);
5378 g_object_unref (header);
5380 g_object_unref (header_list);
5384 * Checks if we need a connection to do the transfer and if the user
5385 * wants to connect to complete it
5388 modest_ui_actions_xfer_messages_check (GtkWindow *parent_window,
5389 TnyFolderStore *src_folder,
5391 TnyFolder *dst_folder,
5392 gboolean delete_originals,
5393 gboolean *need_connection,
5396 TnyAccount *src_account;
5397 gint uncached_msgs = 0;
5399 /* We don't need any further check if
5401 * 1- the source folder is local OR
5402 * 2- the device is already online
5404 if (!modest_tny_folder_store_is_remote (src_folder) ||
5405 tny_device_is_online (modest_runtime_get_device())) {
5406 *need_connection = FALSE;
5411 /* We must ask for a connection when
5413 * - the message(s) is not already cached OR
5414 * - the message(s) is cached but the leave_on_server setting
5415 * is FALSE (because we need to sync the source folder to
5416 * delete the message from the server (for IMAP we could do it
5417 * offline, it'll take place the next time we get a
5420 uncached_msgs = header_list_count_uncached_msgs (headers);
5421 src_account = get_account_from_folder_store (src_folder);
5422 if (uncached_msgs > 0) {
5426 *need_connection = TRUE;
5427 num_headers = tny_list_get_length (headers);
5428 msg = ngettext ("mcen_nc_get_msg", "mcen_nc_get_msgs", num_headers);
5430 if (modest_platform_run_confirmation_dialog (parent_window, msg) ==
5431 GTK_RESPONSE_CANCEL) {
5437 /* The transfer is possible and the user wants to */
5440 if (remote_folder_has_leave_on_server (src_folder) && delete_originals) {
5441 const gchar *account_name;
5442 gboolean leave_on_server;
5444 account_name = modest_tny_account_get_parent_modest_account_name_for_server_account (src_account);
5445 leave_on_server = modest_account_mgr_get_leave_on_server (modest_runtime_get_account_mgr (),
5448 if (leave_on_server == TRUE) {
5449 *need_connection = FALSE;
5451 *need_connection = TRUE;
5454 *need_connection = FALSE;
5459 g_object_unref (src_account);
5463 xfer_messages_error_handler (ModestMailOperation *mail_op,
5467 const GError *error;
5469 win = modest_mail_operation_get_source (mail_op);
5470 error = modest_mail_operation_get_error (mail_op);
5472 if (error && is_memory_full_error ((GError *) error, mail_op))
5473 modest_platform_information_banner ((GtkWidget *) win,
5474 NULL, _KR("cerm_device_memory_full"));
5476 modest_platform_run_information_dialog ((GtkWindow *) win,
5477 _("mail_in_ui_folder_move_target_error"),
5480 g_object_unref (win);
5484 TnyFolderStore *dst_folder;
5489 * Utility function that transfer messages from both the main window
5490 * and the msg view window when using the "Move to" dialog
5493 xfer_messages_performer (gboolean canceled,
5495 GtkWindow *parent_window,
5496 TnyAccount *account,
5499 ModestWindow *win = MODEST_WINDOW (parent_window);
5500 TnyAccount *dst_account = NULL;
5501 gboolean dst_forbids_message_add = FALSE;
5502 XferMsgsHelper *helper;
5503 MoveToHelper *movehelper;
5504 ModestMailOperation *mail_op;
5506 helper = (XferMsgsHelper *) user_data;
5508 if (canceled || err) {
5509 if (!check_memory_full_error ((GtkWidget *) parent_window, err)) {
5510 /* Show the proper error message */
5511 modest_ui_actions_on_account_connection_error (parent_window, account);
5516 dst_account = tny_folder_get_account (TNY_FOLDER (helper->dst_folder));
5518 /* tinymail will return NULL for local folders it seems */
5519 dst_forbids_message_add = modest_protocol_registry_protocol_type_has_tag (modest_runtime_get_protocol_registry (),
5520 modest_tny_account_get_protocol_type (dst_account),
5521 MODEST_PROTOCOL_REGISTRY_STORE_FORBID_MESSAGE_ADD);
5522 g_object_unref (dst_account);
5524 if (dst_forbids_message_add) {
5525 modest_platform_information_banner (GTK_WIDGET (win),
5527 ngettext("mail_in_ui_folder_move_target_error",
5528 "mail_in_ui_folder_move_targets_error",
5529 tny_list_get_length (helper->headers)));
5533 movehelper = g_new0 (MoveToHelper, 1);
5535 #ifndef MODEST_TOOLKIT_HILDON2
5536 movehelper->banner = modest_platform_animation_banner (GTK_WIDGET (win), NULL,
5537 _CS("ckct_nw_pasting"));
5538 if (movehelper->banner != NULL) {
5539 g_object_ref (movehelper->banner);
5540 gtk_widget_show (GTK_WIDGET (movehelper->banner));
5544 if (MODEST_IS_MAIN_WINDOW (win)) {
5545 GtkWidget *header_view =
5546 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
5547 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5548 movehelper->reference = get_next_after_selected_headers (MODEST_HEADER_VIEW (header_view));
5551 /* Perform the mail operation */
5552 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT(win),
5553 xfer_messages_error_handler,
5555 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
5558 modest_mail_operation_xfer_msgs (mail_op,
5560 TNY_FOLDER (helper->dst_folder),
5565 g_object_unref (G_OBJECT (mail_op));
5567 g_object_unref (helper->dst_folder);
5568 g_object_unref (helper->headers);
5569 g_slice_free (XferMsgsHelper, helper);
5573 TnyFolder *src_folder;
5574 TnyFolderStore *dst_folder;
5575 gboolean delete_original;
5576 GtkWidget *folder_view;
5580 on_move_folder_cb (gboolean canceled, GError *err, GtkWindow *parent_window,
5581 TnyAccount *account, gpointer user_data)
5583 MoveFolderInfo *info = (MoveFolderInfo*)user_data;
5584 GtkTreeSelection *sel;
5585 ModestMailOperation *mail_op = NULL;
5587 if (canceled || err || !MODEST_IS_WINDOW (parent_window)) {
5588 g_object_unref (G_OBJECT (info->src_folder));
5589 g_object_unref (G_OBJECT (info->dst_folder));
5594 MoveToHelper *helper = g_new0 (MoveToHelper, 1);
5595 #ifndef MODEST_TOOLKIT_HILDON2
5596 helper->banner = modest_platform_animation_banner (GTK_WIDGET (parent_window), NULL,
5597 _CS("ckct_nw_pasting"));
5598 if (helper->banner != NULL) {
5599 g_object_ref (helper->banner);
5600 gtk_widget_show (GTK_WIDGET(helper->banner));
5603 /* Clean folder on header view before moving it */
5604 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (info->folder_view));
5605 gtk_tree_selection_unselect_all (sel);
5607 /* Let gtk events run. We need that the folder
5608 view frees its reference to the source
5609 folder *before* issuing the mail operation
5610 so we need the signal handler of selection
5611 changed to happen before the mail
5613 while (gtk_events_pending ())
5614 gtk_main_iteration (); */
5617 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
5618 modest_ui_actions_move_folder_error_handler,
5619 info->src_folder, NULL);
5620 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
5623 /* Select *after* the changes */
5624 /* TODO: this function hangs UI after transfer */
5625 /* modest_folder_view_select_folder (MODEST_FOLDER_VIEW(folder_view), */
5626 /* TNY_FOLDER (src_folder), TRUE); */
5628 if (MODEST_IS_MAIN_WINDOW (parent_window)) {
5629 modest_folder_view_select_folder (MODEST_FOLDER_VIEW(info->folder_view),
5630 TNY_FOLDER (info->dst_folder), TRUE);
5632 modest_mail_operation_xfer_folder (mail_op,
5633 TNY_FOLDER (info->src_folder),
5635 info->delete_original,
5638 g_object_unref (G_OBJECT (info->src_folder));
5640 /* if (modest_mail_operation_get_status (mail_op) == MODEST_MAIL_OPERATION_STATUS_SUCCESS) { */
5643 /* Unref mail operation */
5644 g_object_unref (G_OBJECT (mail_op));
5645 g_object_unref (G_OBJECT (info->dst_folder));
5650 get_account_from_folder_store (TnyFolderStore *folder_store)
5652 if (TNY_IS_ACCOUNT (folder_store))
5653 return g_object_ref (folder_store);
5655 return tny_folder_get_account (TNY_FOLDER (folder_store));
5659 * UI handler for the "Move to" action when invoked from the
5663 modest_ui_actions_on_main_window_move_to (GtkAction *action,
5664 GtkWidget *folder_view,
5665 TnyFolderStore *dst_folder,
5666 ModestMainWindow *win)
5668 ModestHeaderView *header_view = NULL;
5669 TnyFolderStore *src_folder = NULL;
5671 g_return_if_fail (MODEST_IS_MAIN_WINDOW (win));
5673 /* Get the source folder */
5674 src_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
5676 /* Get header view */
5677 header_view = (ModestHeaderView *)
5678 modest_main_window_get_child_widget (win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5680 /* Get folder or messages to transfer */
5681 if (gtk_widget_is_focus (folder_view)) {
5682 gboolean do_xfer = TRUE;
5684 /* Allow only to transfer folders to the local root folder */
5685 if (TNY_IS_ACCOUNT (dst_folder) &&
5686 !MODEST_IS_TNY_LOCAL_FOLDERS_ACCOUNT (dst_folder) &&
5687 !modest_tny_account_is_memory_card_account (TNY_ACCOUNT (dst_folder))) {
5689 } else if (!TNY_IS_FOLDER (src_folder)) {
5690 g_warning ("%s: src_folder is not a TnyFolder.\n", __FUNCTION__);
5695 MoveFolderInfo *info = g_new0 (MoveFolderInfo, 1);
5696 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5698 info->src_folder = g_object_ref (src_folder);
5699 info->dst_folder = g_object_ref (dst_folder);
5700 info->delete_original = TRUE;
5701 info->folder_view = folder_view;
5703 connect_info->callback = on_move_folder_cb;
5704 connect_info->dst_account = get_account_from_folder_store (TNY_FOLDER_STORE (dst_folder));
5705 connect_info->data = info;
5707 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5708 TNY_FOLDER_STORE (src_folder),
5711 } else if (gtk_widget_is_focus (GTK_WIDGET(header_view))) {
5714 headers = modest_header_view_get_selected_headers(header_view);
5716 /* Transfer the messages */
5717 modest_ui_actions_transfer_messages_helper (GTK_WINDOW (win), TNY_FOLDER (src_folder),
5718 headers, TNY_FOLDER (dst_folder));
5720 g_object_unref (headers);
5724 g_object_unref (src_folder);
5727 #ifdef MODEST_TOOLKIT_HILDON2
5729 * UI handler for the "Move to" action when invoked from the
5730 * ModestFolderWindow
5733 modest_ui_actions_on_folder_window_move_to (GtkWidget *folder_view,
5734 TnyFolderStore *dst_folder,
5738 TnyFolderStore *src_folder = NULL;
5739 TnyIterator *iterator;
5741 if (tny_list_get_length (selection) != 1)
5744 iterator = tny_list_create_iterator (selection);
5745 src_folder = TNY_FOLDER_STORE (tny_iterator_get_current (iterator));
5746 g_object_unref (iterator);
5749 gboolean do_xfer = TRUE;
5751 /* Allow only to transfer folders to the local root folder */
5752 if (TNY_IS_ACCOUNT (dst_folder) &&
5753 !MODEST_IS_TNY_LOCAL_FOLDERS_ACCOUNT (dst_folder) &&
5754 !modest_tny_account_is_memory_card_account (TNY_ACCOUNT (dst_folder))) {
5757 modest_platform_run_information_dialog (win,
5758 _("mail_in_ui_folder_move_target_error"),
5760 } else if (!TNY_IS_FOLDER (src_folder)) {
5761 g_warning ("%s: src_folder is not a TnyFolder.\n", __FUNCTION__);
5766 MoveFolderInfo *info = g_new0 (MoveFolderInfo, 1);
5767 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5769 info->src_folder = g_object_ref (src_folder);
5770 info->dst_folder = g_object_ref (dst_folder);
5771 info->delete_original = TRUE;
5772 info->folder_view = folder_view;
5774 connect_info->callback = on_move_folder_cb;
5775 connect_info->dst_account = get_account_from_folder_store (TNY_FOLDER_STORE (dst_folder));
5776 connect_info->data = info;
5778 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5779 TNY_FOLDER_STORE (src_folder),
5784 g_object_unref (src_folder);
5790 modest_ui_actions_transfer_messages_helper (GtkWindow *win,
5791 TnyFolder *src_folder,
5793 TnyFolder *dst_folder)
5795 gboolean need_connection = TRUE;
5796 gboolean do_xfer = TRUE;
5797 XferMsgsHelper *helper;
5799 g_return_if_fail (TNY_IS_FOLDER (src_folder));
5800 g_return_if_fail (TNY_IS_FOLDER (dst_folder));
5801 g_return_if_fail (TNY_IS_LIST (headers));
5803 modest_ui_actions_xfer_messages_check (win, TNY_FOLDER_STORE (src_folder),
5804 headers, TNY_FOLDER (dst_folder),
5805 TRUE, &need_connection,
5808 /* If we don't want to transfer just return */
5812 /* Create the helper */
5813 helper = g_slice_new (XferMsgsHelper);
5814 helper->dst_folder = g_object_ref (dst_folder);
5815 helper->headers = g_object_ref (headers);
5817 if (need_connection) {
5818 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5819 connect_info->callback = xfer_messages_performer;
5820 connect_info->dst_account = tny_folder_get_account (TNY_FOLDER (dst_folder));
5821 connect_info->data = helper;
5823 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5824 TNY_FOLDER_STORE (src_folder),
5827 TnyAccount *src_account = get_account_from_folder_store (TNY_FOLDER_STORE (src_folder));
5828 xfer_messages_performer (FALSE, NULL, GTK_WINDOW (win),
5829 src_account, helper);
5830 g_object_unref (src_account);
5835 * UI handler for the "Move to" action when invoked from the
5836 * ModestMsgViewWindow
5839 modest_ui_actions_on_window_move_to (GtkAction *action,
5841 TnyFolderStore *dst_folder,
5844 TnyFolder *src_folder = NULL;
5846 g_return_if_fail (TNY_IS_FOLDER (dst_folder));
5849 TnyHeader *header = NULL;
5852 iter = tny_list_create_iterator (headers);
5853 header = (TnyHeader *) tny_iterator_get_current (iter);
5854 src_folder = tny_header_get_folder (header);
5856 /* Transfer the messages */
5857 modest_ui_actions_transfer_messages_helper (GTK_WINDOW (win), src_folder,
5859 TNY_FOLDER (dst_folder));
5862 g_object_unref (header);
5863 g_object_unref (iter);
5864 g_object_unref (src_folder);
5869 modest_ui_actions_on_move_to (GtkAction *action,
5872 modest_ui_actions_on_edit_mode_move_to (win);
5876 modest_ui_actions_on_edit_mode_move_to (ModestWindow *win)
5878 GtkWidget *dialog = NULL;
5879 MoveToInfo *helper = NULL;
5880 TnyList *list_to_move;
5882 g_return_val_if_fail (MODEST_IS_WINDOW (win), FALSE);
5884 #ifndef MODEST_TOOLKIT_HILDON2
5885 /* Get the main window if exists */
5886 ModestMainWindow *main_window;
5887 if (MODEST_IS_MAIN_WINDOW (win))
5888 main_window = MODEST_MAIN_WINDOW (win);
5891 MODEST_MAIN_WINDOW (modest_window_mgr_get_main_window (modest_runtime_get_window_mgr (),
5892 FALSE)); /* don't create */
5895 list_to_move = modest_platform_get_list_to_move (MODEST_WINDOW (win));
5900 if (tny_list_get_length (list_to_move) < 1) {
5901 g_object_unref (list_to_move);
5905 /* Create and run the dialog */
5906 dialog = create_move_to_dialog (GTK_WINDOW (win), NULL, list_to_move);
5907 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (),
5908 GTK_WINDOW (dialog),
5912 helper = g_slice_new0 (MoveToInfo);
5913 helper->list = list_to_move;
5916 /* Listen to response signal */
5917 g_signal_connect (dialog, "response", G_CALLBACK (on_move_to_dialog_response), helper);
5919 /* Show the dialog */
5920 gtk_widget_show (dialog);
5926 * Calls #HeadersFunc for each header already selected in the main
5927 * window or the message currently being shown in the msg view window
5930 do_headers_action (ModestWindow *win,
5934 TnyList *headers_list = NULL;
5935 TnyIterator *iter = NULL;
5936 TnyHeader *header = NULL;
5937 TnyFolder *folder = NULL;
5940 headers_list = get_selected_headers (win);
5944 /* Get the folder */
5945 iter = tny_list_create_iterator (headers_list);
5946 header = TNY_HEADER (tny_iterator_get_current (iter));
5948 folder = tny_header_get_folder (header);
5949 g_object_unref (header);
5952 /* Call the function for each header */
5953 while (!tny_iterator_is_done (iter)) {
5954 header = TNY_HEADER (tny_iterator_get_current (iter));
5955 func (header, win, user_data);
5956 g_object_unref (header);
5957 tny_iterator_next (iter);
5960 /* Trick: do a poke status in order to speed up the signaling
5963 tny_folder_poke_status (folder);
5964 g_object_unref (folder);
5968 g_object_unref (iter);
5969 g_object_unref (headers_list);
5973 modest_ui_actions_view_attachment (GtkAction *action,
5974 ModestWindow *window)
5976 if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
5977 modest_msg_view_window_view_attachment (MODEST_MSG_VIEW_WINDOW (window), NULL);
5979 /* not supported window for this action */
5980 g_return_if_reached ();
5985 modest_ui_actions_save_attachments (GtkAction *action,
5986 ModestWindow *window)
5988 if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
5990 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
5993 modest_msg_view_window_save_attachments (MODEST_MSG_VIEW_WINDOW (window), NULL);
5995 /* not supported window for this action */
5996 g_return_if_reached ();
6001 modest_ui_actions_remove_attachments (GtkAction *action,
6002 ModestWindow *window)
6004 if (MODEST_IS_MAIN_WINDOW (window)) {
6005 modest_ui_actions_on_main_window_remove_attachments (action, MODEST_MAIN_WINDOW (window));
6006 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
6007 modest_msg_view_window_remove_attachments (MODEST_MSG_VIEW_WINDOW (window), FALSE);
6009 /* not supported window for this action */
6010 g_return_if_reached ();
6015 modest_ui_actions_on_settings (GtkAction *action,
6020 dialog = modest_platform_get_global_settings_dialog ();
6021 gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (win));
6022 gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
6023 gtk_widget_show_all (dialog);
6025 gtk_dialog_run (GTK_DIALOG (dialog));
6027 gtk_widget_destroy (dialog);
6031 modest_ui_actions_on_help (GtkAction *action,
6034 /* Help app is not available at all in fremantle */
6035 #ifndef MODEST_TOOLKIT_HILDON2
6036 const gchar *help_id;
6038 g_return_if_fail (win && GTK_IS_WINDOW(win));
6040 help_id = modest_window_mgr_get_help_id (modest_runtime_get_window_mgr(), win);
6043 modest_platform_show_help (GTK_WINDOW (win), help_id);
6048 modest_ui_actions_on_csm_help (GtkAction *action,
6051 /* Help app is not available at all in fremantle */
6052 #ifndef MODEST_TOOLKIT_HILDON2
6054 const gchar* help_id = NULL;
6055 GtkWidget *folder_view;
6056 TnyFolderStore *folder_store;
6058 g_return_if_fail (win && MODEST_IS_MAIN_WINDOW (win));
6060 /* Get selected folder */
6061 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
6062 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
6063 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
6065 /* Switch help_id */
6066 if (folder_store && TNY_IS_FOLDER (folder_store))
6067 help_id = modest_tny_folder_get_help_id (TNY_FOLDER (folder_store));
6070 g_object_unref (folder_store);
6073 modest_platform_show_help (GTK_WINDOW (win), help_id);
6075 modest_ui_actions_on_help (action, win);
6080 retrieve_contents_cb (ModestMailOperation *mail_op,
6087 /* We only need this callback to show an error in case of
6088 memory low condition */
6089 if (!modest_ui_actions_msg_retrieval_check (mail_op, header, msg)) {
6090 g_debug ("%s: message failed to retrieve. Memory low?", __FUNCTION__);
6095 retrieve_msg_contents_performer (gboolean canceled,
6097 GtkWindow *parent_window,
6098 TnyAccount *account,
6101 ModestMailOperation *mail_op;
6102 TnyList *headers = TNY_LIST (user_data);
6104 if (err || canceled) {
6105 check_memory_full_error ((GtkWidget *) parent_window, err);
6109 /* Create mail operation */
6110 mail_op = modest_mail_operation_new_with_error_handling ((GObject *) parent_window,
6111 modest_ui_actions_disk_operations_error_handler,
6113 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
6114 modest_mail_operation_get_msgs_full (mail_op, headers, retrieve_contents_cb, NULL, NULL);
6117 g_object_unref (mail_op);
6119 g_object_unref (headers);
6120 g_object_unref (account);
6124 modest_ui_actions_on_retrieve_msg_contents (GtkAction *action,
6125 ModestWindow *window)
6127 TnyList *headers = NULL;
6128 TnyAccount *account = NULL;
6129 TnyIterator *iter = NULL;
6130 TnyHeader *header = NULL;
6131 TnyFolder *folder = NULL;
6134 headers = get_selected_headers (window);
6138 /* Pick the account */
6139 iter = tny_list_create_iterator (headers);
6140 header = TNY_HEADER (tny_iterator_get_current (iter));
6141 folder = tny_header_get_folder (header);
6142 account = tny_folder_get_account (folder);
6143 g_object_unref (folder);
6144 g_object_unref (header);
6145 g_object_unref (iter);
6147 /* Connect and perform the message retrieval */
6148 modest_platform_connect_and_perform ((GtkWindow *) window, TRUE,
6149 g_object_ref (account),
6150 retrieve_msg_contents_performer,
6151 g_object_ref (headers));
6154 g_object_unref (account);
6155 g_object_unref (headers);
6159 modest_ui_actions_check_toolbar_dimming_rules (ModestWindow *window)
6161 g_return_if_fail (MODEST_IS_WINDOW (window));
6164 modest_window_check_dimming_rules_group (window, MODEST_DIMMING_RULES_TOOLBAR);
6168 modest_ui_actions_check_menu_dimming_rules (ModestWindow *window)
6170 g_return_if_fail (MODEST_IS_WINDOW (window));
6173 modest_window_check_dimming_rules_group (window, MODEST_DIMMING_RULES_MENU);
6177 modest_ui_actions_on_email_menu_activated (GtkAction *action,
6178 ModestWindow *window)
6180 g_return_if_fail (MODEST_IS_WINDOW (window));
6183 modest_ui_actions_check_menu_dimming_rules (window);
6187 modest_ui_actions_on_edit_menu_activated (GtkAction *action,
6188 ModestWindow *window)
6190 g_return_if_fail (MODEST_IS_WINDOW (window));
6193 modest_ui_actions_check_menu_dimming_rules (window);
6197 modest_ui_actions_on_view_menu_activated (GtkAction *action,
6198 ModestWindow *window)
6200 g_return_if_fail (MODEST_IS_WINDOW (window));
6203 modest_ui_actions_check_menu_dimming_rules (window);
6207 modest_ui_actions_on_format_menu_activated (GtkAction *action,
6208 ModestWindow *window)
6210 g_return_if_fail (MODEST_IS_WINDOW (window));
6213 modest_ui_actions_check_menu_dimming_rules (window);
6217 modest_ui_actions_on_tools_menu_activated (GtkAction *action,
6218 ModestWindow *window)
6220 g_return_if_fail (MODEST_IS_WINDOW (window));
6223 modest_ui_actions_check_menu_dimming_rules (window);
6227 modest_ui_actions_on_attachment_menu_activated (GtkAction *action,
6228 ModestWindow *window)
6230 g_return_if_fail (MODEST_IS_WINDOW (window));
6233 modest_ui_actions_check_menu_dimming_rules (window);
6237 modest_ui_actions_on_toolbar_csm_menu_activated (GtkAction *action,
6238 ModestWindow *window)
6240 g_return_if_fail (MODEST_IS_WINDOW (window));
6243 modest_ui_actions_check_menu_dimming_rules (window);
6247 modest_ui_actions_on_folder_view_csm_menu_activated (GtkAction *action,
6248 ModestWindow *window)
6250 g_return_if_fail (MODEST_IS_WINDOW (window));
6253 modest_ui_actions_check_menu_dimming_rules (window);
6257 modest_ui_actions_on_header_view_csm_menu_activated (GtkAction *action,
6258 ModestWindow *window)
6260 g_return_if_fail (MODEST_IS_WINDOW (window));
6263 modest_ui_actions_check_menu_dimming_rules (window);
6267 modest_ui_actions_on_search_messages (GtkAction *action, ModestWindow *window)
6269 g_return_if_fail (MODEST_IS_WINDOW (window));
6271 /* we check for low-mem; in that case, show a warning, and don't allow
6274 if (modest_platform_check_memory_low (window, TRUE))
6277 modest_platform_show_search_messages (GTK_WINDOW (window));
6281 modest_ui_actions_on_open_addressbook (GtkAction *action, ModestWindow *win)
6283 g_return_if_fail (MODEST_IS_WINDOW (win));
6286 /* we check for low-mem; in that case, show a warning, and don't allow
6287 * for the addressbook
6289 if (modest_platform_check_memory_low (win, TRUE))
6293 modest_platform_show_addressbook (GTK_WINDOW (win));
6298 modest_ui_actions_on_toggle_find_in_page (GtkAction *action,
6299 ModestWindow *window)
6302 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
6304 if (GTK_IS_TOGGLE_ACTION (action))
6305 active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action));
6309 modest_msg_edit_window_toggle_find_toolbar (MODEST_MSG_EDIT_WINDOW (window),
6314 on_send_receive_finished (ModestMailOperation *mail_op,
6317 GtkWidget *header_view, *folder_view;
6318 TnyFolderStore *folder_store;
6319 ModestMainWindow *main_win = MODEST_MAIN_WINDOW (user_data);
6321 /* Set send/receive operation finished */
6322 modest_main_window_notify_send_receive_completed (main_win);
6324 /* Don't refresh the current folder if there were any errors */
6325 if (modest_mail_operation_get_status (mail_op) !=
6326 MODEST_MAIL_OPERATION_STATUS_SUCCESS)
6329 /* Refresh the current folder if we're viewing a window. We do
6330 this because the user won't be able to see the new mails in
6331 the selected folder after a Send&Receive because it only
6332 performs a poke_status, i.e, only the number of read/unread
6333 messages is updated, but the new headers are not
6335 folder_view = modest_main_window_get_child_widget (main_win,
6336 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
6340 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
6342 /* Do not need to refresh INBOX again because the
6343 update_account does it always automatically */
6344 if (folder_store && TNY_IS_FOLDER (folder_store) &&
6345 tny_folder_get_folder_type (TNY_FOLDER (folder_store)) != TNY_FOLDER_TYPE_INBOX) {
6346 ModestMailOperation *refresh_op;
6348 header_view = modest_main_window_get_child_widget (main_win,
6349 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6351 /* We do not need to set the contents style
6352 because it hasn't changed. We also do not
6353 need to save the widget status. Just force
6355 refresh_op = modest_mail_operation_new (G_OBJECT (main_win));
6356 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), refresh_op);
6357 modest_mail_operation_refresh_folder (refresh_op, TNY_FOLDER (folder_store),
6358 folder_refreshed_cb, main_win);
6359 g_object_unref (refresh_op);
6363 g_object_unref (folder_store);
6368 modest_ui_actions_on_send_queue_error_happened (TnySendQueue *self,
6374 const gchar* server_name = NULL;
6375 TnyTransportAccount *transport;
6376 gchar *message = NULL;
6377 ModestProtocol *protocol;
6379 /* Don't show anything if the user cancelled something or the
6380 * send receive request is not interactive. Authentication
6381 * errors are managed by the account store so no need to show
6382 * a dialog here again */
6383 if (err->code == TNY_SYSTEM_ERROR_CANCEL ||
6384 err->code == TNY_SERVICE_ERROR_AUTHENTICATE ||
6385 !modest_tny_send_queue_get_requested_send_receive (MODEST_TNY_SEND_QUEUE (self)))
6389 /* Get the server name. Note that we could be using a
6390 connection specific transport account */
6391 transport = (TnyTransportAccount *)
6392 tny_camel_send_queue_get_transport_account (TNY_CAMEL_SEND_QUEUE (self));
6394 ModestTnyAccountStore *acc_store;
6395 const gchar *acc_name;
6396 TnyTransportAccount *conn_specific;
6398 acc_store = modest_runtime_get_account_store();
6399 acc_name = modest_tny_account_get_parent_modest_account_name_for_server_account (TNY_ACCOUNT (transport));
6400 conn_specific = (TnyTransportAccount *)
6401 modest_tny_account_store_get_transport_account_for_open_connection (acc_store, acc_name);
6402 if (conn_specific) {
6403 server_name = tny_account_get_hostname (TNY_ACCOUNT (conn_specific));
6404 g_object_unref (conn_specific);
6406 server_name = tny_account_get_hostname (TNY_ACCOUNT (transport));
6408 g_object_unref (transport);
6412 protocol = modest_protocol_registry_get_protocol_by_name (modest_runtime_get_protocol_registry (),
6413 MODEST_PROTOCOL_REGISTRY_TRANSPORT_STORE_PROTOCOLS,
6414 tny_account_get_proto (TNY_ACCOUNT (transport)));
6416 g_warning ("%s: Account with no proto", __FUNCTION__);
6420 /* Show the appropriate message text for the GError: */
6421 switch (err->code) {
6422 case TNY_SERVICE_ERROR_CONNECT:
6423 message = modest_protocol_get_translation (protocol,
6424 MODEST_PROTOCOL_TRANSLATION_ACCOUNT_CONNECTION_ERROR,
6427 case TNY_SERVICE_ERROR_SEND:
6428 message = g_strdup (_CS("sfil_ib_unable_to_send"));
6430 case TNY_SERVICE_ERROR_UNAVAILABLE:
6431 message = modest_protocol_get_translation (protocol,
6432 MODEST_PROTOCOL_TRANSLATION_CONNECT_ERROR,
6436 g_warning ("%s: unexpected ERROR %d",
6437 __FUNCTION__, err->code);
6438 message = g_strdup (_CS("sfil_ib_unable_to_send"));
6442 modest_platform_run_information_dialog (NULL, message, FALSE);
6447 modest_ui_actions_on_send_queue_status_changed (ModestTnySendQueue *send_queue,
6452 ModestWindow *top_window = NULL;
6453 ModestWindowMgr *mgr = NULL;
6454 GtkWidget *header_view = NULL;
6455 TnyFolder *selected_folder = NULL;
6456 TnyFolderType folder_type;
6458 mgr = modest_runtime_get_window_mgr ();
6459 top_window = modest_window_mgr_get_current_top (mgr);
6464 #ifndef MODEST_TOOLKIT_HILDON2
6465 if (MODEST_IS_MAIN_WINDOW (top_window)) {
6466 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (top_window),
6467 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6470 if (MODEST_IS_HEADER_WINDOW (top_window)) {
6471 header_view = (GtkWidget *)
6472 modest_header_window_get_header_view (MODEST_HEADER_WINDOW (top_window));
6476 /* Get selected folder */
6478 selected_folder = modest_header_view_get_folder (MODEST_HEADER_VIEW (header_view));
6479 if (!selected_folder)
6482 /* gtk_tree_view_column_queue_resize is only available in GTK+ 2.8 */
6483 #if GTK_CHECK_VERSION(2, 8, 0)
6484 folder_type = modest_tny_folder_guess_folder_type (selected_folder);
6485 if (folder_type == TNY_FOLDER_TYPE_OUTBOX) {
6486 GtkTreeViewColumn *tree_column;
6488 tree_column = gtk_tree_view_get_column (GTK_TREE_VIEW (header_view),
6489 TNY_GTK_HEADER_LIST_MODEL_FROM_COLUMN);
6491 gtk_tree_view_column_queue_resize (tree_column);
6493 #else /* #if GTK_CHECK_VERSION(2, 8, 0) */
6494 gtk_widget_queue_draw (header_view);
6497 #ifndef MODEST_TOOLKIT_HILDON2
6498 /* Rerun dimming rules, because the message could become deletable for example */
6499 modest_window_check_dimming_rules_group (MODEST_WINDOW (top_window),
6500 MODEST_DIMMING_RULES_TOOLBAR);
6501 modest_window_check_dimming_rules_group (MODEST_WINDOW (top_window),
6502 MODEST_DIMMING_RULES_MENU);
6506 g_object_unref (selected_folder);
6510 modest_ui_actions_on_account_connection_error (GtkWindow *parent_window,
6511 TnyAccount *account)
6513 ModestProtocolType protocol_type;
6514 ModestProtocol *protocol;
6515 gchar *error_note = NULL;
6517 protocol_type = modest_tny_account_get_protocol_type (account);
6518 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6521 error_note = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_ACCOUNT_CONNECTION_ERROR, tny_account_get_hostname (account));
6522 if (error_note == NULL) {
6523 g_warning ("%s: This should not be reached", __FUNCTION__);
6525 modest_platform_run_information_dialog (parent_window, error_note, FALSE);
6526 g_free (error_note);
6531 modest_ui_actions_get_msg_already_deleted_error_msg (ModestWindow *win)
6535 TnyFolderStore *folder = NULL;
6536 TnyAccount *account = NULL;
6537 ModestProtocolType proto;
6538 ModestProtocol *protocol;
6539 TnyHeader *header = NULL;
6541 if (MODEST_IS_MAIN_WINDOW (win)) {
6542 GtkWidget *header_view;
6543 TnyList* headers = NULL;
6545 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
6546 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6547 headers = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
6548 if (!headers || tny_list_get_length (headers) == 0) {
6550 g_object_unref (headers);
6553 iter = tny_list_create_iterator (headers);
6554 header = TNY_HEADER (tny_iterator_get_current (iter));
6555 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6556 g_object_unref (iter);
6557 g_object_unref (headers);
6558 #ifdef MODEST_TOOLKIT_HILDON2
6559 } else if (MODEST_IS_HEADER_WINDOW (win)) {
6560 GtkWidget *header_view;
6561 TnyList* headers = NULL;
6563 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
6564 headers = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
6565 if (!headers || tny_list_get_length (headers) == 0) {
6567 g_object_unref (headers);
6570 iter = tny_list_create_iterator (headers);
6571 header = TNY_HEADER (tny_iterator_get_current (iter));
6573 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6575 g_warning ("List should contain headers");
6577 g_object_unref (iter);
6578 g_object_unref (headers);
6580 } else if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
6581 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
6582 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6585 if (!header || !folder)
6588 /* Get the account type */
6589 account = tny_folder_get_account (TNY_FOLDER (folder));
6590 proto = modest_tny_account_get_protocol_type (account);
6591 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6594 subject = tny_header_dup_subject (header);
6595 msg = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
6599 msg = g_strdup_printf (_("mail_ni_ui_folder_get_msg_folder_error"));
6605 g_object_unref (account);
6607 g_object_unref (folder);
6609 g_object_unref (header);
6615 modest_ui_actions_on_delete_account (GtkWindow *parent_window,
6616 const gchar *account_name,
6617 const gchar *account_title)
6619 ModestAccountMgr *account_mgr;
6622 ModestProtocol *protocol;
6623 gboolean removed = FALSE;
6625 g_return_val_if_fail (account_name, FALSE);
6626 g_return_val_if_fail (account_title, FALSE);
6628 account_mgr = modest_runtime_get_account_mgr();
6630 /* The warning text depends on the account type: */
6631 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6632 modest_account_mgr_get_store_protocol (account_mgr,
6634 txt = modest_protocol_get_translation (protocol,
6635 MODEST_PROTOCOL_TRANSLATION_DELETE_MAILBOX,
6638 txt = g_strdup_printf (_("emev_nc_delete_mailbox"), account_title);
6640 response = modest_platform_run_confirmation_dialog (parent_window, txt);
6644 if (response == GTK_RESPONSE_OK) {
6645 /* Remove account. If it succeeds then it also removes
6646 the account from the ModestAccountView: */
6647 gboolean is_default = FALSE;
6648 gchar *default_account_name = modest_account_mgr_get_default_account (account_mgr);
6649 if (default_account_name && (strcmp (default_account_name, account_name) == 0))
6651 g_free (default_account_name);
6653 removed = modest_account_mgr_remove_account (account_mgr, account_name);
6655 g_warning ("%s: modest_account_mgr_remove_account() failed.\n", __FUNCTION__);