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;
112 GtkWidget *parent_window;
114 } ReplyForwardHelper;
116 typedef struct _MoveToHelper {
117 GtkTreeRowReference *reference;
121 typedef struct _PasteAsAttachmentHelper {
122 ModestMsgEditWindow *window;
124 } PasteAsAttachmentHelper;
132 * The do_headers_action uses this kind of functions to perform some
133 * action to each member of a list of headers
135 typedef void (*HeadersFunc) (TnyHeader *header, ModestWindow *win, gpointer user_data);
137 static void do_headers_action (ModestWindow *win,
141 static void open_msg_cb (ModestMailOperation *mail_op,
148 static void reply_forward_cb (ModestMailOperation *mail_op,
155 static void reply_forward (ReplyForwardAction action, ModestWindow *win);
157 static void folder_refreshed_cb (ModestMailOperation *mail_op,
161 static void on_send_receive_finished (ModestMailOperation *mail_op,
164 static gint header_list_count_uncached_msgs (TnyList *header_list);
166 static gboolean connect_to_get_msg (ModestWindow *win,
167 gint num_of_uncached_msgs,
168 TnyAccount *account);
170 static gboolean remote_folder_has_leave_on_server (TnyFolderStore *folder);
172 static void do_create_folder (GtkWindow *window,
173 TnyFolderStore *parent_folder,
174 const gchar *suggested_name);
176 static TnyAccount *get_account_from_folder_store (TnyFolderStore *folder_store);
178 static void modest_ui_actions_on_main_window_move_to (GtkAction *action,
179 GtkWidget *folder_view,
180 TnyFolderStore *dst_folder,
181 ModestMainWindow *win);
182 #ifdef MODEST_TOOLKIT_HILDON2
183 static void modest_ui_actions_on_folder_window_move_to (GtkWidget *folder_view,
184 TnyFolderStore *dst_folder,
189 static void modest_ui_actions_on_window_move_to (GtkAction *action,
190 TnyList *list_to_move,
191 TnyFolderStore *dst_folder,
195 * This function checks whether a TnyFolderStore is a pop account
198 remote_folder_has_leave_on_server (TnyFolderStore *folder)
203 g_return_val_if_fail (TNY_IS_FOLDER_STORE (folder), FALSE);
205 account = get_account_from_folder_store (folder);
206 result = (modest_protocol_registry_protocol_type_has_leave_on_server (modest_runtime_get_protocol_registry (),
207 modest_tny_account_get_protocol_type (account)));
208 g_object_unref (account);
213 /* FIXME: this should be merged with the similar code in modest-account-view-window */
214 /* Show the account creation wizard dialog.
215 * returns: TRUE if an account was created. FALSE if the user cancelled.
218 modest_ui_actions_run_account_setup_wizard (ModestWindow *win)
220 gboolean result = FALSE;
222 gint dialog_response;
224 /* there is no such wizard yet */
225 wizard = GTK_WINDOW (modest_platform_get_account_settings_wizard ());
226 modest_window_mgr_set_modal (modest_runtime_get_window_mgr(), GTK_WINDOW (wizard), (GtkWindow *) win);
228 #ifndef MODEST_TOOLKIT_HILDON2
229 /* always present a main window in the background
230 * we do it here, so we cannot end up with two wizards (as this
231 * function might be called in modest_window_mgr_get_main_window as well */
233 win = modest_window_mgr_get_main_window (modest_runtime_get_window_mgr(),
234 TRUE); /* create if not existent */
238 ModestWindowMgr *mgr;
240 mgr = modest_runtime_get_window_mgr ();
242 window_list = modest_window_mgr_get_window_list (mgr);
243 if (window_list == NULL) {
244 ModestWindow *old_win;
245 win = MODEST_WINDOW (modest_accounts_window_new ());
246 if (modest_window_mgr_register_window (mgr, win, NULL)) {
247 gtk_widget_show_all (GTK_WIDGET (win));
249 gtk_widget_destroy (GTK_WIDGET (win));
254 win = MODEST_WINDOW (modest_folder_window_new (NULL));
255 if (modest_window_mgr_register_window (mgr, win, NULL)) {
256 gtk_widget_show_all (GTK_WIDGET (win));
258 gtk_widget_destroy (GTK_WIDGET (win));
262 g_list_free (window_list);
268 gtk_window_set_transient_for (GTK_WINDOW (wizard), GTK_WINDOW (win));
270 /* make sure the mainwindow is visible. We need to present the
271 wizard again to give it the focus back. show_all are needed
272 in order to get the widgets properly drawn (MainWindow main
273 paned won't be in its right position and the dialog will be
275 #ifndef MODEST_TOOLKIT_HILDON2
276 gtk_widget_show_all (GTK_WIDGET (win));
277 gtk_widget_show_all (GTK_WIDGET (wizard));
278 gtk_window_present (GTK_WINDOW (win));
279 gtk_window_present (GTK_WINDOW (wizard));
282 dialog_response = gtk_dialog_run (GTK_DIALOG (wizard));
283 gtk_widget_destroy (GTK_WIDGET (wizard));
284 if (gtk_events_pending ())
285 gtk_main_iteration ();
287 if (dialog_response == GTK_RESPONSE_CANCEL) {
290 /* Check whether an account was created: */
291 result = modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
298 modest_ui_actions_on_about (GtkAction *action, ModestWindow *win)
301 const gchar *authors[] = {
302 "Dirk-Jan C. Binnema <dirk-jan.binnema@nokia.com>",
305 about = gtk_about_dialog_new ();
306 gtk_about_dialog_set_name (GTK_ABOUT_DIALOG(about), PACKAGE_NAME);
307 gtk_about_dialog_set_version (GTK_ABOUT_DIALOG(about),PACKAGE_VERSION);
308 gtk_about_dialog_set_copyright (GTK_ABOUT_DIALOG(about),
309 _("Copyright (c) 2006, Nokia Corporation\n"
310 "All rights reserved."));
311 gtk_about_dialog_set_comments (GTK_ABOUT_DIALOG(about),
312 _("a modest e-mail client\n\n"
313 "design and implementation: Dirk-Jan C. Binnema\n"
314 "contributions from the fine people at KC and Ig\n"
315 "uses the tinymail email framework written by Philip van Hoof"));
316 gtk_about_dialog_set_authors (GTK_ABOUT_DIALOG(about), authors);
317 gtk_about_dialog_set_website (GTK_ABOUT_DIALOG(about), "http://modest.garage.maemo.org");
318 gtk_window_set_transient_for (GTK_WINDOW (about), GTK_WINDOW (win));
319 gtk_window_set_modal (GTK_WINDOW (about), TRUE);
321 gtk_dialog_run (GTK_DIALOG (about));
322 gtk_widget_destroy(about);
326 * Gets the list of currently selected messages. If the win is the
327 * main window, then it returns a newly allocated list of the headers
328 * selected in the header view. If win is the msg view window, then
329 * the value returned is a list with just a single header.
331 * The caller of this funcion must free the list.
334 get_selected_headers (ModestWindow *win)
336 if (MODEST_IS_MAIN_WINDOW(win)) {
337 GtkWidget *header_view;
339 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
340 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
341 return modest_header_view_get_selected_headers (MODEST_HEADER_VIEW(header_view));
343 } else if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
344 /* for MsgViewWindows, we simply return a list with one element */
346 TnyList *list = NULL;
348 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
349 if (header != NULL) {
350 list = tny_simple_list_new ();
351 tny_list_prepend (list, G_OBJECT(header));
352 g_object_unref (G_OBJECT(header));
357 #ifdef MODEST_TOOLKIT_HILDON2
358 } else if (MODEST_IS_HEADER_WINDOW (win)) {
359 GtkWidget *header_view;
361 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
362 return modest_header_view_get_selected_headers (MODEST_HEADER_VIEW(header_view));
368 static GtkTreeRowReference *
369 get_next_after_selected_headers (ModestHeaderView *header_view)
371 GtkTreeSelection *sel;
372 GList *selected_rows, *node;
374 GtkTreeRowReference *result;
377 model = gtk_tree_view_get_model (GTK_TREE_VIEW (header_view));
378 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (header_view));
379 selected_rows = gtk_tree_selection_get_selected_rows (sel, NULL);
381 if (selected_rows == NULL)
384 node = g_list_last (selected_rows);
385 path = gtk_tree_path_copy ((GtkTreePath *) node->data);
386 gtk_tree_path_next (path);
388 result = gtk_tree_row_reference_new (model, path);
390 gtk_tree_path_free (path);
391 g_list_foreach (selected_rows, (GFunc) gtk_tree_path_free, NULL);
392 g_list_free (selected_rows);
398 headers_action_mark_as_read (TnyHeader *header,
402 TnyHeaderFlags flags;
404 g_return_if_fail (TNY_IS_HEADER(header));
406 flags = tny_header_get_flags (header);
407 if (flags & TNY_HEADER_FLAG_SEEN) return;
408 tny_header_set_flag (header, TNY_HEADER_FLAG_SEEN);
412 headers_action_mark_as_unread (TnyHeader *header,
416 TnyHeaderFlags flags;
418 g_return_if_fail (TNY_IS_HEADER(header));
420 flags = tny_header_get_flags (header);
421 if (flags & TNY_HEADER_FLAG_SEEN) {
422 tny_header_unset_flag (header, TNY_HEADER_FLAG_SEEN);
426 /** After deleing a message that is currently visible in a window,
427 * show the next message from the list, or close the window if there are no more messages.
430 modest_ui_actions_refresh_message_window_after_delete (ModestMsgViewWindow* win)
432 /* Close msg view window or select next */
433 if (!modest_msg_view_window_select_next_message (win) &&
434 !modest_msg_view_window_select_previous_message (win)) {
436 g_signal_emit_by_name (G_OBJECT (win), "delete-event", NULL, &ret_value);
442 modest_ui_actions_on_delete_message (GtkAction *action, ModestWindow *win)
444 modest_ui_actions_on_edit_mode_delete_message (win);
448 modest_ui_actions_on_edit_mode_delete_message (ModestWindow *win)
450 TnyList *header_list = NULL;
451 TnyIterator *iter = NULL;
452 TnyHeader *header = NULL;
453 gchar *message = NULL;
456 ModestWindowMgr *mgr;
457 GtkWidget *header_view = NULL;
458 gboolean retval = TRUE;
460 g_return_val_if_fail (MODEST_IS_WINDOW(win), FALSE);
462 /* Check first if the header view has the focus */
463 if (MODEST_IS_MAIN_WINDOW (win)) {
465 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
466 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
467 if (!gtk_widget_is_focus (header_view))
471 /* Get the headers, either from the header view (if win is the main window),
472 * or from the message view window: */
473 header_list = get_selected_headers (win);
474 if (!header_list) return FALSE;
476 /* Check if any of the headers are already opened, or in the process of being opened */
477 if (MODEST_IS_MAIN_WINDOW (win)) {
478 gint opened_headers = 0;
480 iter = tny_list_create_iterator (header_list);
481 mgr = modest_runtime_get_window_mgr ();
482 while (!tny_iterator_is_done (iter)) {
483 header = TNY_HEADER (tny_iterator_get_current (iter));
485 if (modest_window_mgr_find_registered_header (mgr, header, NULL))
487 g_object_unref (header);
489 tny_iterator_next (iter);
491 g_object_unref (iter);
493 if (opened_headers > 0) {
496 msg = g_strdup_printf (_("mcen_nc_unable_to_delete_n_messages"),
499 modest_platform_run_information_dialog (GTK_WINDOW (win), (const gchar *) msg, FALSE);
502 g_object_unref (header_list);
508 if (tny_list_get_length(header_list) == 1) {
509 iter = tny_list_create_iterator (header_list);
510 header = TNY_HEADER (tny_iterator_get_current (iter));
513 subject = tny_header_dup_subject (header);
515 subject = g_strdup (_("mail_va_no_subject"));
516 desc = g_strdup_printf ("%s", subject);
518 g_object_unref (header);
521 g_object_unref (iter);
523 message = g_strdup_printf(ngettext("emev_nc_delete_message", "emev_nc_delete_messages",
524 tny_list_get_length(header_list)), desc);
526 /* Confirmation dialog */
527 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
531 if (response == GTK_RESPONSE_OK) {
532 ModestWindow *main_window = NULL;
533 ModestWindowMgr *mgr = NULL;
534 GtkTreeModel *model = NULL;
535 GtkTreeSelection *sel = NULL;
536 GList *sel_list = NULL, *tmp = NULL;
537 GtkTreeRowReference *next_row_reference = NULL;
538 GtkTreeRowReference *prev_row_reference = NULL;
539 GtkTreePath *next_path = NULL;
540 GtkTreePath *prev_path = NULL;
541 ModestMailOperation *mail_op = NULL;
543 /* Find last selected row */
544 if (MODEST_IS_MAIN_WINDOW (win)) {
545 model = gtk_tree_view_get_model (GTK_TREE_VIEW (header_view));
546 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (header_view));
547 sel_list = gtk_tree_selection_get_selected_rows (sel, &model);
548 for (tmp=sel_list; tmp; tmp=tmp->next) {
549 if (tmp->next == NULL) {
550 prev_path = gtk_tree_path_copy((GtkTreePath *) tmp->data);
551 next_path = gtk_tree_path_copy((GtkTreePath *) tmp->data);
553 gtk_tree_path_prev (prev_path);
554 gtk_tree_path_next (next_path);
556 prev_row_reference = gtk_tree_row_reference_new (model, prev_path);
557 next_row_reference = gtk_tree_row_reference_new (model, next_path);
562 /* Disable window dimming management */
563 modest_window_disable_dimming (MODEST_WINDOW(win));
565 /* Remove each header. If it's a view window header_view == NULL */
566 mail_op = modest_mail_operation_new ((GObject *) win);
567 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
569 modest_mail_operation_remove_msgs (mail_op, header_list, FALSE);
570 g_object_unref (mail_op);
572 /* Enable window dimming management */
574 gtk_tree_selection_unselect_all (sel);
576 modest_window_enable_dimming (MODEST_WINDOW(win));
578 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
579 modest_ui_actions_refresh_message_window_after_delete (MODEST_MSG_VIEW_WINDOW (win));
581 /* Get main window */
582 mgr = modest_runtime_get_window_mgr ();
583 main_window = modest_window_mgr_get_main_window (mgr, FALSE); /* don't create */
584 } else if (MODEST_IS_MAIN_WINDOW (win)) {
585 /* Move cursor to next row */
588 /* Select next or previous row */
589 if (gtk_tree_row_reference_valid (next_row_reference)) {
590 gtk_tree_selection_select_path (sel, next_path);
592 else if (gtk_tree_row_reference_valid (prev_row_reference)) {
593 gtk_tree_selection_select_path (sel, prev_path);
597 if (gtk_tree_row_reference_valid (next_row_reference))
598 gtk_tree_row_reference_free (next_row_reference);
599 if (next_path != NULL)
600 gtk_tree_path_free (next_path);
601 if (gtk_tree_row_reference_valid (prev_row_reference))
602 gtk_tree_row_reference_free (prev_row_reference);
603 if (prev_path != NULL)
604 gtk_tree_path_free (prev_path);
607 /* Update toolbar dimming state */
609 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
610 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
614 g_list_foreach (sel_list, (GFunc) gtk_tree_path_free, NULL);
615 g_list_free (sel_list);
624 g_object_unref (header_list);
632 /* delete either message or folder, based on where we are */
634 modest_ui_actions_on_delete_message_or_folder (GtkAction *action, ModestWindow *win)
636 g_return_if_fail (MODEST_IS_WINDOW(win));
638 /* Check first if the header view has the focus */
639 if (MODEST_IS_MAIN_WINDOW (win)) {
641 w = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
642 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
643 if (gtk_widget_is_focus (w)) {
644 modest_ui_actions_on_delete_folder (action, MODEST_WINDOW(win));
648 modest_ui_actions_on_delete_message (action, win);
652 modest_ui_actions_on_quit (GtkAction *action, ModestWindow *win)
654 ModestWindowMgr *mgr = NULL;
656 #ifdef MODEST_PLATFORM_MAEMO
657 modest_osso_save_state();
658 #endif /* MODEST_PLATFORM_MAEMO */
660 g_debug ("closing down, clearing %d item(s) from operation queue",
661 modest_mail_operation_queue_num_elements
662 (modest_runtime_get_mail_operation_queue()));
664 /* cancel all outstanding operations */
665 modest_mail_operation_queue_cancel_all
666 (modest_runtime_get_mail_operation_queue());
668 g_debug ("queue has been cleared");
671 /* Check if there are opened editing windows */
672 mgr = modest_runtime_get_window_mgr ();
673 modest_window_mgr_close_all_windows (mgr);
675 /* note: when modest-tny-account-store is finalized,
676 it will automatically set all network connections
679 /* gtk_main_quit (); */
683 modest_ui_actions_on_close_window (GtkAction *action, ModestWindow *win)
687 g_signal_emit_by_name (G_OBJECT (win), "delete-event", NULL, &ret_value);
689 /* if (MODEST_IS_MSG_VIEW_WINDOW (win)) { */
690 /* gtk_widget_destroy (GTK_WIDGET (win)); */
691 /* } else if (MODEST_IS_MSG_EDIT_WINDOW (win)) { */
692 /* gboolean ret_value; */
693 /* g_signal_emit_by_name (G_OBJECT (win), "delete-event", NULL, &ret_value); */
694 /* } else if (MODEST_IS_WINDOW (win)) { */
695 /* gtk_widget_destroy (GTK_WIDGET (win)); */
697 /* g_return_if_reached (); */
702 modest_ui_actions_add_to_contacts (GtkAction *action, ModestWindow *win)
704 g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (win));
706 modest_msg_view_window_add_to_contacts (MODEST_MSG_VIEW_WINDOW (win));
710 modest_ui_actions_on_add_to_contacts (GtkAction *action, ModestWindow *win)
712 GtkClipboard *clipboard = NULL;
713 gchar *selection = NULL;
715 clipboard = gtk_clipboard_get (GDK_SELECTION_PRIMARY);
716 selection = gtk_clipboard_wait_for_text (clipboard);
718 /* Question: why is the clipboard being used here?
719 * It doesn't really make a lot of sense. */
723 modest_address_book_add_address (selection);
729 modest_ui_actions_on_new_account (GtkAction *action,
730 ModestWindow *window)
732 if (!modest_ui_actions_run_account_setup_wizard (window)) {
733 g_debug ("%s: wizard was already running", __FUNCTION__);
738 modest_ui_actions_on_accounts (GtkAction *action,
741 /* This is currently only implemented for Maemo */
742 if (!modest_account_mgr_has_accounts (modest_runtime_get_account_mgr(), TRUE)) {
743 if (!modest_ui_actions_run_account_setup_wizard (win))
744 g_debug ("%s: wizard was already running", __FUNCTION__);
748 /* Show the list of accounts */
749 GtkWindow *account_win = GTK_WINDOW (modest_account_view_window_new ());
751 /* The accounts dialog must be modal */
752 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (), GTK_WINDOW (account_win), (GtkWindow *) win);
753 modest_utils_show_dialog_and_forget (GTK_WINDOW (win), GTK_DIALOG (account_win));
758 modest_ui_actions_on_smtp_servers (GtkAction *action, ModestWindow *win)
760 /* This is currently only implemented for Maemo,
761 * because it requires an API (libconic) to detect different connection
764 #ifndef MODEST_TOOLKIT_GTK /* Defined in config.h */
766 /* Create the window if necessary: */
767 GtkWidget *specific_window = GTK_WIDGET (modest_connection_specific_smtp_window_new ());
768 modest_connection_specific_smtp_window_fill_with_connections (
769 MODEST_CONNECTION_SPECIFIC_SMTP_WINDOW (specific_window),
770 modest_runtime_get_account_mgr());
772 /* Show the window: */
773 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (),
774 GTK_WINDOW (specific_window), (GtkWindow *) win);
775 gtk_widget_show (specific_window);
776 #endif /* !MODEST_TOOLKIT_GTK */
780 modest_ui_actions_compose_msg(ModestWindow *win,
783 const gchar *bcc_str,
784 const gchar *subject_str,
785 const gchar *body_str,
787 gboolean set_as_modified)
789 gchar *account_name = NULL;
790 const gchar *mailbox;
792 TnyAccount *account = NULL;
793 TnyFolder *folder = NULL;
794 gchar *from_str = NULL, *signature = NULL, *body = NULL;
795 gboolean use_signature = FALSE;
796 ModestWindow *msg_win = NULL;
797 ModestAccountMgr *mgr = modest_runtime_get_account_mgr();
798 ModestTnyAccountStore *store = modest_runtime_get_account_store();
799 GnomeVFSFileSize total_size, allowed_size;
801 /* we check for low-mem */
802 if (modest_platform_check_memory_low (win, TRUE))
805 #ifdef MODEST_TOOLKIT_HILDON2
806 account_name = g_strdup (modest_window_get_active_account(win));
809 account_name = modest_account_mgr_get_default_account(mgr);
812 g_printerr ("modest: no account found\n");
816 mailbox = modest_window_get_active_mailbox (win);
817 account = modest_tny_account_store_get_server_account (store, account_name, TNY_ACCOUNT_TYPE_STORE);
819 g_printerr ("modest: failed to get tnyaccount for '%s'\n", account_name);
822 folder = modest_tny_account_get_special_folder (account, TNY_FOLDER_TYPE_DRAFTS);
824 g_printerr ("modest: failed to find Drafts folder\n");
827 from_str = modest_account_mgr_get_from_string (mgr, account_name, mailbox);
829 g_printerr ("modest: failed get from string for '%s'\n", account_name);
833 signature = modest_account_mgr_get_signature (mgr, account_name, &use_signature);
834 if (body_str != NULL) {
835 body = use_signature ? g_strconcat(body_str, "\n--\n", signature, NULL) : g_strdup(body_str);
837 body = use_signature ? g_strconcat("\n--\n", signature, NULL) : g_strdup("");
840 msg = modest_tny_msg_new (to_str, from_str, cc_str, bcc_str, subject_str, NULL, NULL, body, NULL, NULL, NULL);
842 g_printerr ("modest: failed to create new msg\n");
846 /* Create and register edit window */
847 /* This is destroyed by TODO. */
849 allowed_size = MODEST_MAX_ATTACHMENT_SIZE;
850 msg_win = modest_msg_edit_window_new (msg, account_name, mailbox, FALSE);
852 if (!modest_window_mgr_register_window (modest_runtime_get_window_mgr(), msg_win, win)) {
853 gtk_widget_destroy (GTK_WIDGET (msg_win));
856 modest_msg_edit_window_set_modified (MODEST_MSG_EDIT_WINDOW (msg_win), set_as_modified);
857 gtk_widget_show_all (GTK_WIDGET (msg_win));
859 while (attachments) {
861 modest_msg_edit_window_attach_file_one((ModestMsgEditWindow *)msg_win,
862 attachments->data, allowed_size);
864 if (total_size > allowed_size) {
865 g_warning ("%s: total size: %u",
866 __FUNCTION__, (unsigned int)total_size);
869 allowed_size -= total_size;
871 attachments = g_slist_next(attachments);
878 g_free (account_name);
880 g_object_unref (G_OBJECT(account));
882 g_object_unref (G_OBJECT(folder));
884 g_object_unref (G_OBJECT(msg));
888 modest_ui_actions_on_new_msg (GtkAction *action, ModestWindow *win)
890 /* if there are no accounts yet, just show the wizard */
891 if (!modest_account_mgr_has_accounts (modest_runtime_get_account_mgr(), TRUE))
892 if (!modest_ui_actions_run_account_setup_wizard (win))
895 modest_ui_actions_compose_msg(win, NULL, NULL, NULL, NULL, NULL, NULL, FALSE);
900 modest_ui_actions_msg_retrieval_check (ModestMailOperation *mail_op,
904 ModestMailOperationStatus status;
906 /* If there is no message or the operation was not successful */
907 status = modest_mail_operation_get_status (mail_op);
908 if (!msg || status != MODEST_MAIL_OPERATION_STATUS_SUCCESS) {
911 /* If it's a memory low issue, then show a banner */
912 error = modest_mail_operation_get_error (mail_op);
913 if (error && error->domain == MODEST_MAIL_OPERATION_ERROR &&
914 error->code == MODEST_MAIL_OPERATION_ERROR_LOW_MEMORY) {
915 GObject *source = modest_mail_operation_get_source (mail_op);
916 modest_platform_run_information_dialog (GTK_IS_WINDOW (source) ? GTK_WINDOW (source) : NULL,
917 _KR("memr_ib_operation_disabled"),
919 g_object_unref (source);
922 if (error && ((error->code == TNY_SERVICE_ERROR_NO_SUCH_MESSAGE) ||
923 error->code == TNY_SERVICE_ERROR_MESSAGE_NOT_AVAILABLE)) {
924 gchar *subject, *msg, *format = NULL;
926 subject = tny_header_dup_subject (header);
928 subject = g_strdup (_("mail_va_no_subject"));
930 account = modest_mail_operation_get_account (mail_op);
932 ModestProtocol *protocol;
933 ModestProtocolType proto;
934 proto = modest_tny_account_get_protocol_type (account);
935 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (), proto);
937 format = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
938 g_object_unref (account);
942 format = g_strdup (_("emev_ni_ui_imap_message_not_available_in_server"));
944 msg = g_strdup_printf (format, subject);
945 modest_platform_run_information_dialog (NULL, msg, FALSE);
951 /* Remove the header from the preregistered uids */
952 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
970 OpenMsgBannerInfo *banner_info;
971 GtkTreeRowReference *rowref;
975 open_msg_banner_idle (gpointer userdata)
977 OpenMsgBannerInfo *banner_info = (OpenMsgBannerInfo *) userdata;
979 gdk_threads_enter ();
980 banner_info->idle_handler = 0;
981 banner_info->banner = modest_platform_animation_banner (NULL, NULL, banner_info->message);
982 if (banner_info->banner)
983 g_object_ref (banner_info->banner);
985 gdk_threads_leave ();
991 get_header_view_from_window (ModestWindow *window)
993 GtkWidget *header_view;
995 if (MODEST_IS_MAIN_WINDOW (window)) {
996 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
997 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
998 #ifdef MODEST_TOOLKIT_HILDON2
999 } else if (MODEST_IS_HEADER_WINDOW (window)){
1000 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (window)));
1010 get_info_from_header (TnyHeader *header, gboolean *is_draft, gboolean *can_open)
1012 /* TODO: should also retrieve the mailbox from header */
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);
1144 if (helper->rowref && helper->model) {
1145 /* TODO: use mailbox */
1146 win = modest_msg_view_window_new_with_header_model (msg, account, NULL, (const gchar*) uid,
1147 helper->model, helper->rowref);
1149 /* TODO: use mailbox */
1150 win = modest_msg_view_window_new_for_attachment (msg, account, NULL, (const gchar*) uid);
1155 /* Register and show new window */
1157 mgr = modest_runtime_get_window_mgr ();
1158 if (!modest_window_mgr_register_window (mgr, win, NULL)) {
1159 gtk_widget_destroy (GTK_WIDGET (win));
1162 gtk_widget_show_all (GTK_WIDGET(win));
1165 /* Update toolbar dimming state */
1166 if (MODEST_IS_MAIN_WINDOW (parent_win)) {
1167 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (parent_win));
1173 g_object_unref (parent_win);
1177 is_memory_full_error (GError *error, ModestMailOperation *mail_op)
1179 gboolean enough_free_space = TRUE;
1180 GnomeVFSURI *cache_dir_uri;
1181 const gchar *cache_dir = NULL;
1182 GnomeVFSFileSize free_space;
1183 TnyAccountStore *acc_store;
1185 acc_store = TNY_ACCOUNT_STORE (modest_runtime_get_account_store ());
1187 /* Cache dir is different in case we're using an external storage (like MMC account) */
1189 TnyAccount *account = modest_mail_operation_get_account (mail_op);
1191 if (modest_tny_account_is_memory_card_account (account)) {
1192 cache_dir = g_getenv (MODEST_MMC1_VOLUMEPATH_ENV);
1194 g_object_unref (account);
1198 /* Get the default local cache dir */
1200 cache_dir = tny_account_store_get_cache_dir (acc_store);
1202 cache_dir_uri = gnome_vfs_uri_new (cache_dir);
1203 if (cache_dir_uri) {
1204 if (gnome_vfs_get_volume_free_space (cache_dir_uri, &free_space) == GNOME_VFS_OK) {
1205 if (free_space < MIN_FREE_SPACE)
1206 enough_free_space = FALSE;
1208 gnome_vfs_uri_unref (cache_dir_uri);
1211 if ((error->code == TNY_SYSTEM_ERROR_MEMORY ||
1212 /* When asking for a mail and no space left on device
1213 tinymail returns this error */
1214 error->code == TNY_SERVICE_ERROR_MESSAGE_NOT_AVAILABLE ||
1215 /* When the folder summary could not be read or
1217 error->code == TNY_IO_ERROR_WRITE ||
1218 error->code == TNY_IO_ERROR_READ) &&
1219 !enough_free_space) {
1227 check_memory_full_error (GtkWidget *parent_window, GError *err)
1232 if (is_memory_full_error (err, NULL))
1233 modest_platform_information_banner (parent_window,
1234 NULL, _KR("cerm_device_memory_full"));
1235 else if (err->code == TNY_SYSTEM_ERROR_MEMORY)
1236 /* If the account was created in memory full
1237 conditions then tinymail won't be able to
1238 connect so it'll return this error code */
1239 modest_platform_information_banner (parent_window,
1240 NULL, _("emev_ui_imap_inbox_select_error"));
1248 modest_ui_actions_disk_operations_error_handler (ModestMailOperation *mail_op,
1251 const GError *error;
1252 GObject *win = NULL;
1253 ModestMailOperationStatus status;
1255 win = modest_mail_operation_get_source (mail_op);
1256 error = modest_mail_operation_get_error (mail_op);
1257 status = modest_mail_operation_get_status (mail_op);
1259 /* If the mail op has been cancelled then it's not an error:
1260 don't show any message */
1261 if (status != MODEST_MAIL_OPERATION_STATUS_CANCELED) {
1262 if (is_memory_full_error ((GError *) error, mail_op)) {
1263 modest_platform_information_banner ((GtkWidget *) win,
1264 NULL, _KR("cerm_device_memory_full"));
1265 } else if (error->code == TNY_SYSTEM_ERROR_MEMORY) {
1266 modest_platform_information_banner ((GtkWidget *) win,
1267 NULL, _("emev_ui_imap_inbox_select_error"));
1268 } else if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
1269 error->code == MODEST_MAIL_OPERATION_ERROR_FILE_IO) {
1270 modest_platform_information_banner ((GtkWidget *) win,
1271 NULL, _CS ("sfil_ni_unable_to_open_file_not_found"));
1272 } else if (user_data) {
1273 modest_platform_information_banner ((GtkWidget *) win,
1279 g_object_unref (win);
1283 * Returns the account a list of headers belongs to. It returns a
1284 * *new* reference so don't forget to unref it
1287 get_account_from_header_list (TnyList *headers)
1289 TnyAccount *account = NULL;
1291 if (tny_list_get_length (headers) > 0) {
1292 TnyIterator *iter = tny_list_create_iterator (headers);
1293 TnyHeader *header = TNY_HEADER (tny_iterator_get_current (iter));
1294 TnyFolder *folder = tny_header_get_folder (header);
1297 g_object_unref (header);
1299 while (!tny_iterator_is_done (iter)) {
1300 header = TNY_HEADER (tny_iterator_get_current (iter));
1301 folder = tny_header_get_folder (header);
1304 g_object_unref (header);
1306 tny_iterator_next (iter);
1311 account = tny_folder_get_account (folder);
1312 g_object_unref (folder);
1316 g_object_unref (header);
1318 g_object_unref (iter);
1324 get_account_from_header (TnyHeader *header)
1326 TnyAccount *account = NULL;
1329 folder = tny_header_get_folder (header);
1332 account = tny_folder_get_account (folder);
1333 g_object_unref (folder);
1339 open_msg_helper_destroyer (gpointer user_data)
1341 OpenMsgHelper *helper = (OpenMsgHelper *) user_data;
1343 if (helper->banner_info) {
1344 g_free (helper->banner_info->message);
1345 if (helper->banner_info->idle_handler > 0) {
1346 g_source_remove (helper->banner_info->idle_handler);
1347 helper->banner_info->idle_handler = 0;
1349 if (helper->banner_info->banner != NULL) {
1350 gtk_widget_destroy (helper->banner_info->banner);
1351 g_object_unref (helper->banner_info->banner);
1352 helper->banner_info->banner = NULL;
1354 g_slice_free (OpenMsgBannerInfo, helper->banner_info);
1355 helper->banner_info = NULL;
1357 g_object_unref (helper->model);
1358 g_object_unref (helper->header);
1359 gtk_tree_row_reference_free (helper->rowref);
1360 g_slice_free (OpenMsgHelper, helper);
1364 open_msg_performer(gboolean canceled,
1366 GtkWindow *parent_window,
1367 TnyAccount *account,
1370 ModestMailOperation *mail_op = NULL;
1371 gchar *error_msg = NULL;
1372 ModestProtocolType proto;
1373 TnyConnectionStatus status;
1374 OpenMsgHelper *helper = NULL;
1375 ModestProtocol *protocol;
1376 ModestProtocolRegistry *protocol_registry;
1379 helper = (OpenMsgHelper *) user_data;
1381 status = tny_account_get_connection_status (account);
1382 if (err || canceled) {
1383 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (), helper->header);
1384 /* Free the helper */
1385 open_msg_helper_destroyer (helper);
1387 /* In memory full conditions we could get this error here */
1388 check_memory_full_error ((GtkWidget *) parent_window, err);
1393 /* Get the error message depending on the protocol */
1394 proto = modest_tny_account_get_protocol_type (account);
1395 if (proto == MODEST_PROTOCOL_REGISTRY_TYPE_INVALID) {
1396 proto = MODEST_PROTOCOLS_STORE_MAILDIR;
1399 protocol_registry = modest_runtime_get_protocol_registry ();
1400 subject = tny_header_dup_subject (helper->header);
1402 protocol = modest_protocol_registry_get_protocol_by_type (protocol_registry, proto);
1403 error_msg = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
1407 if (error_msg == NULL) {
1408 error_msg = g_strdup (_("mail_ni_ui_folder_get_msg_folder_error"));
1411 #ifndef MODEST_TOOLKIT_HILDON2
1412 gboolean show_open_draft = FALSE;
1413 if (modest_protocol_registry_protocol_type_has_tag (protocol_registry,
1415 MODEST_PROTOCOL_REGISTRY_LOCAL_STORE_PROTOCOLS)) {
1417 TnyFolderType folder_type;
1419 folder = tny_header_get_folder (helper->header);
1420 folder_type = modest_tny_folder_get_local_or_mmc_folder_type (folder);
1421 show_open_draft = (folder_type == TNY_FOLDER_TYPE_DRAFTS);
1422 g_object_unref (folder);
1426 #ifdef MODEST_TOOLKIT_HILDON2
1429 gchar *account_name = get_info_from_header (helper->header, &is_draft, &can_open);
1432 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (), helper->header);
1433 g_free (account_name);
1434 open_msg_helper_destroyer (helper);
1439 ModestWindow *window;
1440 GtkWidget *header_view;
1443 header_view = get_header_view_from_window (MODEST_WINDOW (parent_window));
1444 uid = modest_tny_folder_get_header_unique_id (helper->header);
1446 /* TODO: use mailbox */
1447 window = modest_msg_view_window_new_from_header_view
1448 (MODEST_HEADER_VIEW (header_view), account_name, NULL, uid, helper->rowref);
1449 if (window != NULL) {
1450 if (!modest_window_mgr_register_window (modest_runtime_get_window_mgr (),
1452 gtk_widget_destroy (GTK_WIDGET (window));
1454 gtk_widget_show_all (GTK_WIDGET(window));
1458 g_free (account_name);
1460 open_msg_helper_destroyer (helper);
1463 g_free (account_name);
1465 /* Create the mail operation */
1467 modest_mail_operation_new_with_error_handling ((GObject *) parent_window,
1468 modest_ui_actions_disk_operations_error_handler,
1469 g_strdup (error_msg), g_free);
1470 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
1474 #ifndef MODEST_TOOLKIT_HILDON2
1475 if (show_open_draft) {
1476 helper->banner_info = g_slice_new (OpenMsgBannerInfo);
1477 helper->banner_info->message = g_strdup (_("mail_ib_opening_draft_message"));
1478 helper->banner_info->banner = NULL;
1479 helper->banner_info->idle_handler = g_timeout_add (500, open_msg_banner_idle,
1480 helper->banner_info);
1486 headers = TNY_LIST (tny_simple_list_new ());
1487 tny_list_prepend (headers, G_OBJECT (helper->header));
1488 modest_mail_operation_get_msgs_full (mail_op,
1492 open_msg_helper_destroyer);
1493 g_object_unref (headers);
1500 g_object_unref (mail_op);
1501 g_object_unref (account);
1505 * This function is used by both modest_ui_actions_on_open and
1506 * modest_ui_actions_on_header_activated. This way we always do the
1507 * same when trying to open messages.
1510 open_msg_from_header (TnyHeader *header, GtkTreeRowReference *rowref, ModestWindow *win)
1512 ModestWindowMgr *mgr = NULL;
1513 TnyAccount *account;
1514 gboolean cached = FALSE;
1516 GtkWidget *header_view = NULL;
1517 OpenMsgHelper *helper;
1518 ModestWindow *window;
1520 g_return_if_fail (header != NULL && rowref != NULL);
1522 mgr = modest_runtime_get_window_mgr ();
1525 header_view = get_header_view_from_window (MODEST_WINDOW (win));
1526 if (header_view == NULL)
1529 /* Get the account */
1530 account = get_account_from_header (header);
1535 found = modest_window_mgr_find_registered_header (mgr, header, &window);
1537 /* Do not open again the message and present the
1538 window to the user */
1541 #ifndef MODEST_TOOLKIT_HILDON2
1542 gtk_window_present (GTK_WINDOW (window));
1545 /* the header has been registered already, we don't do
1546 * anything but wait for the window to come up*/
1547 g_debug ("header %p already registered, waiting for window", header);
1552 /* Open each message */
1553 cached = tny_header_get_flags (header) & TNY_HEADER_FLAG_CACHED;
1555 /* Allways download if we are online. */
1556 if (!tny_device_is_online (modest_runtime_get_device ())) {
1559 /* If ask for user permission to download the messages */
1560 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1561 _("mcen_nc_get_msg"));
1563 /* End if the user does not want to continue */
1564 if (response == GTK_RESPONSE_CANCEL) {
1570 /* We register the window for opening */
1571 modest_window_mgr_register_header (mgr, header, NULL);
1573 /* Create the helper. We need to get a reference to the model
1574 here because it could change while the message is readed
1575 (the user could switch between folders) */
1576 helper = g_slice_new (OpenMsgHelper);
1577 helper->model = g_object_ref (gtk_tree_view_get_model (GTK_TREE_VIEW (header_view)));
1578 helper->header = g_object_ref (header);
1579 helper->rowref = gtk_tree_row_reference_copy (rowref);
1580 helper->banner_info = NULL;
1582 /* Connect to the account and perform */
1584 modest_platform_connect_and_perform ((GtkWindow *) win, TRUE, g_object_ref (account),
1585 open_msg_performer, helper);
1587 /* Call directly the performer, do not need to connect */
1588 open_msg_performer (FALSE, NULL, (GtkWindow *) win,
1589 g_object_ref (account), helper);
1594 g_object_unref (account);
1598 modest_ui_actions_on_open (GtkAction *action, ModestWindow *win)
1605 /* we check for low-mem; in that case, show a warning, and don't allow
1608 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
1612 headers = get_selected_headers (win);
1616 headers_count = tny_list_get_length (headers);
1617 if (headers_count != 1) {
1618 if (headers_count > 1) {
1619 /* Don't allow activation if there are more than one message selected */
1620 modest_platform_information_banner (NULL, NULL, _("mcen_ib_select_one_message"));
1623 g_object_unref (headers);
1627 iter = tny_list_create_iterator (headers);
1628 header = TNY_HEADER (tny_iterator_get_current (iter));
1629 g_object_unref (iter);
1633 open_msg_from_header (header, NULL, win);
1634 g_object_unref (header);
1637 g_object_unref(headers);
1641 rf_helper_window_closed (gpointer data,
1644 ReplyForwardHelper *helper = (ReplyForwardHelper *) data;
1646 helper->parent_window = NULL;
1649 static ReplyForwardHelper*
1650 create_reply_forward_helper (ReplyForwardAction action,
1652 guint reply_forward_type,
1655 ReplyForwardHelper *rf_helper = NULL;
1656 const gchar *active_acc = modest_window_get_active_account (win);
1658 rf_helper = g_slice_new0 (ReplyForwardHelper);
1659 rf_helper->reply_forward_type = reply_forward_type;
1660 rf_helper->action = action;
1661 rf_helper->parent_window = (MODEST_IS_WINDOW (win)) ? GTK_WIDGET (win) : NULL;
1662 rf_helper->header = (header) ? g_object_ref (header) : NULL;
1663 rf_helper->account_name = (active_acc) ?
1664 g_strdup (active_acc) :
1665 modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
1667 /* Note that window could be destroyed just AFTER calling
1668 register_window so we must ensure that this pointer does
1669 not hold invalid references */
1670 if (rf_helper->parent_window)
1671 g_object_weak_ref (G_OBJECT (rf_helper->parent_window),
1672 rf_helper_window_closed, rf_helper);
1678 free_reply_forward_helper (gpointer data)
1680 ReplyForwardHelper *helper;
1682 helper = (ReplyForwardHelper *) data;
1683 g_free (helper->account_name);
1685 g_object_unref (helper->header);
1686 if (helper->parent_window)
1687 g_object_weak_unref (G_OBJECT (helper->parent_window),
1688 rf_helper_window_closed, helper);
1689 g_slice_free (ReplyForwardHelper, helper);
1693 reply_forward_cb (ModestMailOperation *mail_op,
1700 TnyMsg *new_msg = NULL;
1701 ReplyForwardHelper *rf_helper;
1702 ModestWindow *msg_win = NULL;
1703 ModestEditType edit_type;
1705 TnyAccount *account = NULL;
1706 ModestWindowMgr *mgr = NULL;
1707 gchar *signature = NULL;
1708 gboolean use_signature;
1710 /* If there was any error. The mail operation could be NULL,
1711 this means that we already have the message downloaded and
1712 that we didn't do a mail operation to retrieve it */
1713 rf_helper = (ReplyForwardHelper *) user_data;
1714 if (mail_op && !modest_ui_actions_msg_retrieval_check (mail_op, header, msg))
1717 from = modest_account_mgr_get_from_string (modest_runtime_get_account_mgr(),
1718 rf_helper->account_name, NULL);
1719 signature = modest_account_mgr_get_signature (modest_runtime_get_account_mgr(),
1720 rf_helper->account_name,
1723 /* Create reply mail */
1724 switch (rf_helper->action) {
1727 modest_tny_msg_create_reply_msg (msg, header, from,
1728 (use_signature) ? signature : NULL,
1729 rf_helper->reply_forward_type,
1730 MODEST_TNY_MSG_REPLY_MODE_SENDER);
1732 case ACTION_REPLY_TO_ALL:
1734 modest_tny_msg_create_reply_msg (msg, header, from,
1735 (use_signature) ? signature : NULL,
1736 rf_helper->reply_forward_type,
1737 MODEST_TNY_MSG_REPLY_MODE_ALL);
1738 edit_type = MODEST_EDIT_TYPE_REPLY;
1740 case ACTION_FORWARD:
1742 modest_tny_msg_create_forward_msg (msg, from, (use_signature) ? signature : NULL,
1743 rf_helper->reply_forward_type);
1744 edit_type = MODEST_EDIT_TYPE_FORWARD;
1747 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1749 g_return_if_reached ();
1757 g_warning ("%s: failed to create message\n", __FUNCTION__);
1761 account = modest_tny_account_store_get_server_account (modest_runtime_get_account_store(),
1762 rf_helper->account_name,
1763 TNY_ACCOUNT_TYPE_STORE);
1765 g_warning ("%s: failed to get tnyaccount for '%s'\n", __FUNCTION__, rf_helper->account_name);
1769 /* Create and register the windows */
1770 /* TODO: fetch mailbox */
1771 msg_win = modest_msg_edit_window_new (new_msg, rf_helper->account_name, NULL, FALSE);
1772 mgr = modest_runtime_get_window_mgr ();
1773 modest_window_mgr_register_window (mgr, msg_win, (ModestWindow *) rf_helper->parent_window);
1775 /* Note that register_window could have deleted the account */
1776 if (MODEST_IS_WINDOW (rf_helper->parent_window)) {
1777 gdouble parent_zoom;
1779 parent_zoom = modest_window_get_zoom (MODEST_WINDOW (rf_helper->parent_window));
1780 modest_window_set_zoom (msg_win, parent_zoom);
1783 /* Show edit window */
1784 gtk_widget_show_all (GTK_WIDGET (msg_win));
1787 /* We always unregister the header because the message is
1788 forwarded or replied so the original one is no longer
1790 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1793 g_object_unref (G_OBJECT (new_msg));
1795 g_object_unref (G_OBJECT (account));
1796 free_reply_forward_helper (rf_helper);
1799 /* Checks a list of headers. If any of them are not currently
1800 * downloaded (CACHED) then returns TRUE else returns FALSE.
1803 header_list_count_uncached_msgs (TnyList *header_list)
1806 gint uncached_messages = 0;
1808 iter = tny_list_create_iterator (header_list);
1809 while (!tny_iterator_is_done (iter)) {
1812 header = TNY_HEADER (tny_iterator_get_current (iter));
1814 if (!(tny_header_get_flags (header) & TNY_HEADER_FLAG_CACHED))
1815 uncached_messages ++;
1816 g_object_unref (header);
1819 tny_iterator_next (iter);
1821 g_object_unref (iter);
1823 return uncached_messages;
1826 /* Returns FALSE if the user does not want to download the
1827 * messages. Returns TRUE if the user allowed the download.
1830 connect_to_get_msg (ModestWindow *win,
1831 gint num_of_uncached_msgs,
1832 TnyAccount *account)
1834 GtkResponseType response;
1836 /* Allways download if we are online. */
1837 if (tny_device_is_online (modest_runtime_get_device ()))
1840 /* If offline, then ask for user permission to download the messages */
1841 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1842 ngettext("mcen_nc_get_msg",
1844 num_of_uncached_msgs));
1846 if (response == GTK_RESPONSE_CANCEL)
1849 return modest_platform_connect_and_wait((GtkWindow *) win, account);
1853 reply_forward_performer (gboolean canceled,
1855 GtkWindow *parent_window,
1856 TnyAccount *account,
1859 ReplyForwardHelper *rf_helper = NULL;
1860 ModestMailOperation *mail_op;
1862 rf_helper = (ReplyForwardHelper *) user_data;
1864 if (canceled || err) {
1865 free_reply_forward_helper (rf_helper);
1869 /* Retrieve the message */
1870 modest_window_mgr_register_header (modest_runtime_get_window_mgr (), rf_helper->header, NULL);
1871 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT (parent_window),
1872 modest_ui_actions_disk_operations_error_handler,
1874 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
1875 modest_mail_operation_get_msg (mail_op, rf_helper->header, TRUE, reply_forward_cb, rf_helper);
1878 g_object_unref(mail_op);
1882 * Common code for the reply and forward actions
1885 reply_forward (ReplyForwardAction action, ModestWindow *win)
1887 ReplyForwardHelper *rf_helper = NULL;
1888 guint reply_forward_type;
1890 g_return_if_fail (MODEST_IS_WINDOW(win));
1892 /* we check for low-mem; in that case, show a warning, and don't allow
1893 * reply/forward (because it could potentially require a lot of memory */
1894 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
1898 /* we need an account when editing */
1899 if (!modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE)) {
1900 if (!modest_ui_actions_run_account_setup_wizard (win))
1904 reply_forward_type =
1905 modest_conf_get_int (modest_runtime_get_conf (),
1906 (action == ACTION_FORWARD) ?
1907 MODEST_CONF_FORWARD_TYPE :
1908 MODEST_CONF_REPLY_TYPE,
1911 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
1913 TnyHeader *header = NULL;
1914 /* Get header and message. Do not free them here, the
1915 reply_forward_cb must do it */
1916 msg = modest_msg_view_window_get_message (MODEST_MSG_VIEW_WINDOW(win));
1917 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
1919 if (msg && header) {
1921 rf_helper = create_reply_forward_helper (action, win,
1922 reply_forward_type, header);
1923 reply_forward_cb (NULL, header, FALSE, msg, NULL, rf_helper);
1925 g_warning("%s: no message or header found in viewer\n", __FUNCTION__);
1929 g_object_unref (msg);
1931 g_object_unref (header);
1933 TnyHeader *header = NULL;
1935 gboolean do_retrieve = TRUE;
1936 TnyList *header_list = NULL;
1938 header_list = get_selected_headers (win);
1941 /* Check that only one message is selected for replying */
1942 if (tny_list_get_length (header_list) != 1) {
1943 modest_platform_information_banner ((win) ? GTK_WIDGET (win) : NULL,
1944 NULL, _("mcen_ib_select_one_message"));
1945 g_object_unref (header_list);
1949 /* Only reply/forward to one message */
1950 iter = tny_list_create_iterator (header_list);
1951 header = TNY_HEADER (tny_iterator_get_current (iter));
1952 g_object_unref (iter);
1954 /* Retrieve messages */
1955 do_retrieve = (action == ACTION_FORWARD) ||
1956 (reply_forward_type != MODEST_TNY_MSG_REPLY_TYPE_CITE);
1959 TnyAccount *account = NULL;
1960 TnyFolder *folder = NULL;
1961 gdouble download = TRUE;
1962 guint uncached_msgs = 0;
1964 folder = tny_header_get_folder (header);
1966 goto do_retrieve_frees;
1967 account = tny_folder_get_account (folder);
1969 goto do_retrieve_frees;
1971 uncached_msgs = header_list_count_uncached_msgs (header_list);
1973 if (uncached_msgs > 0) {
1974 /* Allways download if we are online. */
1975 if (!tny_device_is_online (modest_runtime_get_device ())) {
1978 /* If ask for user permission to download the messages */
1979 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1980 ngettext("mcen_nc_get_msg",
1984 /* End if the user does not want to continue */
1985 if (response == GTK_RESPONSE_CANCEL)
1992 rf_helper = create_reply_forward_helper (action, win,
1993 reply_forward_type, header);
1994 if (uncached_msgs > 0) {
1995 modest_platform_connect_and_perform (GTK_WINDOW (win),
1997 reply_forward_performer,
2000 reply_forward_performer (FALSE, NULL, GTK_WINDOW (win),
2001 account, rf_helper);
2006 g_object_unref (account);
2008 g_object_unref (folder);
2010 reply_forward_cb (NULL, header, FALSE, NULL, NULL, rf_helper);
2013 g_object_unref (header_list);
2014 g_object_unref (header);
2019 modest_ui_actions_on_reply (GtkAction *action, ModestWindow *win)
2021 g_return_if_fail (MODEST_IS_WINDOW(win));
2023 reply_forward (ACTION_REPLY, win);
2027 modest_ui_actions_on_forward (GtkAction *action, ModestWindow *win)
2029 g_return_if_fail (MODEST_IS_WINDOW(win));
2031 reply_forward (ACTION_FORWARD, win);
2035 modest_ui_actions_on_reply_all (GtkAction *action, ModestWindow *win)
2037 g_return_if_fail (MODEST_IS_WINDOW(win));
2039 reply_forward (ACTION_REPLY_TO_ALL, win);
2043 modest_ui_actions_on_next (GtkAction *action,
2044 ModestWindow *window)
2046 if (MODEST_IS_MAIN_WINDOW (window)) {
2047 GtkWidget *header_view;
2049 header_view = modest_main_window_get_child_widget (
2050 MODEST_MAIN_WINDOW(window),
2051 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2055 modest_header_view_select_next (
2056 MODEST_HEADER_VIEW(header_view));
2057 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
2058 modest_msg_view_window_select_next_message (
2059 MODEST_MSG_VIEW_WINDOW (window));
2061 g_return_if_reached ();
2066 modest_ui_actions_on_prev (GtkAction *action,
2067 ModestWindow *window)
2069 g_return_if_fail (MODEST_IS_WINDOW(window));
2071 if (MODEST_IS_MAIN_WINDOW (window)) {
2072 GtkWidget *header_view;
2073 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(window),
2074 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2078 modest_header_view_select_prev (MODEST_HEADER_VIEW(header_view));
2079 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
2080 modest_msg_view_window_select_previous_message (MODEST_MSG_VIEW_WINDOW (window));
2082 g_return_if_reached ();
2087 modest_ui_actions_on_sort (GtkAction *action,
2088 ModestWindow *window)
2090 GtkWidget *header_view = NULL;
2092 g_return_if_fail (MODEST_IS_WINDOW(window));
2094 if (MODEST_IS_MAIN_WINDOW (window)) {
2095 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(window),
2096 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2097 #ifdef MODEST_TOOLKIT_HILDON2
2098 } else if (MODEST_IS_HEADER_WINDOW (window)) {
2099 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (window)));
2104 modest_platform_information_banner (NULL, NULL, _CS("ckdg_ib_nothing_to_sort"));
2109 /* Show sorting dialog */
2110 modest_utils_run_sort_dialog (GTK_WINDOW (window), MODEST_SORT_HEADERS);
2114 new_messages_arrived (ModestMailOperation *self,
2115 TnyList *new_headers,
2119 gboolean show_visual_notifications;
2121 source = modest_mail_operation_get_source (self);
2122 show_visual_notifications = (source) ? FALSE : TRUE;
2124 g_object_unref (source);
2126 /* Notify new messages have been downloaded. If the
2127 send&receive was invoked by the user then do not show any
2128 visual notification, only play a sound and activate the LED
2129 (for the Maemo version) */
2130 if (TNY_IS_LIST(new_headers) && (tny_list_get_length (new_headers)) > 0)
2131 modest_platform_on_new_headers_received (new_headers,
2132 show_visual_notifications);
2137 retrieve_all_messages_cb (GObject *source,
2139 guint retrieve_limit)
2145 window = GTK_WINDOW (source);
2146 msg = g_strdup_printf (_("mail_nc_msg_count_limit_exceeded"),
2147 num_msgs, retrieve_limit);
2149 /* Ask the user if they want to retrieve all the messages */
2151 modest_platform_run_confirmation_dialog_with_buttons (window, msg,
2152 _("mcen_bd_get_all"),
2153 _("mcen_bd_newest_only"));
2154 /* Free and return */
2156 return (response == GTK_RESPONSE_ACCEPT) ? TRUE : FALSE;
2160 TnyAccount *account;
2162 gchar *account_name;
2163 gboolean poke_status;
2164 gboolean interactive;
2165 ModestMailOperation *mail_op;
2169 do_send_receive_performer (gboolean canceled,
2171 GtkWindow *parent_window,
2172 TnyAccount *account,
2175 SendReceiveInfo *info;
2177 info = (SendReceiveInfo *) user_data;
2179 if (err || canceled) {
2180 /* In memory full conditions we could get this error here */
2181 check_memory_full_error ((GtkWidget *) parent_window, err);
2183 if (info->mail_op) {
2184 modest_mail_operation_queue_remove (modest_runtime_get_mail_operation_queue (),
2190 /* Set send/receive operation in progress */
2191 if (info->win && MODEST_IS_MAIN_WINDOW (info->win)) {
2192 modest_main_window_notify_send_receive_initied (MODEST_MAIN_WINDOW (info->win));
2195 if (info->win && MODEST_IS_MAIN_WINDOW (info->win))
2196 g_signal_connect (G_OBJECT (info->mail_op), "operation-finished",
2197 G_CALLBACK (on_send_receive_finished),
2200 /* Send & receive. */
2201 modest_mail_operation_update_account (info->mail_op, info->account_name, info->poke_status, info->interactive,
2202 (info->win) ? retrieve_all_messages_cb : NULL,
2203 new_messages_arrived, info->win);
2208 g_object_unref (G_OBJECT (info->mail_op));
2209 if (info->account_name)
2210 g_free (info->account_name);
2212 g_object_unref (info->win);
2214 g_object_unref (info->account);
2215 g_slice_free (SendReceiveInfo, info);
2219 * This function performs the send & receive required actions. The
2220 * window is used to create the mail operation. Typically it should
2221 * always be the main window, but we pass it as argument in order to
2225 modest_ui_actions_do_send_receive (const gchar *account_name,
2226 gboolean force_connection,
2227 gboolean poke_status,
2228 gboolean interactive,
2231 gchar *acc_name = NULL;
2232 SendReceiveInfo *info;
2233 ModestTnyAccountStore *acc_store;
2235 /* If no account name was provided then get the current account, and if
2236 there is no current account then pick the default one: */
2237 if (!account_name) {
2239 acc_name = g_strdup (modest_window_get_active_account (win));
2241 acc_name = modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
2243 g_printerr ("modest: cannot get default account\n");
2247 acc_name = g_strdup (account_name);
2250 acc_store = modest_runtime_get_account_store ();
2252 /* Create the info for the connect and perform */
2253 info = g_slice_new (SendReceiveInfo);
2254 info->account_name = acc_name;
2255 info->win = (win) ? g_object_ref (win) : NULL;
2256 info->poke_status = poke_status;
2257 info->interactive = interactive;
2258 info->account = modest_tny_account_store_get_server_account (acc_store, acc_name,
2259 TNY_ACCOUNT_TYPE_STORE);
2260 /* We need to create the operation here, because otherwise it
2261 could happen that the queue emits the queue-empty signal
2262 while we're trying to connect the account */
2263 info->mail_op = modest_mail_operation_new_with_error_handling ((info->win) ? G_OBJECT (info->win) : NULL,
2264 modest_ui_actions_disk_operations_error_handler,
2266 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), info->mail_op);
2268 /* Invoke the connect and perform */
2269 modest_platform_connect_and_perform ((win) ? GTK_WINDOW (win) : NULL,
2270 force_connection, info->account,
2271 do_send_receive_performer, info);
2276 modest_ui_actions_do_cancel_send (const gchar *account_name,
2279 TnyTransportAccount *transport_account;
2280 TnySendQueue *send_queue = NULL;
2281 GError *error = NULL;
2283 /* Get transport account */
2285 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
2286 (modest_runtime_get_account_store(),
2288 TNY_ACCOUNT_TYPE_TRANSPORT));
2289 if (!transport_account) {
2290 g_printerr ("modest: no transport account found for '%s'\n", account_name);
2295 send_queue = TNY_SEND_QUEUE (modest_runtime_get_send_queue (transport_account, TRUE));
2296 if (!TNY_IS_SEND_QUEUE(send_queue)) {
2297 g_set_error (&error, MODEST_MAIL_OPERATION_ERROR,
2298 MODEST_MAIL_OPERATION_ERROR_ITEM_NOT_FOUND,
2299 "modest: could not find send queue for account\n");
2301 /* Cancel the current send */
2302 tny_account_cancel (TNY_ACCOUNT (transport_account));
2304 /* Suspend all pending messages */
2305 tny_send_queue_cancel (send_queue, TNY_SEND_QUEUE_CANCEL_ACTION_SUSPEND, &error);
2309 if (transport_account != NULL)
2310 g_object_unref (G_OBJECT (transport_account));
2314 modest_ui_actions_cancel_send_all (ModestWindow *win)
2316 GSList *account_names, *iter;
2318 account_names = modest_account_mgr_account_names (modest_runtime_get_account_mgr(),
2321 iter = account_names;
2323 modest_ui_actions_do_cancel_send ((const char*) iter->data, win);
2324 iter = g_slist_next (iter);
2327 modest_account_mgr_free_account_names (account_names);
2328 account_names = NULL;
2332 modest_ui_actions_cancel_send (GtkAction *action, ModestWindow *win)
2335 /* Check if accounts exist */
2336 gboolean accounts_exist =
2337 modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
2339 /* If not, allow the user to create an account before trying to send/receive. */
2340 if (!accounts_exist)
2341 modest_ui_actions_on_accounts (NULL, win);
2343 /* Cancel all sending operaitons */
2344 modest_ui_actions_cancel_send_all (win);
2348 * Refreshes all accounts. This function will be used by automatic
2352 modest_ui_actions_do_send_receive_all (ModestWindow *win,
2353 gboolean force_connection,
2354 gboolean poke_status,
2355 gboolean interactive)
2357 GSList *account_names, *iter;
2359 account_names = modest_account_mgr_account_names (modest_runtime_get_account_mgr(),
2362 iter = account_names;
2364 modest_ui_actions_do_send_receive ((const char*) iter->data,
2366 poke_status, interactive, win);
2367 iter = g_slist_next (iter);
2370 modest_account_mgr_free_account_names (account_names);
2371 account_names = NULL;
2375 * Handler of the click on Send&Receive button in the main toolbar
2378 modest_ui_actions_on_send_receive (GtkAction *action, ModestWindow *win)
2380 /* Check if accounts exist */
2381 gboolean accounts_exist;
2384 modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
2386 /* If not, allow the user to create an account before trying to send/receive. */
2387 if (!accounts_exist)
2388 modest_ui_actions_on_accounts (NULL, win);
2390 /* Refresh the current folder. The if is always TRUE it's just an extra check */
2391 if (MODEST_IS_MAIN_WINDOW (win)) {
2392 GtkWidget *folder_view;
2393 TnyFolderStore *folder_store;
2396 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
2397 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2401 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
2404 g_object_unref (folder_store);
2405 /* Refresh the active account. Force the connection if needed
2406 and poke the status of all folders */
2407 modest_ui_actions_do_send_receive (NULL, TRUE, TRUE, TRUE, win);
2408 #ifdef MODEST_TOOLKIT_HILDON2
2409 } else if (MODEST_IS_ACCOUNTS_WINDOW (win)) {
2410 modest_ui_actions_do_send_receive_all (win, TRUE, TRUE, TRUE);
2413 const gchar *active_account;
2414 active_account = modest_window_get_active_account (MODEST_WINDOW (win));
2416 modest_ui_actions_do_send_receive (active_account, TRUE, TRUE, TRUE, win);
2423 modest_ui_actions_toggle_header_list_view (GtkAction *action, ModestMainWindow *main_window)
2426 GtkWidget *header_view;
2428 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2430 header_view = modest_main_window_get_child_widget (main_window,
2431 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2435 conf = modest_runtime_get_conf ();
2437 /* what is saved/restored is depending on the style; thus; we save with
2438 * old style, then update the style, and restore for this new style
2440 modest_widget_memory_save (conf, G_OBJECT(header_view), MODEST_CONF_HEADER_VIEW_KEY);
2442 if (modest_header_view_get_style
2443 (MODEST_HEADER_VIEW(header_view)) == MODEST_HEADER_VIEW_STYLE_DETAILS)
2444 modest_header_view_set_style (MODEST_HEADER_VIEW(header_view),
2445 MODEST_HEADER_VIEW_STYLE_TWOLINES);
2447 modest_header_view_set_style (MODEST_HEADER_VIEW(header_view),
2448 MODEST_HEADER_VIEW_STYLE_DETAILS);
2450 modest_widget_memory_restore (conf, G_OBJECT(header_view),
2451 MODEST_CONF_HEADER_VIEW_KEY);
2456 modest_ui_actions_on_header_selected (ModestHeaderView *header_view,
2458 ModestMainWindow *main_window)
2460 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2461 g_return_if_fail (MODEST_IS_HEADER_VIEW (header_view));
2463 /* in the case the folder is empty, show the empty folder message and focus
2465 if (!header && gtk_widget_is_focus (GTK_WIDGET (header_view))) {
2466 if (modest_header_view_is_empty (header_view)) {
2467 TnyFolder *folder = modest_header_view_get_folder (header_view);
2468 GtkWidget *folder_view =
2469 modest_main_window_get_child_widget (main_window,
2470 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2471 if (folder != NULL) {
2472 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view), folder, FALSE);
2473 g_object_unref (folder);
2475 gtk_widget_grab_focus (GTK_WIDGET (folder_view));
2479 /* If no header has been selected then exit */
2484 if (!gtk_widget_is_focus (GTK_WIDGET(header_view)))
2485 gtk_widget_grab_focus (GTK_WIDGET(header_view));
2487 /* Update toolbar dimming state */
2488 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
2489 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
2493 modest_ui_actions_on_header_activated (ModestHeaderView *header_view,
2496 ModestWindow *window)
2498 GtkWidget *open_widget;
2499 GtkTreeRowReference *rowref;
2501 g_return_if_fail (MODEST_IS_WINDOW(window));
2502 g_return_if_fail (MODEST_IS_HEADER_VIEW (header_view));
2503 g_return_if_fail (TNY_IS_HEADER (header));
2505 if (modest_header_view_count_selected_headers (header_view) > 1) {
2506 /* Don't allow activation if there are more than one message selected */
2507 modest_platform_information_banner (NULL, NULL, _("mcen_ib_select_one_message"));
2511 /* we check for low-mem; in that case, show a warning, and don't allow
2512 * activating headers
2514 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
2517 if (MODEST_IS_MAIN_WINDOW (window)) {
2518 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (window));
2519 open_widget = modest_window_get_action_widget (MODEST_WINDOW (window), "/MenuBar/EmailMenu/EmailOpenMenu");
2520 if (!GTK_WIDGET_IS_SENSITIVE (open_widget))
2524 rowref = gtk_tree_row_reference_new (gtk_tree_view_get_model (GTK_TREE_VIEW (header_view)), path);
2525 open_msg_from_header (header, rowref, MODEST_WINDOW (window));
2526 gtk_tree_row_reference_free (rowref);
2530 set_active_account_from_tny_account (TnyAccount *account,
2531 ModestWindow *window)
2533 const gchar *server_acc_name = tny_account_get_id (account);
2535 /* We need the TnyAccount provided by the
2536 account store because that is the one that
2537 knows the name of the Modest account */
2538 TnyAccount *modest_server_account =
2539 modest_tny_account_store_get_tny_account_by (modest_runtime_get_account_store (),
2540 MODEST_TNY_ACCOUNT_STORE_QUERY_ID,
2542 if (!modest_server_account) {
2543 g_warning ("%s: could not get tny account\n", __FUNCTION__);
2547 /* Update active account, but only if it's not a pseudo-account */
2548 if ((!modest_tny_account_is_virtual_local_folders(modest_server_account)) &&
2549 (!modest_tny_account_is_memory_card_account(modest_server_account))) {
2550 const gchar *modest_acc_name =
2551 modest_tny_account_get_parent_modest_account_name_for_server_account (modest_server_account);
2552 if (modest_acc_name)
2553 modest_window_set_active_account (window, modest_acc_name);
2556 g_object_unref (modest_server_account);
2561 folder_refreshed_cb (ModestMailOperation *mail_op,
2565 ModestMainWindow *win = NULL;
2566 GtkWidget *folder_view;
2567 const GError *error;
2569 g_return_if_fail (TNY_IS_FOLDER (folder));
2571 win = MODEST_MAIN_WINDOW (user_data);
2573 /* Check if the operation failed due to memory low conditions */
2574 error = modest_mail_operation_get_error (mail_op);
2575 if (error && error->domain == MODEST_MAIL_OPERATION_ERROR &&
2576 error->code == MODEST_MAIL_OPERATION_ERROR_LOW_MEMORY) {
2577 modest_platform_run_information_dialog (GTK_WINDOW (win),
2578 _KR("memr_ib_operation_disabled"),
2584 modest_main_window_get_child_widget(win, MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2587 TnyFolderStore *current_folder;
2589 current_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
2590 if (current_folder) {
2591 gboolean different = ((TnyFolderStore *) folder != current_folder);
2592 g_object_unref (current_folder);
2598 /* Check if folder is empty and set headers view contents style */
2599 if (tny_folder_get_all_count (folder) == 0)
2600 modest_main_window_set_contents_style (win,
2601 MODEST_MAIN_WINDOW_CONTENTS_STYLE_EMPTY);
2606 modest_ui_actions_on_folder_selection_changed (ModestFolderView *folder_view,
2607 TnyFolderStore *folder_store,
2609 ModestMainWindow *main_window)
2612 GtkWidget *header_view;
2614 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2616 header_view = modest_main_window_get_child_widget(main_window,
2617 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2621 conf = modest_runtime_get_conf ();
2623 if (TNY_IS_ACCOUNT (folder_store)) {
2625 set_active_account_from_tny_account (TNY_ACCOUNT (folder_store), MODEST_WINDOW (main_window));
2627 /* Show account details */
2628 modest_main_window_set_contents_style (main_window, MODEST_MAIN_WINDOW_CONTENTS_STYLE_DETAILS);
2631 if (TNY_IS_FOLDER (folder_store) && selected) {
2632 TnyAccount *account;
2633 const gchar *account_name = NULL;
2635 /* Update the active account */
2636 account = modest_tny_folder_get_account (TNY_FOLDER (folder_store));
2638 set_active_account_from_tny_account (account, MODEST_WINDOW (main_window));
2640 modest_tny_account_get_parent_modest_account_name_for_server_account (account);
2641 g_object_unref (account);
2645 /* Set the header style by default, it could
2646 be changed later by the refresh callback to
2648 modest_main_window_set_contents_style (main_window,
2649 MODEST_MAIN_WINDOW_CONTENTS_STYLE_HEADERS);
2651 /* Set folder on header view. This function
2652 will call tny_folder_refresh_async so we
2653 pass a callback that will be called when
2654 finished. We use that callback to set the
2655 empty view if there are no messages */
2656 modest_header_view_set_folder (MODEST_HEADER_VIEW(header_view),
2657 TNY_FOLDER (folder_store),
2659 MODEST_WINDOW (main_window),
2660 folder_refreshed_cb,
2663 /* Restore configuration. We need to do this
2664 *after* the set_folder because the widget
2665 memory asks the header view about its
2667 modest_widget_memory_restore (modest_runtime_get_conf (),
2668 G_OBJECT(header_view),
2669 MODEST_CONF_HEADER_VIEW_KEY);
2671 /* No need to save the header view
2672 configuration for Maemo because it only
2673 saves the sorting stuff and that it's
2674 already being done by the sort
2675 dialog. Remove it when the GNOME version
2676 has the same behaviour */
2677 #ifdef MODEST_TOOLKIT_GTK
2678 if (modest_main_window_get_contents_style (main_window) ==
2679 MODEST_MAIN_WINDOW_CONTENTS_STYLE_HEADERS)
2680 modest_widget_memory_save (conf, G_OBJECT (header_view),
2681 MODEST_CONF_HEADER_VIEW_KEY);
2683 modest_header_view_clear (MODEST_HEADER_VIEW(header_view));
2687 /* Update dimming state */
2688 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
2689 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
2693 modest_ui_actions_on_item_not_found (ModestHeaderView *header_view,ModestItemType type,
2700 item = (type == MODEST_ITEM_TYPE_FOLDER) ? "folder" : "message";
2702 online = tny_device_is_online (modest_runtime_get_device());
2705 /* already online -- the item is simply not there... */
2706 dialog = gtk_message_dialog_new (GTK_WINDOW (win),
2708 GTK_MESSAGE_WARNING,
2710 _("The %s you selected cannot be found"),
2712 gtk_dialog_add_button (GTK_DIALOG (dialog),_("mcen_bd_dialog_ok"), GTK_RESPONSE_ACCEPT);
2713 gtk_dialog_run (GTK_DIALOG(dialog));
2715 dialog = gtk_dialog_new_with_buttons (_("Connection requested"),
2718 _("mcen_bd_dialog_cancel"),
2719 GTK_RESPONSE_REJECT,
2720 _("mcen_bd_dialog_ok"),
2721 GTK_RESPONSE_ACCEPT,
2723 txt = g_strdup_printf (_("This %s is not available in offline mode.\n"
2724 "Do you want to get online?"), item);
2725 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox),
2726 gtk_label_new (txt), FALSE, FALSE, 0);
2727 gtk_widget_show_all (GTK_WIDGET(GTK_DIALOG(dialog)->vbox));
2730 gtk_window_set_default_size (GTK_WINDOW(dialog), 300, 300);
2731 if (gtk_dialog_run (GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
2732 /* TODO: Comment about why is this commented out: */
2733 /* modest_platform_connect_and_wait (); */
2736 gtk_widget_destroy (dialog);
2740 modest_ui_actions_on_msg_link_hover (ModestMsgView *msgview, const gchar* link,
2743 /* g_message ("%s %s", __FUNCTION__, link); */
2748 modest_ui_actions_on_msg_link_clicked (ModestMsgView *msgview, const gchar* link,
2751 modest_platform_activate_uri (link);
2755 modest_ui_actions_on_msg_link_contextual (ModestMsgView *msgview, const gchar* link,
2758 modest_platform_show_uri_popup (link);
2762 modest_ui_actions_on_msg_attachment_clicked (ModestMsgView *msgview, TnyMimePart *mime_part,
2765 /* we check for low-mem; in that case, show a warning, and don't allow
2766 * viewing attachments
2768 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
2771 modest_msg_view_window_view_attachment (MODEST_MSG_VIEW_WINDOW (win), mime_part);
2775 modest_ui_actions_on_msg_recpt_activated (ModestMsgView *msgview,
2776 const gchar *address,
2779 /* g_message ("%s %s", __FUNCTION__, address); */
2783 on_save_to_drafts_cb (ModestMailOperation *mail_op,
2784 TnyMsg *saved_draft,
2787 ModestMsgEditWindow *edit_window;
2789 /* TODO: in hildon 2 we have to dim and undim the header views while we're saving */
2790 #ifndef MODEST_TOOLKIT_HILDON2
2791 ModestMainWindow *win;
2793 /* FIXME. Make the header view sensitive again. This is a
2794 * temporary hack. See modest_ui_actions_on_save_to_drafts()
2796 win = MODEST_MAIN_WINDOW(modest_window_mgr_get_main_window(
2797 modest_runtime_get_window_mgr(), FALSE));
2799 GtkWidget *hdrview = modest_main_window_get_child_widget(
2800 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2801 if (hdrview) gtk_widget_set_sensitive(hdrview, TRUE);
2805 edit_window = MODEST_MSG_EDIT_WINDOW (user_data);
2807 /* Set draft is there was no error */
2808 if (!modest_mail_operation_get_error (mail_op))
2809 modest_msg_edit_window_set_draft (edit_window, saved_draft);
2811 g_object_unref(edit_window);
2815 enough_space_for_message (ModestMsgEditWindow *edit_window,
2818 TnyAccountStore *acc_store;
2819 guint64 available_disk, expected_size;
2824 acc_store = TNY_ACCOUNT_STORE (modest_runtime_get_account_store());
2825 available_disk = modest_utils_get_available_space (NULL);
2826 modest_msg_edit_window_get_parts_size (edit_window, &parts_count, &parts_size);
2827 expected_size = modest_tny_msg_estimate_size (data->plain_body,
2832 /* Double check: memory full condition or message too big */
2833 if (available_disk < MIN_FREE_SPACE ||
2834 expected_size > available_disk) {
2836 modest_platform_information_banner (NULL, NULL,
2837 _KR("cerm_device_memory_full"));
2842 * djcb: if we're in low-memory state, we only allow for
2843 * saving messages smaller than
2844 * MODEST_MAX_LOW_MEMORY_MESSAGE_SIZE (see modest-defs.h) this
2845 * should still allow for sending anything critical...
2847 if ((expected_size > MODEST_MAX_LOW_MEMORY_MESSAGE_SIZE) &&
2848 modest_platform_check_memory_low (MODEST_WINDOW(edit_window), TRUE))
2852 * djcb: we also make sure that the attachments are smaller than the max size
2853 * this is for the case where we'd try to forward a message with attachments
2854 * bigger than our max allowed size, or sending an message from drafts which
2855 * somehow got past our checks when attaching.
2857 if (expected_size > MODEST_MAX_ATTACHMENT_SIZE) {
2858 modest_platform_run_information_dialog (
2859 GTK_WINDOW(edit_window),
2860 _KR("memr_ib_operation_disabled"),
2869 modest_ui_actions_on_save_to_drafts (GtkWidget *widget, ModestMsgEditWindow *edit_window)
2871 TnyTransportAccount *transport_account;
2872 ModestMailOperation *mail_operation;
2874 gchar *account_name;
2875 ModestAccountMgr *account_mgr;
2876 gboolean had_error = FALSE;
2877 ModestMainWindow *win = NULL;
2879 g_return_val_if_fail (MODEST_IS_MSG_EDIT_WINDOW(edit_window), FALSE);
2881 data = modest_msg_edit_window_get_msg_data (edit_window);
2884 if (!enough_space_for_message (edit_window, data)) {
2885 modest_msg_edit_window_free_msg_data (edit_window, data);
2889 account_name = g_strdup (data->account_name);
2890 account_mgr = modest_runtime_get_account_mgr();
2892 account_name = g_strdup(modest_window_get_active_account (MODEST_WINDOW(edit_window)));
2894 account_name = modest_account_mgr_get_default_account (account_mgr);
2895 if (!account_name) {
2896 g_printerr ("modest: no account found\n");
2897 modest_msg_edit_window_free_msg_data (edit_window, data);
2901 if (!strcmp (account_name, MODEST_LOCAL_FOLDERS_ACCOUNT_ID)) {
2902 account_name = g_strdup (data->account_name);
2906 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
2907 (modest_runtime_get_account_store (),
2909 TNY_ACCOUNT_TYPE_TRANSPORT));
2910 if (!transport_account) {
2911 g_printerr ("modest: no transport account found for '%s'\n", account_name);
2912 g_free (account_name);
2913 modest_msg_edit_window_free_msg_data (edit_window, data);
2917 /* Create the mail operation */
2918 mail_operation = modest_mail_operation_new_with_error_handling (NULL, modest_ui_actions_disk_operations_error_handler,
2920 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_operation);
2922 modest_mail_operation_save_to_drafts (mail_operation,
2934 data->priority_flags,
2937 on_save_to_drafts_cb,
2938 g_object_ref(edit_window));
2940 #ifdef MODEST_TOOLKIT_HILDON2
2941 /* In hildon2 we always show the information banner on saving to drafts.
2942 * It will be a system information banner in this case.
2944 gchar *text = g_strdup_printf (_("mail_va_saved_to_drafts"), _("mcen_me_folder_drafts"));
2945 modest_platform_information_banner (NULL, NULL, text);
2948 /* Use the main window as the parent of the banner, if the
2949 main window does not exist it won't be shown, if the parent
2950 window exists then it's properly shown. We don't use the
2951 editor window because it could be closed (save to drafts
2952 could happen after closing the window */
2953 win = (ModestMainWindow *)
2954 modest_window_mgr_get_main_window( modest_runtime_get_window_mgr(), FALSE);
2956 gchar *text = g_strdup_printf (_("mail_va_saved_to_drafts"), _("mcen_me_folder_drafts"));
2957 modest_platform_information_banner (GTK_WIDGET (win), NULL, text);
2961 modest_msg_edit_window_set_modified (edit_window, FALSE);
2964 g_free (account_name);
2965 g_object_unref (G_OBJECT (transport_account));
2966 g_object_unref (G_OBJECT (mail_operation));
2968 modest_msg_edit_window_free_msg_data (edit_window, data);
2971 * If the drafts folder is selected then make the header view
2972 * insensitive while the message is being saved to drafts
2973 * (it'll be sensitive again in on_save_to_drafts_cb()). This
2974 * is not very clean but it avoids letting the drafts folder
2975 * in an inconsistent state: the user could edit the message
2976 * being saved and undesirable things would happen.
2977 * In the average case the user won't notice anything at
2978 * all. In the worst case (the user is editing a really big
2979 * file from Drafts) the header view will be insensitive
2980 * during the saving process (10 or 20 seconds, depending on
2981 * the message). Anyway this is just a quick workaround: once
2982 * we find a better solution it should be removed
2983 * See NB#65125 (commend #18) for details.
2985 if (!had_error && win != NULL) {
2986 ModestFolderView *view = MODEST_FOLDER_VIEW(modest_main_window_get_child_widget(
2987 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW));
2989 TnyFolder *folder = TNY_FOLDER(modest_folder_view_get_selected(view));
2991 if (modest_tny_folder_is_local_folder(folder)) {
2992 TnyFolderType folder_type;
2993 folder_type = modest_tny_folder_get_local_or_mmc_folder_type(folder);
2994 if (folder_type == TNY_FOLDER_TYPE_DRAFTS) {
2995 GtkWidget *hdrview = modest_main_window_get_child_widget(
2996 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2997 if (hdrview) gtk_widget_set_sensitive(hdrview, FALSE);
3001 if (folder != NULL) g_object_unref(folder);
3008 /* For instance, when clicking the Send toolbar button when editing a message: */
3010 modest_ui_actions_on_send (GtkWidget *widget, ModestMsgEditWindow *edit_window)
3012 TnyTransportAccount *transport_account = NULL;
3013 gboolean had_error = FALSE;
3015 ModestAccountMgr *account_mgr;
3016 gchar *account_name;
3017 ModestMailOperation *mail_operation;
3019 g_return_val_if_fail (MODEST_IS_MSG_EDIT_WINDOW(edit_window), TRUE);
3021 if (!modest_msg_edit_window_check_names (edit_window, TRUE))
3024 data = modest_msg_edit_window_get_msg_data (edit_window);
3027 if (!enough_space_for_message (edit_window, data)) {
3028 modest_msg_edit_window_free_msg_data (edit_window, data);
3032 account_mgr = modest_runtime_get_account_mgr();
3033 account_name = g_strdup (data->account_name);
3035 account_name = g_strdup(modest_window_get_active_account (MODEST_WINDOW(edit_window)));
3038 account_name = modest_account_mgr_get_default_account (account_mgr);
3040 if (!account_name) {
3041 modest_msg_edit_window_free_msg_data (edit_window, data);
3042 /* Run account setup wizard */
3043 if (!modest_ui_actions_run_account_setup_wizard (MODEST_WINDOW(edit_window))) {
3048 /* Get the currently-active transport account for this modest account: */
3049 if (strcmp (account_name, MODEST_LOCAL_FOLDERS_ACCOUNT_ID) != 0) {
3051 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
3052 (modest_runtime_get_account_store (),
3053 account_name, TNY_ACCOUNT_TYPE_TRANSPORT));
3056 if (!transport_account) {
3057 modest_msg_edit_window_free_msg_data (edit_window, data);
3058 /* Run account setup wizard */
3059 if (!modest_ui_actions_run_account_setup_wizard(MODEST_WINDOW(edit_window)))
3064 /* Create the mail operation */
3065 mail_operation = modest_mail_operation_new_with_error_handling (NULL, modest_ui_actions_disk_operations_error_handler, NULL, NULL);
3066 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_operation);
3068 modest_mail_operation_send_new_mail (mail_operation,
3082 data->priority_flags);
3084 if (modest_mail_operation_get_status (mail_operation) == MODEST_MAIL_OPERATION_STATUS_IN_PROGRESS)
3085 modest_platform_information_banner (NULL, NULL, _("mcen_ib_outbox_waiting_to_be_sent"));
3087 if (modest_mail_operation_get_error (mail_operation) != NULL) {
3088 const GError *error = modest_mail_operation_get_error (mail_operation);
3089 if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
3090 error->code == MODEST_MAIL_OPERATION_ERROR_INSTANCE_CREATION_FAILED) {
3091 g_warning ("%s failed: %s\n", __FUNCTION__, (modest_mail_operation_get_error (mail_operation))->message);
3092 modest_platform_information_banner (NULL, NULL, _CS("sfil_ni_not_enough_memory"));
3098 g_free (account_name);
3099 g_object_unref (G_OBJECT (transport_account));
3100 g_object_unref (G_OBJECT (mail_operation));
3102 modest_msg_edit_window_free_msg_data (edit_window, data);
3105 modest_msg_edit_window_set_sent (edit_window, TRUE);
3107 /* Save settings and close the window: */
3108 modest_ui_actions_on_close_window (NULL, MODEST_WINDOW (edit_window));
3115 modest_ui_actions_on_toggle_bold (GtkToggleAction *action,
3116 ModestMsgEditWindow *window)
3118 ModestMsgEditFormatState *format_state = NULL;
3120 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3121 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3123 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW (window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3126 format_state = modest_msg_edit_window_get_format_state (window);
3127 g_return_if_fail (format_state != NULL);
3129 format_state->bold = gtk_toggle_action_get_active (action);
3130 modest_msg_edit_window_set_format_state (window, format_state);
3131 g_free (format_state);
3136 modest_ui_actions_on_toggle_italics (GtkToggleAction *action,
3137 ModestMsgEditWindow *window)
3139 ModestMsgEditFormatState *format_state = NULL;
3141 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3142 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3144 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3147 format_state = modest_msg_edit_window_get_format_state (window);
3148 g_return_if_fail (format_state != NULL);
3150 format_state->italics = gtk_toggle_action_get_active (action);
3151 modest_msg_edit_window_set_format_state (window, format_state);
3152 g_free (format_state);
3157 modest_ui_actions_on_toggle_bullets (GtkToggleAction *action,
3158 ModestMsgEditWindow *window)
3160 ModestMsgEditFormatState *format_state = NULL;
3162 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3163 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3165 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW (window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3168 format_state = modest_msg_edit_window_get_format_state (window);
3169 g_return_if_fail (format_state != NULL);
3171 format_state->bullet = gtk_toggle_action_get_active (action);
3172 modest_msg_edit_window_set_format_state (window, format_state);
3173 g_free (format_state);
3178 modest_ui_actions_on_change_justify (GtkRadioAction *action,
3179 GtkRadioAction *selected,
3180 ModestMsgEditWindow *window)
3182 ModestMsgEditFormatState *format_state = NULL;
3183 GtkJustification value;
3185 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3187 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3190 value = gtk_radio_action_get_current_value (selected);
3192 format_state = modest_msg_edit_window_get_format_state (window);
3193 g_return_if_fail (format_state != NULL);
3195 format_state->justification = value;
3196 modest_msg_edit_window_set_format_state (window, format_state);
3197 g_free (format_state);
3201 modest_ui_actions_on_select_editor_color (GtkAction *action,
3202 ModestMsgEditWindow *window)
3204 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3205 g_return_if_fail (GTK_IS_ACTION (action));
3207 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3210 modest_msg_edit_window_select_color (window);
3214 modest_ui_actions_on_select_editor_background_color (GtkAction *action,
3215 ModestMsgEditWindow *window)
3217 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3218 g_return_if_fail (GTK_IS_ACTION (action));
3220 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3223 modest_msg_edit_window_select_background_color (window);
3227 modest_ui_actions_on_insert_image (GObject *object,
3228 ModestMsgEditWindow *window)
3230 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3233 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
3236 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3239 modest_msg_edit_window_insert_image (window);
3243 modest_ui_actions_on_attach_file (GtkAction *action,
3244 ModestMsgEditWindow *window)
3246 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3247 g_return_if_fail (GTK_IS_ACTION (action));
3249 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
3252 modest_msg_edit_window_offer_attach_file (window);
3256 modest_ui_actions_on_remove_attachments (GtkAction *action,
3257 ModestMsgEditWindow *window)
3259 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3261 modest_msg_edit_window_remove_attachments (window, NULL);
3265 do_create_folder_cb (ModestMailOperation *mail_op,
3266 TnyFolderStore *parent_folder,
3267 TnyFolder *new_folder,
3270 gchar *suggested_name = (gchar *) user_data;
3271 GtkWindow *source_win = (GtkWindow *) modest_mail_operation_get_source (mail_op);
3272 const GError *error;
3274 error = modest_mail_operation_get_error (mail_op);
3277 /* Show an error. If there was some problem writing to
3278 disk, show it, otherwise show the generic folder
3279 create error. We do it here and not in an error
3280 handler because the call to do_create_folder will
3281 stop the main loop in a gtk_dialog_run and then,
3282 the message won't be shown until that dialog is
3284 modest_ui_actions_disk_operations_error_handler (mail_op,
3285 _("mail_in_ui_folder_create_error"));
3287 if (!is_memory_full_error ((GError *) error, mail_op)) {
3288 /* Try again if there is no full memory condition */
3289 do_create_folder (source_win, parent_folder, (const gchar *) suggested_name);
3292 /* the 'source_win' is either the ModestMainWindow, or the 'Move to folder'-dialog
3293 * FIXME: any other? */
3294 GtkWidget *folder_view;
3296 if (MODEST_IS_MAIN_WINDOW(source_win))
3298 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (source_win),
3299 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3301 folder_view = GTK_WIDGET(g_object_get_data (G_OBJECT (source_win),
3302 MODEST_MOVE_TO_DIALOG_FOLDER_VIEW));
3304 /* Select the newly created folder. It could happen
3305 that the widget is no longer there (i.e. the window
3306 has been destroyed, so we need to check this */
3308 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view),
3310 g_object_unref (new_folder);
3312 /* Free. Note that the first time it'll be NULL so noop */
3313 g_free (suggested_name);
3314 g_object_unref (source_win);
3319 TnyFolderStore *parent;
3320 } CreateFolderConnect;
3323 do_create_folder_performer (gboolean canceled,
3325 GtkWindow *parent_window,
3326 TnyAccount *account,
3329 CreateFolderConnect *helper = (CreateFolderConnect *) user_data;
3330 ModestMailOperation *mail_op;
3332 if (canceled || err) {
3333 /* In memory full conditions we could get this error here */
3334 check_memory_full_error ((GtkWidget *) parent_window, err);
3338 mail_op = modest_mail_operation_new ((GObject *) parent_window);
3339 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3341 modest_mail_operation_create_folder (mail_op,
3343 (const gchar *) helper->folder_name,
3344 do_create_folder_cb,
3345 g_strdup (helper->folder_name));
3346 g_object_unref (mail_op);
3350 g_object_unref (helper->parent);
3351 if (helper->folder_name)
3352 g_free (helper->folder_name);
3353 g_slice_free (CreateFolderConnect, helper);
3358 do_create_folder (GtkWindow *parent_window,
3359 TnyFolderStore *suggested_parent,
3360 const gchar *suggested_name)
3363 gchar *folder_name = NULL;
3364 TnyFolderStore *parent_folder = NULL;
3366 result = modest_platform_run_new_folder_dialog (GTK_WINDOW (parent_window),
3368 (gchar *) suggested_name,
3372 if (result == GTK_RESPONSE_ACCEPT && parent_folder) {
3373 CreateFolderConnect *helper = (CreateFolderConnect *) g_slice_new0 (CreateFolderConnect);
3374 helper->folder_name = g_strdup (folder_name);
3375 helper->parent = g_object_ref (parent_folder);
3377 modest_platform_connect_if_remote_and_perform (GTK_WINDOW (parent_window),
3380 do_create_folder_performer,
3385 g_free (folder_name);
3387 g_object_unref (parent_folder);
3391 modest_ui_actions_create_folder(GtkWidget *parent_window,
3392 GtkWidget *folder_view)
3394 TnyFolderStore *parent_folder;
3396 #ifdef MODEST_TOOLKIT_HILDON2
3397 ModestTnyAccountStore *acc_store;
3399 acc_store = modest_runtime_get_account_store ();
3401 parent_folder = (TnyFolderStore *)
3402 modest_tny_account_store_get_local_folders_account (acc_store);
3404 parent_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW(folder_view));
3407 if (parent_folder) {
3408 do_create_folder (GTK_WINDOW (parent_window), parent_folder, NULL);
3409 g_object_unref (parent_folder);
3414 modest_ui_actions_on_new_folder (GtkAction *action, ModestWindow *window)
3417 g_return_if_fail (MODEST_IS_WINDOW(window));
3419 if (MODEST_IS_MAIN_WINDOW (window)) {
3420 GtkWidget *folder_view;
3422 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3423 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3427 modest_ui_actions_create_folder (GTK_WIDGET (window), folder_view);
3428 #ifdef MODEST_TOOLKIT_HILDON2
3429 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3430 GtkWidget *folder_view;
3432 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3433 modest_ui_actions_create_folder (GTK_WIDGET (window), folder_view);
3436 g_assert_not_reached ();
3441 modest_ui_actions_rename_folder_error_handler (ModestMailOperation *mail_op,
3444 const GError *error = NULL;
3445 const gchar *message = NULL;
3447 /* Get error message */
3448 error = modest_mail_operation_get_error (mail_op);
3450 g_return_if_reached ();
3452 if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
3453 error->code == MODEST_MAIL_OPERATION_ERROR_FOLDER_EXISTS) {
3454 message = _CS("ckdg_ib_folder_already_exists");
3455 } else if (error->domain == TNY_ERROR_DOMAIN &&
3456 error->code == TNY_SERVICE_ERROR_STATE) {
3457 /* This means that the folder is already in use (a
3458 message is opened for example */
3459 message = _("emev_ni_internal_error");
3461 message = _CS("ckdg_ib_unable_to_rename");
3464 /* We don't set a parent for the dialog because the dialog
3465 will be destroyed so the banner won't appear */
3466 modest_platform_information_banner (NULL, NULL, message);
3470 TnyFolderStore *folder;
3475 on_rename_folder_cb (ModestMailOperation *mail_op,
3476 TnyFolder *new_folder,
3479 ModestFolderView *folder_view;
3481 /* If the window was closed when renaming a folder, or if
3482 * it's not a main window this will happen */
3483 if (!MODEST_IS_FOLDER_VIEW (user_data))
3486 folder_view = MODEST_FOLDER_VIEW (user_data);
3487 /* Note that if the rename fails new_folder will be NULL */
3489 modest_folder_view_select_folder (folder_view, new_folder, FALSE);
3491 modest_folder_view_select_first_inbox_or_local (folder_view);
3493 gtk_widget_grab_focus (GTK_WIDGET (folder_view));
3497 on_rename_folder_performer (gboolean canceled,
3499 GtkWindow *parent_window,
3500 TnyAccount *account,
3503 ModestMailOperation *mail_op = NULL;
3504 GtkTreeSelection *sel = NULL;
3505 GtkWidget *folder_view = NULL;
3506 RenameFolderInfo *data = (RenameFolderInfo*)user_data;
3508 if (canceled || err) {
3509 /* In memory full conditions we could get this error here */
3510 check_memory_full_error ((GtkWidget *) parent_window, err);
3514 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
3515 modest_ui_actions_rename_folder_error_handler,
3516 parent_window, NULL);
3518 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3521 if (MODEST_IS_MAIN_WINDOW(parent_window)) {
3523 folder_view = modest_main_window_get_child_widget (
3524 MODEST_MAIN_WINDOW (parent_window),
3525 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3527 #ifdef MODEST_TOOLKIT_HILDON2
3528 else if (MODEST_IS_FOLDER_WINDOW (parent_window)) {
3529 ModestFolderWindow *folder_window = (ModestFolderWindow *) parent_window;
3530 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (folder_window));
3534 /* Clear the folders view */
3535 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_view));
3536 gtk_tree_selection_unselect_all (sel);
3538 /* Actually rename the folder */
3539 modest_mail_operation_rename_folder (mail_op,
3540 TNY_FOLDER (data->folder),
3541 (const gchar *) (data->new_name),
3542 on_rename_folder_cb,
3544 g_object_unref (mail_op);
3547 g_object_unref (data->folder);
3548 g_free (data->new_name);
3553 modest_ui_actions_on_rename_folder (GtkAction *action,
3554 ModestWindow *window)
3556 modest_ui_actions_on_edit_mode_rename_folder (window);
3560 modest_ui_actions_on_edit_mode_rename_folder (ModestWindow *window)
3562 TnyFolderStore *folder;
3563 GtkWidget *folder_view;
3564 gboolean do_rename = TRUE;
3566 g_return_val_if_fail (MODEST_IS_WINDOW(window), FALSE);
3568 if (MODEST_IS_MAIN_WINDOW (window)) {
3569 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3570 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3574 #ifdef MODEST_TOOLKIT_HILDON2
3575 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3576 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3582 folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW(folder_view));
3587 if (TNY_IS_FOLDER (folder)) {
3588 gchar *folder_name = NULL;
3590 const gchar *current_name;
3591 TnyFolderStore *parent;
3593 current_name = tny_folder_get_name (TNY_FOLDER (folder));
3594 parent = tny_folder_get_folder_store (TNY_FOLDER (folder));
3595 response = modest_platform_run_rename_folder_dialog (GTK_WINDOW (window),
3596 parent, current_name,
3598 g_object_unref (parent);
3600 if (response != GTK_RESPONSE_ACCEPT || strlen (folder_name) == 0) {
3603 RenameFolderInfo *rename_folder_data = g_new0 (RenameFolderInfo, 1);
3604 rename_folder_data->folder = g_object_ref (folder);
3605 rename_folder_data->new_name = folder_name;
3606 modest_platform_connect_if_remote_and_perform (GTK_WINDOW(window), TRUE,
3607 folder, on_rename_folder_performer, rename_folder_data);
3610 g_object_unref (folder);
3615 modest_ui_actions_delete_folder_error_handler (ModestMailOperation *mail_op,
3618 GObject *win = modest_mail_operation_get_source (mail_op);
3620 modest_platform_run_information_dialog ((win) ? GTK_WINDOW (win) : NULL,
3621 _("mail_in_ui_folder_delete_error"),
3623 g_object_unref (win);
3627 TnyFolderStore *folder;
3628 gboolean move_to_trash;
3632 on_delete_folder_cb (gboolean canceled,
3634 GtkWindow *parent_window,
3635 TnyAccount *account,
3638 DeleteFolderInfo *info = (DeleteFolderInfo*) user_data;
3639 GtkWidget *folder_view;
3640 ModestMailOperation *mail_op;
3641 GtkTreeSelection *sel;
3643 if (!MODEST_IS_WINDOW(parent_window) || canceled || (err!=NULL)) {
3644 g_object_unref (G_OBJECT (info->folder));
3649 if (MODEST_IS_MAIN_WINDOW (parent_window)) {
3650 folder_view = modest_main_window_get_child_widget (
3651 MODEST_MAIN_WINDOW (parent_window),
3652 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3653 #ifdef MODEST_TOOLKIT_HILDON2
3654 } else if (MODEST_IS_FOLDER_WINDOW (parent_window)) {
3655 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (parent_window)));
3658 g_object_unref (G_OBJECT (info->folder));
3663 /* Unselect the folder before deleting it to free the headers */
3664 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_view));
3665 gtk_tree_selection_unselect_all (sel);
3667 /* Create the mail operation */
3669 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
3670 modest_ui_actions_delete_folder_error_handler,
3673 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3675 modest_mail_operation_remove_folder (mail_op, TNY_FOLDER (info->folder), info->move_to_trash);
3677 modest_folder_view_select_first_inbox_or_local (MODEST_FOLDER_VIEW (folder_view));
3679 g_object_unref (G_OBJECT (mail_op));
3680 g_object_unref (G_OBJECT (info->folder));
3685 delete_folder (ModestWindow *window, gboolean move_to_trash)
3687 TnyFolderStore *folder;
3688 GtkWidget *folder_view;
3692 g_return_val_if_fail (MODEST_IS_WINDOW(window), FALSE);
3694 if (MODEST_IS_MAIN_WINDOW (window)) {
3696 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3697 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3698 #ifdef MODEST_TOOLKIT_HILDON2
3699 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3700 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3708 folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
3713 /* Show an error if it's an account */
3714 if (!TNY_IS_FOLDER (folder)) {
3715 modest_platform_run_information_dialog (GTK_WINDOW (window),
3716 _("mail_in_ui_folder_delete_error"),
3718 g_object_unref (G_OBJECT (folder));
3723 message = g_strdup_printf (_("mcen_nc_delete_folder_text"),
3724 tny_folder_get_name (TNY_FOLDER (folder)));
3725 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (window),
3726 (const gchar *) message);
3729 if (response == GTK_RESPONSE_OK) {
3730 DeleteFolderInfo *info;
3731 info = g_new0(DeleteFolderInfo, 1);
3732 info->folder = folder;
3733 info->move_to_trash = move_to_trash;
3734 g_object_ref (G_OBJECT (info->folder));
3735 TnyAccount *account = tny_folder_get_account (TNY_FOLDER (folder));
3736 modest_platform_connect_if_remote_and_perform (GTK_WINDOW (window),
3738 TNY_FOLDER_STORE (account),
3739 on_delete_folder_cb, info);
3740 g_object_unref (account);
3745 g_object_unref (G_OBJECT (folder));
3749 modest_ui_actions_on_delete_folder (GtkAction *action,
3750 ModestWindow *window)
3752 modest_ui_actions_on_edit_mode_delete_folder (window);
3756 modest_ui_actions_on_edit_mode_delete_folder (ModestWindow *window)
3758 g_return_val_if_fail (MODEST_IS_WINDOW(window), TRUE);
3760 return delete_folder (window, FALSE);
3764 modest_ui_actions_on_move_folder_to_trash_folder (GtkAction *action, ModestMainWindow *main_window)
3766 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
3768 delete_folder (MODEST_WINDOW (main_window), TRUE);
3772 typedef struct _PasswordDialogFields {
3773 GtkWidget *username;
3774 GtkWidget *password;
3776 } PasswordDialogFields;
3779 password_dialog_check_field (GtkEditable *editable,
3780 PasswordDialogFields *fields)
3783 gboolean any_value_empty = FALSE;
3785 #ifdef MODEST_TOOLKIT_HILDON2
3786 value = hildon_entry_get_text (HILDON_ENTRY (fields->username));
3788 value = gtk_entry_get_text (GTK_ENTRY (fields->username));
3790 if ((value == NULL) || value[0] == '\0') {
3791 any_value_empty = TRUE;
3793 #ifdef MODEST_TOOLKIT_HILDON2
3794 value = hildon_entry_get_text (HILDON_ENTRY (fields->password));
3796 value = gtk_entry_get_text (GTK_ENTRY (fields->password));
3798 if ((value == NULL) || value[0] == '\0') {
3799 any_value_empty = TRUE;
3801 gtk_dialog_set_response_sensitive (GTK_DIALOG (fields->dialog), GTK_RESPONSE_ACCEPT, !any_value_empty);
3805 modest_ui_actions_on_password_requested (TnyAccountStore *account_store,
3806 const gchar* server_account_name,
3811 ModestMainWindow *main_window)
3813 g_return_if_fail(server_account_name);
3814 gboolean completed = FALSE;
3815 PasswordDialogFields *fields = NULL;
3817 /* Initalize output parameters: */
3824 #ifndef MODEST_TOOLKIT_GTK
3825 /* Maemo uses a different (awkward) button order,
3826 * It should probably just use gtk_alternative_dialog_button_order ().
3828 #ifdef MODEST_TOOLKIT_HILDON2
3830 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3833 _HL("wdgt_bd_done"),
3834 GTK_RESPONSE_ACCEPT,
3836 gtk_container_set_border_width (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox),
3837 HILDON_MARGIN_DOUBLE);
3840 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3843 _("mcen_bd_dialog_ok"),
3844 GTK_RESPONSE_ACCEPT,
3845 _("mcen_bd_dialog_cancel"),
3846 GTK_RESPONSE_REJECT,
3848 #endif /* MODEST_TOOLKIT_HILDON2 */
3851 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3855 GTK_RESPONSE_REJECT,
3857 GTK_RESPONSE_ACCEPT,
3859 #endif /* MODEST_TOOLKIT_GTK */
3861 modest_window_mgr_set_modal (modest_runtime_get_window_mgr(), GTK_WINDOW (dialog), NULL);
3863 gchar *server_name = modest_account_mgr_get_server_account_hostname (
3864 modest_runtime_get_account_mgr(), server_account_name);
3865 if (!server_name) {/* This happened once, though I don't know why. murrayc. */
3866 g_warning("%s: Could not get server name for server account '%s'", __FUNCTION__, server_account_name);
3869 gtk_widget_destroy (dialog);
3873 gchar *txt = g_strdup_printf (_("mail_ia_password_info"), server_name);
3874 GtkWidget *label = gtk_label_new (txt);
3875 gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
3877 g_free (server_name);
3878 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), label,
3883 gchar *initial_username = modest_account_mgr_get_server_account_username (
3884 modest_runtime_get_account_mgr(), server_account_name);
3886 #ifdef MODEST_TOOLKIT_HILDON2
3887 GtkWidget *entry_username = hildon_entry_new (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH);
3888 if (initial_username)
3889 hildon_entry_set_text (HILDON_ENTRY (entry_username), initial_username);
3891 GtkWidget *entry_username = gtk_entry_new ();
3892 if (initial_username)
3893 gtk_entry_set_text (GTK_ENTRY (entry_username), initial_username);
3895 /* Dim this if a connection has ever succeeded with this username,
3896 * as per the UI spec: */
3897 /* const gboolean username_known = */
3898 /* modest_account_mgr_get_server_account_username_has_succeeded( */
3899 /* modest_runtime_get_account_mgr(), server_account_name); */
3900 /* gtk_widget_set_sensitive (entry_username, !username_known); */
3902 /* We drop the username sensitive code and disallow changing it here
3903 * as tinymail does not support really changing the username in the callback
3905 gtk_widget_set_sensitive (entry_username, FALSE);
3907 #ifndef MODEST_TOOLKIT_GTK
3908 /* Auto-capitalization is the default, so let's turn it off: */
3909 hildon_gtk_entry_set_input_mode (GTK_ENTRY (entry_username), HILDON_GTK_INPUT_MODE_FULL);
3911 /* Create a size group to be used by all captions.
3912 * Note that HildonCaption does not create a default size group if we do not specify one.
3913 * We use GTK_SIZE_GROUP_HORIZONTAL, so that the widths are the same. */
3914 GtkSizeGroup *sizegroup = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
3916 #ifdef MODEST_TOOLKIT_HILDON2
3917 GtkWidget *caption = modest_maemo_utils_create_captioned (sizegroup, NULL,
3918 _("mail_fi_username"), FALSE,
3921 GtkWidget *caption = hildon_caption_new (sizegroup,
3922 _("mail_fi_username"), entry_username, NULL, HILDON_CAPTION_MANDATORY);
3924 gtk_widget_show (entry_username);
3925 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), caption,
3926 FALSE, FALSE, MODEST_MARGIN_HALF);
3927 gtk_widget_show (caption);
3929 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), entry_username,
3931 #endif /* !MODEST_TOOLKIT_GTK */
3934 #ifdef MODEST_TOOLKIT_HILDON2
3935 GtkWidget *entry_password = hildon_entry_new (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH);
3937 GtkWidget *entry_password = gtk_entry_new ();
3939 gtk_entry_set_visibility (GTK_ENTRY(entry_password), FALSE);
3940 /* gtk_entry_set_invisible_char (GTK_ENTRY(entry_password), "*"); */
3942 #ifndef MODEST_TOOLKIT_GTK
3943 /* Auto-capitalization is the default, so let's turn it off: */
3944 hildon_gtk_entry_set_input_mode (GTK_ENTRY (entry_password),
3945 HILDON_GTK_INPUT_MODE_FULL | HILDON_GTK_INPUT_MODE_INVISIBLE);
3947 #ifdef MODEST_TOOLKIT_HILDON2
3948 caption = modest_maemo_utils_create_captioned (sizegroup, NULL,
3949 _("mail_fi_password"), FALSE,
3952 caption = hildon_caption_new (sizegroup,
3953 _("mail_fi_password"), entry_password, NULL, HILDON_CAPTION_MANDATORY);
3955 gtk_widget_show (entry_password);
3956 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), caption,
3957 FALSE, FALSE, MODEST_MARGIN_HALF);
3958 gtk_widget_show (caption);
3959 g_object_unref (sizegroup);
3961 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), entry_password,
3963 #endif /* !MODEST_TOOLKIT_GTK */
3965 if (initial_username != NULL)
3966 gtk_widget_grab_focus (GTK_WIDGET (entry_password));
3968 /* This is not in the Maemo UI spec:
3969 remember_pass_check = gtk_check_button_new_with_label (_("Remember password"));
3970 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), remember_pass_check,
3974 fields = g_slice_new0 (PasswordDialogFields);
3975 fields->username = entry_username;
3976 fields->password = entry_password;
3977 fields->dialog = dialog;
3979 g_signal_connect (entry_username, "changed", G_CALLBACK (password_dialog_check_field), fields);
3980 g_signal_connect (entry_password, "changed", G_CALLBACK (password_dialog_check_field), fields);
3981 password_dialog_check_field (NULL, fields);
3983 gtk_widget_show_all (GTK_WIDGET(GTK_DIALOG(dialog)->vbox));
3985 while (!completed) {
3987 if (gtk_dialog_run (GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
3989 #ifdef MODEST_TOOLKIT_HILDON2
3990 *username = g_strdup (hildon_entry_get_text (HILDON_ENTRY(entry_username)));
3992 *username = g_strdup (gtk_entry_get_text (GTK_ENTRY(entry_username)));
3995 /* Note that an empty field becomes the "" string */
3996 if (*username && strlen (*username) > 0) {
3997 modest_account_mgr_set_server_account_username (modest_runtime_get_account_mgr(),
3998 server_account_name,
4002 const gboolean username_was_changed =
4003 (strcmp (*username, initial_username) != 0);
4004 if (username_was_changed) {
4005 g_warning ("%s: tinymail does not yet support changing the "
4006 "username in the get_password() callback.\n", __FUNCTION__);
4012 modest_platform_information_banner (GTK_WIDGET (dialog), NULL,
4013 _("mcen_ib_username_pw_incorrect"));
4019 #ifdef MODEST_TOOLKIT_HILDON2
4020 *password = g_strdup (hildon_entry_get_text (HILDON_ENTRY(entry_password)));
4022 *password = g_strdup (gtk_entry_get_text (GTK_ENTRY(entry_password)));
4025 /* We do not save the password in the configuration,
4026 * because this function is only called for passwords that should
4027 * not be remembered:
4028 modest_server_account_set_password (
4029 modest_runtime_get_account_mgr(), server_account_name,
4036 #ifndef MODEST_TOOLKIT_HILDON2
4037 /* Set parent to NULL or the banner will disappear with its parent dialog */
4038 modest_platform_information_banner(NULL, NULL, _("mail_ib_login_cancelled"));
4050 /* This is not in the Maemo UI spec:
4051 if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (remember_pass_check)))
4057 g_free (initial_username);
4058 gtk_widget_destroy (dialog);
4059 g_slice_free (PasswordDialogFields, fields);
4061 /* printf ("DEBUG: %s: cancel=%d\n", __FUNCTION__, *cancel); */
4065 modest_ui_actions_on_cut (GtkAction *action,
4066 ModestWindow *window)
4068 GtkWidget *focused_widget;
4069 GtkClipboard *clipboard;
4071 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4072 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4073 if (GTK_IS_EDITABLE (focused_widget)) {
4074 gtk_editable_cut_clipboard (GTK_EDITABLE(focused_widget));
4075 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4076 gtk_clipboard_store (clipboard);
4077 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4078 GtkTextBuffer *buffer;
4080 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4081 if (modest_text_utils_buffer_selection_is_valid (buffer)) {
4082 gtk_text_buffer_cut_clipboard (buffer, clipboard, TRUE);
4083 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4084 gtk_clipboard_store (clipboard);
4086 } else if (MODEST_IS_HEADER_VIEW (focused_widget)) {
4087 TnyList *header_list = modest_header_view_get_selected_headers (
4088 MODEST_HEADER_VIEW (focused_widget));
4089 gboolean continue_download = FALSE;
4090 gint num_of_unc_msgs;
4092 num_of_unc_msgs = header_list_count_uncached_msgs(header_list);
4094 if (num_of_unc_msgs) {
4095 TnyAccount *account = get_account_from_header_list (header_list);
4097 continue_download = connect_to_get_msg (window, num_of_unc_msgs, account);
4098 g_object_unref (account);
4102 if (num_of_unc_msgs == 0 || continue_download) {
4103 /* modest_platform_information_banner (
4104 NULL, NULL, _CS("mcen_ib_getting_items"));*/
4105 modest_header_view_cut_selection (
4106 MODEST_HEADER_VIEW (focused_widget));
4109 g_object_unref (header_list);
4110 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4111 modest_folder_view_cut_selection (MODEST_FOLDER_VIEW (focused_widget));
4116 modest_ui_actions_on_copy (GtkAction *action,
4117 ModestWindow *window)
4119 GtkClipboard *clipboard;
4120 GtkWidget *focused_widget;
4121 gboolean copied = TRUE;
4123 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4124 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4126 if (GTK_IS_LABEL (focused_widget)) {
4128 selection = modest_text_utils_label_get_selection (GTK_LABEL (focused_widget));
4129 gtk_clipboard_set_text (clipboard, selection, -1);
4131 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4132 gtk_clipboard_store (clipboard);
4133 } else if (GTK_IS_EDITABLE (focused_widget)) {
4134 gtk_editable_copy_clipboard (GTK_EDITABLE(focused_widget));
4135 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4136 gtk_clipboard_store (clipboard);
4137 } else if (GTK_IS_HTML (focused_widget)) {
4140 sel = gtk_html_get_selection_html (GTK_HTML (focused_widget), &len);
4141 if ((sel == NULL) || (sel[0] == '\0')) {
4144 gtk_html_copy (GTK_HTML (focused_widget));
4145 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4146 gtk_clipboard_store (clipboard);
4148 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4149 GtkTextBuffer *buffer;
4150 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4151 if (modest_text_utils_buffer_selection_is_valid (buffer)) {
4152 gtk_text_buffer_copy_clipboard (buffer, clipboard);
4153 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4154 gtk_clipboard_store (clipboard);
4156 } else if (MODEST_IS_HEADER_VIEW (focused_widget)) {
4157 TnyList *header_list = modest_header_view_get_selected_headers (
4158 MODEST_HEADER_VIEW (focused_widget));
4159 gboolean continue_download = FALSE;
4160 gint num_of_unc_msgs;
4162 num_of_unc_msgs = header_list_count_uncached_msgs(header_list);
4164 if (num_of_unc_msgs) {
4165 TnyAccount *account = get_account_from_header_list (header_list);
4167 continue_download = connect_to_get_msg (window, num_of_unc_msgs, account);
4168 g_object_unref (account);
4172 if (num_of_unc_msgs == 0 || continue_download) {
4173 modest_platform_information_banner (
4174 NULL, NULL, _CS("mcen_ib_getting_items"));
4175 modest_header_view_copy_selection (
4176 MODEST_HEADER_VIEW (focused_widget));
4180 g_object_unref (header_list);
4182 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4183 modest_folder_view_copy_selection (MODEST_FOLDER_VIEW (focused_widget));
4186 /* Show information banner if there was a copy to clipboard */
4188 modest_platform_information_banner (
4189 NULL, NULL, _CS("ecoc_ib_edwin_copied"));
4193 modest_ui_actions_on_undo (GtkAction *action,
4194 ModestWindow *window)
4196 ModestEmailClipboard *clipboard = NULL;
4198 if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4199 modest_msg_edit_window_undo (MODEST_MSG_EDIT_WINDOW (window));
4200 } else if (MODEST_IS_MAIN_WINDOW (window)) {
4201 /* Clear clipboard source */
4202 clipboard = modest_runtime_get_email_clipboard ();
4203 modest_email_clipboard_clear (clipboard);
4206 g_return_if_reached ();
4211 modest_ui_actions_on_redo (GtkAction *action,
4212 ModestWindow *window)
4214 if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4215 modest_msg_edit_window_redo (MODEST_MSG_EDIT_WINDOW (window));
4218 g_return_if_reached ();
4224 destroy_information_note (ModestMailOperation *mail_op,
4227 /* destroy information note */
4228 gtk_widget_destroy (GTK_WIDGET(user_data));
4232 destroy_folder_information_note (ModestMailOperation *mail_op,
4233 TnyFolder *new_folder,
4236 /* destroy information note */
4237 gtk_widget_destroy (GTK_WIDGET(user_data));
4242 paste_as_attachment_free (gpointer data)
4244 PasteAsAttachmentHelper *helper = (PasteAsAttachmentHelper *) data;
4246 if (helper->banner) {
4247 gtk_widget_destroy (helper->banner);
4248 g_object_unref (helper->banner);
4254 paste_msg_as_attachment_cb (ModestMailOperation *mail_op,
4259 PasteAsAttachmentHelper *helper = (PasteAsAttachmentHelper *) userdata;
4260 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (helper->window));
4265 modest_msg_edit_window_add_part (MODEST_MSG_EDIT_WINDOW (helper->window), TNY_MIME_PART (msg));
4270 modest_ui_actions_on_paste (GtkAction *action,
4271 ModestWindow *window)
4273 GtkWidget *focused_widget = NULL;
4274 GtkWidget *inf_note = NULL;
4275 ModestMailOperation *mail_op = NULL;
4277 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4278 if (GTK_IS_EDITABLE (focused_widget)) {
4279 gtk_editable_paste_clipboard (GTK_EDITABLE(focused_widget));
4280 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4281 ModestEmailClipboard *e_clipboard = NULL;
4282 e_clipboard = modest_runtime_get_email_clipboard ();
4283 if (modest_email_clipboard_cleared (e_clipboard)) {
4284 GtkTextBuffer *buffer;
4285 GtkClipboard *clipboard;
4287 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4288 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4289 gtk_text_buffer_paste_clipboard (buffer, clipboard, NULL, TRUE);
4290 } else if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4291 ModestMailOperation *mail_op;
4292 TnyFolder *src_folder = NULL;
4293 TnyList *data = NULL;
4295 PasteAsAttachmentHelper *helper = g_new0 (PasteAsAttachmentHelper, 1);
4296 helper->window = MODEST_MSG_EDIT_WINDOW (window);
4297 helper->banner = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4298 _CS("ckct_nw_pasting"));
4299 modest_email_clipboard_get_data (e_clipboard, &src_folder, &data, &delete);
4300 mail_op = modest_mail_operation_new (G_OBJECT (window));
4301 if (helper->banner != NULL) {
4302 g_object_ref (G_OBJECT (helper->banner));
4303 gtk_widget_show (GTK_WIDGET (helper->banner));
4307 modest_mail_operation_get_msgs_full (mail_op,
4309 (GetMsgAsyncUserCallback) paste_msg_as_attachment_cb,
4311 paste_as_attachment_free);
4315 g_object_unref (data);
4317 g_object_unref (src_folder);
4320 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4321 ModestEmailClipboard *clipboard = NULL;
4322 TnyFolder *src_folder = NULL;
4323 TnyFolderStore *folder_store = NULL;
4324 TnyList *data = NULL;
4325 gboolean delete = FALSE;
4327 /* Check clipboard source */
4328 clipboard = modest_runtime_get_email_clipboard ();
4329 if (modest_email_clipboard_cleared (clipboard))
4332 /* Get elements to paste */
4333 modest_email_clipboard_get_data (clipboard, &src_folder, &data, &delete);
4335 /* Create a new mail operation */
4336 mail_op = modest_mail_operation_new (G_OBJECT(window));
4338 /* Get destination folder */
4339 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (focused_widget));
4341 /* transfer messages */
4345 /* Ask for user confirmation */
4347 modest_ui_actions_msgs_move_to_confirmation (window,
4348 TNY_FOLDER (folder_store),
4352 if (response == GTK_RESPONSE_OK) {
4353 /* Launch notification */
4354 inf_note = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4355 _CS("ckct_nw_pasting"));
4356 if (inf_note != NULL) {
4357 gtk_window_set_modal (GTK_WINDOW(inf_note), FALSE);
4358 gtk_widget_show (GTK_WIDGET(inf_note));
4361 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
4362 modest_mail_operation_xfer_msgs (mail_op,
4364 TNY_FOLDER (folder_store),
4366 destroy_information_note,
4369 g_object_unref (mail_op);
4372 } else if (src_folder != NULL) {
4373 /* Launch notification */
4374 inf_note = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4375 _CS("ckct_nw_pasting"));
4376 if (inf_note != NULL) {
4377 gtk_window_set_modal (GTK_WINDOW(inf_note), FALSE);
4378 gtk_widget_show (GTK_WIDGET(inf_note));
4381 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
4382 modest_mail_operation_xfer_folder (mail_op,
4386 destroy_folder_information_note,
4392 g_object_unref (data);
4393 if (src_folder != NULL)
4394 g_object_unref (src_folder);
4395 if (folder_store != NULL)
4396 g_object_unref (folder_store);
4402 modest_ui_actions_on_select_all (GtkAction *action,
4403 ModestWindow *window)
4405 GtkWidget *focused_widget;
4407 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4408 if (MODEST_IS_ATTACHMENTS_VIEW (focused_widget)) {
4409 modest_attachments_view_select_all (MODEST_ATTACHMENTS_VIEW (focused_widget));
4410 } else if (GTK_IS_LABEL (focused_widget)) {
4411 gtk_label_select_region (GTK_LABEL (focused_widget), 0, -1);
4412 } else if (GTK_IS_EDITABLE (focused_widget)) {
4413 gtk_editable_select_region (GTK_EDITABLE(focused_widget), 0, -1);
4414 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4415 GtkTextBuffer *buffer;
4416 GtkTextIter start, end;
4418 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4419 gtk_text_buffer_get_start_iter (buffer, &start);
4420 gtk_text_buffer_get_end_iter (buffer, &end);
4421 gtk_text_buffer_select_range (buffer, &start, &end);
4422 } else if (GTK_IS_HTML (focused_widget)) {
4423 gtk_html_select_all (GTK_HTML (focused_widget));
4424 } else if (MODEST_IS_MAIN_WINDOW (window)) {
4425 GtkWidget *header_view = focused_widget;
4426 GtkTreeSelection *selection = NULL;
4428 if (!(MODEST_IS_HEADER_VIEW (focused_widget))) {
4429 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
4430 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
4433 /* Disable window dimming management */
4434 modest_window_disable_dimming (MODEST_WINDOW(window));
4436 /* Select all messages */
4437 selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(header_view));
4438 gtk_tree_selection_select_all (selection);
4440 /* Set focuse on header view */
4441 gtk_widget_grab_focus (header_view);
4443 /* Enable window dimming management */
4444 modest_window_enable_dimming (MODEST_WINDOW(window));
4445 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (window));
4446 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (window));
4452 modest_ui_actions_on_mark_as_read (GtkAction *action,
4453 ModestWindow *window)
4455 g_return_if_fail (MODEST_IS_WINDOW(window));
4457 /* Mark each header as read */
4458 do_headers_action (window, headers_action_mark_as_read, NULL);
4462 modest_ui_actions_on_mark_as_unread (GtkAction *action,
4463 ModestWindow *window)
4465 g_return_if_fail (MODEST_IS_WINDOW(window));
4467 /* Mark each header as read */
4468 do_headers_action (window, headers_action_mark_as_unread, NULL);
4472 modest_ui_actions_on_change_zoom (GtkRadioAction *action,
4473 GtkRadioAction *selected,
4474 ModestWindow *window)
4478 value = gtk_radio_action_get_current_value (selected);
4479 if (MODEST_IS_WINDOW (window)) {
4480 modest_window_set_zoom (MODEST_WINDOW (window), ((gdouble)value)/100);
4485 modest_ui_actions_msg_edit_on_change_priority (GtkRadioAction *action,
4486 GtkRadioAction *selected,
4487 ModestWindow *window)
4489 TnyHeaderFlags flags;
4490 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4492 flags = gtk_radio_action_get_current_value (selected);
4493 modest_msg_edit_window_set_priority_flags (MODEST_MSG_EDIT_WINDOW (window), flags);
4497 modest_ui_actions_msg_edit_on_change_file_format (GtkRadioAction *action,
4498 GtkRadioAction *selected,
4499 ModestWindow *window)
4503 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4505 file_format = gtk_radio_action_get_current_value (selected);
4506 modest_msg_edit_window_set_file_format (MODEST_MSG_EDIT_WINDOW (window), file_format);
4511 modest_ui_actions_on_zoom_plus (GtkAction *action,
4512 ModestWindow *window)
4514 g_return_if_fail (MODEST_IS_WINDOW (window));
4516 modest_window_zoom_plus (MODEST_WINDOW (window));
4520 modest_ui_actions_on_zoom_minus (GtkAction *action,
4521 ModestWindow *window)
4523 g_return_if_fail (MODEST_IS_WINDOW (window));
4525 modest_window_zoom_minus (MODEST_WINDOW (window));
4529 modest_ui_actions_on_toggle_fullscreen (GtkToggleAction *toggle,
4530 ModestWindow *window)
4532 ModestWindowMgr *mgr;
4533 gboolean fullscreen, active;
4534 g_return_if_fail (MODEST_IS_WINDOW (window));
4536 mgr = modest_runtime_get_window_mgr ();
4538 active = (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (toggle)))?1:0;
4539 fullscreen = modest_window_mgr_get_fullscreen_mode (mgr);
4541 if (active != fullscreen) {
4542 modest_window_mgr_set_fullscreen_mode (mgr, active);
4543 #ifndef MODEST_TOOLKIT_HILDON2
4544 gtk_window_present (GTK_WINDOW (window));
4550 modest_ui_actions_on_change_fullscreen (GtkAction *action,
4551 ModestWindow *window)
4553 ModestWindowMgr *mgr;
4554 gboolean fullscreen;
4556 g_return_if_fail (MODEST_IS_WINDOW (window));
4558 mgr = modest_runtime_get_window_mgr ();
4559 fullscreen = modest_window_mgr_get_fullscreen_mode (mgr);
4560 modest_window_mgr_set_fullscreen_mode (mgr, !fullscreen);
4562 #ifndef MODEST_TOOLKIT_HILDON2
4563 gtk_window_present (GTK_WINDOW (window));
4568 * Used by modest_ui_actions_on_details to call do_headers_action
4571 headers_action_show_details (TnyHeader *header,
4572 ModestWindow *window,
4576 modest_platform_run_header_details_dialog (GTK_WINDOW (window), header);
4580 * Show the header details in a ModestDetailsDialog widget
4583 modest_ui_actions_on_details (GtkAction *action,
4586 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
4590 msg = modest_msg_view_window_get_message (MODEST_MSG_VIEW_WINDOW (win));
4594 header = tny_msg_get_header (msg);
4596 headers_action_show_details (header, win, NULL);
4597 g_object_unref (header);
4599 g_object_unref (msg);
4601 } else if (MODEST_IS_MAIN_WINDOW (win)) {
4602 GtkWidget *folder_view, *header_view;
4604 /* Check which widget has the focus */
4605 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
4606 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
4607 if (gtk_widget_is_focus (folder_view)) {
4608 TnyFolderStore *folder_store
4609 = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
4610 if (!folder_store) {
4611 g_warning ("%s: No item was selected.\n", __FUNCTION__);
4614 /* Show only when it's a folder */
4615 /* This function should not be called for account items,
4616 * because we dim the menu item for them. */
4617 if (TNY_IS_FOLDER (folder_store)) {
4618 modest_platform_run_folder_details_dialog (GTK_WINDOW (win),
4619 TNY_FOLDER (folder_store));
4622 g_object_unref (folder_store);
4625 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
4626 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
4627 /* Show details of each header */
4628 do_headers_action (win, headers_action_show_details, header_view);
4630 #ifdef MODEST_TOOLKIT_HILDON2
4631 } else if (MODEST_IS_HEADER_WINDOW (win)) {
4633 GtkWidget *header_view;
4635 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
4636 folder = modest_header_view_get_folder (MODEST_HEADER_VIEW (header_view));
4638 modest_platform_run_folder_details_dialog (GTK_WINDOW (win),
4640 g_object_unref (folder);
4647 modest_ui_actions_on_toggle_show_cc (GtkToggleAction *toggle,
4648 ModestMsgEditWindow *window)
4650 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4652 modest_msg_edit_window_show_cc (window, gtk_toggle_action_get_active (toggle));
4656 modest_ui_actions_on_toggle_show_bcc (GtkToggleAction *toggle,
4657 ModestMsgEditWindow *window)
4659 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4661 modest_msg_edit_window_show_bcc (window, gtk_toggle_action_get_active (toggle));
4665 modest_ui_actions_toggle_folders_view (GtkAction *action,
4666 ModestMainWindow *main_window)
4668 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
4670 if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)))
4671 modest_main_window_set_style (main_window, MODEST_MAIN_WINDOW_STYLE_SPLIT);
4673 modest_main_window_set_style (main_window, MODEST_MAIN_WINDOW_STYLE_SIMPLE);
4677 modest_ui_actions_on_toggle_toolbar (GtkToggleAction *toggle,
4678 ModestWindow *window)
4680 gboolean active, fullscreen = FALSE;
4681 ModestWindowMgr *mgr;
4683 active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (toggle));
4685 /* Check if we want to toggle the toolbar view in fullscreen
4687 if (!strcmp (gtk_action_get_name (GTK_ACTION (toggle)),
4688 "ViewShowToolbarFullScreen")) {
4692 /* Toggle toolbar */
4693 mgr = modest_runtime_get_window_mgr ();
4694 modest_window_mgr_show_toolbars (mgr, G_TYPE_FROM_INSTANCE (window), active, fullscreen);
4698 modest_ui_actions_msg_edit_on_select_font (GtkAction *action,
4699 ModestMsgEditWindow *window)
4701 modest_msg_edit_window_select_font (window);
4706 modest_ui_actions_on_folder_display_name_changed (ModestFolderView *folder_view,
4707 const gchar *display_name,
4710 /* don't update the display name if it was already set;
4711 * updating the display name apparently is expensive */
4712 const gchar* old_name = gtk_window_get_title (window);
4714 if (display_name == NULL)
4717 if (old_name && display_name && strcmp (old_name, display_name) == 0)
4718 return; /* don't do anything */
4720 /* This is usually used to change the title of the main window, which
4721 * is the one that holds the folder view. Note that this change can
4722 * happen even when the widget doesn't have the focus. */
4723 gtk_window_set_title (window, display_name);
4728 modest_ui_actions_on_select_contacts (GtkAction *action, ModestMsgEditWindow *window)
4730 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4731 modest_msg_edit_window_select_contacts (window);
4735 modest_ui_actions_on_check_names (GtkAction *action, ModestMsgEditWindow *window)
4737 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4738 modest_msg_edit_window_check_names (window, FALSE);
4741 #ifndef MODEST_TOOLKIT_HILDON2
4743 * This function is used to track changes in the selection of the
4744 * folder view that is inside the "move to" dialog to enable/disable
4745 * the OK button because we do not want the user to select a disallowed
4746 * destination for a folder.
4747 * The user also not desired to be able to use NEW button on items where
4748 * folder creation is not possibel.
4751 on_move_to_dialog_folder_selection_changed (ModestFolderView* self,
4752 TnyFolderStore *folder_store,
4756 GtkWidget *dialog = NULL;
4757 gboolean ok_sensitive = TRUE, new_sensitive = TRUE;
4758 gboolean moving_folder = FALSE;
4759 gboolean is_local_account = TRUE;
4760 GtkWidget *folder_view = NULL;
4761 ModestTnyFolderRules rules;
4763 g_return_if_fail (MODEST_IS_FOLDER_VIEW(self));
4768 dialog = gtk_widget_get_ancestor (GTK_WIDGET (self), GTK_TYPE_DIALOG);
4772 /* check if folder_store is an remote account */
4773 if (TNY_IS_ACCOUNT (folder_store)) {
4774 TnyAccount *local_account = NULL;
4775 TnyAccount *mmc_account = NULL;
4776 ModestTnyAccountStore *account_store = NULL;
4778 account_store = modest_runtime_get_account_store ();
4779 local_account = modest_tny_account_store_get_local_folders_account (account_store);
4780 mmc_account = modest_tny_account_store_get_mmc_folders_account (account_store);
4782 if ((gpointer) local_account != (gpointer) folder_store &&
4783 (gpointer) mmc_account != (gpointer) folder_store) {
4784 ModestProtocolType proto;
4785 proto = modest_tny_account_get_protocol_type (TNY_ACCOUNT (folder_store));
4786 if (proto == MODEST_PROTOCOL_REGISTRY_TYPE_INVALID) {
4787 proto = MODEST_PROTOCOLS_STORE_MAILDIR;
4789 is_local_account = FALSE;
4790 /* New button should be dimmed on remote
4792 new_sensitive = (modest_protocol_registry_protocol_type_has_tag (modest_runtime_get_protocol_registry (),
4794 MODEST_PROTOCOL_REGISTRY_STORE_HAS_FOLDERS));
4796 g_object_unref (local_account);
4798 /* It could not exist */
4800 g_object_unref (mmc_account);
4803 /* Check the target folder rules */
4804 if (TNY_IS_FOLDER (folder_store)) {
4805 rules = modest_tny_folder_get_rules (TNY_FOLDER (folder_store));
4806 if (rules & MODEST_FOLDER_RULES_FOLDER_NON_WRITEABLE) {
4807 ok_sensitive = FALSE;
4808 new_sensitive = FALSE;
4813 /* Check if we're moving a folder */
4814 if (MODEST_IS_MAIN_WINDOW (user_data)) {
4815 /* Get the widgets */
4816 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (user_data),
4817 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
4818 if (gtk_widget_is_focus (folder_view))
4819 moving_folder = TRUE;
4822 if (moving_folder) {
4823 TnyFolderStore *moved_folder = NULL, *parent = NULL;
4825 /* Get the folder to move */
4826 moved_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
4828 /* Check that we're not moving to the same folder */
4829 if (TNY_IS_FOLDER (moved_folder)) {
4830 parent = tny_folder_get_folder_store (TNY_FOLDER (moved_folder));
4831 if (parent == folder_store)
4832 ok_sensitive = FALSE;
4833 g_object_unref (parent);
4836 if (ok_sensitive && TNY_IS_ACCOUNT (folder_store)) {
4837 /* Do not allow to move to an account unless it's the
4838 local folders account */
4839 if (!is_local_account)
4840 ok_sensitive = FALSE;
4843 if (ok_sensitive && (moved_folder == folder_store)) {
4844 /* Do not allow to move to itself */
4845 ok_sensitive = FALSE;
4847 g_object_unref (moved_folder);
4849 TnyFolder *src_folder = NULL;
4851 /* Moving a message */
4852 if (MODEST_IS_MSG_VIEW_WINDOW (user_data)) {
4854 TnyHeader *header = NULL;
4855 header = modest_msg_view_window_get_header
4856 (MODEST_MSG_VIEW_WINDOW (user_data));
4857 if (!TNY_IS_HEADER(header))
4858 g_warning ("%s: could not get source header", __FUNCTION__);
4860 src_folder = tny_header_get_folder (header);
4863 g_object_unref (header);
4866 TNY_FOLDER (modest_folder_view_get_selected
4867 (MODEST_FOLDER_VIEW (folder_view)));
4870 if (TNY_IS_FOLDER(src_folder)) {
4871 /* Do not allow to move the msg to the same folder */
4872 /* Do not allow to move the msg to an account */
4873 if ((gpointer) src_folder == (gpointer) folder_store ||
4874 TNY_IS_ACCOUNT (folder_store))
4875 ok_sensitive = FALSE;
4876 g_object_unref (src_folder);
4878 g_warning ("%s: could not get source folder", __FUNCTION__);
4882 /* Set sensitivity of the OK and NEW button */
4883 gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), GTK_RESPONSE_OK, ok_sensitive);
4884 gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), MODEST_GTK_RESPONSE_NEW_FOLDER, new_sensitive);
4889 on_move_to_dialog_response (GtkDialog *dialog,
4893 GtkWidget *parent_win;
4894 MoveToInfo *helper = NULL;
4895 ModestFolderView *folder_view;
4897 helper = (MoveToInfo *) user_data;
4899 parent_win = (GtkWidget *) helper->win;
4900 folder_view = MODEST_FOLDER_VIEW (g_object_get_data (G_OBJECT (dialog),
4901 MODEST_MOVE_TO_DIALOG_FOLDER_VIEW));
4904 TnyFolderStore *dst_folder;
4906 case MODEST_GTK_RESPONSE_NEW_FOLDER:
4907 modest_ui_actions_create_folder (GTK_WIDGET (dialog), GTK_WIDGET (folder_view));
4909 case GTK_RESPONSE_NONE:
4910 case GTK_RESPONSE_CANCEL:
4911 case GTK_RESPONSE_DELETE_EVENT:
4913 case GTK_RESPONSE_OK:
4914 dst_folder = modest_folder_view_get_selected (folder_view);
4916 if (MODEST_IS_MAIN_WINDOW (parent_win)) {
4917 /* Clean list to move used for filtering */
4918 modest_folder_view_set_list_to_move (folder_view, NULL);
4920 modest_ui_actions_on_main_window_move_to (NULL,
4921 GTK_WIDGET (folder_view),
4923 MODEST_MAIN_WINDOW (parent_win));
4924 #ifdef MODEST_TOOLKIT_HILDON2
4925 } else if (MODEST_IS_FOLDER_WINDOW (parent_win)) {
4926 /* Clean list to move used for filtering */
4927 modest_folder_view_set_list_to_move (folder_view, NULL);
4929 modest_ui_actions_on_folder_window_move_to (GTK_WIDGET (folder_view),
4932 GTK_WINDOW (parent_win));
4935 /* if the user selected a root folder
4936 (account) then do not perform any action */
4937 if (TNY_IS_ACCOUNT (dst_folder)) {
4938 g_signal_stop_emission_by_name (dialog, "response");
4942 /* Clean list to move used for filtering */
4943 modest_folder_view_set_list_to_move (folder_view, NULL);
4945 /* Moving from headers window in edit mode */
4946 modest_ui_actions_on_window_move_to (NULL, helper->list,
4948 MODEST_WINDOW (parent_win));
4952 g_object_unref (dst_folder);
4956 g_warning ("%s unexpected response id %d", __FUNCTION__, response);
4959 /* Free the helper and exit */
4961 g_object_unref (helper->list);
4962 g_slice_free (MoveToInfo, helper);
4963 gtk_widget_destroy (GTK_WIDGET (dialog));
4967 create_move_to_dialog (GtkWindow *win,
4968 GtkWidget *folder_view,
4969 TnyList *list_to_move)
4971 GtkWidget *dialog, *tree_view = NULL;
4973 dialog = modest_platform_create_move_to_dialog (win, &tree_view);
4975 #ifndef MODEST_TOOLKIT_HILDON2
4976 /* Track changes in the selection to
4977 * disable the OK button whenever "Move to" is not possible
4978 * disbale NEW button whenever New is not possible */
4979 g_signal_connect (tree_view,
4980 "folder_selection_changed",
4981 G_CALLBACK (on_move_to_dialog_folder_selection_changed),
4985 /* It could happen that we're trying to move a message from a
4986 window (msg window for example) after the main window was
4987 closed, so we can not just get the model of the folder
4989 if (MODEST_IS_FOLDER_VIEW (folder_view)) {
4990 const gchar *visible_id = NULL;
4992 modest_folder_view_set_style (MODEST_FOLDER_VIEW (tree_view),
4993 MODEST_FOLDER_VIEW_STYLE_SHOW_ALL);
4994 modest_folder_view_copy_model (MODEST_FOLDER_VIEW(folder_view),
4995 MODEST_FOLDER_VIEW(tree_view));
4998 modest_folder_view_get_account_id_of_visible_server_account (MODEST_FOLDER_VIEW(folder_view));
5000 /* Show the same account than the one that is shown in the main window */
5001 modest_folder_view_set_account_id_of_visible_server_account (MODEST_FOLDER_VIEW(tree_view),
5004 const gchar *active_account_name = NULL;
5005 ModestAccountMgr *mgr = NULL;
5006 ModestAccountSettings *settings = NULL;
5007 ModestServerAccountSettings *store_settings = NULL;
5009 modest_folder_view_set_style (MODEST_FOLDER_VIEW (tree_view),
5010 MODEST_FOLDER_VIEW_STYLE_SHOW_ALL);
5011 modest_folder_view_update_model (MODEST_FOLDER_VIEW (tree_view),
5012 TNY_ACCOUNT_STORE (modest_runtime_get_account_store ()));
5014 active_account_name = modest_window_get_active_account (MODEST_WINDOW (win));
5015 mgr = modest_runtime_get_account_mgr ();
5016 settings = modest_account_mgr_load_account_settings (mgr, active_account_name);
5019 const gchar *store_account_name;
5020 store_settings = modest_account_settings_get_store_settings (settings);
5021 store_account_name = modest_server_account_settings_get_account_name (store_settings);
5023 modest_folder_view_set_account_id_of_visible_server_account (MODEST_FOLDER_VIEW (tree_view),
5024 store_account_name);
5025 g_object_unref (store_settings);
5026 g_object_unref (settings);
5030 /* we keep a pointer to the embedded folder view, so we can
5031 * retrieve it with get_folder_view_from_move_to_dialog (see
5032 * above) later (needed for focus handling)
5034 g_object_set_data (G_OBJECT(dialog), MODEST_MOVE_TO_DIALOG_FOLDER_VIEW, tree_view);
5036 /* Hide special folders */
5037 modest_folder_view_show_non_move_folders (MODEST_FOLDER_VIEW (tree_view), FALSE);
5039 modest_folder_view_set_list_to_move (MODEST_FOLDER_VIEW (tree_view), list_to_move);
5040 #ifndef MODEST_TOOLKIT_HILDON2
5041 modest_folder_view_select_first_inbox_or_local (MODEST_FOLDER_VIEW (tree_view));
5044 gtk_widget_show (GTK_WIDGET (tree_view));
5050 * Shows a confirmation dialog to the user when we're moving messages
5051 * from a remote server to the local storage. Returns the dialog
5052 * response. If it's other kind of movement then it always returns
5055 * This one is used by the next functions:
5056 * modest_ui_actions_on_paste - commented out
5057 * drag_and_drop_from_header_view (for d&d in modest_folder_view.c)
5060 modest_ui_actions_msgs_move_to_confirmation (ModestWindow *win,
5061 TnyFolder *dest_folder,
5065 gint response = GTK_RESPONSE_OK;
5066 TnyAccount *account = NULL;
5067 TnyFolder *src_folder = NULL;
5068 TnyIterator *iter = NULL;
5069 TnyHeader *header = NULL;
5071 /* return with OK if the destination is a remote folder */
5072 if (modest_tny_folder_is_remote_folder (dest_folder))
5073 return GTK_RESPONSE_OK;
5075 /* Get source folder */
5076 iter = tny_list_create_iterator (headers);
5077 header = TNY_HEADER (tny_iterator_get_current (iter));
5079 src_folder = tny_header_get_folder (header);
5080 g_object_unref (header);
5082 g_object_unref (iter);
5084 /* if no src_folder, message may be an attahcment */
5085 if (src_folder == NULL)
5086 return GTK_RESPONSE_CANCEL;
5088 /* If the source is a local or MMC folder */
5089 if (!modest_tny_folder_is_remote_folder (src_folder)) {
5090 g_object_unref (src_folder);
5091 return GTK_RESPONSE_OK;
5094 /* Get the account */
5095 account = tny_folder_get_account (src_folder);
5097 /* now if offline we ask the user */
5098 if(connect_to_get_msg (win, tny_list_get_length (headers), account))
5099 response = GTK_RESPONSE_OK;
5101 response = GTK_RESPONSE_CANCEL;
5104 g_object_unref (src_folder);
5105 g_object_unref (account);
5111 move_to_helper_destroyer (gpointer user_data)
5113 MoveToHelper *helper = (MoveToHelper *) user_data;
5115 /* Close the "Pasting" information banner */
5116 if (helper->banner) {
5117 gtk_widget_destroy (GTK_WIDGET (helper->banner));
5118 g_object_unref (helper->banner);
5120 if (gtk_tree_row_reference_valid (helper->reference)) {
5121 gtk_tree_row_reference_free (helper->reference);
5122 helper->reference = NULL;
5128 move_to_cb (ModestMailOperation *mail_op,
5131 MoveToHelper *helper = (MoveToHelper *) user_data;
5132 GObject *object = modest_mail_operation_get_source (mail_op);
5134 /* Note that the operation could have failed, in that case do
5136 if (modest_mail_operation_get_status (mail_op) !=
5137 MODEST_MAIL_OPERATION_STATUS_SUCCESS)
5140 if (MODEST_IS_MSG_VIEW_WINDOW (object)) {
5141 ModestMsgViewWindow *self = MODEST_MSG_VIEW_WINDOW (object);
5143 if (!modest_msg_view_window_select_next_message (self) &&
5144 !modest_msg_view_window_select_previous_message (self)) {
5145 /* No more messages to view, so close this window */
5146 modest_ui_actions_on_close_window (NULL, MODEST_WINDOW(self));
5148 } else if (MODEST_IS_MAIN_WINDOW (object) &&
5149 gtk_tree_row_reference_valid (helper->reference)) {
5150 GtkWidget *header_view;
5152 GtkTreeSelection *sel;
5154 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(object),
5155 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5156 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (header_view));
5157 path = gtk_tree_row_reference_get_path (helper->reference);
5158 /* We need to unselect the previous one
5159 because we could be copying instead of
5161 gtk_tree_selection_unselect_all (sel);
5162 gtk_tree_selection_select_path (sel, path);
5163 gtk_tree_path_free (path);
5165 g_object_unref (object);
5168 /* Destroy the helper */
5169 move_to_helper_destroyer (helper);
5173 folder_move_to_cb (ModestMailOperation *mail_op,
5174 TnyFolder *new_folder,
5177 GtkWidget *folder_view;
5180 object = modest_mail_operation_get_source (mail_op);
5181 if (MODEST_IS_MAIN_WINDOW (object)) {
5182 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(object),
5183 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
5184 g_object_ref (folder_view);
5185 g_object_unref (object);
5186 move_to_cb (mail_op, user_data);
5187 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view), new_folder, FALSE);
5188 g_object_unref (folder_view);
5190 move_to_cb (mail_op, user_data);
5195 msgs_move_to_cb (ModestMailOperation *mail_op,
5198 move_to_cb (mail_op, user_data);
5202 modest_ui_actions_move_folder_error_handler (ModestMailOperation *mail_op,
5205 GObject *win = NULL;
5207 #ifndef MODEST_TOOLKIT_HILDON2
5208 ModestWindow *main_window = NULL;
5210 /* Disable next automatic folder selection */
5211 main_window = modest_window_mgr_get_main_window (modest_runtime_get_window_mgr (),
5212 FALSE); /* don't create */
5214 GtkWidget *folder_view = NULL;
5216 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (main_window),
5217 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
5218 modest_folder_view_disable_next_folder_selection (MODEST_FOLDER_VIEW(folder_view));
5220 if (user_data && TNY_IS_FOLDER (user_data)) {
5221 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view),
5222 TNY_FOLDER (user_data), FALSE);
5226 /* Show notification dialog only if the main window exists */
5227 win = modest_mail_operation_get_source (mail_op);
5228 modest_platform_run_information_dialog ((GtkWindow *) win,
5229 _("mail_in_ui_folder_move_target_error"),
5232 g_object_unref (win);
5236 open_msg_for_purge_cb (ModestMailOperation *mail_op,
5245 gint pending_purges = 0;
5246 gboolean some_purged = FALSE;
5247 ModestWindow *win = MODEST_WINDOW (user_data);
5248 ModestWindowMgr *mgr = modest_runtime_get_window_mgr ();
5250 /* If there was any error */
5251 if (!modest_ui_actions_msg_retrieval_check (mail_op, header, msg)) {
5252 modest_window_mgr_unregister_header (mgr, header);
5256 /* Once the message has been retrieved for purging, we check if
5257 * it's all ok for purging */
5259 parts = tny_simple_list_new ();
5260 tny_mime_part_get_parts (TNY_MIME_PART (msg), parts);
5261 iter = tny_list_create_iterator (parts);
5263 while (!tny_iterator_is_done (iter)) {
5265 part = TNY_MIME_PART (tny_iterator_get_current (iter));
5266 if (part && (tny_mime_part_is_attachment (part) || TNY_IS_MSG (part))) {
5267 if (tny_mime_part_is_purged (part))
5274 g_object_unref (part);
5276 tny_iterator_next (iter);
5278 g_object_unref (iter);
5281 if (pending_purges>0) {
5283 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),_("mcen_nc_purge_file_text_inbox"));
5285 if (response == GTK_RESPONSE_OK) {
5288 modest_platform_animation_banner (GTK_WIDGET (win), NULL, _("mcen_me_inbox_remove_attachments"));
5289 iter = tny_list_create_iterator (parts);
5290 while (!tny_iterator_is_done (iter)) {
5293 part = TNY_MIME_PART (tny_iterator_get_current (iter));
5294 if (part && (tny_mime_part_is_attachment (part) || TNY_IS_MSG (part)))
5295 tny_mime_part_set_purged (part);
5298 g_object_unref (part);
5300 tny_iterator_next (iter);
5302 g_object_unref (iter);
5304 tny_msg_rewrite_cache (msg);
5306 gtk_widget_destroy (info);
5310 modest_window_mgr_unregister_header (mgr, header);
5312 g_object_unref (parts);
5316 modest_ui_actions_on_main_window_remove_attachments (GtkAction *action,
5317 ModestMainWindow *win)
5319 GtkWidget *header_view;
5320 TnyList *header_list;
5322 TnyHeaderFlags flags;
5323 ModestWindow *msg_view_window = NULL;
5326 g_return_if_fail (MODEST_IS_MAIN_WINDOW (win));
5328 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
5329 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5331 header_list = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
5333 g_warning ("%s: no header selected", __FUNCTION__);
5337 if (tny_list_get_length (header_list) == 1) {
5338 TnyIterator *iter = tny_list_create_iterator (header_list);
5339 header = TNY_HEADER (tny_iterator_get_current (iter));
5340 g_object_unref (iter);
5344 if (!header || !TNY_IS_HEADER(header)) {
5345 g_warning ("%s: header is not valid", __FUNCTION__);
5349 found = modest_window_mgr_find_registered_header (modest_runtime_get_window_mgr (),
5350 header, &msg_view_window);
5351 flags = tny_header_get_flags (header);
5352 if (!(flags & TNY_HEADER_FLAG_CACHED))
5355 if (msg_view_window != NULL)
5356 modest_msg_view_window_remove_attachments (MODEST_MSG_VIEW_WINDOW (msg_view_window), TRUE);
5358 /* do nothing; uid was registered before, so window is probably on it's way */
5359 g_warning ("debug: header %p has already been registered", header);
5362 ModestMailOperation *mail_op = NULL;
5363 modest_window_mgr_register_header (modest_runtime_get_window_mgr (), header, NULL);
5364 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT (win),
5365 modest_ui_actions_disk_operations_error_handler,
5367 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
5368 modest_mail_operation_get_msg (mail_op, header, FALSE, open_msg_for_purge_cb, win);
5370 g_object_unref (mail_op);
5373 g_object_unref (header);
5375 g_object_unref (header_list);
5379 * Checks if we need a connection to do the transfer and if the user
5380 * wants to connect to complete it
5383 modest_ui_actions_xfer_messages_check (GtkWindow *parent_window,
5384 TnyFolderStore *src_folder,
5386 TnyFolder *dst_folder,
5387 gboolean delete_originals,
5388 gboolean *need_connection,
5391 TnyAccount *src_account;
5392 gint uncached_msgs = 0;
5394 /* We don't need any further check if
5396 * 1- the source folder is local OR
5397 * 2- the device is already online
5399 if (!modest_tny_folder_store_is_remote (src_folder) ||
5400 tny_device_is_online (modest_runtime_get_device())) {
5401 *need_connection = FALSE;
5406 /* We must ask for a connection when
5408 * - the message(s) is not already cached OR
5409 * - the message(s) is cached but the leave_on_server setting
5410 * is FALSE (because we need to sync the source folder to
5411 * delete the message from the server (for IMAP we could do it
5412 * offline, it'll take place the next time we get a
5415 uncached_msgs = header_list_count_uncached_msgs (headers);
5416 src_account = get_account_from_folder_store (src_folder);
5417 if (uncached_msgs > 0) {
5421 *need_connection = TRUE;
5422 num_headers = tny_list_get_length (headers);
5423 msg = ngettext ("mcen_nc_get_msg", "mcen_nc_get_msgs", num_headers);
5425 if (modest_platform_run_confirmation_dialog (parent_window, msg) ==
5426 GTK_RESPONSE_CANCEL) {
5432 /* The transfer is possible and the user wants to */
5435 if (remote_folder_has_leave_on_server (src_folder) && delete_originals) {
5436 const gchar *account_name;
5437 gboolean leave_on_server;
5439 account_name = modest_tny_account_get_parent_modest_account_name_for_server_account (src_account);
5440 leave_on_server = modest_account_mgr_get_leave_on_server (modest_runtime_get_account_mgr (),
5443 if (leave_on_server == TRUE) {
5444 *need_connection = FALSE;
5446 *need_connection = TRUE;
5449 *need_connection = FALSE;
5454 g_object_unref (src_account);
5458 xfer_messages_error_handler (ModestMailOperation *mail_op,
5462 const GError *error;
5464 win = modest_mail_operation_get_source (mail_op);
5465 error = modest_mail_operation_get_error (mail_op);
5467 if (error && is_memory_full_error ((GError *) error, mail_op))
5468 modest_platform_information_banner ((GtkWidget *) win,
5469 NULL, _KR("cerm_device_memory_full"));
5471 modest_platform_run_information_dialog ((GtkWindow *) win,
5472 _("mail_in_ui_folder_move_target_error"),
5475 g_object_unref (win);
5479 TnyFolderStore *dst_folder;
5484 * Utility function that transfer messages from both the main window
5485 * and the msg view window when using the "Move to" dialog
5488 xfer_messages_performer (gboolean canceled,
5490 GtkWindow *parent_window,
5491 TnyAccount *account,
5494 ModestWindow *win = MODEST_WINDOW (parent_window);
5495 TnyAccount *dst_account = NULL;
5496 gboolean dst_forbids_message_add = FALSE;
5497 XferMsgsHelper *helper;
5498 MoveToHelper *movehelper;
5499 ModestMailOperation *mail_op;
5501 helper = (XferMsgsHelper *) user_data;
5503 if (canceled || err) {
5504 if (!check_memory_full_error ((GtkWidget *) parent_window, err)) {
5505 /* Show the proper error message */
5506 modest_ui_actions_on_account_connection_error (parent_window, account);
5511 dst_account = tny_folder_get_account (TNY_FOLDER (helper->dst_folder));
5513 /* tinymail will return NULL for local folders it seems */
5514 dst_forbids_message_add = modest_protocol_registry_protocol_type_has_tag (modest_runtime_get_protocol_registry (),
5515 modest_tny_account_get_protocol_type (dst_account),
5516 MODEST_PROTOCOL_REGISTRY_STORE_FORBID_MESSAGE_ADD);
5517 g_object_unref (dst_account);
5519 if (dst_forbids_message_add) {
5520 modest_platform_information_banner (GTK_WIDGET (win),
5522 ngettext("mail_in_ui_folder_move_target_error",
5523 "mail_in_ui_folder_move_targets_error",
5524 tny_list_get_length (helper->headers)));
5528 movehelper = g_new0 (MoveToHelper, 1);
5530 #ifndef MODEST_TOOLKIT_HILDON2
5531 movehelper->banner = modest_platform_animation_banner (GTK_WIDGET (win), NULL,
5532 _CS("ckct_nw_pasting"));
5533 if (movehelper->banner != NULL) {
5534 g_object_ref (movehelper->banner);
5535 gtk_widget_show (GTK_WIDGET (movehelper->banner));
5539 if (MODEST_IS_MAIN_WINDOW (win)) {
5540 GtkWidget *header_view =
5541 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
5542 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5543 movehelper->reference = get_next_after_selected_headers (MODEST_HEADER_VIEW (header_view));
5546 /* Perform the mail operation */
5547 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT(win),
5548 xfer_messages_error_handler,
5550 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
5553 modest_mail_operation_xfer_msgs (mail_op,
5555 TNY_FOLDER (helper->dst_folder),
5560 g_object_unref (G_OBJECT (mail_op));
5562 g_object_unref (helper->dst_folder);
5563 g_object_unref (helper->headers);
5564 g_slice_free (XferMsgsHelper, helper);
5568 TnyFolder *src_folder;
5569 TnyFolderStore *dst_folder;
5570 gboolean delete_original;
5571 GtkWidget *folder_view;
5575 on_move_folder_cb (gboolean canceled, GError *err, GtkWindow *parent_window,
5576 TnyAccount *account, gpointer user_data)
5578 MoveFolderInfo *info = (MoveFolderInfo*)user_data;
5579 GtkTreeSelection *sel;
5580 ModestMailOperation *mail_op = NULL;
5582 if (canceled || err || !MODEST_IS_WINDOW (parent_window)) {
5583 g_object_unref (G_OBJECT (info->src_folder));
5584 g_object_unref (G_OBJECT (info->dst_folder));
5589 MoveToHelper *helper = g_new0 (MoveToHelper, 1);
5590 #ifndef MODEST_TOOLKIT_HILDON2
5591 helper->banner = modest_platform_animation_banner (GTK_WIDGET (parent_window), NULL,
5592 _CS("ckct_nw_pasting"));
5593 if (helper->banner != NULL) {
5594 g_object_ref (helper->banner);
5595 gtk_widget_show (GTK_WIDGET(helper->banner));
5598 /* Clean folder on header view before moving it */
5599 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (info->folder_view));
5600 gtk_tree_selection_unselect_all (sel);
5602 /* Let gtk events run. We need that the folder
5603 view frees its reference to the source
5604 folder *before* issuing the mail operation
5605 so we need the signal handler of selection
5606 changed to happen before the mail
5608 while (gtk_events_pending ())
5609 gtk_main_iteration (); */
5612 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
5613 modest_ui_actions_move_folder_error_handler,
5614 info->src_folder, NULL);
5615 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
5618 /* Select *after* the changes */
5619 /* TODO: this function hangs UI after transfer */
5620 /* modest_folder_view_select_folder (MODEST_FOLDER_VIEW(folder_view), */
5621 /* TNY_FOLDER (src_folder), TRUE); */
5623 if (MODEST_IS_MAIN_WINDOW (parent_window)) {
5624 modest_folder_view_select_folder (MODEST_FOLDER_VIEW(info->folder_view),
5625 TNY_FOLDER (info->dst_folder), TRUE);
5627 modest_mail_operation_xfer_folder (mail_op,
5628 TNY_FOLDER (info->src_folder),
5630 info->delete_original,
5633 g_object_unref (G_OBJECT (info->src_folder));
5635 /* if (modest_mail_operation_get_status (mail_op) == MODEST_MAIL_OPERATION_STATUS_SUCCESS) { */
5638 /* Unref mail operation */
5639 g_object_unref (G_OBJECT (mail_op));
5640 g_object_unref (G_OBJECT (info->dst_folder));
5645 get_account_from_folder_store (TnyFolderStore *folder_store)
5647 if (TNY_IS_ACCOUNT (folder_store))
5648 return g_object_ref (folder_store);
5650 return tny_folder_get_account (TNY_FOLDER (folder_store));
5654 * UI handler for the "Move to" action when invoked from the
5658 modest_ui_actions_on_main_window_move_to (GtkAction *action,
5659 GtkWidget *folder_view,
5660 TnyFolderStore *dst_folder,
5661 ModestMainWindow *win)
5663 ModestHeaderView *header_view = NULL;
5664 TnyFolderStore *src_folder = NULL;
5666 g_return_if_fail (MODEST_IS_MAIN_WINDOW (win));
5668 /* Get the source folder */
5669 src_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
5671 /* Get header view */
5672 header_view = (ModestHeaderView *)
5673 modest_main_window_get_child_widget (win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5675 /* Get folder or messages to transfer */
5676 if (gtk_widget_is_focus (folder_view)) {
5677 gboolean do_xfer = TRUE;
5679 /* Allow only to transfer folders to the local root folder */
5680 if (TNY_IS_ACCOUNT (dst_folder) &&
5681 !MODEST_IS_TNY_LOCAL_FOLDERS_ACCOUNT (dst_folder) &&
5682 !modest_tny_account_is_memory_card_account (TNY_ACCOUNT (dst_folder))) {
5684 } else if (!TNY_IS_FOLDER (src_folder)) {
5685 g_warning ("%s: src_folder is not a TnyFolder.\n", __FUNCTION__);
5690 MoveFolderInfo *info = g_new0 (MoveFolderInfo, 1);
5691 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5693 info->src_folder = g_object_ref (src_folder);
5694 info->dst_folder = g_object_ref (dst_folder);
5695 info->delete_original = TRUE;
5696 info->folder_view = folder_view;
5698 connect_info->callback = on_move_folder_cb;
5699 connect_info->dst_account = get_account_from_folder_store (TNY_FOLDER_STORE (dst_folder));
5700 connect_info->data = info;
5702 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5703 TNY_FOLDER_STORE (src_folder),
5706 } else if (gtk_widget_is_focus (GTK_WIDGET(header_view))) {
5709 headers = modest_header_view_get_selected_headers(header_view);
5711 /* Transfer the messages */
5712 modest_ui_actions_transfer_messages_helper (GTK_WINDOW (win), TNY_FOLDER (src_folder),
5713 headers, TNY_FOLDER (dst_folder));
5715 g_object_unref (headers);
5719 g_object_unref (src_folder);
5722 #ifdef MODEST_TOOLKIT_HILDON2
5724 * UI handler for the "Move to" action when invoked from the
5725 * ModestFolderWindow
5728 modest_ui_actions_on_folder_window_move_to (GtkWidget *folder_view,
5729 TnyFolderStore *dst_folder,
5733 TnyFolderStore *src_folder = NULL;
5734 TnyIterator *iterator;
5736 if (tny_list_get_length (selection) != 1)
5739 iterator = tny_list_create_iterator (selection);
5740 src_folder = TNY_FOLDER_STORE (tny_iterator_get_current (iterator));
5741 g_object_unref (iterator);
5744 gboolean do_xfer = TRUE;
5746 /* Allow only to transfer folders to the local root folder */
5747 if (TNY_IS_ACCOUNT (dst_folder) &&
5748 !MODEST_IS_TNY_LOCAL_FOLDERS_ACCOUNT (dst_folder) &&
5749 !modest_tny_account_is_memory_card_account (TNY_ACCOUNT (dst_folder))) {
5752 modest_platform_run_information_dialog (win,
5753 _("mail_in_ui_folder_move_target_error"),
5755 } else if (!TNY_IS_FOLDER (src_folder)) {
5756 g_warning ("%s: src_folder is not a TnyFolder.\n", __FUNCTION__);
5761 MoveFolderInfo *info = g_new0 (MoveFolderInfo, 1);
5762 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5764 info->src_folder = g_object_ref (src_folder);
5765 info->dst_folder = g_object_ref (dst_folder);
5766 info->delete_original = TRUE;
5767 info->folder_view = folder_view;
5769 connect_info->callback = on_move_folder_cb;
5770 connect_info->dst_account = get_account_from_folder_store (TNY_FOLDER_STORE (dst_folder));
5771 connect_info->data = info;
5773 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5774 TNY_FOLDER_STORE (src_folder),
5779 g_object_unref (src_folder);
5785 modest_ui_actions_transfer_messages_helper (GtkWindow *win,
5786 TnyFolder *src_folder,
5788 TnyFolder *dst_folder)
5790 gboolean need_connection = TRUE;
5791 gboolean do_xfer = TRUE;
5792 XferMsgsHelper *helper;
5794 g_return_if_fail (TNY_IS_FOLDER (src_folder));
5795 g_return_if_fail (TNY_IS_FOLDER (dst_folder));
5796 g_return_if_fail (TNY_IS_LIST (headers));
5798 modest_ui_actions_xfer_messages_check (win, TNY_FOLDER_STORE (src_folder),
5799 headers, TNY_FOLDER (dst_folder),
5800 TRUE, &need_connection,
5803 /* If we don't want to transfer just return */
5807 /* Create the helper */
5808 helper = g_slice_new (XferMsgsHelper);
5809 helper->dst_folder = g_object_ref (dst_folder);
5810 helper->headers = g_object_ref (headers);
5812 if (need_connection) {
5813 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5814 connect_info->callback = xfer_messages_performer;
5815 connect_info->dst_account = tny_folder_get_account (TNY_FOLDER (dst_folder));
5816 connect_info->data = helper;
5818 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5819 TNY_FOLDER_STORE (src_folder),
5822 TnyAccount *src_account = get_account_from_folder_store (TNY_FOLDER_STORE (src_folder));
5823 xfer_messages_performer (FALSE, NULL, GTK_WINDOW (win),
5824 src_account, helper);
5825 g_object_unref (src_account);
5830 * UI handler for the "Move to" action when invoked from the
5831 * ModestMsgViewWindow
5834 modest_ui_actions_on_window_move_to (GtkAction *action,
5836 TnyFolderStore *dst_folder,
5839 TnyFolder *src_folder = NULL;
5841 g_return_if_fail (TNY_IS_FOLDER (dst_folder));
5844 TnyHeader *header = NULL;
5847 iter = tny_list_create_iterator (headers);
5848 header = (TnyHeader *) tny_iterator_get_current (iter);
5849 src_folder = tny_header_get_folder (header);
5851 /* Transfer the messages */
5852 modest_ui_actions_transfer_messages_helper (GTK_WINDOW (win), src_folder,
5854 TNY_FOLDER (dst_folder));
5857 g_object_unref (header);
5858 g_object_unref (iter);
5859 g_object_unref (src_folder);
5864 modest_ui_actions_on_move_to (GtkAction *action,
5867 modest_ui_actions_on_edit_mode_move_to (win);
5871 modest_ui_actions_on_edit_mode_move_to (ModestWindow *win)
5873 GtkWidget *dialog = NULL;
5874 MoveToInfo *helper = NULL;
5875 TnyList *list_to_move;
5877 g_return_val_if_fail (MODEST_IS_WINDOW (win), FALSE);
5879 #ifndef MODEST_TOOLKIT_HILDON2
5880 /* Get the main window if exists */
5881 ModestMainWindow *main_window;
5882 if (MODEST_IS_MAIN_WINDOW (win))
5883 main_window = MODEST_MAIN_WINDOW (win);
5886 MODEST_MAIN_WINDOW (modest_window_mgr_get_main_window (modest_runtime_get_window_mgr (),
5887 FALSE)); /* don't create */
5890 list_to_move = modest_platform_get_list_to_move (MODEST_WINDOW (win));
5895 if (tny_list_get_length (list_to_move) < 1) {
5896 g_object_unref (list_to_move);
5900 /* Create and run the dialog */
5901 dialog = create_move_to_dialog (GTK_WINDOW (win), NULL, list_to_move);
5902 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (),
5903 GTK_WINDOW (dialog),
5907 helper = g_slice_new0 (MoveToInfo);
5908 helper->list = list_to_move;
5911 /* Listen to response signal */
5912 g_signal_connect (dialog, "response", G_CALLBACK (on_move_to_dialog_response), helper);
5914 /* Show the dialog */
5915 gtk_widget_show (dialog);
5921 * Calls #HeadersFunc for each header already selected in the main
5922 * window or the message currently being shown in the msg view window
5925 do_headers_action (ModestWindow *win,
5929 TnyList *headers_list = NULL;
5930 TnyIterator *iter = NULL;
5931 TnyHeader *header = NULL;
5932 TnyFolder *folder = NULL;
5935 headers_list = get_selected_headers (win);
5939 /* Get the folder */
5940 iter = tny_list_create_iterator (headers_list);
5941 header = TNY_HEADER (tny_iterator_get_current (iter));
5943 folder = tny_header_get_folder (header);
5944 g_object_unref (header);
5947 /* Call the function for each header */
5948 while (!tny_iterator_is_done (iter)) {
5949 header = TNY_HEADER (tny_iterator_get_current (iter));
5950 func (header, win, user_data);
5951 g_object_unref (header);
5952 tny_iterator_next (iter);
5955 /* Trick: do a poke status in order to speed up the signaling
5958 tny_folder_poke_status (folder);
5959 g_object_unref (folder);
5963 g_object_unref (iter);
5964 g_object_unref (headers_list);
5968 modest_ui_actions_view_attachment (GtkAction *action,
5969 ModestWindow *window)
5971 if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
5972 modest_msg_view_window_view_attachment (MODEST_MSG_VIEW_WINDOW (window), NULL);
5974 /* not supported window for this action */
5975 g_return_if_reached ();
5980 modest_ui_actions_save_attachments (GtkAction *action,
5981 ModestWindow *window)
5983 if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
5985 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
5988 modest_msg_view_window_save_attachments (MODEST_MSG_VIEW_WINDOW (window), NULL);
5990 /* not supported window for this action */
5991 g_return_if_reached ();
5996 modest_ui_actions_remove_attachments (GtkAction *action,
5997 ModestWindow *window)
5999 if (MODEST_IS_MAIN_WINDOW (window)) {
6000 modest_ui_actions_on_main_window_remove_attachments (action, MODEST_MAIN_WINDOW (window));
6001 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
6002 modest_msg_view_window_remove_attachments (MODEST_MSG_VIEW_WINDOW (window), FALSE);
6004 /* not supported window for this action */
6005 g_return_if_reached ();
6010 modest_ui_actions_on_settings (GtkAction *action,
6015 dialog = modest_platform_get_global_settings_dialog ();
6016 gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (win));
6017 gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
6018 gtk_widget_show_all (dialog);
6020 gtk_dialog_run (GTK_DIALOG (dialog));
6022 gtk_widget_destroy (dialog);
6026 modest_ui_actions_on_help (GtkAction *action,
6029 /* Help app is not available at all in fremantle */
6030 #ifndef MODEST_TOOLKIT_HILDON2
6031 const gchar *help_id;
6033 g_return_if_fail (win && GTK_IS_WINDOW(win));
6035 help_id = modest_window_mgr_get_help_id (modest_runtime_get_window_mgr(), win);
6038 modest_platform_show_help (GTK_WINDOW (win), help_id);
6043 modest_ui_actions_on_csm_help (GtkAction *action,
6046 /* Help app is not available at all in fremantle */
6047 #ifndef MODEST_TOOLKIT_HILDON2
6049 const gchar* help_id = NULL;
6050 GtkWidget *folder_view;
6051 TnyFolderStore *folder_store;
6053 g_return_if_fail (win && MODEST_IS_MAIN_WINDOW (win));
6055 /* Get selected folder */
6056 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
6057 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
6058 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
6060 /* Switch help_id */
6061 if (folder_store && TNY_IS_FOLDER (folder_store))
6062 help_id = modest_tny_folder_get_help_id (TNY_FOLDER (folder_store));
6065 g_object_unref (folder_store);
6068 modest_platform_show_help (GTK_WINDOW (win), help_id);
6070 modest_ui_actions_on_help (action, win);
6075 retrieve_contents_cb (ModestMailOperation *mail_op,
6082 /* We only need this callback to show an error in case of
6083 memory low condition */
6084 if (!modest_ui_actions_msg_retrieval_check (mail_op, header, msg)) {
6085 g_debug ("%s: message failed to retrieve. Memory low?", __FUNCTION__);
6090 retrieve_msg_contents_performer (gboolean canceled,
6092 GtkWindow *parent_window,
6093 TnyAccount *account,
6096 ModestMailOperation *mail_op;
6097 TnyList *headers = TNY_LIST (user_data);
6099 if (err || canceled) {
6100 check_memory_full_error ((GtkWidget *) parent_window, err);
6104 /* Create mail operation */
6105 mail_op = modest_mail_operation_new_with_error_handling ((GObject *) parent_window,
6106 modest_ui_actions_disk_operations_error_handler,
6108 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
6109 modest_mail_operation_get_msgs_full (mail_op, headers, retrieve_contents_cb, NULL, NULL);
6112 g_object_unref (mail_op);
6114 g_object_unref (headers);
6115 g_object_unref (account);
6119 modest_ui_actions_on_retrieve_msg_contents (GtkAction *action,
6120 ModestWindow *window)
6122 TnyList *headers = NULL;
6123 TnyAccount *account = NULL;
6124 TnyIterator *iter = NULL;
6125 TnyHeader *header = NULL;
6126 TnyFolder *folder = NULL;
6129 headers = get_selected_headers (window);
6133 /* Pick the account */
6134 iter = tny_list_create_iterator (headers);
6135 header = TNY_HEADER (tny_iterator_get_current (iter));
6136 folder = tny_header_get_folder (header);
6137 account = tny_folder_get_account (folder);
6138 g_object_unref (folder);
6139 g_object_unref (header);
6140 g_object_unref (iter);
6142 /* Connect and perform the message retrieval */
6143 modest_platform_connect_and_perform ((GtkWindow *) window, TRUE,
6144 g_object_ref (account),
6145 retrieve_msg_contents_performer,
6146 g_object_ref (headers));
6149 g_object_unref (account);
6150 g_object_unref (headers);
6154 modest_ui_actions_check_toolbar_dimming_rules (ModestWindow *window)
6156 g_return_if_fail (MODEST_IS_WINDOW (window));
6159 modest_window_check_dimming_rules_group (window, MODEST_DIMMING_RULES_TOOLBAR);
6163 modest_ui_actions_check_menu_dimming_rules (ModestWindow *window)
6165 g_return_if_fail (MODEST_IS_WINDOW (window));
6168 modest_window_check_dimming_rules_group (window, MODEST_DIMMING_RULES_MENU);
6172 modest_ui_actions_on_email_menu_activated (GtkAction *action,
6173 ModestWindow *window)
6175 g_return_if_fail (MODEST_IS_WINDOW (window));
6178 modest_ui_actions_check_menu_dimming_rules (window);
6182 modest_ui_actions_on_edit_menu_activated (GtkAction *action,
6183 ModestWindow *window)
6185 g_return_if_fail (MODEST_IS_WINDOW (window));
6188 modest_ui_actions_check_menu_dimming_rules (window);
6192 modest_ui_actions_on_view_menu_activated (GtkAction *action,
6193 ModestWindow *window)
6195 g_return_if_fail (MODEST_IS_WINDOW (window));
6198 modest_ui_actions_check_menu_dimming_rules (window);
6202 modest_ui_actions_on_format_menu_activated (GtkAction *action,
6203 ModestWindow *window)
6205 g_return_if_fail (MODEST_IS_WINDOW (window));
6208 modest_ui_actions_check_menu_dimming_rules (window);
6212 modest_ui_actions_on_tools_menu_activated (GtkAction *action,
6213 ModestWindow *window)
6215 g_return_if_fail (MODEST_IS_WINDOW (window));
6218 modest_ui_actions_check_menu_dimming_rules (window);
6222 modest_ui_actions_on_attachment_menu_activated (GtkAction *action,
6223 ModestWindow *window)
6225 g_return_if_fail (MODEST_IS_WINDOW (window));
6228 modest_ui_actions_check_menu_dimming_rules (window);
6232 modest_ui_actions_on_toolbar_csm_menu_activated (GtkAction *action,
6233 ModestWindow *window)
6235 g_return_if_fail (MODEST_IS_WINDOW (window));
6238 modest_ui_actions_check_menu_dimming_rules (window);
6242 modest_ui_actions_on_folder_view_csm_menu_activated (GtkAction *action,
6243 ModestWindow *window)
6245 g_return_if_fail (MODEST_IS_WINDOW (window));
6248 modest_ui_actions_check_menu_dimming_rules (window);
6252 modest_ui_actions_on_header_view_csm_menu_activated (GtkAction *action,
6253 ModestWindow *window)
6255 g_return_if_fail (MODEST_IS_WINDOW (window));
6258 modest_ui_actions_check_menu_dimming_rules (window);
6262 modest_ui_actions_on_search_messages (GtkAction *action, ModestWindow *window)
6264 g_return_if_fail (MODEST_IS_WINDOW (window));
6266 /* we check for low-mem; in that case, show a warning, and don't allow
6269 if (modest_platform_check_memory_low (window, TRUE))
6272 modest_platform_show_search_messages (GTK_WINDOW (window));
6276 modest_ui_actions_on_open_addressbook (GtkAction *action, ModestWindow *win)
6278 g_return_if_fail (MODEST_IS_WINDOW (win));
6281 /* we check for low-mem; in that case, show a warning, and don't allow
6282 * for the addressbook
6284 if (modest_platform_check_memory_low (win, TRUE))
6288 modest_platform_show_addressbook (GTK_WINDOW (win));
6293 modest_ui_actions_on_toggle_find_in_page (GtkAction *action,
6294 ModestWindow *window)
6297 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
6299 if (GTK_IS_TOGGLE_ACTION (action))
6300 active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action));
6304 modest_msg_edit_window_toggle_find_toolbar (MODEST_MSG_EDIT_WINDOW (window),
6309 on_send_receive_finished (ModestMailOperation *mail_op,
6312 GtkWidget *header_view, *folder_view;
6313 TnyFolderStore *folder_store;
6314 ModestMainWindow *main_win = MODEST_MAIN_WINDOW (user_data);
6316 /* Set send/receive operation finished */
6317 modest_main_window_notify_send_receive_completed (main_win);
6319 /* Don't refresh the current folder if there were any errors */
6320 if (modest_mail_operation_get_status (mail_op) !=
6321 MODEST_MAIL_OPERATION_STATUS_SUCCESS)
6324 /* Refresh the current folder if we're viewing a window. We do
6325 this because the user won't be able to see the new mails in
6326 the selected folder after a Send&Receive because it only
6327 performs a poke_status, i.e, only the number of read/unread
6328 messages is updated, but the new headers are not
6330 folder_view = modest_main_window_get_child_widget (main_win,
6331 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
6335 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
6337 /* Do not need to refresh INBOX again because the
6338 update_account does it always automatically */
6339 if (folder_store && TNY_IS_FOLDER (folder_store) &&
6340 tny_folder_get_folder_type (TNY_FOLDER (folder_store)) != TNY_FOLDER_TYPE_INBOX) {
6341 ModestMailOperation *refresh_op;
6343 header_view = modest_main_window_get_child_widget (main_win,
6344 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6346 /* We do not need to set the contents style
6347 because it hasn't changed. We also do not
6348 need to save the widget status. Just force
6350 refresh_op = modest_mail_operation_new (G_OBJECT (main_win));
6351 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), refresh_op);
6352 modest_mail_operation_refresh_folder (refresh_op, TNY_FOLDER (folder_store),
6353 folder_refreshed_cb, main_win);
6354 g_object_unref (refresh_op);
6358 g_object_unref (folder_store);
6363 modest_ui_actions_on_send_queue_error_happened (TnySendQueue *self,
6369 const gchar* server_name = NULL;
6370 TnyTransportAccount *transport;
6371 gchar *message = NULL;
6372 ModestProtocol *protocol;
6374 /* Don't show anything if the user cancelled something or the
6375 * send receive request is not interactive. Authentication
6376 * errors are managed by the account store so no need to show
6377 * a dialog here again */
6378 if (err->code == TNY_SYSTEM_ERROR_CANCEL ||
6379 err->code == TNY_SERVICE_ERROR_AUTHENTICATE ||
6380 !modest_tny_send_queue_get_requested_send_receive (MODEST_TNY_SEND_QUEUE (self)))
6384 /* Get the server name. Note that we could be using a
6385 connection specific transport account */
6386 transport = (TnyTransportAccount *)
6387 tny_camel_send_queue_get_transport_account (TNY_CAMEL_SEND_QUEUE (self));
6389 ModestTnyAccountStore *acc_store;
6390 const gchar *acc_name;
6391 TnyTransportAccount *conn_specific;
6393 acc_store = modest_runtime_get_account_store();
6394 acc_name = modest_tny_account_get_parent_modest_account_name_for_server_account (TNY_ACCOUNT (transport));
6395 conn_specific = (TnyTransportAccount *)
6396 modest_tny_account_store_get_transport_account_for_open_connection (acc_store, acc_name);
6397 if (conn_specific) {
6398 server_name = tny_account_get_hostname (TNY_ACCOUNT (conn_specific));
6399 g_object_unref (conn_specific);
6401 server_name = tny_account_get_hostname (TNY_ACCOUNT (transport));
6403 g_object_unref (transport);
6407 protocol = modest_protocol_registry_get_protocol_by_name (modest_runtime_get_protocol_registry (),
6408 MODEST_PROTOCOL_REGISTRY_TRANSPORT_STORE_PROTOCOLS,
6409 tny_account_get_proto (TNY_ACCOUNT (transport)));
6411 g_warning ("%s: Account with no proto", __FUNCTION__);
6415 /* Show the appropriate message text for the GError: */
6416 switch (err->code) {
6417 case TNY_SERVICE_ERROR_CONNECT:
6418 message = modest_protocol_get_translation (protocol,
6419 MODEST_PROTOCOL_TRANSLATION_ACCOUNT_CONNECTION_ERROR,
6422 case TNY_SERVICE_ERROR_SEND:
6423 message = g_strdup (_CS("sfil_ib_unable_to_send"));
6425 case TNY_SERVICE_ERROR_UNAVAILABLE:
6426 message = modest_protocol_get_translation (protocol,
6427 MODEST_PROTOCOL_TRANSLATION_CONNECT_ERROR,
6431 g_warning ("%s: unexpected ERROR %d",
6432 __FUNCTION__, err->code);
6433 message = g_strdup (_CS("sfil_ib_unable_to_send"));
6437 modest_platform_run_information_dialog (NULL, message, FALSE);
6442 modest_ui_actions_on_send_queue_status_changed (ModestTnySendQueue *send_queue,
6447 ModestWindow *top_window = NULL;
6448 ModestWindowMgr *mgr = NULL;
6449 GtkWidget *header_view = NULL;
6450 TnyFolder *selected_folder = NULL;
6451 TnyFolderType folder_type;
6453 mgr = modest_runtime_get_window_mgr ();
6454 top_window = modest_window_mgr_get_current_top (mgr);
6459 #ifndef MODEST_TOOLKIT_HILDON2
6460 if (MODEST_IS_MAIN_WINDOW (top_window)) {
6461 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (top_window),
6462 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6465 if (MODEST_IS_HEADER_WINDOW (top_window)) {
6466 header_view = (GtkWidget *)
6467 modest_header_window_get_header_view (MODEST_HEADER_WINDOW (top_window));
6471 /* Get selected folder */
6473 selected_folder = modest_header_view_get_folder (MODEST_HEADER_VIEW (header_view));
6474 if (!selected_folder)
6477 /* gtk_tree_view_column_queue_resize is only available in GTK+ 2.8 */
6478 #if GTK_CHECK_VERSION(2, 8, 0)
6479 folder_type = modest_tny_folder_guess_folder_type (selected_folder);
6480 if (folder_type == TNY_FOLDER_TYPE_OUTBOX) {
6481 GtkTreeViewColumn *tree_column;
6483 tree_column = gtk_tree_view_get_column (GTK_TREE_VIEW (header_view),
6484 TNY_GTK_HEADER_LIST_MODEL_FROM_COLUMN);
6486 gtk_tree_view_column_queue_resize (tree_column);
6488 #else /* #if GTK_CHECK_VERSION(2, 8, 0) */
6489 gtk_widget_queue_draw (header_view);
6492 #ifndef MODEST_TOOLKIT_HILDON2
6493 /* Rerun dimming rules, because the message could become deletable for example */
6494 modest_window_check_dimming_rules_group (MODEST_WINDOW (top_window),
6495 MODEST_DIMMING_RULES_TOOLBAR);
6496 modest_window_check_dimming_rules_group (MODEST_WINDOW (top_window),
6497 MODEST_DIMMING_RULES_MENU);
6501 g_object_unref (selected_folder);
6505 modest_ui_actions_on_account_connection_error (GtkWindow *parent_window,
6506 TnyAccount *account)
6508 ModestProtocolType protocol_type;
6509 ModestProtocol *protocol;
6510 gchar *error_note = NULL;
6512 protocol_type = modest_tny_account_get_protocol_type (account);
6513 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6516 error_note = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_ACCOUNT_CONNECTION_ERROR, tny_account_get_hostname (account));
6517 if (error_note == NULL) {
6518 g_warning ("%s: This should not be reached", __FUNCTION__);
6520 modest_platform_run_information_dialog (parent_window, error_note, FALSE);
6521 g_free (error_note);
6526 modest_ui_actions_get_msg_already_deleted_error_msg (ModestWindow *win)
6530 TnyFolderStore *folder = NULL;
6531 TnyAccount *account = NULL;
6532 ModestProtocolType proto;
6533 ModestProtocol *protocol;
6534 TnyHeader *header = NULL;
6536 if (MODEST_IS_MAIN_WINDOW (win)) {
6537 GtkWidget *header_view;
6538 TnyList* headers = NULL;
6540 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
6541 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6542 headers = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
6543 if (!headers || tny_list_get_length (headers) == 0) {
6545 g_object_unref (headers);
6548 iter = tny_list_create_iterator (headers);
6549 header = TNY_HEADER (tny_iterator_get_current (iter));
6550 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6551 g_object_unref (iter);
6552 g_object_unref (headers);
6553 #ifdef MODEST_TOOLKIT_HILDON2
6554 } else if (MODEST_IS_HEADER_WINDOW (win)) {
6555 GtkWidget *header_view;
6556 TnyList* headers = NULL;
6558 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
6559 headers = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
6560 if (!headers || tny_list_get_length (headers) == 0) {
6562 g_object_unref (headers);
6565 iter = tny_list_create_iterator (headers);
6566 header = TNY_HEADER (tny_iterator_get_current (iter));
6568 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6570 g_warning ("List should contain headers");
6572 g_object_unref (iter);
6573 g_object_unref (headers);
6575 } else if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
6576 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
6577 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6580 if (!header || !folder)
6583 /* Get the account type */
6584 account = tny_folder_get_account (TNY_FOLDER (folder));
6585 proto = modest_tny_account_get_protocol_type (account);
6586 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6589 subject = tny_header_dup_subject (header);
6590 msg = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
6594 msg = g_strdup_printf (_("mail_ni_ui_folder_get_msg_folder_error"));
6600 g_object_unref (account);
6602 g_object_unref (folder);
6604 g_object_unref (header);
6610 modest_ui_actions_on_delete_account (GtkWindow *parent_window,
6611 const gchar *account_name,
6612 const gchar *account_title)
6614 ModestAccountMgr *account_mgr;
6617 ModestProtocol *protocol;
6618 gboolean removed = FALSE;
6620 g_return_val_if_fail (account_name, FALSE);
6621 g_return_val_if_fail (account_title, FALSE);
6623 account_mgr = modest_runtime_get_account_mgr();
6625 /* The warning text depends on the account type: */
6626 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6627 modest_account_mgr_get_store_protocol (account_mgr,
6629 txt = modest_protocol_get_translation (protocol,
6630 MODEST_PROTOCOL_TRANSLATION_DELETE_MAILBOX,
6633 txt = g_strdup_printf (_("emev_nc_delete_mailbox"), account_title);
6635 response = modest_platform_run_confirmation_dialog (parent_window, txt);
6639 if (response == GTK_RESPONSE_OK) {
6640 /* Remove account. If it succeeds then it also removes
6641 the account from the ModestAccountView: */
6642 gboolean is_default = FALSE;
6643 gchar *default_account_name = modest_account_mgr_get_default_account (account_mgr);
6644 if (default_account_name && (strcmp (default_account_name, account_name) == 0))
6646 g_free (default_account_name);
6648 removed = modest_account_mgr_remove_account (account_mgr, account_name);
6650 g_warning ("%s: modest_account_mgr_remove_account() failed.\n", __FUNCTION__);