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 modest_ui_actions_run_account_setup_wizard (window);
736 modest_ui_actions_on_accounts (GtkAction *action,
739 /* This is currently only implemented for Maemo */
740 if (!modest_account_mgr_has_accounts (modest_runtime_get_account_mgr(), TRUE)) {
741 if (!modest_ui_actions_run_account_setup_wizard (win))
742 g_debug ("%s: wizard was already running", __FUNCTION__);
746 /* Show the list of accounts */
747 GtkWindow *account_win = GTK_WINDOW (modest_account_view_window_new ());
749 /* The accounts dialog must be modal */
750 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (), GTK_WINDOW (account_win), (GtkWindow *) win);
751 modest_utils_show_dialog_and_forget (GTK_WINDOW (win), GTK_DIALOG (account_win));
756 modest_ui_actions_on_smtp_servers (GtkAction *action, ModestWindow *win)
758 /* This is currently only implemented for Maemo,
759 * because it requires an API (libconic) to detect different connection
762 #ifndef MODEST_TOOLKIT_GTK /* Defined in config.h */
764 /* Create the window if necessary: */
765 GtkWidget *specific_window = GTK_WIDGET (modest_connection_specific_smtp_window_new ());
766 modest_connection_specific_smtp_window_fill_with_connections (
767 MODEST_CONNECTION_SPECIFIC_SMTP_WINDOW (specific_window),
768 modest_runtime_get_account_mgr());
770 /* Show the window: */
771 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (),
772 GTK_WINDOW (specific_window), (GtkWindow *) win);
773 gtk_widget_show (specific_window);
774 #endif /* !MODEST_TOOLKIT_GTK */
778 modest_ui_actions_compose_msg(ModestWindow *win,
781 const gchar *bcc_str,
782 const gchar *subject_str,
783 const gchar *body_str,
785 gboolean set_as_modified)
787 gchar *account_name = NULL;
789 TnyAccount *account = NULL;
790 TnyFolder *folder = NULL;
791 gchar *from_str = NULL, *signature = NULL, *body = NULL;
792 gboolean use_signature = FALSE;
793 ModestWindow *msg_win = NULL;
794 ModestAccountMgr *mgr = modest_runtime_get_account_mgr();
795 ModestTnyAccountStore *store = modest_runtime_get_account_store();
796 GnomeVFSFileSize total_size, allowed_size;
798 /* we check for low-mem */
799 if (modest_platform_check_memory_low (win, TRUE))
802 #ifdef MODEST_TOOLKIT_HILDON2
803 account_name = g_strdup (modest_window_get_active_account(win));
806 account_name = modest_account_mgr_get_default_account(mgr);
809 g_printerr ("modest: no account found\n");
812 account = modest_tny_account_store_get_server_account (store, account_name, TNY_ACCOUNT_TYPE_STORE);
814 g_printerr ("modest: failed to get tnyaccount for '%s'\n", account_name);
817 folder = modest_tny_account_get_special_folder (account, TNY_FOLDER_TYPE_DRAFTS);
819 g_printerr ("modest: failed to find Drafts folder\n");
822 from_str = modest_account_mgr_get_from_string (mgr, account_name);
824 g_printerr ("modest: failed get from string for '%s'\n", account_name);
828 signature = modest_account_mgr_get_signature (mgr, account_name, &use_signature);
829 if (body_str != NULL) {
830 body = use_signature ? g_strconcat(body_str, "\n--\n", signature, NULL) : g_strdup(body_str);
832 body = use_signature ? g_strconcat("\n--\n", signature, NULL) : g_strdup("");
835 msg = modest_tny_msg_new (to_str, from_str, cc_str, bcc_str, subject_str, body, NULL, NULL, NULL);
837 g_printerr ("modest: failed to create new msg\n");
841 /* Create and register edit window */
842 /* This is destroyed by TODO. */
844 allowed_size = MODEST_MAX_ATTACHMENT_SIZE;
845 msg_win = modest_msg_edit_window_new (msg, account_name, FALSE);
847 if (!modest_window_mgr_register_window (modest_runtime_get_window_mgr(), msg_win, win)) {
848 gtk_widget_destroy (GTK_WIDGET (msg_win));
851 modest_msg_edit_window_set_modified (MODEST_MSG_EDIT_WINDOW (msg_win), set_as_modified);
852 gtk_widget_show_all (GTK_WIDGET (msg_win));
854 while (attachments) {
856 modest_msg_edit_window_attach_file_one((ModestMsgEditWindow *)msg_win,
857 attachments->data, allowed_size);
859 if (total_size > allowed_size) {
860 g_warning ("%s: total size: %u",
861 __FUNCTION__, (unsigned int)total_size);
864 allowed_size -= total_size;
866 attachments = g_slist_next(attachments);
873 g_free (account_name);
875 g_object_unref (G_OBJECT(account));
877 g_object_unref (G_OBJECT(folder));
879 g_object_unref (G_OBJECT(msg));
883 modest_ui_actions_on_new_msg (GtkAction *action, ModestWindow *win)
885 /* if there are no accounts yet, just show the wizard */
886 if (!modest_account_mgr_has_accounts (modest_runtime_get_account_mgr(), TRUE))
887 if (!modest_ui_actions_run_account_setup_wizard (win))
890 modest_ui_actions_compose_msg(win, NULL, NULL, NULL, NULL, NULL, NULL, FALSE);
895 modest_ui_actions_msg_retrieval_check (ModestMailOperation *mail_op,
899 ModestMailOperationStatus status;
901 /* If there is no message or the operation was not successful */
902 status = modest_mail_operation_get_status (mail_op);
903 if (!msg || status != MODEST_MAIL_OPERATION_STATUS_SUCCESS) {
906 /* If it's a memory low issue, then show a banner */
907 error = modest_mail_operation_get_error (mail_op);
908 if (error && error->domain == MODEST_MAIL_OPERATION_ERROR &&
909 error->code == MODEST_MAIL_OPERATION_ERROR_LOW_MEMORY) {
910 GObject *source = modest_mail_operation_get_source (mail_op);
911 modest_platform_run_information_dialog (GTK_IS_WINDOW (source) ? GTK_WINDOW (source) : NULL,
912 _KR("memr_ib_operation_disabled"),
914 g_object_unref (source);
917 if (error && ((error->code == TNY_SERVICE_ERROR_NO_SUCH_MESSAGE) ||
918 error->code == TNY_SERVICE_ERROR_MESSAGE_NOT_AVAILABLE)) {
919 gchar *subject, *msg, *format = NULL;
921 subject = tny_header_dup_subject (header);
923 subject = g_strdup (_("mail_va_no_subject"));
925 account = modest_mail_operation_get_account (mail_op);
927 ModestProtocol *protocol;
928 ModestProtocolType proto;
929 proto = modest_tny_account_get_protocol_type (account);
930 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (), proto);
932 format = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
933 g_object_unref (account);
937 format = g_strdup (_("emev_ni_ui_imap_message_not_available_in_server"));
939 msg = g_strdup_printf (format, subject);
940 modest_platform_run_information_dialog (NULL, msg, FALSE);
946 /* Remove the header from the preregistered uids */
947 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
965 OpenMsgBannerInfo *banner_info;
966 GtkTreeRowReference *rowref;
970 open_msg_banner_idle (gpointer userdata)
972 OpenMsgBannerInfo *banner_info = (OpenMsgBannerInfo *) userdata;
974 gdk_threads_enter ();
975 banner_info->idle_handler = 0;
976 banner_info->banner = modest_platform_animation_banner (NULL, NULL, banner_info->message);
978 g_object_ref (banner_info->banner);
980 gdk_threads_leave ();
987 get_header_view_from_window (ModestWindow *window)
989 GtkWidget *header_view;
991 if (MODEST_IS_MAIN_WINDOW (window)) {
992 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
993 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
994 #ifdef MODEST_TOOLKIT_HILDON2
995 } else if (MODEST_IS_HEADER_WINDOW (window)){
996 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (window)));
1006 get_info_from_header (TnyHeader *header, gboolean *is_draft, gboolean *can_open)
1009 gchar *account = NULL;
1010 TnyFolderType folder_type = TNY_FOLDER_TYPE_UNKNOWN;
1015 folder = tny_header_get_folder (header);
1016 /* Gets folder type (OUTBOX headers will be opened in edit window */
1017 if (modest_tny_folder_is_local_folder (folder)) {
1018 folder_type = modest_tny_folder_get_local_or_mmc_folder_type (folder);
1019 if (folder_type == TNY_FOLDER_TYPE_INVALID)
1020 g_warning ("%s: BUG: TNY_FOLDER_TYPE_INVALID", __FUNCTION__);
1023 if (folder_type == TNY_FOLDER_TYPE_OUTBOX) {
1024 TnyTransportAccount *traccount = NULL;
1025 ModestTnyAccountStore *accstore = modest_runtime_get_account_store();
1026 traccount = modest_tny_account_store_get_transport_account_from_outbox_header(accstore, header);
1028 ModestTnySendQueue *send_queue = NULL;
1029 ModestTnySendQueueStatus status;
1031 account = g_strdup(modest_tny_account_get_parent_modest_account_name_for_server_account(
1032 TNY_ACCOUNT(traccount)));
1033 send_queue = modest_runtime_get_send_queue(traccount, TRUE);
1034 if (TNY_IS_SEND_QUEUE (send_queue)) {
1035 msg_id = modest_tny_send_queue_get_msg_id (header);
1036 status = modest_tny_send_queue_get_msg_status(send_queue, msg_id);
1038 /* Only open messages in outbox with the editor if they are in Failed state */
1039 if (status == MODEST_TNY_SEND_QUEUE_FAILED) {
1042 #ifdef MODEST_TOOLKIT_HILDON2
1044 /* In Fremantle we can not
1045 open any message from
1046 outbox which is not in
1052 g_object_unref(traccount);
1054 g_warning("Cannot get transport account for message in outbox!!");
1056 } else if (folder_type == TNY_FOLDER_TYPE_DRAFTS) {
1057 *is_draft = TRUE; /* Open in editor if the message is in the Drafts folder */
1061 TnyAccount *acc = tny_folder_get_account (folder);
1064 g_strdup (modest_tny_account_get_parent_modest_account_name_for_server_account (acc));
1065 g_object_unref (acc);
1069 g_object_unref (folder);
1075 open_msg_cb (ModestMailOperation *mail_op,
1082 ModestWindowMgr *mgr = NULL;
1083 ModestWindow *parent_win = NULL;
1084 ModestWindow *win = NULL;
1085 gchar *account = NULL;
1086 gboolean open_in_editor = FALSE;
1088 OpenMsgHelper *helper = (OpenMsgHelper *) user_data;
1090 /* Do nothing if there was any problem with the mail
1091 operation. The error will be shown by the error_handler of
1092 the mail operation */
1093 if (!modest_ui_actions_msg_retrieval_check (mail_op, header, msg))
1096 parent_win = (ModestWindow *) modest_mail_operation_get_source (mail_op);
1098 /* Mark header as read */
1099 headers_action_mark_as_read (header, MODEST_WINDOW(parent_win), NULL);
1101 account = get_info_from_header (header, &open_in_editor, &can_open);
1105 account = g_strdup (modest_window_get_active_account (MODEST_WINDOW (parent_win)));
1107 account = modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
1109 if (open_in_editor) {
1110 ModestAccountMgr *mgr = modest_runtime_get_account_mgr ();
1111 gchar *from_header = NULL, *acc_name;
1113 from_header = tny_header_dup_from (header);
1115 /* we cannot edit without a valid account... */
1116 if (!modest_account_mgr_has_accounts(mgr, TRUE)) {
1117 if (!modest_ui_actions_run_account_setup_wizard(parent_win)) {
1118 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1120 g_free (from_header);
1125 acc_name = modest_utils_get_account_name_from_recipient (from_header);
1126 g_free (from_header);
1132 win = modest_msg_edit_window_new (msg, account, TRUE);
1134 gchar *uid = modest_tny_folder_get_header_unique_id (header);
1136 if (helper->rowref && helper->model) {
1137 win = modest_msg_view_window_new_with_header_model (msg, account, (const gchar*) uid,
1138 helper->model, helper->rowref);
1140 win = modest_msg_view_window_new_for_attachment (msg, account, (const gchar*) uid);
1145 /* Register and show new window */
1147 mgr = modest_runtime_get_window_mgr ();
1148 if (!modest_window_mgr_register_window (mgr, win, NULL)) {
1149 gtk_widget_destroy (GTK_WIDGET (win));
1152 gtk_widget_show_all (GTK_WIDGET(win));
1155 /* Update toolbar dimming state */
1156 if (MODEST_IS_MAIN_WINDOW (parent_win)) {
1157 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (parent_win));
1163 g_object_unref (parent_win);
1167 is_memory_full_error (GError *error, ModestMailOperation *mail_op)
1169 gboolean enough_free_space = TRUE;
1170 GnomeVFSURI *cache_dir_uri;
1171 const gchar *cache_dir = NULL;
1172 GnomeVFSFileSize free_space;
1173 TnyAccountStore *acc_store;
1175 acc_store = TNY_ACCOUNT_STORE (modest_runtime_get_account_store ());
1177 /* Cache dir is different in case we're using an external storage (like MMC account) */
1179 TnyAccount *account = modest_mail_operation_get_account (mail_op);
1181 if (modest_tny_account_is_memory_card_account (account)) {
1182 cache_dir = g_getenv (MODEST_MMC1_VOLUMEPATH_ENV);
1184 g_object_unref (account);
1188 /* Get the default local cache dir */
1190 cache_dir = tny_account_store_get_cache_dir (acc_store);
1192 cache_dir_uri = gnome_vfs_uri_new (cache_dir);
1193 if (cache_dir_uri) {
1194 if (gnome_vfs_get_volume_free_space (cache_dir_uri, &free_space) == GNOME_VFS_OK) {
1195 if (free_space < MIN_FREE_SPACE)
1196 enough_free_space = FALSE;
1198 gnome_vfs_uri_unref (cache_dir_uri);
1201 if ((error->code == TNY_SYSTEM_ERROR_MEMORY ||
1202 /* When asking for a mail and no space left on device
1203 tinymail returns this error */
1204 error->code == TNY_SERVICE_ERROR_MESSAGE_NOT_AVAILABLE ||
1205 /* When the folder summary could not be read or
1207 error->code == TNY_IO_ERROR_WRITE ||
1208 error->code == TNY_IO_ERROR_READ) &&
1209 !enough_free_space) {
1217 check_memory_full_error (GtkWidget *parent_window, GError *err)
1222 if (is_memory_full_error (err, NULL))
1223 modest_platform_information_banner (parent_window,
1224 NULL, _KR("cerm_device_memory_full"));
1225 else if (err->code == TNY_SYSTEM_ERROR_MEMORY)
1226 /* If the account was created in memory full
1227 conditions then tinymail won't be able to
1228 connect so it'll return this error code */
1229 modest_platform_information_banner (parent_window,
1230 NULL, _("emev_ui_imap_inbox_select_error"));
1238 modest_ui_actions_disk_operations_error_handler (ModestMailOperation *mail_op,
1241 const GError *error;
1242 GObject *win = NULL;
1243 ModestMailOperationStatus status;
1245 win = modest_mail_operation_get_source (mail_op);
1246 error = modest_mail_operation_get_error (mail_op);
1247 status = modest_mail_operation_get_status (mail_op);
1249 /* If the mail op has been cancelled then it's not an error:
1250 don't show any message */
1251 if (status != MODEST_MAIL_OPERATION_STATUS_CANCELED) {
1252 if (is_memory_full_error ((GError *) error, mail_op)) {
1253 modest_platform_information_banner ((GtkWidget *) win,
1254 NULL, _KR("cerm_device_memory_full"));
1255 } else if (error->code == TNY_SYSTEM_ERROR_MEMORY) {
1256 modest_platform_information_banner ((GtkWidget *) win,
1257 NULL, _("emev_ui_imap_inbox_select_error"));
1258 } else if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
1259 error->code == MODEST_MAIL_OPERATION_ERROR_FILE_IO) {
1260 modest_platform_information_banner ((GtkWidget *) win,
1261 NULL, _CS ("sfil_ni_unable_to_open_file_not_found"));
1262 } else if (user_data) {
1263 modest_platform_information_banner ((GtkWidget *) win,
1269 g_object_unref (win);
1273 * Returns the account a list of headers belongs to. It returns a
1274 * *new* reference so don't forget to unref it
1277 get_account_from_header_list (TnyList *headers)
1279 TnyAccount *account = NULL;
1281 if (tny_list_get_length (headers) > 0) {
1282 TnyIterator *iter = tny_list_create_iterator (headers);
1283 TnyHeader *header = TNY_HEADER (tny_iterator_get_current (iter));
1284 TnyFolder *folder = tny_header_get_folder (header);
1287 g_object_unref (header);
1289 while (!tny_iterator_is_done (iter)) {
1290 header = TNY_HEADER (tny_iterator_get_current (iter));
1291 folder = tny_header_get_folder (header);
1294 g_object_unref (header);
1296 tny_iterator_next (iter);
1301 account = tny_folder_get_account (folder);
1302 g_object_unref (folder);
1306 g_object_unref (header);
1308 g_object_unref (iter);
1314 get_account_from_header (TnyHeader *header)
1316 TnyAccount *account = NULL;
1319 folder = tny_header_get_folder (header);
1322 account = tny_folder_get_account (folder);
1323 g_object_unref (folder);
1329 open_msg_helper_destroyer (gpointer user_data)
1331 OpenMsgHelper *helper = (OpenMsgHelper *) user_data;
1333 if (helper->banner_info) {
1334 g_free (helper->banner_info->message);
1335 if (helper->banner_info->idle_handler > 0) {
1336 g_source_remove (helper->banner_info->idle_handler);
1337 helper->banner_info->idle_handler = 0;
1339 if (helper->banner_info->banner != NULL) {
1340 gtk_widget_destroy (helper->banner_info->banner);
1341 g_object_unref (helper->banner_info->banner);
1342 helper->banner_info->banner = NULL;
1344 g_slice_free (OpenMsgBannerInfo, helper->banner_info);
1345 helper->banner_info = NULL;
1347 g_object_unref (helper->model);
1348 g_object_unref (helper->header);
1349 gtk_tree_row_reference_free (helper->rowref);
1350 g_slice_free (OpenMsgHelper, helper);
1354 open_msg_performer(gboolean canceled,
1356 GtkWindow *parent_window,
1357 TnyAccount *account,
1360 ModestMailOperation *mail_op = NULL;
1361 gchar *error_msg = NULL;
1362 ModestProtocolType proto;
1363 TnyConnectionStatus status;
1364 OpenMsgHelper *helper = NULL;
1365 ModestProtocol *protocol;
1366 ModestProtocolRegistry *protocol_registry;
1369 helper = (OpenMsgHelper *) user_data;
1371 status = tny_account_get_connection_status (account);
1372 if (err || canceled) {
1373 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (), helper->header);
1374 /* Free the helper */
1375 open_msg_helper_destroyer (helper);
1377 /* In memory full conditions we could get this error here */
1378 check_memory_full_error ((GtkWidget *) parent_window, err);
1383 /* Get the error message depending on the protocol */
1384 proto = modest_tny_account_get_protocol_type (account);
1385 if (proto == MODEST_PROTOCOL_REGISTRY_TYPE_INVALID) {
1386 proto = MODEST_PROTOCOLS_STORE_MAILDIR;
1389 protocol_registry = modest_runtime_get_protocol_registry ();
1390 subject = tny_header_dup_subject (helper->header);
1392 protocol = modest_protocol_registry_get_protocol_by_type (protocol_registry, proto);
1393 error_msg = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
1397 if (error_msg == NULL) {
1398 error_msg = g_strdup (_("mail_ni_ui_folder_get_msg_folder_error"));
1401 #ifndef MODEST_TOOLKIT_HILDON2
1402 gboolean show_open_draft = FALSE;
1403 if (modest_protocol_registry_protocol_type_has_tag (protocol_registry,
1405 MODEST_PROTOCOL_REGISTRY_LOCAL_STORE_PROTOCOLS)) {
1407 TnyFolderType folder_type;
1409 folder = tny_header_get_folder (helper->header);
1410 folder_type = modest_tny_folder_get_local_or_mmc_folder_type (folder);
1411 show_open_draft = (folder_type == TNY_FOLDER_TYPE_DRAFTS);
1412 g_object_unref (folder);
1416 #ifdef MODEST_TOOLKIT_HILDON2
1419 gchar *account_name = get_info_from_header (helper->header, &is_draft, &can_open);
1422 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (), helper->header);
1423 g_free (account_name);
1424 open_msg_helper_destroyer (helper);
1429 ModestWindow *window;
1430 GtkWidget *header_view;
1433 header_view = get_header_view_from_window (MODEST_WINDOW (parent_window));
1434 uid = modest_tny_folder_get_header_unique_id (helper->header);
1436 window = modest_msg_view_window_new_from_header_view
1437 (MODEST_HEADER_VIEW (header_view), account_name, uid, helper->rowref);
1438 if (window != NULL) {
1439 if (!modest_window_mgr_register_window (modest_runtime_get_window_mgr (),
1441 gtk_widget_destroy (GTK_WIDGET (window));
1443 gtk_widget_show_all (GTK_WIDGET(window));
1447 g_free (account_name);
1449 open_msg_helper_destroyer (helper);
1452 g_free (account_name);
1454 /* Create the mail operation */
1456 modest_mail_operation_new_with_error_handling ((GObject *) parent_window,
1457 modest_ui_actions_disk_operations_error_handler,
1458 g_strdup (error_msg), g_free);
1459 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
1463 #ifndef MODEST_TOOLKIT_HILDON2
1464 if (show_open_draft) {
1465 helper->banner_info = g_slice_new (OpenMsgBannerInfo);
1466 helper->banner_info->message = g_strdup (_("mail_ib_opening_draft_message"));
1467 helper->banner_info->banner = NULL;
1468 helper->banner_info->idle_handler = g_timeout_add (500, open_msg_banner_idle,
1469 helper->banner_info);
1475 headers = TNY_LIST (tny_simple_list_new ());
1476 tny_list_prepend (headers, G_OBJECT (helper->header));
1477 modest_mail_operation_get_msgs_full (mail_op,
1481 open_msg_helper_destroyer);
1482 g_object_unref (headers);
1489 g_object_unref (mail_op);
1490 g_object_unref (account);
1494 * This function is used by both modest_ui_actions_on_open and
1495 * modest_ui_actions_on_header_activated. This way we always do the
1496 * same when trying to open messages.
1499 open_msg_from_header (TnyHeader *header, GtkTreeRowReference *rowref, ModestWindow *win)
1501 ModestWindowMgr *mgr = NULL;
1502 TnyAccount *account;
1503 gboolean cached = FALSE;
1505 GtkWidget *header_view = NULL;
1506 OpenMsgHelper *helper;
1507 ModestWindow *window;
1509 g_return_if_fail (header != NULL && rowref != NULL);
1511 mgr = modest_runtime_get_window_mgr ();
1514 header_view = get_header_view_from_window (MODEST_WINDOW (win));
1515 if (header_view == NULL)
1518 /* Get the account */
1519 account = get_account_from_header (header);
1524 found = modest_window_mgr_find_registered_header (mgr, header, &window);
1526 /* Do not open again the message and present the
1527 window to the user */
1530 #ifndef MODEST_TOOLKIT_HILDON2
1531 gtk_window_present (GTK_WINDOW (window));
1534 /* the header has been registered already, we don't do
1535 * anything but wait for the window to come up*/
1536 g_debug ("header %p already registered, waiting for window", header);
1541 /* Open each message */
1542 cached = tny_header_get_flags (header) & TNY_HEADER_FLAG_CACHED;
1544 /* Allways download if we are online. */
1545 if (!tny_device_is_online (modest_runtime_get_device ())) {
1548 /* If ask for user permission to download the messages */
1549 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1550 _("mcen_nc_get_msg"));
1552 /* End if the user does not want to continue */
1553 if (response == GTK_RESPONSE_CANCEL) {
1559 /* We register the window for opening */
1560 modest_window_mgr_register_header (mgr, header, NULL);
1562 /* Create the helper. We need to get a reference to the model
1563 here because it could change while the message is readed
1564 (the user could switch between folders) */
1565 helper = g_slice_new (OpenMsgHelper);
1566 helper->model = g_object_ref (gtk_tree_view_get_model (GTK_TREE_VIEW (header_view)));
1567 helper->header = g_object_ref (header);
1568 helper->rowref = gtk_tree_row_reference_copy (rowref);
1569 helper->banner_info = NULL;
1571 /* Connect to the account and perform */
1573 modest_platform_connect_and_perform ((GtkWindow *) win, TRUE, g_object_ref (account),
1574 open_msg_performer, helper);
1576 /* Call directly the performer, do not need to connect */
1577 open_msg_performer (FALSE, NULL, (GtkWindow *) win,
1578 g_object_ref (account), helper);
1583 g_object_unref (account);
1587 modest_ui_actions_on_open (GtkAction *action, ModestWindow *win)
1594 /* we check for low-mem; in that case, show a warning, and don't allow
1597 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
1601 headers = get_selected_headers (win);
1605 headers_count = tny_list_get_length (headers);
1606 if (headers_count != 1) {
1607 if (headers_count > 1) {
1608 /* Don't allow activation if there are more than one message selected */
1609 modest_platform_information_banner (NULL, NULL, _("mcen_ib_select_one_message"));
1612 g_object_unref (headers);
1616 iter = tny_list_create_iterator (headers);
1617 header = TNY_HEADER (tny_iterator_get_current (iter));
1618 g_object_unref (iter);
1622 open_msg_from_header (header, NULL, win);
1623 g_object_unref (header);
1626 g_object_unref(headers);
1630 rf_helper_window_closed (gpointer data,
1633 ReplyForwardHelper *helper = (ReplyForwardHelper *) data;
1635 helper->parent_window = NULL;
1638 static ReplyForwardHelper*
1639 create_reply_forward_helper (ReplyForwardAction action,
1641 guint reply_forward_type,
1644 ReplyForwardHelper *rf_helper = NULL;
1645 const gchar *active_acc = modest_window_get_active_account (win);
1647 rf_helper = g_slice_new0 (ReplyForwardHelper);
1648 rf_helper->reply_forward_type = reply_forward_type;
1649 rf_helper->action = action;
1650 rf_helper->parent_window = (MODEST_IS_WINDOW (win)) ? GTK_WIDGET (win) : NULL;
1651 rf_helper->header = (header) ? g_object_ref (header) : NULL;
1652 rf_helper->account_name = (active_acc) ?
1653 g_strdup (active_acc) :
1654 modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
1656 /* Note that window could be destroyed just AFTER calling
1657 register_window so we must ensure that this pointer does
1658 not hold invalid references */
1659 if (rf_helper->parent_window)
1660 g_object_weak_ref (G_OBJECT (rf_helper->parent_window),
1661 rf_helper_window_closed, rf_helper);
1667 free_reply_forward_helper (gpointer data)
1669 ReplyForwardHelper *helper;
1671 helper = (ReplyForwardHelper *) data;
1672 g_free (helper->account_name);
1674 g_object_unref (helper->header);
1675 if (helper->parent_window)
1676 g_object_weak_unref (G_OBJECT (helper->parent_window),
1677 rf_helper_window_closed, helper);
1678 g_slice_free (ReplyForwardHelper, helper);
1682 reply_forward_cb (ModestMailOperation *mail_op,
1689 TnyMsg *new_msg = NULL;
1690 ReplyForwardHelper *rf_helper;
1691 ModestWindow *msg_win = NULL;
1692 ModestEditType edit_type;
1694 TnyAccount *account = NULL;
1695 ModestWindowMgr *mgr = NULL;
1696 gchar *signature = NULL;
1697 gboolean use_signature;
1699 /* If there was any error. The mail operation could be NULL,
1700 this means that we already have the message downloaded and
1701 that we didn't do a mail operation to retrieve it */
1702 rf_helper = (ReplyForwardHelper *) user_data;
1703 if (mail_op && !modest_ui_actions_msg_retrieval_check (mail_op, header, msg))
1706 from = modest_account_mgr_get_from_string (modest_runtime_get_account_mgr(),
1707 rf_helper->account_name);
1708 signature = modest_account_mgr_get_signature (modest_runtime_get_account_mgr(),
1709 rf_helper->account_name,
1712 /* Create reply mail */
1713 switch (rf_helper->action) {
1716 modest_tny_msg_create_reply_msg (msg, header, from,
1717 (use_signature) ? signature : NULL,
1718 rf_helper->reply_forward_type,
1719 MODEST_TNY_MSG_REPLY_MODE_SENDER);
1721 case ACTION_REPLY_TO_ALL:
1723 modest_tny_msg_create_reply_msg (msg, header, from,
1724 (use_signature) ? signature : NULL,
1725 rf_helper->reply_forward_type,
1726 MODEST_TNY_MSG_REPLY_MODE_ALL);
1727 edit_type = MODEST_EDIT_TYPE_REPLY;
1729 case ACTION_FORWARD:
1731 modest_tny_msg_create_forward_msg (msg, from, (use_signature) ? signature : NULL,
1732 rf_helper->reply_forward_type);
1733 edit_type = MODEST_EDIT_TYPE_FORWARD;
1736 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1738 g_return_if_reached ();
1746 g_warning ("%s: failed to create message\n", __FUNCTION__);
1750 account = modest_tny_account_store_get_server_account (modest_runtime_get_account_store(),
1751 rf_helper->account_name,
1752 TNY_ACCOUNT_TYPE_STORE);
1754 g_warning ("%s: failed to get tnyaccount for '%s'\n", __FUNCTION__, rf_helper->account_name);
1758 /* Create and register the windows */
1759 msg_win = modest_msg_edit_window_new (new_msg, rf_helper->account_name, FALSE);
1760 mgr = modest_runtime_get_window_mgr ();
1761 modest_window_mgr_register_window (mgr, msg_win, (ModestWindow *) rf_helper->parent_window);
1763 /* Note that register_window could have deleted the account */
1764 if (MODEST_IS_WINDOW (rf_helper->parent_window)) {
1765 gdouble parent_zoom;
1767 parent_zoom = modest_window_get_zoom (MODEST_WINDOW (rf_helper->parent_window));
1768 modest_window_set_zoom (msg_win, parent_zoom);
1771 /* Show edit window */
1772 gtk_widget_show_all (GTK_WIDGET (msg_win));
1775 /* We always unregister the header because the message is
1776 forwarded or replied so the original one is no longer
1778 modest_window_mgr_unregister_header (modest_runtime_get_window_mgr (),
1781 g_object_unref (G_OBJECT (new_msg));
1783 g_object_unref (G_OBJECT (account));
1784 free_reply_forward_helper (rf_helper);
1787 /* Checks a list of headers. If any of them are not currently
1788 * downloaded (CACHED) then returns TRUE else returns FALSE.
1791 header_list_count_uncached_msgs (TnyList *header_list)
1794 gint uncached_messages = 0;
1796 iter = tny_list_create_iterator (header_list);
1797 while (!tny_iterator_is_done (iter)) {
1800 header = TNY_HEADER (tny_iterator_get_current (iter));
1802 if (!(tny_header_get_flags (header) & TNY_HEADER_FLAG_CACHED))
1803 uncached_messages ++;
1804 g_object_unref (header);
1807 tny_iterator_next (iter);
1809 g_object_unref (iter);
1811 return uncached_messages;
1814 /* Returns FALSE if the user does not want to download the
1815 * messages. Returns TRUE if the user allowed the download.
1818 connect_to_get_msg (ModestWindow *win,
1819 gint num_of_uncached_msgs,
1820 TnyAccount *account)
1822 GtkResponseType response;
1824 /* Allways download if we are online. */
1825 if (tny_device_is_online (modest_runtime_get_device ()))
1828 /* If offline, then ask for user permission to download the messages */
1829 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1830 ngettext("mcen_nc_get_msg",
1832 num_of_uncached_msgs));
1834 if (response == GTK_RESPONSE_CANCEL)
1837 return modest_platform_connect_and_wait((GtkWindow *) win, account);
1841 reply_forward_performer (gboolean canceled,
1843 GtkWindow *parent_window,
1844 TnyAccount *account,
1847 ReplyForwardHelper *rf_helper = NULL;
1848 ModestMailOperation *mail_op;
1850 rf_helper = (ReplyForwardHelper *) user_data;
1852 if (canceled || err) {
1853 free_reply_forward_helper (rf_helper);
1857 /* Retrieve the message */
1858 modest_window_mgr_register_header (modest_runtime_get_window_mgr (), rf_helper->header, NULL);
1859 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT (parent_window),
1860 modest_ui_actions_disk_operations_error_handler,
1862 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
1863 modest_mail_operation_get_msg (mail_op, rf_helper->header, TRUE, reply_forward_cb, rf_helper);
1866 g_object_unref(mail_op);
1870 * Common code for the reply and forward actions
1873 reply_forward (ReplyForwardAction action, ModestWindow *win)
1875 ReplyForwardHelper *rf_helper = NULL;
1876 guint reply_forward_type;
1878 g_return_if_fail (MODEST_IS_WINDOW(win));
1880 /* we check for low-mem; in that case, show a warning, and don't allow
1881 * reply/forward (because it could potentially require a lot of memory */
1882 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
1886 /* we need an account when editing */
1887 if (!modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE)) {
1888 if (!modest_ui_actions_run_account_setup_wizard (win))
1892 reply_forward_type =
1893 modest_conf_get_int (modest_runtime_get_conf (),
1894 (action == ACTION_FORWARD) ?
1895 MODEST_CONF_FORWARD_TYPE :
1896 MODEST_CONF_REPLY_TYPE,
1899 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
1901 TnyHeader *header = NULL;
1902 /* Get header and message. Do not free them here, the
1903 reply_forward_cb must do it */
1904 msg = modest_msg_view_window_get_message (MODEST_MSG_VIEW_WINDOW(win));
1905 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
1907 if (msg && header) {
1909 rf_helper = create_reply_forward_helper (action, win,
1910 reply_forward_type, header);
1911 reply_forward_cb (NULL, header, FALSE, msg, NULL, rf_helper);
1913 g_warning("%s: no message or header found in viewer\n", __FUNCTION__);
1917 g_object_unref (msg);
1919 g_object_unref (header);
1921 TnyHeader *header = NULL;
1923 gboolean do_retrieve = TRUE;
1924 TnyList *header_list = NULL;
1926 header_list = get_selected_headers (win);
1929 /* Check that only one message is selected for replying */
1930 if (tny_list_get_length (header_list) != 1) {
1931 modest_platform_information_banner ((win) ? GTK_WIDGET (win) : NULL,
1932 NULL, _("mcen_ib_select_one_message"));
1933 g_object_unref (header_list);
1937 /* Only reply/forward to one message */
1938 iter = tny_list_create_iterator (header_list);
1939 header = TNY_HEADER (tny_iterator_get_current (iter));
1940 g_object_unref (iter);
1942 /* Retrieve messages */
1943 do_retrieve = (action == ACTION_FORWARD) ||
1944 (reply_forward_type != MODEST_TNY_MSG_REPLY_TYPE_CITE);
1947 TnyAccount *account = NULL;
1948 TnyFolder *folder = NULL;
1949 gdouble download = TRUE;
1950 guint uncached_msgs = 0;
1952 folder = tny_header_get_folder (header);
1954 goto do_retrieve_frees;
1955 account = tny_folder_get_account (folder);
1957 goto do_retrieve_frees;
1959 uncached_msgs = header_list_count_uncached_msgs (header_list);
1961 if (uncached_msgs > 0) {
1962 /* Allways download if we are online. */
1963 if (!tny_device_is_online (modest_runtime_get_device ())) {
1966 /* If ask for user permission to download the messages */
1967 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),
1968 ngettext("mcen_nc_get_msg",
1972 /* End if the user does not want to continue */
1973 if (response == GTK_RESPONSE_CANCEL)
1980 rf_helper = create_reply_forward_helper (action, win,
1981 reply_forward_type, header);
1982 if (uncached_msgs > 0) {
1983 modest_platform_connect_and_perform (GTK_WINDOW (win),
1985 reply_forward_performer,
1988 reply_forward_performer (FALSE, NULL, GTK_WINDOW (win),
1989 account, rf_helper);
1994 g_object_unref (account);
1996 g_object_unref (folder);
1998 reply_forward_cb (NULL, header, FALSE, NULL, NULL, rf_helper);
2001 g_object_unref (header_list);
2002 g_object_unref (header);
2007 modest_ui_actions_on_reply (GtkAction *action, ModestWindow *win)
2009 g_return_if_fail (MODEST_IS_WINDOW(win));
2011 reply_forward (ACTION_REPLY, win);
2015 modest_ui_actions_on_forward (GtkAction *action, ModestWindow *win)
2017 g_return_if_fail (MODEST_IS_WINDOW(win));
2019 reply_forward (ACTION_FORWARD, win);
2023 modest_ui_actions_on_reply_all (GtkAction *action, ModestWindow *win)
2025 g_return_if_fail (MODEST_IS_WINDOW(win));
2027 reply_forward (ACTION_REPLY_TO_ALL, win);
2031 modest_ui_actions_on_next (GtkAction *action,
2032 ModestWindow *window)
2034 if (MODEST_IS_MAIN_WINDOW (window)) {
2035 GtkWidget *header_view;
2037 header_view = modest_main_window_get_child_widget (
2038 MODEST_MAIN_WINDOW(window),
2039 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2043 modest_header_view_select_next (
2044 MODEST_HEADER_VIEW(header_view));
2045 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
2046 modest_msg_view_window_select_next_message (
2047 MODEST_MSG_VIEW_WINDOW (window));
2049 g_return_if_reached ();
2054 modest_ui_actions_on_prev (GtkAction *action,
2055 ModestWindow *window)
2057 g_return_if_fail (MODEST_IS_WINDOW(window));
2059 if (MODEST_IS_MAIN_WINDOW (window)) {
2060 GtkWidget *header_view;
2061 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(window),
2062 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2066 modest_header_view_select_prev (MODEST_HEADER_VIEW(header_view));
2067 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
2068 modest_msg_view_window_select_previous_message (MODEST_MSG_VIEW_WINDOW (window));
2070 g_return_if_reached ();
2075 modest_ui_actions_on_sort (GtkAction *action,
2076 ModestWindow *window)
2078 GtkWidget *header_view = NULL;
2080 g_return_if_fail (MODEST_IS_WINDOW(window));
2082 if (MODEST_IS_MAIN_WINDOW (window)) {
2083 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(window),
2084 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2085 #ifdef MODEST_TOOLKIT_HILDON2
2086 } else if (MODEST_IS_HEADER_WINDOW (window)) {
2087 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (window)));
2092 modest_platform_information_banner (NULL, NULL, _CS("ckdg_ib_nothing_to_sort"));
2097 /* Show sorting dialog */
2098 modest_utils_run_sort_dialog (GTK_WINDOW (window), MODEST_SORT_HEADERS);
2102 new_messages_arrived (ModestMailOperation *self,
2103 TnyList *new_headers,
2107 gboolean show_visual_notifications;
2109 source = modest_mail_operation_get_source (self);
2110 show_visual_notifications = (source) ? FALSE : TRUE;
2112 g_object_unref (source);
2114 /* Notify new messages have been downloaded. If the
2115 send&receive was invoked by the user then do not show any
2116 visual notification, only play a sound and activate the LED
2117 (for the Maemo version) */
2118 if (TNY_IS_LIST(new_headers) && (tny_list_get_length (new_headers)) > 0)
2119 modest_platform_on_new_headers_received (new_headers,
2120 show_visual_notifications);
2125 retrieve_all_messages_cb (GObject *source,
2127 guint retrieve_limit)
2133 window = GTK_WINDOW (source);
2134 msg = g_strdup_printf (_("mail_nc_msg_count_limit_exceeded"),
2135 num_msgs, retrieve_limit);
2137 /* Ask the user if they want to retrieve all the messages */
2139 modest_platform_run_confirmation_dialog_with_buttons (window, msg,
2140 _("mcen_bd_get_all"),
2141 _("mcen_bd_newest_only"));
2142 /* Free and return */
2144 return (response == GTK_RESPONSE_ACCEPT) ? TRUE : FALSE;
2148 TnyAccount *account;
2150 gchar *account_name;
2151 gboolean poke_status;
2152 gboolean interactive;
2153 ModestMailOperation *mail_op;
2157 do_send_receive_performer (gboolean canceled,
2159 GtkWindow *parent_window,
2160 TnyAccount *account,
2163 SendReceiveInfo *info;
2165 info = (SendReceiveInfo *) user_data;
2167 if (err || canceled) {
2168 /* In memory full conditions we could get this error here */
2169 check_memory_full_error ((GtkWidget *) parent_window, err);
2171 if (info->mail_op) {
2172 modest_mail_operation_queue_remove (modest_runtime_get_mail_operation_queue (),
2178 /* Set send/receive operation in progress */
2179 if (info->win && MODEST_IS_MAIN_WINDOW (info->win)) {
2180 modest_main_window_notify_send_receive_initied (MODEST_MAIN_WINDOW (info->win));
2183 if (info->win && MODEST_IS_MAIN_WINDOW (info->win))
2184 g_signal_connect (G_OBJECT (info->mail_op), "operation-finished",
2185 G_CALLBACK (on_send_receive_finished),
2188 /* Send & receive. */
2189 modest_mail_operation_update_account (info->mail_op, info->account_name, info->poke_status, info->interactive,
2190 (info->win) ? retrieve_all_messages_cb : NULL,
2191 new_messages_arrived, info->win);
2196 g_object_unref (G_OBJECT (info->mail_op));
2197 if (info->account_name)
2198 g_free (info->account_name);
2200 g_object_unref (info->win);
2202 g_object_unref (info->account);
2203 g_slice_free (SendReceiveInfo, info);
2207 * This function performs the send & receive required actions. The
2208 * window is used to create the mail operation. Typically it should
2209 * always be the main window, but we pass it as argument in order to
2213 modest_ui_actions_do_send_receive (const gchar *account_name,
2214 gboolean force_connection,
2215 gboolean poke_status,
2216 gboolean interactive,
2219 gchar *acc_name = NULL;
2220 SendReceiveInfo *info;
2221 ModestTnyAccountStore *acc_store;
2223 /* If no account name was provided then get the current account, and if
2224 there is no current account then pick the default one: */
2225 if (!account_name) {
2227 acc_name = g_strdup (modest_window_get_active_account (win));
2229 acc_name = modest_account_mgr_get_default_account (modest_runtime_get_account_mgr());
2231 g_printerr ("modest: cannot get default account\n");
2235 acc_name = g_strdup (account_name);
2238 acc_store = modest_runtime_get_account_store ();
2240 /* Create the info for the connect and perform */
2241 info = g_slice_new (SendReceiveInfo);
2242 info->account_name = acc_name;
2243 info->win = (win) ? g_object_ref (win) : NULL;
2244 info->poke_status = poke_status;
2245 info->interactive = interactive;
2246 info->account = modest_tny_account_store_get_server_account (acc_store, acc_name,
2247 TNY_ACCOUNT_TYPE_STORE);
2248 /* We need to create the operation here, because otherwise it
2249 could happen that the queue emits the queue-empty signal
2250 while we're trying to connect the account */
2251 info->mail_op = modest_mail_operation_new_with_error_handling ((info->win) ? G_OBJECT (info->win) : NULL,
2252 modest_ui_actions_disk_operations_error_handler,
2254 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), info->mail_op);
2256 /* Invoke the connect and perform */
2257 modest_platform_connect_and_perform ((win) ? GTK_WINDOW (win) : NULL,
2258 force_connection, info->account,
2259 do_send_receive_performer, info);
2264 modest_ui_actions_do_cancel_send (const gchar *account_name,
2267 TnyTransportAccount *transport_account;
2268 TnySendQueue *send_queue = NULL;
2269 GError *error = NULL;
2271 /* Get transport account */
2273 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
2274 (modest_runtime_get_account_store(),
2276 TNY_ACCOUNT_TYPE_TRANSPORT));
2277 if (!transport_account) {
2278 g_printerr ("modest: no transport account found for '%s'\n", account_name);
2283 send_queue = TNY_SEND_QUEUE (modest_runtime_get_send_queue (transport_account, TRUE));
2284 if (!TNY_IS_SEND_QUEUE(send_queue)) {
2285 g_set_error (&error, MODEST_MAIL_OPERATION_ERROR,
2286 MODEST_MAIL_OPERATION_ERROR_ITEM_NOT_FOUND,
2287 "modest: could not find send queue for account\n");
2289 /* Cancel the current send */
2290 tny_account_cancel (TNY_ACCOUNT (transport_account));
2292 /* Suspend all pending messages */
2293 tny_send_queue_cancel (send_queue, TNY_SEND_QUEUE_CANCEL_ACTION_SUSPEND, &error);
2297 if (transport_account != NULL)
2298 g_object_unref (G_OBJECT (transport_account));
2302 modest_ui_actions_cancel_send_all (ModestWindow *win)
2304 GSList *account_names, *iter;
2306 account_names = modest_account_mgr_account_names (modest_runtime_get_account_mgr(),
2309 iter = account_names;
2311 modest_ui_actions_do_cancel_send ((const char*) iter->data, win);
2312 iter = g_slist_next (iter);
2315 modest_account_mgr_free_account_names (account_names);
2316 account_names = NULL;
2320 modest_ui_actions_cancel_send (GtkAction *action, ModestWindow *win)
2323 /* Check if accounts exist */
2324 gboolean accounts_exist =
2325 modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
2327 /* If not, allow the user to create an account before trying to send/receive. */
2328 if (!accounts_exist)
2329 modest_ui_actions_on_accounts (NULL, win);
2331 /* Cancel all sending operaitons */
2332 modest_ui_actions_cancel_send_all (win);
2336 * Refreshes all accounts. This function will be used by automatic
2340 modest_ui_actions_do_send_receive_all (ModestWindow *win,
2341 gboolean force_connection,
2342 gboolean poke_status,
2343 gboolean interactive)
2345 GSList *account_names, *iter;
2347 account_names = modest_account_mgr_account_names (modest_runtime_get_account_mgr(),
2350 iter = account_names;
2352 modest_ui_actions_do_send_receive ((const char*) iter->data,
2354 poke_status, interactive, win);
2355 iter = g_slist_next (iter);
2358 modest_account_mgr_free_account_names (account_names);
2359 account_names = NULL;
2363 * Handler of the click on Send&Receive button in the main toolbar
2366 modest_ui_actions_on_send_receive (GtkAction *action, ModestWindow *win)
2368 /* Check if accounts exist */
2369 gboolean accounts_exist;
2372 modest_account_mgr_has_accounts(modest_runtime_get_account_mgr(), TRUE);
2374 /* If not, allow the user to create an account before trying to send/receive. */
2375 if (!accounts_exist)
2376 modest_ui_actions_on_accounts (NULL, win);
2378 /* Refresh the current folder. The if is always TRUE it's just an extra check */
2379 if (MODEST_IS_MAIN_WINDOW (win)) {
2380 GtkWidget *folder_view;
2381 TnyFolderStore *folder_store;
2384 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
2385 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2389 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
2392 g_object_unref (folder_store);
2393 /* Refresh the active account. Force the connection if needed
2394 and poke the status of all folders */
2395 modest_ui_actions_do_send_receive (NULL, TRUE, TRUE, TRUE, win);
2396 #ifdef MODEST_TOOLKIT_HILDON2
2397 } else if (MODEST_IS_ACCOUNTS_WINDOW (win)) {
2398 modest_ui_actions_do_send_receive_all (win, TRUE, TRUE, TRUE);
2401 const gchar *active_account;
2402 active_account = modest_window_get_active_account (MODEST_WINDOW (win));
2404 modest_ui_actions_do_send_receive (active_account, TRUE, TRUE, TRUE, win);
2411 modest_ui_actions_toggle_header_list_view (GtkAction *action, ModestMainWindow *main_window)
2414 GtkWidget *header_view;
2416 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2418 header_view = modest_main_window_get_child_widget (main_window,
2419 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2423 conf = modest_runtime_get_conf ();
2425 /* what is saved/restored is depending on the style; thus; we save with
2426 * old style, then update the style, and restore for this new style
2428 modest_widget_memory_save (conf, G_OBJECT(header_view), MODEST_CONF_HEADER_VIEW_KEY);
2430 if (modest_header_view_get_style
2431 (MODEST_HEADER_VIEW(header_view)) == MODEST_HEADER_VIEW_STYLE_DETAILS)
2432 modest_header_view_set_style (MODEST_HEADER_VIEW(header_view),
2433 MODEST_HEADER_VIEW_STYLE_TWOLINES);
2435 modest_header_view_set_style (MODEST_HEADER_VIEW(header_view),
2436 MODEST_HEADER_VIEW_STYLE_DETAILS);
2438 modest_widget_memory_restore (conf, G_OBJECT(header_view),
2439 MODEST_CONF_HEADER_VIEW_KEY);
2444 modest_ui_actions_on_header_selected (ModestHeaderView *header_view,
2446 ModestMainWindow *main_window)
2448 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2449 g_return_if_fail (MODEST_IS_HEADER_VIEW (header_view));
2451 /* in the case the folder is empty, show the empty folder message and focus
2453 if (!header && gtk_widget_is_focus (GTK_WIDGET (header_view))) {
2454 if (modest_header_view_is_empty (header_view)) {
2455 TnyFolder *folder = modest_header_view_get_folder (header_view);
2456 GtkWidget *folder_view =
2457 modest_main_window_get_child_widget (main_window,
2458 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2459 if (folder != NULL) {
2460 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view), folder, FALSE);
2461 g_object_unref (folder);
2463 gtk_widget_grab_focus (GTK_WIDGET (folder_view));
2467 /* If no header has been selected then exit */
2472 if (!gtk_widget_is_focus (GTK_WIDGET(header_view)))
2473 gtk_widget_grab_focus (GTK_WIDGET(header_view));
2475 /* Update toolbar dimming state */
2476 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
2477 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
2481 modest_ui_actions_on_header_activated (ModestHeaderView *header_view,
2484 ModestWindow *window)
2486 GtkWidget *open_widget;
2487 GtkTreeRowReference *rowref;
2489 g_return_if_fail (MODEST_IS_WINDOW(window));
2490 g_return_if_fail (MODEST_IS_HEADER_VIEW (header_view));
2491 g_return_if_fail (TNY_IS_HEADER (header));
2493 if (modest_header_view_count_selected_headers (header_view) > 1) {
2494 /* Don't allow activation if there are more than one message selected */
2495 modest_platform_information_banner (NULL, NULL, _("mcen_ib_select_one_message"));
2499 /* we check for low-mem; in that case, show a warning, and don't allow
2500 * activating headers
2502 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
2505 if (MODEST_IS_MAIN_WINDOW (window)) {
2506 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (window));
2507 open_widget = modest_window_get_action_widget (MODEST_WINDOW (window), "/MenuBar/EmailMenu/EmailOpenMenu");
2508 if (!GTK_WIDGET_IS_SENSITIVE (open_widget))
2512 rowref = gtk_tree_row_reference_new (gtk_tree_view_get_model (GTK_TREE_VIEW (header_view)), path);
2513 open_msg_from_header (header, rowref, MODEST_WINDOW (window));
2514 gtk_tree_row_reference_free (rowref);
2518 set_active_account_from_tny_account (TnyAccount *account,
2519 ModestWindow *window)
2521 const gchar *server_acc_name = tny_account_get_id (account);
2523 /* We need the TnyAccount provided by the
2524 account store because that is the one that
2525 knows the name of the Modest account */
2526 TnyAccount *modest_server_account = modest_server_account =
2527 modest_tny_account_store_get_tny_account_by (modest_runtime_get_account_store (),
2528 MODEST_TNY_ACCOUNT_STORE_QUERY_ID,
2530 if (!modest_server_account) {
2531 g_warning ("%s: could not get tny account\n", __FUNCTION__);
2535 /* Update active account, but only if it's not a pseudo-account */
2536 if ((!modest_tny_account_is_virtual_local_folders(modest_server_account)) &&
2537 (!modest_tny_account_is_memory_card_account(modest_server_account))) {
2538 const gchar *modest_acc_name =
2539 modest_tny_account_get_parent_modest_account_name_for_server_account (modest_server_account);
2540 if (modest_acc_name)
2541 modest_window_set_active_account (window, modest_acc_name);
2544 g_object_unref (modest_server_account);
2549 folder_refreshed_cb (ModestMailOperation *mail_op,
2553 ModestMainWindow *win = NULL;
2554 GtkWidget *folder_view;
2555 const GError *error;
2557 g_return_if_fail (TNY_IS_FOLDER (folder));
2559 win = MODEST_MAIN_WINDOW (user_data);
2561 /* Check if the operation failed due to memory low conditions */
2562 error = modest_mail_operation_get_error (mail_op);
2563 if (error && error->domain == MODEST_MAIL_OPERATION_ERROR &&
2564 error->code == MODEST_MAIL_OPERATION_ERROR_LOW_MEMORY) {
2565 modest_platform_run_information_dialog (GTK_WINDOW (win),
2566 _KR("memr_ib_operation_disabled"),
2572 modest_main_window_get_child_widget(win, MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
2575 TnyFolderStore *current_folder;
2577 current_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
2578 if (current_folder) {
2579 gboolean different = ((TnyFolderStore *) folder != current_folder);
2580 g_object_unref (current_folder);
2586 /* Check if folder is empty and set headers view contents style */
2587 if (tny_folder_get_all_count (folder) == 0)
2588 modest_main_window_set_contents_style (win,
2589 MODEST_MAIN_WINDOW_CONTENTS_STYLE_EMPTY);
2594 modest_ui_actions_on_folder_selection_changed (ModestFolderView *folder_view,
2595 TnyFolderStore *folder_store,
2597 ModestMainWindow *main_window)
2600 GtkWidget *header_view;
2602 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
2604 header_view = modest_main_window_get_child_widget(main_window,
2605 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2609 conf = modest_runtime_get_conf ();
2611 if (TNY_IS_ACCOUNT (folder_store)) {
2613 set_active_account_from_tny_account (TNY_ACCOUNT (folder_store), MODEST_WINDOW (main_window));
2615 /* Show account details */
2616 modest_main_window_set_contents_style (main_window, MODEST_MAIN_WINDOW_CONTENTS_STYLE_DETAILS);
2619 if (TNY_IS_FOLDER (folder_store) && selected) {
2620 TnyAccount *account;
2621 const gchar *account_name = NULL;
2623 /* Update the active account */
2624 account = modest_tny_folder_get_account (TNY_FOLDER (folder_store));
2626 set_active_account_from_tny_account (account, MODEST_WINDOW (main_window));
2628 modest_tny_account_get_parent_modest_account_name_for_server_account (account);
2629 g_object_unref (account);
2633 /* Set the header style by default, it could
2634 be changed later by the refresh callback to
2636 modest_main_window_set_contents_style (main_window,
2637 MODEST_MAIN_WINDOW_CONTENTS_STYLE_HEADERS);
2639 /* Set folder on header view. This function
2640 will call tny_folder_refresh_async so we
2641 pass a callback that will be called when
2642 finished. We use that callback to set the
2643 empty view if there are no messages */
2644 modest_header_view_set_folder (MODEST_HEADER_VIEW(header_view),
2645 TNY_FOLDER (folder_store),
2647 MODEST_WINDOW (main_window),
2648 folder_refreshed_cb,
2651 /* Restore configuration. We need to do this
2652 *after* the set_folder because the widget
2653 memory asks the header view about its
2655 modest_widget_memory_restore (modest_runtime_get_conf (),
2656 G_OBJECT(header_view),
2657 MODEST_CONF_HEADER_VIEW_KEY);
2659 /* No need to save the header view
2660 configuration for Maemo because it only
2661 saves the sorting stuff and that it's
2662 already being done by the sort
2663 dialog. Remove it when the GNOME version
2664 has the same behaviour */
2665 #ifdef MODEST_TOOLKIT_GTK
2666 if (modest_main_window_get_contents_style (main_window) ==
2667 MODEST_MAIN_WINDOW_CONTENTS_STYLE_HEADERS)
2668 modest_widget_memory_save (conf, G_OBJECT (header_view),
2669 MODEST_CONF_HEADER_VIEW_KEY);
2671 modest_header_view_clear (MODEST_HEADER_VIEW(header_view));
2675 /* Update dimming state */
2676 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (main_window));
2677 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (main_window));
2681 modest_ui_actions_on_item_not_found (ModestHeaderView *header_view,ModestItemType type,
2688 item = (type == MODEST_ITEM_TYPE_FOLDER) ? "folder" : "message";
2690 online = tny_device_is_online (modest_runtime_get_device());
2693 /* already online -- the item is simply not there... */
2694 dialog = gtk_message_dialog_new (GTK_WINDOW (win),
2696 GTK_MESSAGE_WARNING,
2698 _("The %s you selected cannot be found"),
2700 gtk_dialog_add_button (GTK_DIALOG (dialog),_("mcen_bd_dialog_ok"), GTK_RESPONSE_ACCEPT);
2701 gtk_dialog_run (GTK_DIALOG(dialog));
2703 dialog = gtk_dialog_new_with_buttons (_("Connection requested"),
2706 _("mcen_bd_dialog_cancel"),
2707 GTK_RESPONSE_REJECT,
2708 _("mcen_bd_dialog_ok"),
2709 GTK_RESPONSE_ACCEPT,
2711 txt = g_strdup_printf (_("This %s is not available in offline mode.\n"
2712 "Do you want to get online?"), item);
2713 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox),
2714 gtk_label_new (txt), FALSE, FALSE, 0);
2715 gtk_widget_show_all (GTK_WIDGET(GTK_DIALOG(dialog)->vbox));
2718 gtk_window_set_default_size (GTK_WINDOW(dialog), 300, 300);
2719 if (gtk_dialog_run (GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
2720 /* TODO: Comment about why is this commented out: */
2721 /* modest_platform_connect_and_wait (); */
2724 gtk_widget_destroy (dialog);
2728 modest_ui_actions_on_msg_link_hover (ModestMsgView *msgview, const gchar* link,
2731 /* g_message ("%s %s", __FUNCTION__, link); */
2736 modest_ui_actions_on_msg_link_clicked (ModestMsgView *msgview, const gchar* link,
2739 modest_platform_activate_uri (link);
2743 modest_ui_actions_on_msg_link_contextual (ModestMsgView *msgview, const gchar* link,
2746 modest_platform_show_uri_popup (link);
2750 modest_ui_actions_on_msg_attachment_clicked (ModestMsgView *msgview, TnyMimePart *mime_part,
2753 /* we check for low-mem; in that case, show a warning, and don't allow
2754 * viewing attachments
2756 if (modest_platform_check_memory_low (MODEST_WINDOW(win), TRUE))
2759 modest_msg_view_window_view_attachment (MODEST_MSG_VIEW_WINDOW (win), mime_part);
2763 modest_ui_actions_on_msg_recpt_activated (ModestMsgView *msgview,
2764 const gchar *address,
2767 /* g_message ("%s %s", __FUNCTION__, address); */
2771 on_save_to_drafts_cb (ModestMailOperation *mail_op,
2772 TnyMsg *saved_draft,
2775 ModestMsgEditWindow *edit_window;
2777 /* TODO: in hildon 2 we have to dim and undim the header views while we're saving */
2778 #ifndef MODEST_TOOLKIT_HILDON2
2779 ModestMainWindow *win;
2781 /* FIXME. Make the header view sensitive again. This is a
2782 * temporary hack. See modest_ui_actions_on_save_to_drafts()
2784 win = MODEST_MAIN_WINDOW(modest_window_mgr_get_main_window(
2785 modest_runtime_get_window_mgr(), FALSE));
2787 GtkWidget *hdrview = modest_main_window_get_child_widget(
2788 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2789 if (hdrview) gtk_widget_set_sensitive(hdrview, TRUE);
2793 edit_window = MODEST_MSG_EDIT_WINDOW (user_data);
2795 /* Set draft is there was no error */
2796 if (!modest_mail_operation_get_error (mail_op))
2797 modest_msg_edit_window_set_draft (edit_window, saved_draft);
2799 g_object_unref(edit_window);
2803 enough_space_for_message (ModestMsgEditWindow *edit_window,
2806 TnyAccountStore *acc_store;
2807 guint64 available_disk, expected_size;
2812 acc_store = TNY_ACCOUNT_STORE (modest_runtime_get_account_store());
2813 available_disk = modest_utils_get_available_space (NULL);
2814 modest_msg_edit_window_get_parts_size (edit_window, &parts_count, &parts_size);
2815 expected_size = modest_tny_msg_estimate_size (data->plain_body,
2820 /* Double check: memory full condition or message too big */
2821 if (available_disk < MIN_FREE_SPACE ||
2822 expected_size > available_disk) {
2824 modest_platform_information_banner (NULL, NULL,
2825 _KR("cerm_device_memory_full"));
2830 * djcb: if we're in low-memory state, we only allow for
2831 * saving messages smaller than
2832 * MODEST_MAX_LOW_MEMORY_MESSAGE_SIZE (see modest-defs.h) this
2833 * should still allow for sending anything critical...
2835 if ((expected_size > MODEST_MAX_LOW_MEMORY_MESSAGE_SIZE) &&
2836 modest_platform_check_memory_low (MODEST_WINDOW(edit_window), TRUE))
2840 * djcb: we also make sure that the attachments are smaller than the max size
2841 * this is for the case where we'd try to forward a message with attachments
2842 * bigger than our max allowed size, or sending an message from drafts which
2843 * somehow got past our checks when attaching.
2845 if (expected_size > MODEST_MAX_ATTACHMENT_SIZE) {
2846 modest_platform_run_information_dialog (
2847 GTK_WINDOW(edit_window),
2848 _KR("memr_ib_operation_disabled"),
2857 modest_ui_actions_on_save_to_drafts (GtkWidget *widget, ModestMsgEditWindow *edit_window)
2859 TnyTransportAccount *transport_account;
2860 ModestMailOperation *mail_operation;
2862 gchar *account_name, *from;
2863 ModestAccountMgr *account_mgr;
2864 gboolean had_error = FALSE;
2865 ModestMainWindow *win = NULL;
2867 g_return_val_if_fail (MODEST_IS_MSG_EDIT_WINDOW(edit_window), FALSE);
2869 data = modest_msg_edit_window_get_msg_data (edit_window);
2872 if (!enough_space_for_message (edit_window, data)) {
2873 modest_msg_edit_window_free_msg_data (edit_window, data);
2877 account_name = g_strdup (data->account_name);
2878 account_mgr = modest_runtime_get_account_mgr();
2880 account_name = g_strdup(modest_window_get_active_account (MODEST_WINDOW(edit_window)));
2882 account_name = modest_account_mgr_get_default_account (account_mgr);
2883 if (!account_name) {
2884 g_printerr ("modest: no account found\n");
2885 modest_msg_edit_window_free_msg_data (edit_window, data);
2889 if (!strcmp (account_name, MODEST_LOCAL_FOLDERS_ACCOUNT_ID)) {
2890 account_name = g_strdup (data->account_name);
2894 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
2895 (modest_runtime_get_account_store (),
2897 TNY_ACCOUNT_TYPE_TRANSPORT));
2898 if (!transport_account) {
2899 g_printerr ("modest: no transport account found for '%s'\n", account_name);
2900 g_free (account_name);
2901 modest_msg_edit_window_free_msg_data (edit_window, data);
2904 from = modest_account_mgr_get_from_string (account_mgr, account_name);
2906 /* Create the mail operation */
2907 mail_operation = modest_mail_operation_new_with_error_handling (NULL, modest_ui_actions_disk_operations_error_handler,
2909 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_operation);
2911 modest_mail_operation_save_to_drafts (mail_operation,
2923 data->priority_flags,
2924 on_save_to_drafts_cb,
2925 g_object_ref(edit_window));
2927 #ifdef MODEST_TOOLKIT_HILDON2
2928 /* In hildon2 we always show the information banner on saving to drafts.
2929 * It will be a system information banner in this case.
2931 gchar *text = g_strdup_printf (_("mail_va_saved_to_drafts"), _("mcen_me_folder_drafts"));
2932 modest_platform_information_banner (NULL, NULL, text);
2935 /* Use the main window as the parent of the banner, if the
2936 main window does not exist it won't be shown, if the parent
2937 window exists then it's properly shown. We don't use the
2938 editor window because it could be closed (save to drafts
2939 could happen after closing the window */
2940 win = (ModestMainWindow *)
2941 modest_window_mgr_get_main_window( modest_runtime_get_window_mgr(), FALSE);
2943 gchar *text = g_strdup_printf (_("mail_va_saved_to_drafts"), _("mcen_me_folder_drafts"));
2944 modest_platform_information_banner (GTK_WIDGET (win), NULL, text);
2948 modest_msg_edit_window_set_modified (edit_window, FALSE);
2952 g_free (account_name);
2953 g_object_unref (G_OBJECT (transport_account));
2954 g_object_unref (G_OBJECT (mail_operation));
2956 modest_msg_edit_window_free_msg_data (edit_window, data);
2959 * If the drafts folder is selected then make the header view
2960 * insensitive while the message is being saved to drafts
2961 * (it'll be sensitive again in on_save_to_drafts_cb()). This
2962 * is not very clean but it avoids letting the drafts folder
2963 * in an inconsistent state: the user could edit the message
2964 * being saved and undesirable things would happen.
2965 * In the average case the user won't notice anything at
2966 * all. In the worst case (the user is editing a really big
2967 * file from Drafts) the header view will be insensitive
2968 * during the saving process (10 or 20 seconds, depending on
2969 * the message). Anyway this is just a quick workaround: once
2970 * we find a better solution it should be removed
2971 * See NB#65125 (commend #18) for details.
2973 if (!had_error && win != NULL) {
2974 ModestFolderView *view = MODEST_FOLDER_VIEW(modest_main_window_get_child_widget(
2975 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW));
2977 TnyFolder *folder = TNY_FOLDER(modest_folder_view_get_selected(view));
2979 if (modest_tny_folder_is_local_folder(folder)) {
2980 TnyFolderType folder_type;
2981 folder_type = modest_tny_folder_get_local_or_mmc_folder_type(folder);
2982 if (folder_type == TNY_FOLDER_TYPE_DRAFTS) {
2983 GtkWidget *hdrview = modest_main_window_get_child_widget(
2984 win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
2985 if (hdrview) gtk_widget_set_sensitive(hdrview, FALSE);
2989 if (folder != NULL) g_object_unref(folder);
2996 /* For instance, when clicking the Send toolbar button when editing a message: */
2998 modest_ui_actions_on_send (GtkWidget *widget, ModestMsgEditWindow *edit_window)
3000 TnyTransportAccount *transport_account = NULL;
3001 gboolean had_error = FALSE;
3003 ModestAccountMgr *account_mgr;
3004 gchar *account_name;
3006 ModestMailOperation *mail_operation;
3008 g_return_val_if_fail (MODEST_IS_MSG_EDIT_WINDOW(edit_window), TRUE);
3010 if (!modest_msg_edit_window_check_names (edit_window, TRUE))
3013 data = modest_msg_edit_window_get_msg_data (edit_window);
3016 if (!enough_space_for_message (edit_window, data)) {
3017 modest_msg_edit_window_free_msg_data (edit_window, data);
3021 account_mgr = modest_runtime_get_account_mgr();
3022 account_name = g_strdup (data->account_name);
3024 account_name = g_strdup(modest_window_get_active_account (MODEST_WINDOW(edit_window)));
3027 account_name = modest_account_mgr_get_default_account (account_mgr);
3029 if (!account_name) {
3030 modest_msg_edit_window_free_msg_data (edit_window, data);
3031 /* Run account setup wizard */
3032 if (!modest_ui_actions_run_account_setup_wizard (MODEST_WINDOW(edit_window))) {
3037 /* Get the currently-active transport account for this modest account: */
3038 if (strcmp (account_name, MODEST_LOCAL_FOLDERS_ACCOUNT_ID) != 0) {
3040 TNY_TRANSPORT_ACCOUNT(modest_tny_account_store_get_server_account
3041 (modest_runtime_get_account_store (),
3042 account_name, TNY_ACCOUNT_TYPE_TRANSPORT));
3045 if (!transport_account) {
3046 modest_msg_edit_window_free_msg_data (edit_window, data);
3047 /* Run account setup wizard */
3048 if (!modest_ui_actions_run_account_setup_wizard(MODEST_WINDOW(edit_window)))
3053 /* Create the mail operation */
3054 from = modest_account_mgr_get_from_string (account_mgr, account_name);
3055 mail_operation = modest_mail_operation_new_with_error_handling (NULL, modest_ui_actions_disk_operations_error_handler, NULL, NULL);
3056 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_operation);
3058 modest_mail_operation_send_new_mail (mail_operation,
3070 data->priority_flags);
3072 if (modest_mail_operation_get_status (mail_operation) == MODEST_MAIL_OPERATION_STATUS_IN_PROGRESS)
3073 modest_platform_information_banner (NULL, NULL, _("mcen_ib_outbox_waiting_to_be_sent"));
3076 if (modest_mail_operation_get_error (mail_operation) != NULL) {
3077 const GError *error = modest_mail_operation_get_error (mail_operation);
3078 if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
3079 error->code == MODEST_MAIL_OPERATION_ERROR_INSTANCE_CREATION_FAILED) {
3080 g_warning ("%s failed: %s\n", __FUNCTION__, (modest_mail_operation_get_error (mail_operation))->message);
3081 modest_platform_information_banner (NULL, NULL, _CS("sfil_ni_not_enough_memory"));
3088 g_free (account_name);
3089 g_object_unref (G_OBJECT (transport_account));
3090 g_object_unref (G_OBJECT (mail_operation));
3092 modest_msg_edit_window_free_msg_data (edit_window, data);
3095 modest_msg_edit_window_set_sent (edit_window, TRUE);
3097 /* Save settings and close the window: */
3098 modest_ui_actions_on_close_window (NULL, MODEST_WINDOW (edit_window));
3105 modest_ui_actions_on_toggle_bold (GtkToggleAction *action,
3106 ModestMsgEditWindow *window)
3108 ModestMsgEditFormatState *format_state = NULL;
3110 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3111 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3113 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW (window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3116 format_state = modest_msg_edit_window_get_format_state (window);
3117 g_return_if_fail (format_state != NULL);
3119 format_state->bold = gtk_toggle_action_get_active (action);
3120 modest_msg_edit_window_set_format_state (window, format_state);
3121 g_free (format_state);
3126 modest_ui_actions_on_toggle_italics (GtkToggleAction *action,
3127 ModestMsgEditWindow *window)
3129 ModestMsgEditFormatState *format_state = NULL;
3131 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3132 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3134 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3137 format_state = modest_msg_edit_window_get_format_state (window);
3138 g_return_if_fail (format_state != NULL);
3140 format_state->italics = gtk_toggle_action_get_active (action);
3141 modest_msg_edit_window_set_format_state (window, format_state);
3142 g_free (format_state);
3147 modest_ui_actions_on_toggle_bullets (GtkToggleAction *action,
3148 ModestMsgEditWindow *window)
3150 ModestMsgEditFormatState *format_state = NULL;
3152 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3153 g_return_if_fail (GTK_IS_TOGGLE_ACTION (action));
3155 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW (window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3158 format_state = modest_msg_edit_window_get_format_state (window);
3159 g_return_if_fail (format_state != NULL);
3161 format_state->bullet = gtk_toggle_action_get_active (action);
3162 modest_msg_edit_window_set_format_state (window, format_state);
3163 g_free (format_state);
3168 modest_ui_actions_on_change_justify (GtkRadioAction *action,
3169 GtkRadioAction *selected,
3170 ModestMsgEditWindow *window)
3172 ModestMsgEditFormatState *format_state = NULL;
3173 GtkJustification value;
3175 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3177 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3180 value = gtk_radio_action_get_current_value (selected);
3182 format_state = modest_msg_edit_window_get_format_state (window);
3183 g_return_if_fail (format_state != NULL);
3185 format_state->justification = value;
3186 modest_msg_edit_window_set_format_state (window, format_state);
3187 g_free (format_state);
3191 modest_ui_actions_on_select_editor_color (GtkAction *action,
3192 ModestMsgEditWindow *window)
3194 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3195 g_return_if_fail (GTK_IS_ACTION (action));
3197 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3200 modest_msg_edit_window_select_color (window);
3204 modest_ui_actions_on_select_editor_background_color (GtkAction *action,
3205 ModestMsgEditWindow *window)
3207 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3208 g_return_if_fail (GTK_IS_ACTION (action));
3210 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3213 modest_msg_edit_window_select_background_color (window);
3217 modest_ui_actions_on_insert_image (GObject *object,
3218 ModestMsgEditWindow *window)
3220 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3223 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
3226 if (modest_msg_edit_window_get_format (MODEST_MSG_EDIT_WINDOW(window)) == MODEST_MSG_EDIT_FORMAT_TEXT)
3229 modest_msg_edit_window_insert_image (window);
3233 modest_ui_actions_on_attach_file (GtkAction *action,
3234 ModestMsgEditWindow *window)
3236 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3237 g_return_if_fail (GTK_IS_ACTION (action));
3239 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
3242 modest_msg_edit_window_offer_attach_file (window);
3246 modest_ui_actions_on_remove_attachments (GtkAction *action,
3247 ModestMsgEditWindow *window)
3249 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
3251 modest_msg_edit_window_remove_attachments (window, NULL);
3255 do_create_folder_cb (ModestMailOperation *mail_op,
3256 TnyFolderStore *parent_folder,
3257 TnyFolder *new_folder,
3260 gchar *suggested_name = (gchar *) user_data;
3261 GtkWindow *source_win = (GtkWindow *) modest_mail_operation_get_source (mail_op);
3262 const GError *error;
3264 error = modest_mail_operation_get_error (mail_op);
3267 /* Show an error. If there was some problem writing to
3268 disk, show it, otherwise show the generic folder
3269 create error. We do it here and not in an error
3270 handler because the call to do_create_folder will
3271 stop the main loop in a gtk_dialog_run and then,
3272 the message won't be shown until that dialog is
3274 modest_ui_actions_disk_operations_error_handler (mail_op,
3275 _("mail_in_ui_folder_create_error"));
3277 if (!is_memory_full_error ((GError *) error, mail_op)) {
3278 /* Try again if there is no full memory condition */
3279 do_create_folder (source_win, parent_folder, (const gchar *) suggested_name);
3282 /* the 'source_win' is either the ModestMainWindow, or the 'Move to folder'-dialog
3283 * FIXME: any other? */
3284 GtkWidget *folder_view;
3286 if (MODEST_IS_MAIN_WINDOW(source_win))
3288 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (source_win),
3289 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3291 folder_view = GTK_WIDGET(g_object_get_data (G_OBJECT (source_win),
3292 MODEST_MOVE_TO_DIALOG_FOLDER_VIEW));
3294 /* Select the newly created folder. It could happen
3295 that the widget is no longer there (i.e. the window
3296 has been destroyed, so we need to check this */
3298 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view),
3300 g_object_unref (new_folder);
3302 /* Free. Note that the first time it'll be NULL so noop */
3303 g_free (suggested_name);
3304 g_object_unref (source_win);
3309 TnyFolderStore *parent;
3310 } CreateFolderConnect;
3313 do_create_folder_performer (gboolean canceled,
3315 GtkWindow *parent_window,
3316 TnyAccount *account,
3319 CreateFolderConnect *helper = (CreateFolderConnect *) user_data;
3320 ModestMailOperation *mail_op;
3322 if (canceled || err) {
3323 /* In memory full conditions we could get this error here */
3324 check_memory_full_error ((GtkWidget *) parent_window, err);
3328 mail_op = modest_mail_operation_new ((GObject *) parent_window);
3329 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3331 modest_mail_operation_create_folder (mail_op,
3333 (const gchar *) helper->folder_name,
3334 do_create_folder_cb,
3335 g_strdup (helper->folder_name));
3336 g_object_unref (mail_op);
3340 g_object_unref (helper->parent);
3341 if (helper->folder_name)
3342 g_free (helper->folder_name);
3343 g_slice_free (CreateFolderConnect, helper);
3348 do_create_folder (GtkWindow *parent_window,
3349 TnyFolderStore *suggested_parent,
3350 const gchar *suggested_name)
3353 gchar *folder_name = NULL;
3354 TnyFolderStore *parent_folder = NULL;
3356 result = modest_platform_run_new_folder_dialog (GTK_WINDOW (parent_window),
3358 (gchar *) suggested_name,
3362 if (result == GTK_RESPONSE_ACCEPT && parent_folder) {
3363 CreateFolderConnect *helper = (CreateFolderConnect *) g_slice_new0 (CreateFolderConnect);
3364 helper->folder_name = g_strdup (folder_name);
3365 helper->parent = g_object_ref (parent_folder);
3367 modest_platform_connect_if_remote_and_perform (GTK_WINDOW (parent_window),
3370 do_create_folder_performer,
3375 g_free (folder_name);
3377 g_object_unref (parent_folder);
3381 modest_ui_actions_create_folder(GtkWidget *parent_window,
3382 GtkWidget *folder_view)
3384 TnyFolderStore *parent_folder;
3386 #ifdef MODEST_TOOLKIT_HILDON2
3387 ModestTnyAccountStore *acc_store;
3389 acc_store = modest_runtime_get_account_store ();
3391 parent_folder = (TnyFolderStore *)
3392 modest_tny_account_store_get_local_folders_account (acc_store);
3394 parent_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW(folder_view));
3397 if (parent_folder) {
3398 do_create_folder (GTK_WINDOW (parent_window), parent_folder, NULL);
3399 g_object_unref (parent_folder);
3404 modest_ui_actions_on_new_folder (GtkAction *action, ModestWindow *window)
3407 g_return_if_fail (MODEST_IS_WINDOW(window));
3409 if (MODEST_IS_MAIN_WINDOW (window)) {
3410 GtkWidget *folder_view;
3412 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3413 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3417 modest_ui_actions_create_folder (GTK_WIDGET (window), folder_view);
3418 #ifdef MODEST_TOOLKIT_HILDON2
3419 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3420 GtkWidget *folder_view;
3422 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3423 modest_ui_actions_create_folder (GTK_WIDGET (window), folder_view);
3426 g_assert_not_reached ();
3431 modest_ui_actions_rename_folder_error_handler (ModestMailOperation *mail_op,
3434 const GError *error = NULL;
3435 const gchar *message = NULL;
3437 /* Get error message */
3438 error = modest_mail_operation_get_error (mail_op);
3440 g_return_if_reached ();
3442 if (error->domain == MODEST_MAIL_OPERATION_ERROR &&
3443 error->code == MODEST_MAIL_OPERATION_ERROR_FOLDER_EXISTS) {
3444 message = _CS("ckdg_ib_folder_already_exists");
3445 } else if (error->domain == TNY_ERROR_DOMAIN &&
3446 error->code == TNY_SERVICE_ERROR_STATE) {
3447 /* This means that the folder is already in use (a
3448 message is opened for example */
3449 message = _("emev_ni_internal_error");
3451 message = _CS("ckdg_ib_unable_to_rename");
3454 /* We don't set a parent for the dialog because the dialog
3455 will be destroyed so the banner won't appear */
3456 modest_platform_information_banner (NULL, NULL, message);
3460 TnyFolderStore *folder;
3465 on_rename_folder_cb (ModestMailOperation *mail_op,
3466 TnyFolder *new_folder,
3469 ModestFolderView *folder_view;
3471 /* If the window was closed when renaming a folder, or if
3472 * it's not a main window this will happen */
3473 if (!MODEST_IS_FOLDER_VIEW (user_data))
3476 folder_view = MODEST_FOLDER_VIEW (user_data);
3477 /* Note that if the rename fails new_folder will be NULL */
3479 modest_folder_view_select_folder (folder_view, new_folder, FALSE);
3481 modest_folder_view_select_first_inbox_or_local (folder_view);
3483 gtk_widget_grab_focus (GTK_WIDGET (folder_view));
3487 on_rename_folder_performer (gboolean canceled,
3489 GtkWindow *parent_window,
3490 TnyAccount *account,
3493 ModestMailOperation *mail_op = NULL;
3494 GtkTreeSelection *sel = NULL;
3495 GtkWidget *folder_view = NULL;
3496 RenameFolderInfo *data = (RenameFolderInfo*)user_data;
3498 if (canceled || err) {
3499 /* In memory full conditions we could get this error here */
3500 check_memory_full_error ((GtkWidget *) parent_window, err);
3504 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
3505 modest_ui_actions_rename_folder_error_handler,
3506 parent_window, NULL);
3508 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3511 if (MODEST_IS_MAIN_WINDOW(parent_window)) {
3513 folder_view = modest_main_window_get_child_widget (
3514 MODEST_MAIN_WINDOW (parent_window),
3515 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3517 #ifdef MODEST_TOOLKIT_HILDON2
3518 else if (MODEST_IS_FOLDER_WINDOW (parent_window)) {
3519 ModestFolderWindow *folder_window = (ModestFolderWindow *) parent_window;
3520 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (folder_window));
3524 /* Clear the folders view */
3525 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_view));
3526 gtk_tree_selection_unselect_all (sel);
3528 /* Actually rename the folder */
3529 modest_mail_operation_rename_folder (mail_op,
3530 TNY_FOLDER (data->folder),
3531 (const gchar *) (data->new_name),
3532 on_rename_folder_cb,
3534 g_object_unref (mail_op);
3537 g_object_unref (data->folder);
3538 g_free (data->new_name);
3543 modest_ui_actions_on_rename_folder (GtkAction *action,
3544 ModestWindow *window)
3546 modest_ui_actions_on_edit_mode_rename_folder (window);
3550 modest_ui_actions_on_edit_mode_rename_folder (ModestWindow *window)
3552 TnyFolderStore *folder;
3553 GtkWidget *folder_view;
3554 gboolean do_rename = TRUE;
3556 g_return_val_if_fail (MODEST_IS_WINDOW(window), FALSE);
3558 if (MODEST_IS_MAIN_WINDOW (window)) {
3559 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3560 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3564 #ifdef MODEST_TOOLKIT_HILDON2
3565 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3566 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3572 folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW(folder_view));
3577 if (TNY_IS_FOLDER (folder)) {
3578 gchar *folder_name = NULL;
3580 const gchar *current_name;
3581 TnyFolderStore *parent;
3583 current_name = tny_folder_get_name (TNY_FOLDER (folder));
3584 parent = tny_folder_get_folder_store (TNY_FOLDER (folder));
3585 response = modest_platform_run_rename_folder_dialog (GTK_WINDOW (window),
3586 parent, current_name,
3588 g_object_unref (parent);
3590 if (response != GTK_RESPONSE_ACCEPT || strlen (folder_name) == 0) {
3593 RenameFolderInfo *rename_folder_data = g_new0 (RenameFolderInfo, 1);
3594 rename_folder_data->folder = g_object_ref (folder);
3595 rename_folder_data->new_name = folder_name;
3596 modest_platform_connect_if_remote_and_perform (GTK_WINDOW(window), TRUE,
3597 folder, on_rename_folder_performer, rename_folder_data);
3600 g_object_unref (folder);
3605 modest_ui_actions_delete_folder_error_handler (ModestMailOperation *mail_op,
3608 GObject *win = modest_mail_operation_get_source (mail_op);
3610 modest_platform_run_information_dialog ((win) ? GTK_WINDOW (win) : NULL,
3611 _("mail_in_ui_folder_delete_error"),
3613 g_object_unref (win);
3617 TnyFolderStore *folder;
3618 gboolean move_to_trash;
3622 on_delete_folder_cb (gboolean canceled,
3624 GtkWindow *parent_window,
3625 TnyAccount *account,
3628 DeleteFolderInfo *info = (DeleteFolderInfo*) user_data;
3629 GtkWidget *folder_view;
3630 ModestMailOperation *mail_op;
3631 GtkTreeSelection *sel;
3633 if (!MODEST_IS_WINDOW(parent_window) || canceled || (err!=NULL)) {
3634 g_object_unref (G_OBJECT (info->folder));
3639 if (MODEST_IS_MAIN_WINDOW (parent_window)) {
3640 folder_view = modest_main_window_get_child_widget (
3641 MODEST_MAIN_WINDOW (parent_window),
3642 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3643 #ifdef MODEST_TOOLKIT_HILDON2
3644 } else if (MODEST_IS_FOLDER_WINDOW (parent_window)) {
3645 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (parent_window)));
3648 g_object_unref (G_OBJECT (info->folder));
3653 /* Unselect the folder before deleting it to free the headers */
3654 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_view));
3655 gtk_tree_selection_unselect_all (sel);
3657 /* Create the mail operation */
3659 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
3660 modest_ui_actions_delete_folder_error_handler,
3663 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
3665 modest_mail_operation_remove_folder (mail_op, TNY_FOLDER (info->folder), info->move_to_trash);
3667 modest_folder_view_select_first_inbox_or_local (MODEST_FOLDER_VIEW (folder_view));
3669 g_object_unref (G_OBJECT (mail_op));
3670 g_object_unref (G_OBJECT (info->folder));
3675 delete_folder (ModestWindow *window, gboolean move_to_trash)
3677 TnyFolderStore *folder;
3678 GtkWidget *folder_view;
3682 g_return_val_if_fail (MODEST_IS_WINDOW(window), FALSE);
3684 if (MODEST_IS_MAIN_WINDOW (window)) {
3686 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
3687 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
3688 #ifdef MODEST_TOOLKIT_HILDON2
3689 } else if (MODEST_IS_FOLDER_WINDOW (window)) {
3690 folder_view = GTK_WIDGET (modest_folder_window_get_folder_view (MODEST_FOLDER_WINDOW (window)));
3698 folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
3703 /* Show an error if it's an account */
3704 if (!TNY_IS_FOLDER (folder)) {
3705 modest_platform_run_information_dialog (GTK_WINDOW (window),
3706 _("mail_in_ui_folder_delete_error"),
3708 g_object_unref (G_OBJECT (folder));
3713 message = g_strdup_printf (_("mcen_nc_delete_folder_text"),
3714 tny_folder_get_name (TNY_FOLDER (folder)));
3715 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (window),
3716 (const gchar *) message);
3719 if (response == GTK_RESPONSE_OK) {
3720 DeleteFolderInfo *info;
3721 info = g_new0(DeleteFolderInfo, 1);
3722 info->folder = folder;
3723 info->move_to_trash = move_to_trash;
3724 g_object_ref (G_OBJECT (info->folder));
3725 TnyAccount *account = tny_folder_get_account (TNY_FOLDER (folder));
3726 modest_platform_connect_if_remote_and_perform (GTK_WINDOW (window),
3728 TNY_FOLDER_STORE (account),
3729 on_delete_folder_cb, info);
3730 g_object_unref (account);
3735 g_object_unref (G_OBJECT (folder));
3739 modest_ui_actions_on_delete_folder (GtkAction *action,
3740 ModestWindow *window)
3742 modest_ui_actions_on_edit_mode_delete_folder (window);
3746 modest_ui_actions_on_edit_mode_delete_folder (ModestWindow *window)
3748 g_return_val_if_fail (MODEST_IS_WINDOW(window), TRUE);
3750 return delete_folder (window, FALSE);
3754 modest_ui_actions_on_move_folder_to_trash_folder (GtkAction *action, ModestMainWindow *main_window)
3756 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
3758 delete_folder (MODEST_WINDOW (main_window), TRUE);
3762 typedef struct _PasswordDialogFields {
3763 GtkWidget *username;
3764 GtkWidget *password;
3766 } PasswordDialogFields;
3769 password_dialog_check_field (GtkEditable *editable,
3770 PasswordDialogFields *fields)
3773 gboolean any_value_empty = FALSE;
3775 #ifdef MODEST_TOOLKIT_HILDON2
3776 value = hildon_entry_get_text (HILDON_ENTRY (fields->username));
3778 value = gtk_entry_get_text (GTK_ENTRY (fields->username));
3780 if ((value == NULL) || value[0] == '\0') {
3781 any_value_empty = TRUE;
3783 #ifdef MODEST_TOOLKIT_HILDON2
3784 value = hildon_entry_get_text (HILDON_ENTRY (fields->password));
3786 value = gtk_entry_get_text (GTK_ENTRY (fields->password));
3788 if ((value == NULL) || value[0] == '\0') {
3789 any_value_empty = TRUE;
3791 gtk_dialog_set_response_sensitive (GTK_DIALOG (fields->dialog), GTK_RESPONSE_ACCEPT, !any_value_empty);
3795 modest_ui_actions_on_password_requested (TnyAccountStore *account_store,
3796 const gchar* server_account_name,
3801 ModestMainWindow *main_window)
3803 g_return_if_fail(server_account_name);
3804 gboolean completed = FALSE;
3805 PasswordDialogFields *fields = NULL;
3807 /* Initalize output parameters: */
3814 #ifndef MODEST_TOOLKIT_GTK
3815 /* Maemo uses a different (awkward) button order,
3816 * It should probably just use gtk_alternative_dialog_button_order ().
3818 #ifdef MODEST_TOOLKIT_HILDON2
3820 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3823 _HL("wdgt_bd_done"),
3824 GTK_RESPONSE_ACCEPT,
3826 gtk_container_set_border_width (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox),
3827 HILDON_MARGIN_DOUBLE);
3830 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3833 _("mcen_bd_dialog_ok"),
3834 GTK_RESPONSE_ACCEPT,
3835 _("mcen_bd_dialog_cancel"),
3836 GTK_RESPONSE_REJECT,
3838 #endif /* MODEST_TOOLKIT_HILDON2 */
3841 gtk_dialog_new_with_buttons (_("mail_ti_password_protected"),
3845 GTK_RESPONSE_REJECT,
3847 GTK_RESPONSE_ACCEPT,
3849 #endif /* MODEST_TOOLKIT_GTK */
3851 modest_window_mgr_set_modal (modest_runtime_get_window_mgr(), GTK_WINDOW (dialog), NULL);
3853 gchar *server_name = modest_account_mgr_get_server_account_hostname (
3854 modest_runtime_get_account_mgr(), server_account_name);
3855 if (!server_name) {/* This happened once, though I don't know why. murrayc. */
3856 g_warning("%s: Could not get server name for server account '%s'", __FUNCTION__, server_account_name);
3859 gtk_widget_destroy (dialog);
3863 gchar *txt = g_strdup_printf (_("mail_ia_password_info"), server_name);
3864 GtkWidget *label = gtk_label_new (txt);
3865 gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
3867 g_free (server_name);
3868 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), label,
3873 gchar *initial_username = modest_account_mgr_get_server_account_username (
3874 modest_runtime_get_account_mgr(), server_account_name);
3876 #ifdef MODEST_TOOLKIT_HILDON2
3877 GtkWidget *entry_username = hildon_entry_new (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH);
3878 if (initial_username)
3879 hildon_entry_set_text (HILDON_ENTRY (entry_username), initial_username);
3881 GtkWidget *entry_username = gtk_entry_new ();
3882 if (initial_username)
3883 gtk_entry_set_text (GTK_ENTRY (entry_username), initial_username);
3885 /* Dim this if a connection has ever succeeded with this username,
3886 * as per the UI spec: */
3887 /* const gboolean username_known = */
3888 /* modest_account_mgr_get_server_account_username_has_succeeded( */
3889 /* modest_runtime_get_account_mgr(), server_account_name); */
3890 /* gtk_widget_set_sensitive (entry_username, !username_known); */
3892 /* We drop the username sensitive code and disallow changing it here
3893 * as tinymail does not support really changing the username in the callback
3895 gtk_widget_set_sensitive (entry_username, FALSE);
3897 #ifndef MODEST_TOOLKIT_GTK
3898 /* Auto-capitalization is the default, so let's turn it off: */
3899 hildon_gtk_entry_set_input_mode (GTK_ENTRY (entry_username), HILDON_GTK_INPUT_MODE_FULL);
3901 /* Create a size group to be used by all captions.
3902 * Note that HildonCaption does not create a default size group if we do not specify one.
3903 * We use GTK_SIZE_GROUP_HORIZONTAL, so that the widths are the same. */
3904 GtkSizeGroup *sizegroup = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
3906 #ifdef MODEST_TOOLKIT_HILDON2
3907 GtkWidget *caption = modest_maemo_utils_create_captioned (sizegroup, NULL,
3908 _("mail_fi_username"), FALSE,
3911 GtkWidget *caption = hildon_caption_new (sizegroup,
3912 _("mail_fi_username"), entry_username, NULL, HILDON_CAPTION_MANDATORY);
3914 gtk_widget_show (entry_username);
3915 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), caption,
3916 FALSE, FALSE, MODEST_MARGIN_HALF);
3917 gtk_widget_show (caption);
3919 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), entry_username,
3921 #endif /* !MODEST_TOOLKIT_GTK */
3924 #ifdef MODEST_TOOLKIT_HILDON2
3925 GtkWidget *entry_password = hildon_entry_new (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH);
3927 GtkWidget *entry_password = gtk_entry_new ();
3929 gtk_entry_set_visibility (GTK_ENTRY(entry_password), FALSE);
3930 /* gtk_entry_set_invisible_char (GTK_ENTRY(entry_password), "*"); */
3932 #ifndef MODEST_TOOLKIT_GTK
3933 /* Auto-capitalization is the default, so let's turn it off: */
3934 hildon_gtk_entry_set_input_mode (GTK_ENTRY (entry_password),
3935 HILDON_GTK_INPUT_MODE_FULL | HILDON_GTK_INPUT_MODE_INVISIBLE);
3937 #ifdef MODEST_TOOLKIT_HILDON2
3938 caption = modest_maemo_utils_create_captioned (sizegroup, NULL,
3939 _("mail_fi_password"), FALSE,
3942 caption = hildon_caption_new (sizegroup,
3943 _("mail_fi_password"), entry_password, NULL, HILDON_CAPTION_MANDATORY);
3945 gtk_widget_show (entry_password);
3946 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), caption,
3947 FALSE, FALSE, MODEST_MARGIN_HALF);
3948 gtk_widget_show (caption);
3949 g_object_unref (sizegroup);
3951 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), entry_password,
3953 #endif /* !MODEST_TOOLKIT_GTK */
3955 if (initial_username != NULL)
3956 gtk_widget_grab_focus (GTK_WIDGET (entry_password));
3958 /* This is not in the Maemo UI spec:
3959 remember_pass_check = gtk_check_button_new_with_label (_("Remember password"));
3960 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), remember_pass_check,
3964 fields = g_slice_new0 (PasswordDialogFields);
3965 fields->username = entry_username;
3966 fields->password = entry_password;
3967 fields->dialog = dialog;
3969 g_signal_connect (entry_username, "changed", G_CALLBACK (password_dialog_check_field), fields);
3970 g_signal_connect (entry_password, "changed", G_CALLBACK (password_dialog_check_field), fields);
3971 password_dialog_check_field (NULL, fields);
3973 gtk_widget_show_all (GTK_WIDGET(GTK_DIALOG(dialog)->vbox));
3975 while (!completed) {
3977 if (gtk_dialog_run (GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
3979 #ifdef MODEST_TOOLKIT_HILDON2
3980 *username = g_strdup (hildon_entry_get_text (HILDON_ENTRY(entry_username)));
3982 *username = g_strdup (gtk_entry_get_text (GTK_ENTRY(entry_username)));
3985 /* Note that an empty field becomes the "" string */
3986 if (*username && strlen (*username) > 0) {
3987 modest_account_mgr_set_server_account_username (modest_runtime_get_account_mgr(),
3988 server_account_name,
3992 const gboolean username_was_changed =
3993 (strcmp (*username, initial_username) != 0);
3994 if (username_was_changed) {
3995 g_warning ("%s: tinymail does not yet support changing the "
3996 "username in the get_password() callback.\n", __FUNCTION__);
4002 modest_platform_information_banner (GTK_WIDGET (dialog), NULL,
4003 _("mcen_ib_username_pw_incorrect"));
4009 #ifdef MODEST_TOOLKIT_HILDON2
4010 *password = g_strdup (hildon_entry_get_text (HILDON_ENTRY(entry_password)));
4012 *password = g_strdup (gtk_entry_get_text (GTK_ENTRY(entry_password)));
4015 /* We do not save the password in the configuration,
4016 * because this function is only called for passwords that should
4017 * not be remembered:
4018 modest_server_account_set_password (
4019 modest_runtime_get_account_mgr(), server_account_name,
4026 #ifndef MODEST_TOOLKIT_HILDON2
4027 /* Set parent to NULL or the banner will disappear with its parent dialog */
4028 modest_platform_information_banner(NULL, NULL, _("mail_ib_login_cancelled"));
4040 /* This is not in the Maemo UI spec:
4041 if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (remember_pass_check)))
4047 g_free (initial_username);
4048 gtk_widget_destroy (dialog);
4049 g_slice_free (PasswordDialogFields, fields);
4051 /* printf ("DEBUG: %s: cancel=%d\n", __FUNCTION__, *cancel); */
4055 modest_ui_actions_on_cut (GtkAction *action,
4056 ModestWindow *window)
4058 GtkWidget *focused_widget;
4059 GtkClipboard *clipboard;
4061 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4062 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4063 if (GTK_IS_EDITABLE (focused_widget)) {
4064 gtk_editable_cut_clipboard (GTK_EDITABLE(focused_widget));
4065 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4066 gtk_clipboard_store (clipboard);
4067 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4068 GtkTextBuffer *buffer;
4070 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4071 if (modest_text_utils_buffer_selection_is_valid (buffer)) {
4072 gtk_text_buffer_cut_clipboard (buffer, clipboard, TRUE);
4073 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4074 gtk_clipboard_store (clipboard);
4076 } else if (MODEST_IS_HEADER_VIEW (focused_widget)) {
4077 TnyList *header_list = modest_header_view_get_selected_headers (
4078 MODEST_HEADER_VIEW (focused_widget));
4079 gboolean continue_download = FALSE;
4080 gint num_of_unc_msgs;
4082 num_of_unc_msgs = header_list_count_uncached_msgs(header_list);
4084 if (num_of_unc_msgs) {
4085 TnyAccount *account = get_account_from_header_list (header_list);
4087 continue_download = connect_to_get_msg (window, num_of_unc_msgs, account);
4088 g_object_unref (account);
4092 if (num_of_unc_msgs == 0 || continue_download) {
4093 /* modest_platform_information_banner (
4094 NULL, NULL, _CS("mcen_ib_getting_items"));*/
4095 modest_header_view_cut_selection (
4096 MODEST_HEADER_VIEW (focused_widget));
4099 g_object_unref (header_list);
4100 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4101 modest_folder_view_cut_selection (MODEST_FOLDER_VIEW (focused_widget));
4106 modest_ui_actions_on_copy (GtkAction *action,
4107 ModestWindow *window)
4109 GtkClipboard *clipboard;
4110 GtkWidget *focused_widget;
4111 gboolean copied = TRUE;
4113 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4114 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4116 if (GTK_IS_LABEL (focused_widget)) {
4118 selection = modest_text_utils_label_get_selection (GTK_LABEL (focused_widget));
4119 gtk_clipboard_set_text (clipboard, selection, -1);
4121 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4122 gtk_clipboard_store (clipboard);
4123 } else if (GTK_IS_EDITABLE (focused_widget)) {
4124 gtk_editable_copy_clipboard (GTK_EDITABLE(focused_widget));
4125 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4126 gtk_clipboard_store (clipboard);
4127 } else if (GTK_IS_HTML (focused_widget)) {
4130 sel = gtk_html_get_selection_html (GTK_HTML (focused_widget), &len);
4131 if ((sel == NULL) || (sel[0] == '\0')) {
4134 gtk_html_copy (GTK_HTML (focused_widget));
4135 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4136 gtk_clipboard_store (clipboard);
4138 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4139 GtkTextBuffer *buffer;
4140 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4141 if (modest_text_utils_buffer_selection_is_valid (buffer)) {
4142 gtk_text_buffer_copy_clipboard (buffer, clipboard);
4143 gtk_clipboard_set_can_store (clipboard, NULL, 0);
4144 gtk_clipboard_store (clipboard);
4146 } else if (MODEST_IS_HEADER_VIEW (focused_widget)) {
4147 TnyList *header_list = modest_header_view_get_selected_headers (
4148 MODEST_HEADER_VIEW (focused_widget));
4149 gboolean continue_download = FALSE;
4150 gint num_of_unc_msgs;
4152 num_of_unc_msgs = header_list_count_uncached_msgs(header_list);
4154 if (num_of_unc_msgs) {
4155 TnyAccount *account = get_account_from_header_list (header_list);
4157 continue_download = connect_to_get_msg (window, num_of_unc_msgs, account);
4158 g_object_unref (account);
4162 if (num_of_unc_msgs == 0 || continue_download) {
4163 modest_platform_information_banner (
4164 NULL, NULL, _CS("mcen_ib_getting_items"));
4165 modest_header_view_copy_selection (
4166 MODEST_HEADER_VIEW (focused_widget));
4170 g_object_unref (header_list);
4172 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4173 modest_folder_view_copy_selection (MODEST_FOLDER_VIEW (focused_widget));
4176 /* Show information banner if there was a copy to clipboard */
4178 modest_platform_information_banner (
4179 NULL, NULL, _CS("ecoc_ib_edwin_copied"));
4183 modest_ui_actions_on_undo (GtkAction *action,
4184 ModestWindow *window)
4186 ModestEmailClipboard *clipboard = NULL;
4188 if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4189 modest_msg_edit_window_undo (MODEST_MSG_EDIT_WINDOW (window));
4190 } else if (MODEST_IS_MAIN_WINDOW (window)) {
4191 /* Clear clipboard source */
4192 clipboard = modest_runtime_get_email_clipboard ();
4193 modest_email_clipboard_clear (clipboard);
4196 g_return_if_reached ();
4201 modest_ui_actions_on_redo (GtkAction *action,
4202 ModestWindow *window)
4204 if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4205 modest_msg_edit_window_redo (MODEST_MSG_EDIT_WINDOW (window));
4208 g_return_if_reached ();
4214 destroy_information_note (ModestMailOperation *mail_op,
4217 /* destroy information note */
4218 gtk_widget_destroy (GTK_WIDGET(user_data));
4222 destroy_folder_information_note (ModestMailOperation *mail_op,
4223 TnyFolder *new_folder,
4226 /* destroy information note */
4227 gtk_widget_destroy (GTK_WIDGET(user_data));
4232 paste_as_attachment_free (gpointer data)
4234 PasteAsAttachmentHelper *helper = (PasteAsAttachmentHelper *) data;
4236 if (helper->banner) {
4237 gtk_widget_destroy (helper->banner);
4238 g_object_unref (helper->banner);
4244 paste_msg_as_attachment_cb (ModestMailOperation *mail_op,
4249 PasteAsAttachmentHelper *helper = (PasteAsAttachmentHelper *) userdata;
4250 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (helper->window));
4255 modest_msg_edit_window_add_part (MODEST_MSG_EDIT_WINDOW (helper->window), TNY_MIME_PART (msg));
4260 modest_ui_actions_on_paste (GtkAction *action,
4261 ModestWindow *window)
4263 GtkWidget *focused_widget = NULL;
4264 GtkWidget *inf_note = NULL;
4265 ModestMailOperation *mail_op = NULL;
4267 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4268 if (GTK_IS_EDITABLE (focused_widget)) {
4269 gtk_editable_paste_clipboard (GTK_EDITABLE(focused_widget));
4270 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4271 ModestEmailClipboard *e_clipboard = NULL;
4272 e_clipboard = modest_runtime_get_email_clipboard ();
4273 if (modest_email_clipboard_cleared (e_clipboard)) {
4274 GtkTextBuffer *buffer;
4275 GtkClipboard *clipboard;
4277 clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
4278 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4279 gtk_text_buffer_paste_clipboard (buffer, clipboard, NULL, TRUE);
4280 } else if (MODEST_IS_MSG_EDIT_WINDOW (window)) {
4281 ModestMailOperation *mail_op;
4282 TnyFolder *src_folder = NULL;
4283 TnyList *data = NULL;
4285 PasteAsAttachmentHelper *helper = g_new0 (PasteAsAttachmentHelper, 1);
4286 helper->window = MODEST_MSG_EDIT_WINDOW (window);
4287 helper->banner = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4288 _CS("ckct_nw_pasting"));
4289 modest_email_clipboard_get_data (e_clipboard, &src_folder, &data, &delete);
4290 mail_op = modest_mail_operation_new (G_OBJECT (window));
4291 if (helper->banner != NULL) {
4292 g_object_ref (G_OBJECT (helper->banner));
4293 gtk_widget_show (GTK_WIDGET (helper->banner));
4297 modest_mail_operation_get_msgs_full (mail_op,
4299 (GetMsgAsyncUserCallback) paste_msg_as_attachment_cb,
4301 paste_as_attachment_free);
4305 g_object_unref (data);
4307 g_object_unref (src_folder);
4310 } else if (MODEST_IS_FOLDER_VIEW (focused_widget)) {
4311 ModestEmailClipboard *clipboard = NULL;
4312 TnyFolder *src_folder = NULL;
4313 TnyFolderStore *folder_store = NULL;
4314 TnyList *data = NULL;
4315 gboolean delete = FALSE;
4317 /* Check clipboard source */
4318 clipboard = modest_runtime_get_email_clipboard ();
4319 if (modest_email_clipboard_cleared (clipboard))
4322 /* Get elements to paste */
4323 modest_email_clipboard_get_data (clipboard, &src_folder, &data, &delete);
4325 /* Create a new mail operation */
4326 mail_op = modest_mail_operation_new (G_OBJECT(window));
4328 /* Get destination folder */
4329 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (focused_widget));
4331 /* transfer messages */
4335 /* Ask for user confirmation */
4337 modest_ui_actions_msgs_move_to_confirmation (window,
4338 TNY_FOLDER (folder_store),
4342 if (response == GTK_RESPONSE_OK) {
4343 /* Launch notification */
4344 inf_note = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4345 _CS("ckct_nw_pasting"));
4346 if (inf_note != NULL) {
4347 gtk_window_set_modal (GTK_WINDOW(inf_note), FALSE);
4348 gtk_widget_show (GTK_WIDGET(inf_note));
4351 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
4352 modest_mail_operation_xfer_msgs (mail_op,
4354 TNY_FOLDER (folder_store),
4356 destroy_information_note,
4359 g_object_unref (mail_op);
4362 } else if (src_folder != NULL) {
4363 /* Launch notification */
4364 inf_note = modest_platform_animation_banner (GTK_WIDGET (window), NULL,
4365 _CS("ckct_nw_pasting"));
4366 if (inf_note != NULL) {
4367 gtk_window_set_modal (GTK_WINDOW(inf_note), FALSE);
4368 gtk_widget_show (GTK_WIDGET(inf_note));
4371 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
4372 modest_mail_operation_xfer_folder (mail_op,
4376 destroy_folder_information_note,
4382 g_object_unref (data);
4383 if (src_folder != NULL)
4384 g_object_unref (src_folder);
4385 if (folder_store != NULL)
4386 g_object_unref (folder_store);
4392 modest_ui_actions_on_select_all (GtkAction *action,
4393 ModestWindow *window)
4395 GtkWidget *focused_widget;
4397 focused_widget = gtk_window_get_focus (GTK_WINDOW (window));
4398 if (MODEST_IS_ATTACHMENTS_VIEW (focused_widget)) {
4399 modest_attachments_view_select_all (MODEST_ATTACHMENTS_VIEW (focused_widget));
4400 } else if (GTK_IS_LABEL (focused_widget)) {
4401 gtk_label_select_region (GTK_LABEL (focused_widget), 0, -1);
4402 } else if (GTK_IS_EDITABLE (focused_widget)) {
4403 gtk_editable_select_region (GTK_EDITABLE(focused_widget), 0, -1);
4404 } else if (GTK_IS_TEXT_VIEW (focused_widget)) {
4405 GtkTextBuffer *buffer;
4406 GtkTextIter start, end;
4408 buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (focused_widget));
4409 gtk_text_buffer_get_start_iter (buffer, &start);
4410 gtk_text_buffer_get_end_iter (buffer, &end);
4411 gtk_text_buffer_select_range (buffer, &start, &end);
4412 } else if (GTK_IS_HTML (focused_widget)) {
4413 gtk_html_select_all (GTK_HTML (focused_widget));
4414 } else if (MODEST_IS_MAIN_WINDOW (window)) {
4415 GtkWidget *header_view = focused_widget;
4416 GtkTreeSelection *selection = NULL;
4418 if (!(MODEST_IS_HEADER_VIEW (focused_widget))) {
4419 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (window),
4420 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
4423 /* Disable window dimming management */
4424 modest_window_disable_dimming (MODEST_WINDOW(window));
4426 /* Select all messages */
4427 selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(header_view));
4428 gtk_tree_selection_select_all (selection);
4430 /* Set focuse on header view */
4431 gtk_widget_grab_focus (header_view);
4433 /* Enable window dimming management */
4434 modest_window_enable_dimming (MODEST_WINDOW(window));
4435 modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (window));
4436 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (window));
4442 modest_ui_actions_on_mark_as_read (GtkAction *action,
4443 ModestWindow *window)
4445 g_return_if_fail (MODEST_IS_WINDOW(window));
4447 /* Mark each header as read */
4448 do_headers_action (window, headers_action_mark_as_read, NULL);
4452 modest_ui_actions_on_mark_as_unread (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_unread, NULL);
4462 modest_ui_actions_on_change_zoom (GtkRadioAction *action,
4463 GtkRadioAction *selected,
4464 ModestWindow *window)
4468 value = gtk_radio_action_get_current_value (selected);
4469 if (MODEST_IS_WINDOW (window)) {
4470 modest_window_set_zoom (MODEST_WINDOW (window), ((gdouble)value)/100);
4475 modest_ui_actions_msg_edit_on_change_priority (GtkRadioAction *action,
4476 GtkRadioAction *selected,
4477 ModestWindow *window)
4479 TnyHeaderFlags flags;
4480 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4482 flags = gtk_radio_action_get_current_value (selected);
4483 modest_msg_edit_window_set_priority_flags (MODEST_MSG_EDIT_WINDOW (window), flags);
4487 modest_ui_actions_msg_edit_on_change_file_format (GtkRadioAction *action,
4488 GtkRadioAction *selected,
4489 ModestWindow *window)
4493 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4495 file_format = gtk_radio_action_get_current_value (selected);
4496 modest_msg_edit_window_set_file_format (MODEST_MSG_EDIT_WINDOW (window), file_format);
4501 modest_ui_actions_on_zoom_plus (GtkAction *action,
4502 ModestWindow *window)
4504 g_return_if_fail (MODEST_IS_WINDOW (window));
4506 modest_window_zoom_plus (MODEST_WINDOW (window));
4510 modest_ui_actions_on_zoom_minus (GtkAction *action,
4511 ModestWindow *window)
4513 g_return_if_fail (MODEST_IS_WINDOW (window));
4515 modest_window_zoom_minus (MODEST_WINDOW (window));
4519 modest_ui_actions_on_toggle_fullscreen (GtkToggleAction *toggle,
4520 ModestWindow *window)
4522 ModestWindowMgr *mgr;
4523 gboolean fullscreen, active;
4524 g_return_if_fail (MODEST_IS_WINDOW (window));
4526 mgr = modest_runtime_get_window_mgr ();
4528 active = (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (toggle)))?1:0;
4529 fullscreen = modest_window_mgr_get_fullscreen_mode (mgr);
4531 if (active != fullscreen) {
4532 modest_window_mgr_set_fullscreen_mode (mgr, active);
4533 #ifndef MODEST_TOOLKIT_HILDON2
4534 gtk_window_present (GTK_WINDOW (window));
4540 modest_ui_actions_on_change_fullscreen (GtkAction *action,
4541 ModestWindow *window)
4543 ModestWindowMgr *mgr;
4544 gboolean fullscreen;
4546 g_return_if_fail (MODEST_IS_WINDOW (window));
4548 mgr = modest_runtime_get_window_mgr ();
4549 fullscreen = modest_window_mgr_get_fullscreen_mode (mgr);
4550 modest_window_mgr_set_fullscreen_mode (mgr, !fullscreen);
4552 #ifndef MODEST_TOOLKIT_HILDON2
4553 gtk_window_present (GTK_WINDOW (window));
4558 * Used by modest_ui_actions_on_details to call do_headers_action
4561 headers_action_show_details (TnyHeader *header,
4562 ModestWindow *window,
4566 modest_platform_run_header_details_dialog (GTK_WINDOW (window), header);
4570 * Show the header details in a ModestDetailsDialog widget
4573 modest_ui_actions_on_details (GtkAction *action,
4576 if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
4580 msg = modest_msg_view_window_get_message (MODEST_MSG_VIEW_WINDOW (win));
4584 header = tny_msg_get_header (msg);
4586 headers_action_show_details (header, win, NULL);
4587 g_object_unref (header);
4589 g_object_unref (msg);
4591 } else if (MODEST_IS_MAIN_WINDOW (win)) {
4592 GtkWidget *folder_view, *header_view;
4594 /* Check which widget has the focus */
4595 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
4596 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
4597 if (gtk_widget_is_focus (folder_view)) {
4598 TnyFolderStore *folder_store
4599 = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
4600 if (!folder_store) {
4601 g_warning ("%s: No item was selected.\n", __FUNCTION__);
4604 /* Show only when it's a folder */
4605 /* This function should not be called for account items,
4606 * because we dim the menu item for them. */
4607 if (TNY_IS_FOLDER (folder_store)) {
4608 modest_platform_run_folder_details_dialog (GTK_WINDOW (win),
4609 TNY_FOLDER (folder_store));
4612 g_object_unref (folder_store);
4615 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
4616 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
4617 /* Show details of each header */
4618 do_headers_action (win, headers_action_show_details, header_view);
4620 #ifdef MODEST_TOOLKIT_HILDON2
4621 } else if (MODEST_IS_HEADER_WINDOW (win)) {
4623 GtkWidget *header_view;
4625 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
4626 folder = modest_header_view_get_folder (MODEST_HEADER_VIEW (header_view));
4628 modest_platform_run_folder_details_dialog (GTK_WINDOW (win),
4630 g_object_unref (folder);
4637 modest_ui_actions_on_toggle_show_cc (GtkToggleAction *toggle,
4638 ModestMsgEditWindow *window)
4640 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4642 modest_msg_edit_window_show_cc (window, gtk_toggle_action_get_active (toggle));
4646 modest_ui_actions_on_toggle_show_bcc (GtkToggleAction *toggle,
4647 ModestMsgEditWindow *window)
4649 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4651 modest_msg_edit_window_show_bcc (window, gtk_toggle_action_get_active (toggle));
4655 modest_ui_actions_toggle_folders_view (GtkAction *action,
4656 ModestMainWindow *main_window)
4658 g_return_if_fail (MODEST_IS_MAIN_WINDOW(main_window));
4660 if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)))
4661 modest_main_window_set_style (main_window, MODEST_MAIN_WINDOW_STYLE_SPLIT);
4663 modest_main_window_set_style (main_window, MODEST_MAIN_WINDOW_STYLE_SIMPLE);
4667 modest_ui_actions_on_toggle_toolbar (GtkToggleAction *toggle,
4668 ModestWindow *window)
4670 gboolean active, fullscreen = FALSE;
4671 ModestWindowMgr *mgr;
4673 active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (toggle));
4675 /* Check if we want to toggle the toolbar view in fullscreen
4677 if (!strcmp (gtk_action_get_name (GTK_ACTION (toggle)),
4678 "ViewShowToolbarFullScreen")) {
4682 /* Toggle toolbar */
4683 mgr = modest_runtime_get_window_mgr ();
4684 modest_window_mgr_show_toolbars (mgr, G_TYPE_FROM_INSTANCE (window), active, fullscreen);
4688 modest_ui_actions_msg_edit_on_select_font (GtkAction *action,
4689 ModestMsgEditWindow *window)
4691 modest_msg_edit_window_select_font (window);
4696 modest_ui_actions_on_folder_display_name_changed (ModestFolderView *folder_view,
4697 const gchar *display_name,
4700 /* don't update the display name if it was already set;
4701 * updating the display name apparently is expensive */
4702 const gchar* old_name = gtk_window_get_title (window);
4704 if (display_name == NULL)
4707 if (old_name && display_name && strcmp (old_name, display_name) == 0)
4708 return; /* don't do anything */
4710 /* This is usually used to change the title of the main window, which
4711 * is the one that holds the folder view. Note that this change can
4712 * happen even when the widget doesn't have the focus. */
4713 gtk_window_set_title (window, display_name);
4718 modest_ui_actions_on_select_contacts (GtkAction *action, ModestMsgEditWindow *window)
4720 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4721 modest_msg_edit_window_select_contacts (window);
4725 modest_ui_actions_on_check_names (GtkAction *action, ModestMsgEditWindow *window)
4727 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
4728 modest_msg_edit_window_check_names (window, FALSE);
4731 #ifndef MODEST_TOOLKIT_HILDON2
4733 * This function is used to track changes in the selection of the
4734 * folder view that is inside the "move to" dialog to enable/disable
4735 * the OK button because we do not want the user to select a disallowed
4736 * destination for a folder.
4737 * The user also not desired to be able to use NEW button on items where
4738 * folder creation is not possibel.
4741 on_move_to_dialog_folder_selection_changed (ModestFolderView* self,
4742 TnyFolderStore *folder_store,
4746 GtkWidget *dialog = NULL;
4747 gboolean ok_sensitive = TRUE, new_sensitive = TRUE;
4748 gboolean moving_folder = FALSE;
4749 gboolean is_local_account = TRUE;
4750 GtkWidget *folder_view = NULL;
4751 ModestTnyFolderRules rules;
4753 g_return_if_fail (MODEST_IS_FOLDER_VIEW(self));
4758 dialog = gtk_widget_get_ancestor (GTK_WIDGET (self), GTK_TYPE_DIALOG);
4762 /* check if folder_store is an remote account */
4763 if (TNY_IS_ACCOUNT (folder_store)) {
4764 TnyAccount *local_account = NULL;
4765 TnyAccount *mmc_account = NULL;
4766 ModestTnyAccountStore *account_store = NULL;
4768 account_store = modest_runtime_get_account_store ();
4769 local_account = modest_tny_account_store_get_local_folders_account (account_store);
4770 mmc_account = modest_tny_account_store_get_mmc_folders_account (account_store);
4772 if ((gpointer) local_account != (gpointer) folder_store &&
4773 (gpointer) mmc_account != (gpointer) folder_store) {
4774 ModestProtocolType proto;
4775 proto = modest_tny_account_get_protocol_type (TNY_ACCOUNT (folder_store));
4776 if (proto == MODEST_PROTOCOL_REGISTRY_TYPE_INVALID) {
4777 proto = MODEST_PROTOCOLS_STORE_MAILDIR;
4779 is_local_account = FALSE;
4780 /* New button should be dimmed on remote
4782 new_sensitive = (modest_protocol_registry_protocol_type_has_tag (modest_runtime_get_protocol_registry (),
4784 MODEST_PROTOCOL_REGISTRY_STORE_HAS_FOLDERS));
4786 g_object_unref (local_account);
4788 /* It could not exist */
4790 g_object_unref (mmc_account);
4793 /* Check the target folder rules */
4794 if (TNY_IS_FOLDER (folder_store)) {
4795 rules = modest_tny_folder_get_rules (TNY_FOLDER (folder_store));
4796 if (rules & MODEST_FOLDER_RULES_FOLDER_NON_WRITEABLE) {
4797 ok_sensitive = FALSE;
4798 new_sensitive = FALSE;
4803 /* Check if we're moving a folder */
4804 if (MODEST_IS_MAIN_WINDOW (user_data)) {
4805 /* Get the widgets */
4806 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (user_data),
4807 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
4808 if (gtk_widget_is_focus (folder_view))
4809 moving_folder = TRUE;
4812 if (moving_folder) {
4813 TnyFolderStore *moved_folder = NULL, *parent = NULL;
4815 /* Get the folder to move */
4816 moved_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
4818 /* Check that we're not moving to the same folder */
4819 if (TNY_IS_FOLDER (moved_folder)) {
4820 parent = tny_folder_get_folder_store (TNY_FOLDER (moved_folder));
4821 if (parent == folder_store)
4822 ok_sensitive = FALSE;
4823 g_object_unref (parent);
4826 if (ok_sensitive && TNY_IS_ACCOUNT (folder_store)) {
4827 /* Do not allow to move to an account unless it's the
4828 local folders account */
4829 if (!is_local_account)
4830 ok_sensitive = FALSE;
4833 if (ok_sensitive && (moved_folder == folder_store)) {
4834 /* Do not allow to move to itself */
4835 ok_sensitive = FALSE;
4837 g_object_unref (moved_folder);
4839 TnyFolder *src_folder = NULL;
4841 /* Moving a message */
4842 if (MODEST_IS_MSG_VIEW_WINDOW (user_data)) {
4844 TnyHeader *header = NULL;
4845 header = modest_msg_view_window_get_header
4846 (MODEST_MSG_VIEW_WINDOW (user_data));
4847 if (!TNY_IS_HEADER(header))
4848 g_warning ("%s: could not get source header", __FUNCTION__);
4850 src_folder = tny_header_get_folder (header);
4853 g_object_unref (header);
4856 TNY_FOLDER (modest_folder_view_get_selected
4857 (MODEST_FOLDER_VIEW (folder_view)));
4860 if (TNY_IS_FOLDER(src_folder)) {
4861 /* Do not allow to move the msg to the same folder */
4862 /* Do not allow to move the msg to an account */
4863 if ((gpointer) src_folder == (gpointer) folder_store ||
4864 TNY_IS_ACCOUNT (folder_store))
4865 ok_sensitive = FALSE;
4866 g_object_unref (src_folder);
4868 g_warning ("%s: could not get source folder", __FUNCTION__);
4872 /* Set sensitivity of the OK and NEW button */
4873 gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), GTK_RESPONSE_OK, ok_sensitive);
4874 gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), MODEST_GTK_RESPONSE_NEW_FOLDER, new_sensitive);
4879 on_move_to_dialog_response (GtkDialog *dialog,
4883 GtkWidget *parent_win;
4884 MoveToInfo *helper = NULL;
4885 ModestFolderView *folder_view;
4887 helper = (MoveToInfo *) user_data;
4889 parent_win = (GtkWidget *) helper->win;
4890 folder_view = MODEST_FOLDER_VIEW (g_object_get_data (G_OBJECT (dialog),
4891 MODEST_MOVE_TO_DIALOG_FOLDER_VIEW));
4894 TnyFolderStore *dst_folder;
4896 case MODEST_GTK_RESPONSE_NEW_FOLDER:
4897 modest_ui_actions_create_folder (GTK_WIDGET (dialog), GTK_WIDGET (folder_view));
4899 case GTK_RESPONSE_NONE:
4900 case GTK_RESPONSE_CANCEL:
4901 case GTK_RESPONSE_DELETE_EVENT:
4903 case GTK_RESPONSE_OK:
4904 dst_folder = modest_folder_view_get_selected (folder_view);
4906 if (MODEST_IS_MAIN_WINDOW (parent_win)) {
4907 /* Clean list to move used for filtering */
4908 modest_folder_view_set_list_to_move (folder_view, NULL);
4910 modest_ui_actions_on_main_window_move_to (NULL,
4911 GTK_WIDGET (folder_view),
4913 MODEST_MAIN_WINDOW (parent_win));
4914 #ifdef MODEST_TOOLKIT_HILDON2
4915 } else if (MODEST_IS_FOLDER_WINDOW (parent_win)) {
4916 /* Clean list to move used for filtering */
4917 modest_folder_view_set_list_to_move (folder_view, NULL);
4919 modest_ui_actions_on_folder_window_move_to (GTK_WIDGET (folder_view),
4922 GTK_WINDOW (parent_win));
4925 /* if the user selected a root folder
4926 (account) then do not perform any action */
4927 if (TNY_IS_ACCOUNT (dst_folder)) {
4928 g_signal_stop_emission_by_name (dialog, "response");
4932 /* Clean list to move used for filtering */
4933 modest_folder_view_set_list_to_move (folder_view, NULL);
4935 /* Moving from headers window in edit mode */
4936 modest_ui_actions_on_window_move_to (NULL, helper->list,
4938 MODEST_WINDOW (parent_win));
4942 g_object_unref (dst_folder);
4946 g_warning ("%s unexpected response id %d", __FUNCTION__, response);
4949 /* Free the helper and exit */
4951 g_object_unref (helper->list);
4952 g_slice_free (MoveToInfo, helper);
4953 gtk_widget_destroy (GTK_WIDGET (dialog));
4957 create_move_to_dialog (GtkWindow *win,
4958 GtkWidget *folder_view,
4959 TnyList *list_to_move)
4961 GtkWidget *dialog, *tree_view = NULL;
4963 dialog = modest_platform_create_move_to_dialog (win, &tree_view);
4965 #ifndef MODEST_TOOLKIT_HILDON2
4966 /* Track changes in the selection to
4967 * disable the OK button whenever "Move to" is not possible
4968 * disbale NEW button whenever New is not possible */
4969 g_signal_connect (tree_view,
4970 "folder_selection_changed",
4971 G_CALLBACK (on_move_to_dialog_folder_selection_changed),
4975 /* It could happen that we're trying to move a message from a
4976 window (msg window for example) after the main window was
4977 closed, so we can not just get the model of the folder
4979 if (MODEST_IS_FOLDER_VIEW (folder_view)) {
4980 const gchar *visible_id = NULL;
4982 modest_folder_view_set_style (MODEST_FOLDER_VIEW (tree_view),
4983 MODEST_FOLDER_VIEW_STYLE_SHOW_ALL);
4984 modest_folder_view_copy_model (MODEST_FOLDER_VIEW(folder_view),
4985 MODEST_FOLDER_VIEW(tree_view));
4988 modest_folder_view_get_account_id_of_visible_server_account (MODEST_FOLDER_VIEW(folder_view));
4990 /* Show the same account than the one that is shown in the main window */
4991 modest_folder_view_set_account_id_of_visible_server_account (MODEST_FOLDER_VIEW(tree_view),
4994 const gchar *active_account_name = NULL;
4995 ModestAccountMgr *mgr = NULL;
4996 ModestAccountSettings *settings = NULL;
4997 ModestServerAccountSettings *store_settings = NULL;
4999 modest_folder_view_set_style (MODEST_FOLDER_VIEW (tree_view),
5000 MODEST_FOLDER_VIEW_STYLE_SHOW_ALL);
5001 modest_folder_view_update_model (MODEST_FOLDER_VIEW (tree_view),
5002 TNY_ACCOUNT_STORE (modest_runtime_get_account_store ()));
5004 active_account_name = modest_window_get_active_account (MODEST_WINDOW (win));
5005 mgr = modest_runtime_get_account_mgr ();
5006 settings = modest_account_mgr_load_account_settings (mgr, active_account_name);
5009 const gchar *store_account_name;
5010 store_settings = modest_account_settings_get_store_settings (settings);
5011 store_account_name = modest_server_account_settings_get_account_name (store_settings);
5013 modest_folder_view_set_account_id_of_visible_server_account (MODEST_FOLDER_VIEW (tree_view),
5014 store_account_name);
5015 g_object_unref (store_settings);
5016 g_object_unref (settings);
5020 /* we keep a pointer to the embedded folder view, so we can
5021 * retrieve it with get_folder_view_from_move_to_dialog (see
5022 * above) later (needed for focus handling)
5024 g_object_set_data (G_OBJECT(dialog), MODEST_MOVE_TO_DIALOG_FOLDER_VIEW, tree_view);
5026 /* Hide special folders */
5027 modest_folder_view_show_non_move_folders (MODEST_FOLDER_VIEW (tree_view), FALSE);
5029 modest_folder_view_set_list_to_move (MODEST_FOLDER_VIEW (tree_view), list_to_move);
5030 #ifndef MODEST_TOOLKIT_HILDON2
5031 modest_folder_view_select_first_inbox_or_local (MODEST_FOLDER_VIEW (tree_view));
5034 gtk_widget_show (GTK_WIDGET (tree_view));
5040 * Shows a confirmation dialog to the user when we're moving messages
5041 * from a remote server to the local storage. Returns the dialog
5042 * response. If it's other kind of movement then it always returns
5045 * This one is used by the next functions:
5046 * modest_ui_actions_on_paste - commented out
5047 * drag_and_drop_from_header_view (for d&d in modest_folder_view.c)
5050 modest_ui_actions_msgs_move_to_confirmation (ModestWindow *win,
5051 TnyFolder *dest_folder,
5055 gint response = GTK_RESPONSE_OK;
5056 TnyAccount *account = NULL;
5057 TnyFolder *src_folder = NULL;
5058 TnyIterator *iter = NULL;
5059 TnyHeader *header = NULL;
5061 /* return with OK if the destination is a remote folder */
5062 if (modest_tny_folder_is_remote_folder (dest_folder))
5063 return GTK_RESPONSE_OK;
5065 /* Get source folder */
5066 iter = tny_list_create_iterator (headers);
5067 header = TNY_HEADER (tny_iterator_get_current (iter));
5069 src_folder = tny_header_get_folder (header);
5070 g_object_unref (header);
5072 g_object_unref (iter);
5074 /* if no src_folder, message may be an attahcment */
5075 if (src_folder == NULL)
5076 return GTK_RESPONSE_CANCEL;
5078 /* If the source is a local or MMC folder */
5079 if (!modest_tny_folder_is_remote_folder (src_folder)) {
5080 g_object_unref (src_folder);
5081 return GTK_RESPONSE_OK;
5084 /* Get the account */
5085 account = tny_folder_get_account (src_folder);
5087 /* now if offline we ask the user */
5088 if(connect_to_get_msg (win, tny_list_get_length (headers), account))
5089 response = GTK_RESPONSE_OK;
5091 response = GTK_RESPONSE_CANCEL;
5094 g_object_unref (src_folder);
5095 g_object_unref (account);
5101 move_to_helper_destroyer (gpointer user_data)
5103 MoveToHelper *helper = (MoveToHelper *) user_data;
5105 /* Close the "Pasting" information banner */
5106 if (helper->banner) {
5107 gtk_widget_destroy (GTK_WIDGET (helper->banner));
5108 g_object_unref (helper->banner);
5110 if (gtk_tree_row_reference_valid (helper->reference)) {
5111 gtk_tree_row_reference_free (helper->reference);
5112 helper->reference = NULL;
5118 move_to_cb (ModestMailOperation *mail_op,
5121 MoveToHelper *helper = (MoveToHelper *) user_data;
5122 GObject *object = modest_mail_operation_get_source (mail_op);
5124 /* Note that the operation could have failed, in that case do
5126 if (modest_mail_operation_get_status (mail_op) !=
5127 MODEST_MAIL_OPERATION_STATUS_SUCCESS)
5130 if (MODEST_IS_MSG_VIEW_WINDOW (object)) {
5131 ModestMsgViewWindow *self = MODEST_MSG_VIEW_WINDOW (object);
5133 if (!modest_msg_view_window_select_next_message (self) &&
5134 !modest_msg_view_window_select_previous_message (self)) {
5135 /* No more messages to view, so close this window */
5136 modest_ui_actions_on_close_window (NULL, MODEST_WINDOW(self));
5138 } else if (MODEST_IS_MAIN_WINDOW (object) &&
5139 gtk_tree_row_reference_valid (helper->reference)) {
5140 GtkWidget *header_view;
5142 GtkTreeSelection *sel;
5144 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(object),
5145 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5146 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (header_view));
5147 path = gtk_tree_row_reference_get_path (helper->reference);
5148 /* We need to unselect the previous one
5149 because we could be copying instead of
5151 gtk_tree_selection_unselect_all (sel);
5152 gtk_tree_selection_select_path (sel, path);
5153 gtk_tree_path_free (path);
5155 g_object_unref (object);
5158 /* Destroy the helper */
5159 move_to_helper_destroyer (helper);
5163 folder_move_to_cb (ModestMailOperation *mail_op,
5164 TnyFolder *new_folder,
5167 GtkWidget *folder_view;
5170 object = modest_mail_operation_get_source (mail_op);
5171 if (MODEST_IS_MAIN_WINDOW (object)) {
5172 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(object),
5173 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
5174 g_object_ref (folder_view);
5175 g_object_unref (object);
5176 move_to_cb (mail_op, user_data);
5177 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view), new_folder, FALSE);
5178 g_object_unref (folder_view);
5180 move_to_cb (mail_op, user_data);
5185 msgs_move_to_cb (ModestMailOperation *mail_op,
5188 move_to_cb (mail_op, user_data);
5192 modest_ui_actions_move_folder_error_handler (ModestMailOperation *mail_op,
5195 GObject *win = NULL;
5197 #ifndef MODEST_TOOLKIT_HILDON2
5198 ModestWindow *main_window = NULL;
5200 /* Disable next automatic folder selection */
5201 main_window = modest_window_mgr_get_main_window (modest_runtime_get_window_mgr (),
5202 FALSE); /* don't create */
5204 GtkWidget *folder_view = NULL;
5206 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (main_window),
5207 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
5208 modest_folder_view_disable_next_folder_selection (MODEST_FOLDER_VIEW(folder_view));
5210 if (user_data && TNY_IS_FOLDER (user_data)) {
5211 modest_folder_view_select_folder (MODEST_FOLDER_VIEW (folder_view),
5212 TNY_FOLDER (user_data), FALSE);
5216 /* Show notification dialog only if the main window exists */
5217 win = modest_mail_operation_get_source (mail_op);
5218 modest_platform_run_information_dialog ((GtkWindow *) win,
5219 _("mail_in_ui_folder_move_target_error"),
5222 g_object_unref (win);
5226 open_msg_for_purge_cb (ModestMailOperation *mail_op,
5235 gint pending_purges = 0;
5236 gboolean some_purged = FALSE;
5237 ModestWindow *win = MODEST_WINDOW (user_data);
5238 ModestWindowMgr *mgr = modest_runtime_get_window_mgr ();
5240 /* If there was any error */
5241 if (!modest_ui_actions_msg_retrieval_check (mail_op, header, msg)) {
5242 modest_window_mgr_unregister_header (mgr, header);
5246 /* Once the message has been retrieved for purging, we check if
5247 * it's all ok for purging */
5249 parts = tny_simple_list_new ();
5250 tny_mime_part_get_parts (TNY_MIME_PART (msg), parts);
5251 iter = tny_list_create_iterator (parts);
5253 while (!tny_iterator_is_done (iter)) {
5255 part = TNY_MIME_PART (tny_iterator_get_current (iter));
5256 if (part && (tny_mime_part_is_attachment (part) || TNY_IS_MSG (part))) {
5257 if (tny_mime_part_is_purged (part))
5264 g_object_unref (part);
5266 tny_iterator_next (iter);
5268 g_object_unref (iter);
5271 if (pending_purges>0) {
5273 response = modest_platform_run_confirmation_dialog (GTK_WINDOW (win),_("mcen_nc_purge_file_text_inbox"));
5275 if (response == GTK_RESPONSE_OK) {
5278 modest_platform_animation_banner (GTK_WIDGET (win), NULL, _("mcen_me_inbox_remove_attachments"));
5279 iter = tny_list_create_iterator (parts);
5280 while (!tny_iterator_is_done (iter)) {
5283 part = TNY_MIME_PART (tny_iterator_get_current (iter));
5284 if (part && (tny_mime_part_is_attachment (part) || TNY_IS_MSG (part)))
5285 tny_mime_part_set_purged (part);
5288 g_object_unref (part);
5290 tny_iterator_next (iter);
5292 g_object_unref (iter);
5294 tny_msg_rewrite_cache (msg);
5296 gtk_widget_destroy (info);
5300 modest_window_mgr_unregister_header (mgr, header);
5302 g_object_unref (parts);
5306 modest_ui_actions_on_main_window_remove_attachments (GtkAction *action,
5307 ModestMainWindow *win)
5309 GtkWidget *header_view;
5310 TnyList *header_list;
5312 TnyHeaderFlags flags;
5313 ModestWindow *msg_view_window = NULL;
5316 g_return_if_fail (MODEST_IS_MAIN_WINDOW (win));
5318 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
5319 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5321 header_list = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
5323 g_warning ("%s: no header selected", __FUNCTION__);
5327 if (tny_list_get_length (header_list) == 1) {
5328 TnyIterator *iter = tny_list_create_iterator (header_list);
5329 header = TNY_HEADER (tny_iterator_get_current (iter));
5330 g_object_unref (iter);
5334 if (!header || !TNY_IS_HEADER(header)) {
5335 g_warning ("%s: header is not valid", __FUNCTION__);
5339 found = modest_window_mgr_find_registered_header (modest_runtime_get_window_mgr (),
5340 header, &msg_view_window);
5341 flags = tny_header_get_flags (header);
5342 if (!(flags & TNY_HEADER_FLAG_CACHED))
5345 if (msg_view_window != NULL)
5346 modest_msg_view_window_remove_attachments (MODEST_MSG_VIEW_WINDOW (msg_view_window), TRUE);
5348 /* do nothing; uid was registered before, so window is probably on it's way */
5349 g_warning ("debug: header %p has already been registered", header);
5352 ModestMailOperation *mail_op = NULL;
5353 modest_window_mgr_register_header (modest_runtime_get_window_mgr (), header, NULL);
5354 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT (win),
5355 modest_ui_actions_disk_operations_error_handler,
5357 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
5358 modest_mail_operation_get_msg (mail_op, header, FALSE, open_msg_for_purge_cb, win);
5360 g_object_unref (mail_op);
5363 g_object_unref (header);
5365 g_object_unref (header_list);
5369 * Checks if we need a connection to do the transfer and if the user
5370 * wants to connect to complete it
5373 modest_ui_actions_xfer_messages_check (GtkWindow *parent_window,
5374 TnyFolderStore *src_folder,
5376 TnyFolder *dst_folder,
5377 gboolean delete_originals,
5378 gboolean *need_connection,
5381 TnyAccount *src_account;
5382 gint uncached_msgs = 0;
5384 /* We don't need any further check if
5386 * 1- the source folder is local OR
5387 * 2- the device is already online
5389 if (!modest_tny_folder_store_is_remote (src_folder) ||
5390 tny_device_is_online (modest_runtime_get_device())) {
5391 *need_connection = FALSE;
5396 /* We must ask for a connection when
5398 * - the message(s) is not already cached OR
5399 * - the message(s) is cached but the leave_on_server setting
5400 * is FALSE (because we need to sync the source folder to
5401 * delete the message from the server (for IMAP we could do it
5402 * offline, it'll take place the next time we get a
5405 uncached_msgs = header_list_count_uncached_msgs (headers);
5406 src_account = get_account_from_folder_store (src_folder);
5407 if (uncached_msgs > 0) {
5411 *need_connection = TRUE;
5412 num_headers = tny_list_get_length (headers);
5413 msg = ngettext ("mcen_nc_get_msg", "mcen_nc_get_msgs", num_headers);
5415 if (modest_platform_run_confirmation_dialog (parent_window, msg) ==
5416 GTK_RESPONSE_CANCEL) {
5422 /* The transfer is possible and the user wants to */
5425 if (remote_folder_has_leave_on_server (src_folder) && delete_originals) {
5426 const gchar *account_name;
5427 gboolean leave_on_server;
5429 account_name = modest_tny_account_get_parent_modest_account_name_for_server_account (src_account);
5430 leave_on_server = modest_account_mgr_get_leave_on_server (modest_runtime_get_account_mgr (),
5433 if (leave_on_server == TRUE) {
5434 *need_connection = FALSE;
5436 *need_connection = TRUE;
5439 *need_connection = FALSE;
5444 g_object_unref (src_account);
5448 xfer_messages_error_handler (ModestMailOperation *mail_op,
5452 const GError *error;
5454 win = modest_mail_operation_get_source (mail_op);
5455 error = modest_mail_operation_get_error (mail_op);
5457 if (error && is_memory_full_error ((GError *) error, mail_op))
5458 modest_platform_information_banner ((GtkWidget *) win,
5459 NULL, _KR("cerm_device_memory_full"));
5461 modest_platform_run_information_dialog ((GtkWindow *) win,
5462 _("mail_in_ui_folder_move_target_error"),
5465 g_object_unref (win);
5469 TnyFolderStore *dst_folder;
5474 * Utility function that transfer messages from both the main window
5475 * and the msg view window when using the "Move to" dialog
5478 xfer_messages_performer (gboolean canceled,
5480 GtkWindow *parent_window,
5481 TnyAccount *account,
5484 ModestWindow *win = MODEST_WINDOW (parent_window);
5485 TnyAccount *dst_account = NULL;
5486 gboolean dst_forbids_message_add = FALSE;
5487 XferMsgsHelper *helper;
5488 MoveToHelper *movehelper;
5489 ModestMailOperation *mail_op;
5491 helper = (XferMsgsHelper *) user_data;
5493 if (canceled || err) {
5494 if (!check_memory_full_error ((GtkWidget *) parent_window, err)) {
5495 /* Show the proper error message */
5496 modest_ui_actions_on_account_connection_error (parent_window, account);
5501 dst_account = tny_folder_get_account (TNY_FOLDER (helper->dst_folder));
5503 /* tinymail will return NULL for local folders it seems */
5504 dst_forbids_message_add = modest_protocol_registry_protocol_type_has_tag (modest_runtime_get_protocol_registry (),
5505 modest_tny_account_get_protocol_type (dst_account),
5506 MODEST_PROTOCOL_REGISTRY_STORE_FORBID_MESSAGE_ADD);
5507 g_object_unref (dst_account);
5509 if (dst_forbids_message_add) {
5510 modest_platform_information_banner (GTK_WIDGET (win),
5512 ngettext("mail_in_ui_folder_move_target_error",
5513 "mail_in_ui_folder_move_targets_error",
5514 tny_list_get_length (helper->headers)));
5518 movehelper = g_new0 (MoveToHelper, 1);
5520 #ifndef MODEST_TOOLKIT_HILDON2
5521 movehelper->banner = modest_platform_animation_banner (GTK_WIDGET (win), NULL,
5522 _CS("ckct_nw_pasting"));
5523 if (movehelper->banner != NULL) {
5524 g_object_ref (movehelper->banner);
5525 gtk_widget_show (GTK_WIDGET (movehelper->banner));
5529 if (MODEST_IS_MAIN_WINDOW (win)) {
5530 GtkWidget *header_view =
5531 modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
5532 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5533 movehelper->reference = get_next_after_selected_headers (MODEST_HEADER_VIEW (header_view));
5536 /* Perform the mail operation */
5537 mail_op = modest_mail_operation_new_with_error_handling (G_OBJECT(win),
5538 xfer_messages_error_handler,
5540 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
5543 modest_mail_operation_xfer_msgs (mail_op,
5545 TNY_FOLDER (helper->dst_folder),
5550 g_object_unref (G_OBJECT (mail_op));
5552 g_object_unref (helper->dst_folder);
5553 g_object_unref (helper->headers);
5554 g_slice_free (XferMsgsHelper, helper);
5558 TnyFolder *src_folder;
5559 TnyFolderStore *dst_folder;
5560 gboolean delete_original;
5561 GtkWidget *folder_view;
5565 on_move_folder_cb (gboolean canceled, GError *err, GtkWindow *parent_window,
5566 TnyAccount *account, gpointer user_data)
5568 MoveFolderInfo *info = (MoveFolderInfo*)user_data;
5569 GtkTreeSelection *sel;
5570 ModestMailOperation *mail_op = NULL;
5572 if (canceled || err || !MODEST_IS_WINDOW (parent_window)) {
5573 g_object_unref (G_OBJECT (info->src_folder));
5574 g_object_unref (G_OBJECT (info->dst_folder));
5579 MoveToHelper *helper = g_new0 (MoveToHelper, 1);
5580 #ifndef MODEST_TOOLKIT_HILDON2
5581 helper->banner = modest_platform_animation_banner (GTK_WIDGET (parent_window), NULL,
5582 _CS("ckct_nw_pasting"));
5583 if (helper->banner != NULL) {
5584 g_object_ref (helper->banner);
5585 gtk_widget_show (GTK_WIDGET(helper->banner));
5588 /* Clean folder on header view before moving it */
5589 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (info->folder_view));
5590 gtk_tree_selection_unselect_all (sel);
5592 /* Let gtk events run. We need that the folder
5593 view frees its reference to the source
5594 folder *before* issuing the mail operation
5595 so we need the signal handler of selection
5596 changed to happen before the mail
5598 while (gtk_events_pending ())
5599 gtk_main_iteration (); */
5602 modest_mail_operation_new_with_error_handling (G_OBJECT(parent_window),
5603 modest_ui_actions_move_folder_error_handler,
5604 info->src_folder, NULL);
5605 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
5608 /* Select *after* the changes */
5609 /* TODO: this function hangs UI after transfer */
5610 /* modest_folder_view_select_folder (MODEST_FOLDER_VIEW(folder_view), */
5611 /* TNY_FOLDER (src_folder), TRUE); */
5613 if (MODEST_IS_MAIN_WINDOW (parent_window)) {
5614 modest_folder_view_select_folder (MODEST_FOLDER_VIEW(info->folder_view),
5615 TNY_FOLDER (info->dst_folder), TRUE);
5617 modest_mail_operation_xfer_folder (mail_op,
5618 TNY_FOLDER (info->src_folder),
5620 info->delete_original,
5623 g_object_unref (G_OBJECT (info->src_folder));
5625 /* if (modest_mail_operation_get_status (mail_op) == MODEST_MAIL_OPERATION_STATUS_SUCCESS) { */
5628 /* Unref mail operation */
5629 g_object_unref (G_OBJECT (mail_op));
5630 g_object_unref (G_OBJECT (info->dst_folder));
5635 get_account_from_folder_store (TnyFolderStore *folder_store)
5637 if (TNY_IS_ACCOUNT (folder_store))
5638 return g_object_ref (folder_store);
5640 return tny_folder_get_account (TNY_FOLDER (folder_store));
5644 * UI handler for the "Move to" action when invoked from the
5648 modest_ui_actions_on_main_window_move_to (GtkAction *action,
5649 GtkWidget *folder_view,
5650 TnyFolderStore *dst_folder,
5651 ModestMainWindow *win)
5653 ModestHeaderView *header_view = NULL;
5654 TnyFolderStore *src_folder = NULL;
5656 g_return_if_fail (MODEST_IS_MAIN_WINDOW (win));
5658 /* Get the source folder */
5659 src_folder = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
5661 /* Get header view */
5662 header_view = (ModestHeaderView *)
5663 modest_main_window_get_child_widget (win, MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
5665 /* Get folder or messages to transfer */
5666 if (gtk_widget_is_focus (folder_view)) {
5667 gboolean do_xfer = TRUE;
5669 /* Allow only to transfer folders to the local root folder */
5670 if (TNY_IS_ACCOUNT (dst_folder) &&
5671 !MODEST_IS_TNY_LOCAL_FOLDERS_ACCOUNT (dst_folder) &&
5672 !modest_tny_account_is_memory_card_account (TNY_ACCOUNT (dst_folder))) {
5674 } else if (!TNY_IS_FOLDER (src_folder)) {
5675 g_warning ("%s: src_folder is not a TnyFolder.\n", __FUNCTION__);
5680 MoveFolderInfo *info = g_new0 (MoveFolderInfo, 1);
5681 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5683 info->src_folder = g_object_ref (src_folder);
5684 info->dst_folder = g_object_ref (dst_folder);
5685 info->delete_original = TRUE;
5686 info->folder_view = folder_view;
5688 connect_info->callback = on_move_folder_cb;
5689 connect_info->dst_account = get_account_from_folder_store (TNY_FOLDER_STORE (dst_folder));
5690 connect_info->data = info;
5692 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5693 TNY_FOLDER_STORE (src_folder),
5696 } else if (gtk_widget_is_focus (GTK_WIDGET(header_view))) {
5699 headers = modest_header_view_get_selected_headers(header_view);
5701 /* Transfer the messages */
5702 modest_ui_actions_transfer_messages_helper (GTK_WINDOW (win), TNY_FOLDER (src_folder),
5703 headers, TNY_FOLDER (dst_folder));
5705 g_object_unref (headers);
5709 g_object_unref (src_folder);
5712 #ifdef MODEST_TOOLKIT_HILDON2
5714 * UI handler for the "Move to" action when invoked from the
5715 * ModestFolderWindow
5718 modest_ui_actions_on_folder_window_move_to (GtkWidget *folder_view,
5719 TnyFolderStore *dst_folder,
5723 TnyFolderStore *src_folder = NULL;
5724 TnyIterator *iterator;
5726 if (tny_list_get_length (selection) != 1)
5729 iterator = tny_list_create_iterator (selection);
5730 src_folder = TNY_FOLDER_STORE (tny_iterator_get_current (iterator));
5731 g_object_unref (iterator);
5734 gboolean do_xfer = TRUE;
5736 /* Allow only to transfer folders to the local root folder */
5737 if (TNY_IS_ACCOUNT (dst_folder) &&
5738 !MODEST_IS_TNY_LOCAL_FOLDERS_ACCOUNT (dst_folder) &&
5739 !modest_tny_account_is_memory_card_account (TNY_ACCOUNT (dst_folder))) {
5742 modest_platform_run_information_dialog (win,
5743 _("mail_in_ui_folder_move_target_error"),
5745 } else if (!TNY_IS_FOLDER (src_folder)) {
5746 g_warning ("%s: src_folder is not a TnyFolder.\n", __FUNCTION__);
5751 MoveFolderInfo *info = g_new0 (MoveFolderInfo, 1);
5752 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5754 info->src_folder = g_object_ref (src_folder);
5755 info->dst_folder = g_object_ref (dst_folder);
5756 info->delete_original = TRUE;
5757 info->folder_view = folder_view;
5759 connect_info->callback = on_move_folder_cb;
5760 connect_info->dst_account = get_account_from_folder_store (TNY_FOLDER_STORE (dst_folder));
5761 connect_info->data = info;
5763 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5764 TNY_FOLDER_STORE (src_folder),
5769 g_object_unref (src_folder);
5775 modest_ui_actions_transfer_messages_helper (GtkWindow *win,
5776 TnyFolder *src_folder,
5778 TnyFolder *dst_folder)
5780 gboolean need_connection = TRUE;
5781 gboolean do_xfer = TRUE;
5782 XferMsgsHelper *helper;
5784 g_return_if_fail (TNY_IS_FOLDER (src_folder));
5785 g_return_if_fail (TNY_IS_FOLDER (dst_folder));
5786 g_return_if_fail (TNY_IS_LIST (headers));
5788 modest_ui_actions_xfer_messages_check (win, TNY_FOLDER_STORE (src_folder),
5789 headers, TNY_FOLDER (dst_folder),
5790 TRUE, &need_connection,
5793 /* If we don't want to transfer just return */
5797 /* Create the helper */
5798 helper = g_slice_new (XferMsgsHelper);
5799 helper->dst_folder = g_object_ref (dst_folder);
5800 helper->headers = g_object_ref (headers);
5802 if (need_connection) {
5803 DoubleConnectionInfo *connect_info = g_slice_new (DoubleConnectionInfo);
5804 connect_info->callback = xfer_messages_performer;
5805 connect_info->dst_account = tny_folder_get_account (TNY_FOLDER (dst_folder));
5806 connect_info->data = helper;
5808 modest_platform_double_connect_and_perform(GTK_WINDOW (win), TRUE,
5809 TNY_FOLDER_STORE (src_folder),
5812 TnyAccount *src_account = get_account_from_folder_store (TNY_FOLDER_STORE (src_folder));
5813 xfer_messages_performer (FALSE, NULL, GTK_WINDOW (win),
5814 src_account, helper);
5815 g_object_unref (src_account);
5820 * UI handler for the "Move to" action when invoked from the
5821 * ModestMsgViewWindow
5824 modest_ui_actions_on_window_move_to (GtkAction *action,
5826 TnyFolderStore *dst_folder,
5829 TnyFolder *src_folder = NULL;
5831 g_return_if_fail (TNY_IS_FOLDER (dst_folder));
5834 TnyHeader *header = NULL;
5837 iter = tny_list_create_iterator (headers);
5838 header = (TnyHeader *) tny_iterator_get_current (iter);
5839 src_folder = tny_header_get_folder (header);
5841 /* Transfer the messages */
5842 modest_ui_actions_transfer_messages_helper (GTK_WINDOW (win), src_folder,
5844 TNY_FOLDER (dst_folder));
5847 g_object_unref (header);
5848 g_object_unref (iter);
5849 g_object_unref (src_folder);
5854 modest_ui_actions_on_move_to (GtkAction *action,
5857 modest_ui_actions_on_edit_mode_move_to (win);
5861 modest_ui_actions_on_edit_mode_move_to (ModestWindow *win)
5863 GtkWidget *dialog = NULL;
5864 MoveToInfo *helper = NULL;
5865 TnyList *list_to_move;
5867 g_return_val_if_fail (MODEST_IS_WINDOW (win), FALSE);
5869 #ifndef MODEST_TOOLKIT_HILDON2
5870 /* Get the main window if exists */
5871 ModestMainWindow *main_window;
5872 if (MODEST_IS_MAIN_WINDOW (win))
5873 main_window = MODEST_MAIN_WINDOW (win);
5876 MODEST_MAIN_WINDOW (modest_window_mgr_get_main_window (modest_runtime_get_window_mgr (),
5877 FALSE)); /* don't create */
5880 list_to_move = modest_platform_get_list_to_move (MODEST_WINDOW (win));
5885 if (tny_list_get_length (list_to_move) < 1) {
5886 g_object_unref (list_to_move);
5890 /* Create and run the dialog */
5891 dialog = create_move_to_dialog (GTK_WINDOW (win), NULL, list_to_move);
5892 modest_window_mgr_set_modal (modest_runtime_get_window_mgr (),
5893 GTK_WINDOW (dialog),
5897 helper = g_slice_new0 (MoveToInfo);
5898 helper->list = list_to_move;
5901 /* Listen to response signal */
5902 g_signal_connect (dialog, "response", G_CALLBACK (on_move_to_dialog_response), helper);
5904 /* Show the dialog */
5905 gtk_widget_show (dialog);
5911 * Calls #HeadersFunc for each header already selected in the main
5912 * window or the message currently being shown in the msg view window
5915 do_headers_action (ModestWindow *win,
5919 TnyList *headers_list = NULL;
5920 TnyIterator *iter = NULL;
5921 TnyHeader *header = NULL;
5922 TnyFolder *folder = NULL;
5925 headers_list = get_selected_headers (win);
5929 /* Get the folder */
5930 iter = tny_list_create_iterator (headers_list);
5931 header = TNY_HEADER (tny_iterator_get_current (iter));
5933 folder = tny_header_get_folder (header);
5934 g_object_unref (header);
5937 /* Call the function for each header */
5938 while (!tny_iterator_is_done (iter)) {
5939 header = TNY_HEADER (tny_iterator_get_current (iter));
5940 func (header, win, user_data);
5941 g_object_unref (header);
5942 tny_iterator_next (iter);
5945 /* Trick: do a poke status in order to speed up the signaling
5948 tny_folder_poke_status (folder);
5949 g_object_unref (folder);
5953 g_object_unref (iter);
5954 g_object_unref (headers_list);
5958 modest_ui_actions_view_attachment (GtkAction *action,
5959 ModestWindow *window)
5961 if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
5962 modest_msg_view_window_view_attachment (MODEST_MSG_VIEW_WINDOW (window), NULL);
5964 /* not supported window for this action */
5965 g_return_if_reached ();
5970 modest_ui_actions_save_attachments (GtkAction *action,
5971 ModestWindow *window)
5973 if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
5975 if (modest_platform_check_memory_low (MODEST_WINDOW(window), TRUE))
5978 modest_msg_view_window_save_attachments (MODEST_MSG_VIEW_WINDOW (window), NULL);
5980 /* not supported window for this action */
5981 g_return_if_reached ();
5986 modest_ui_actions_remove_attachments (GtkAction *action,
5987 ModestWindow *window)
5989 if (MODEST_IS_MAIN_WINDOW (window)) {
5990 modest_ui_actions_on_main_window_remove_attachments (action, MODEST_MAIN_WINDOW (window));
5991 } else if (MODEST_IS_MSG_VIEW_WINDOW (window)) {
5992 modest_msg_view_window_remove_attachments (MODEST_MSG_VIEW_WINDOW (window), FALSE);
5994 /* not supported window for this action */
5995 g_return_if_reached ();
6000 modest_ui_actions_on_settings (GtkAction *action,
6005 dialog = modest_platform_get_global_settings_dialog ();
6006 gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (win));
6007 gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
6008 gtk_widget_show_all (dialog);
6010 gtk_dialog_run (GTK_DIALOG (dialog));
6012 gtk_widget_destroy (dialog);
6016 modest_ui_actions_on_help (GtkAction *action,
6019 /* Help app is not available at all in fremantle */
6020 #ifndef MODEST_TOOLKIT_HILDON2
6021 const gchar *help_id;
6023 g_return_if_fail (win && GTK_IS_WINDOW(win));
6025 help_id = modest_window_mgr_get_help_id (modest_runtime_get_window_mgr(), win);
6028 modest_platform_show_help (GTK_WINDOW (win), help_id);
6033 modest_ui_actions_on_csm_help (GtkAction *action,
6036 /* Help app is not available at all in fremantle */
6037 #ifndef MODEST_TOOLKIT_HILDON2
6039 const gchar* help_id = NULL;
6040 GtkWidget *folder_view;
6041 TnyFolderStore *folder_store;
6043 g_return_if_fail (win && MODEST_IS_MAIN_WINDOW (win));
6045 /* Get selected folder */
6046 folder_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (win),
6047 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
6048 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
6050 /* Switch help_id */
6051 if (folder_store && TNY_IS_FOLDER (folder_store))
6052 help_id = modest_tny_folder_get_help_id (TNY_FOLDER (folder_store));
6055 g_object_unref (folder_store);
6058 modest_platform_show_help (GTK_WINDOW (win), help_id);
6060 modest_ui_actions_on_help (action, win);
6065 retrieve_contents_cb (ModestMailOperation *mail_op,
6072 /* We only need this callback to show an error in case of
6073 memory low condition */
6074 modest_ui_actions_msg_retrieval_check (mail_op, header, msg);
6078 retrieve_msg_contents_performer (gboolean canceled,
6080 GtkWindow *parent_window,
6081 TnyAccount *account,
6084 ModestMailOperation *mail_op;
6085 TnyList *headers = TNY_LIST (user_data);
6087 if (err || canceled) {
6088 check_memory_full_error ((GtkWidget *) parent_window, err);
6092 /* Create mail operation */
6093 mail_op = modest_mail_operation_new_with_error_handling ((GObject *) parent_window,
6094 modest_ui_actions_disk_operations_error_handler,
6096 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
6097 modest_mail_operation_get_msgs_full (mail_op, headers, retrieve_contents_cb, NULL, NULL);
6100 g_object_unref (mail_op);
6102 g_object_unref (headers);
6103 g_object_unref (account);
6107 modest_ui_actions_on_retrieve_msg_contents (GtkAction *action,
6108 ModestWindow *window)
6110 TnyList *headers = NULL;
6111 TnyAccount *account = NULL;
6112 TnyIterator *iter = NULL;
6113 TnyHeader *header = NULL;
6114 TnyFolder *folder = NULL;
6117 headers = get_selected_headers (window);
6121 /* Pick the account */
6122 iter = tny_list_create_iterator (headers);
6123 header = TNY_HEADER (tny_iterator_get_current (iter));
6124 folder = tny_header_get_folder (header);
6125 account = tny_folder_get_account (folder);
6126 g_object_unref (folder);
6127 g_object_unref (header);
6128 g_object_unref (iter);
6130 /* Connect and perform the message retrieval */
6131 modest_platform_connect_and_perform ((GtkWindow *) window, TRUE,
6132 g_object_ref (account),
6133 retrieve_msg_contents_performer,
6134 g_object_ref (headers));
6137 g_object_unref (account);
6138 g_object_unref (headers);
6142 modest_ui_actions_check_toolbar_dimming_rules (ModestWindow *window)
6144 g_return_if_fail (MODEST_IS_WINDOW (window));
6147 modest_window_check_dimming_rules_group (window, MODEST_DIMMING_RULES_TOOLBAR);
6151 modest_ui_actions_check_menu_dimming_rules (ModestWindow *window)
6153 g_return_if_fail (MODEST_IS_WINDOW (window));
6156 modest_window_check_dimming_rules_group (window, MODEST_DIMMING_RULES_MENU);
6160 modest_ui_actions_on_email_menu_activated (GtkAction *action,
6161 ModestWindow *window)
6163 g_return_if_fail (MODEST_IS_WINDOW (window));
6166 modest_ui_actions_check_menu_dimming_rules (window);
6170 modest_ui_actions_on_edit_menu_activated (GtkAction *action,
6171 ModestWindow *window)
6173 g_return_if_fail (MODEST_IS_WINDOW (window));
6176 modest_ui_actions_check_menu_dimming_rules (window);
6180 modest_ui_actions_on_view_menu_activated (GtkAction *action,
6181 ModestWindow *window)
6183 g_return_if_fail (MODEST_IS_WINDOW (window));
6186 modest_ui_actions_check_menu_dimming_rules (window);
6190 modest_ui_actions_on_format_menu_activated (GtkAction *action,
6191 ModestWindow *window)
6193 g_return_if_fail (MODEST_IS_WINDOW (window));
6196 modest_ui_actions_check_menu_dimming_rules (window);
6200 modest_ui_actions_on_tools_menu_activated (GtkAction *action,
6201 ModestWindow *window)
6203 g_return_if_fail (MODEST_IS_WINDOW (window));
6206 modest_ui_actions_check_menu_dimming_rules (window);
6210 modest_ui_actions_on_attachment_menu_activated (GtkAction *action,
6211 ModestWindow *window)
6213 g_return_if_fail (MODEST_IS_WINDOW (window));
6216 modest_ui_actions_check_menu_dimming_rules (window);
6220 modest_ui_actions_on_toolbar_csm_menu_activated (GtkAction *action,
6221 ModestWindow *window)
6223 g_return_if_fail (MODEST_IS_WINDOW (window));
6226 modest_ui_actions_check_menu_dimming_rules (window);
6230 modest_ui_actions_on_folder_view_csm_menu_activated (GtkAction *action,
6231 ModestWindow *window)
6233 g_return_if_fail (MODEST_IS_WINDOW (window));
6236 modest_ui_actions_check_menu_dimming_rules (window);
6240 modest_ui_actions_on_header_view_csm_menu_activated (GtkAction *action,
6241 ModestWindow *window)
6243 g_return_if_fail (MODEST_IS_WINDOW (window));
6246 modest_ui_actions_check_menu_dimming_rules (window);
6250 modest_ui_actions_on_search_messages (GtkAction *action, ModestWindow *window)
6252 g_return_if_fail (MODEST_IS_WINDOW (window));
6254 /* we check for low-mem; in that case, show a warning, and don't allow
6257 if (modest_platform_check_memory_low (window, TRUE))
6260 modest_platform_show_search_messages (GTK_WINDOW (window));
6264 modest_ui_actions_on_open_addressbook (GtkAction *action, ModestWindow *win)
6266 g_return_if_fail (MODEST_IS_WINDOW (win));
6269 /* we check for low-mem; in that case, show a warning, and don't allow
6270 * for the addressbook
6272 if (modest_platform_check_memory_low (win, TRUE))
6276 modest_platform_show_addressbook (GTK_WINDOW (win));
6281 modest_ui_actions_on_toggle_find_in_page (GtkAction *action,
6282 ModestWindow *window)
6285 g_return_if_fail (MODEST_IS_MSG_EDIT_WINDOW (window));
6287 if (GTK_IS_TOGGLE_ACTION (action))
6288 active = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action));
6292 modest_msg_edit_window_toggle_find_toolbar (MODEST_MSG_EDIT_WINDOW (window),
6297 on_send_receive_finished (ModestMailOperation *mail_op,
6300 GtkWidget *header_view, *folder_view;
6301 TnyFolderStore *folder_store;
6302 ModestMainWindow *main_win = MODEST_MAIN_WINDOW (user_data);
6304 /* Set send/receive operation finished */
6305 modest_main_window_notify_send_receive_completed (main_win);
6307 /* Don't refresh the current folder if there were any errors */
6308 if (modest_mail_operation_get_status (mail_op) !=
6309 MODEST_MAIL_OPERATION_STATUS_SUCCESS)
6312 /* Refresh the current folder if we're viewing a window. We do
6313 this because the user won't be able to see the new mails in
6314 the selected folder after a Send&Receive because it only
6315 performs a poke_status, i.e, only the number of read/unread
6316 messages is updated, but the new headers are not
6318 folder_view = modest_main_window_get_child_widget (main_win,
6319 MODEST_MAIN_WINDOW_WIDGET_TYPE_FOLDER_VIEW);
6323 folder_store = modest_folder_view_get_selected (MODEST_FOLDER_VIEW (folder_view));
6325 /* Do not need to refresh INBOX again because the
6326 update_account does it always automatically */
6327 if (folder_store && TNY_IS_FOLDER (folder_store) &&
6328 tny_folder_get_folder_type (TNY_FOLDER (folder_store)) != TNY_FOLDER_TYPE_INBOX) {
6329 ModestMailOperation *refresh_op;
6331 header_view = modest_main_window_get_child_widget (main_win,
6332 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6334 /* We do not need to set the contents style
6335 because it hasn't changed. We also do not
6336 need to save the widget status. Just force
6338 refresh_op = modest_mail_operation_new (G_OBJECT (main_win));
6339 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), refresh_op);
6340 modest_mail_operation_refresh_folder (refresh_op, TNY_FOLDER (folder_store),
6341 folder_refreshed_cb, main_win);
6342 g_object_unref (refresh_op);
6346 g_object_unref (folder_store);
6351 modest_ui_actions_on_send_queue_error_happened (TnySendQueue *self,
6357 const gchar* server_name = NULL;
6358 TnyTransportAccount *transport;
6359 gchar *message = NULL;
6360 ModestProtocol *protocol;
6362 /* Don't show anything if the user cancelled something or the
6363 * send receive request is not interactive. Authentication
6364 * errors are managed by the account store so no need to show
6365 * a dialog here again */
6366 if (err->code == TNY_SYSTEM_ERROR_CANCEL ||
6367 err->code == TNY_SERVICE_ERROR_AUTHENTICATE ||
6368 !modest_tny_send_queue_get_requested_send_receive (MODEST_TNY_SEND_QUEUE (self)))
6372 /* Get the server name. Note that we could be using a
6373 connection specific transport account */
6374 transport = (TnyTransportAccount *)
6375 tny_camel_send_queue_get_transport_account (TNY_CAMEL_SEND_QUEUE (self));
6377 ModestTnyAccountStore *acc_store;
6378 const gchar *acc_name;
6379 TnyTransportAccount *conn_specific;
6381 acc_store = modest_runtime_get_account_store();
6382 acc_name = modest_tny_account_get_parent_modest_account_name_for_server_account (TNY_ACCOUNT (transport));
6383 conn_specific = (TnyTransportAccount *)
6384 modest_tny_account_store_get_transport_account_for_open_connection (acc_store, acc_name);
6385 if (conn_specific) {
6386 server_name = tny_account_get_hostname (TNY_ACCOUNT (conn_specific));
6387 g_object_unref (conn_specific);
6389 server_name = tny_account_get_hostname (TNY_ACCOUNT (transport));
6391 g_object_unref (transport);
6395 protocol = modest_protocol_registry_get_protocol_by_name (modest_runtime_get_protocol_registry (),
6396 MODEST_PROTOCOL_REGISTRY_TRANSPORT_STORE_PROTOCOLS,
6397 tny_account_get_proto (TNY_ACCOUNT (transport)));
6399 g_warning ("%s: Account with no proto", __FUNCTION__);
6403 /* Show the appropriate message text for the GError: */
6404 switch (err->code) {
6405 case TNY_SERVICE_ERROR_CONNECT:
6406 message = modest_protocol_get_translation (protocol,
6407 MODEST_PROTOCOL_TRANSLATION_ACCOUNT_CONNECTION_ERROR,
6410 case TNY_SERVICE_ERROR_SEND:
6411 message = g_strdup (_CS("sfil_ib_unable_to_send"));
6413 case TNY_SERVICE_ERROR_UNAVAILABLE:
6414 message = modest_protocol_get_translation (protocol,
6415 MODEST_PROTOCOL_TRANSLATION_CONNECT_ERROR,
6419 g_warning ("%s: unexpected ERROR %d",
6420 __FUNCTION__, err->code);
6421 message = g_strdup (_CS("sfil_ib_unable_to_send"));
6425 modest_platform_run_information_dialog (NULL, message, FALSE);
6430 modest_ui_actions_on_send_queue_status_changed (ModestTnySendQueue *send_queue,
6435 ModestWindow *top_window = NULL;
6436 ModestWindowMgr *mgr = NULL;
6437 GtkWidget *header_view = NULL;
6438 TnyFolder *selected_folder = NULL;
6439 TnyFolderType folder_type;
6441 mgr = modest_runtime_get_window_mgr ();
6442 top_window = modest_window_mgr_get_current_top (mgr);
6447 #ifndef MODEST_TOOLKIT_HILDON2
6448 if (MODEST_IS_MAIN_WINDOW (top_window)) {
6449 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW (top_window),
6450 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6453 if (MODEST_IS_HEADER_WINDOW (top_window)) {
6454 header_view = (GtkWidget *)
6455 modest_header_window_get_header_view (MODEST_HEADER_WINDOW (top_window));
6459 /* Get selected folder */
6461 selected_folder = modest_header_view_get_folder (MODEST_HEADER_VIEW (header_view));
6462 if (!selected_folder)
6465 /* gtk_tree_view_column_queue_resize is only available in GTK+ 2.8 */
6466 #if GTK_CHECK_VERSION(2, 8, 0)
6467 folder_type = modest_tny_folder_guess_folder_type (selected_folder);
6468 if (folder_type == TNY_FOLDER_TYPE_OUTBOX) {
6469 GtkTreeViewColumn *tree_column;
6471 tree_column = gtk_tree_view_get_column (GTK_TREE_VIEW (header_view),
6472 TNY_GTK_HEADER_LIST_MODEL_FROM_COLUMN);
6474 gtk_tree_view_column_queue_resize (tree_column);
6476 #else /* #if GTK_CHECK_VERSION(2, 8, 0) */
6477 gtk_widget_queue_draw (header_view);
6480 #ifndef MODEST_TOOLKIT_HILDON2
6481 /* Rerun dimming rules, because the message could become deletable for example */
6482 modest_window_check_dimming_rules_group (MODEST_WINDOW (top_window),
6483 MODEST_DIMMING_RULES_TOOLBAR);
6484 modest_window_check_dimming_rules_group (MODEST_WINDOW (top_window),
6485 MODEST_DIMMING_RULES_MENU);
6489 g_object_unref (selected_folder);
6493 modest_ui_actions_on_account_connection_error (GtkWindow *parent_window,
6494 TnyAccount *account)
6496 ModestProtocolType protocol_type;
6497 ModestProtocol *protocol;
6498 gchar *error_note = NULL;
6500 protocol_type = modest_tny_account_get_protocol_type (account);
6501 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6504 error_note = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_ACCOUNT_CONNECTION_ERROR, tny_account_get_hostname (account));
6505 if (error_note == NULL) {
6506 g_warning ("%s: This should not be reached", __FUNCTION__);
6508 modest_platform_run_information_dialog (parent_window, error_note, FALSE);
6509 g_free (error_note);
6514 modest_ui_actions_get_msg_already_deleted_error_msg (ModestWindow *win)
6518 TnyFolderStore *folder = NULL;
6519 TnyAccount *account = NULL;
6520 ModestProtocolType proto;
6521 ModestProtocol *protocol;
6522 TnyHeader *header = NULL;
6524 if (MODEST_IS_MAIN_WINDOW (win)) {
6525 GtkWidget *header_view;
6526 TnyList* headers = NULL;
6528 header_view = modest_main_window_get_child_widget (MODEST_MAIN_WINDOW(win),
6529 MODEST_MAIN_WINDOW_WIDGET_TYPE_HEADER_VIEW);
6530 headers = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
6531 if (!headers || tny_list_get_length (headers) == 0) {
6533 g_object_unref (headers);
6536 iter = tny_list_create_iterator (headers);
6537 header = TNY_HEADER (tny_iterator_get_current (iter));
6538 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6539 g_object_unref (iter);
6540 g_object_unref (headers);
6541 #ifdef MODEST_TOOLKIT_HILDON2
6542 } else if (MODEST_IS_HEADER_WINDOW (win)) {
6543 GtkWidget *header_view;
6544 TnyList* headers = NULL;
6546 header_view = GTK_WIDGET (modest_header_window_get_header_view (MODEST_HEADER_WINDOW (win)));
6547 headers = modest_header_view_get_selected_headers (MODEST_HEADER_VIEW (header_view));
6548 if (!headers || tny_list_get_length (headers) == 0) {
6550 g_object_unref (headers);
6553 iter = tny_list_create_iterator (headers);
6554 header = TNY_HEADER (tny_iterator_get_current (iter));
6555 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6556 g_object_unref (iter);
6557 g_object_unref (headers);
6559 } else if (MODEST_IS_MSG_VIEW_WINDOW (win)) {
6560 header = modest_msg_view_window_get_header (MODEST_MSG_VIEW_WINDOW (win));
6561 folder = TNY_FOLDER_STORE (tny_header_get_folder (header));
6564 if (!header || !folder)
6567 /* Get the account type */
6568 account = tny_folder_get_account (TNY_FOLDER (folder));
6569 proto = modest_tny_account_get_protocol_type (account);
6570 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6573 subject = tny_header_dup_subject (header);
6574 msg = modest_protocol_get_translation (protocol, MODEST_PROTOCOL_TRANSLATION_MSG_NOT_AVAILABLE, subject);
6578 msg = g_strdup_printf (_("mail_ni_ui_folder_get_msg_folder_error"));
6584 g_object_unref (account);
6586 g_object_unref (folder);
6588 g_object_unref (header);
6594 modest_ui_actions_on_delete_account (GtkWindow *parent_window,
6595 const gchar *account_name,
6596 const gchar *account_title)
6598 ModestAccountMgr *account_mgr;
6601 ModestProtocol *protocol;
6602 gboolean removed = FALSE;
6604 g_return_val_if_fail (account_name, FALSE);
6605 g_return_val_if_fail (account_title, FALSE);
6607 account_mgr = modest_runtime_get_account_mgr();
6609 /* The warning text depends on the account type: */
6610 protocol = modest_protocol_registry_get_protocol_by_type (modest_runtime_get_protocol_registry (),
6611 modest_account_mgr_get_store_protocol (account_mgr,
6613 txt = modest_protocol_get_translation (protocol,
6614 MODEST_PROTOCOL_TRANSLATION_DELETE_MAILBOX,
6617 txt = g_strdup_printf (_("emev_nc_delete_mailbox"), account_title);
6619 response = modest_platform_run_confirmation_dialog (parent_window, txt);
6623 if (response == GTK_RESPONSE_OK) {
6624 /* Remove account. If it succeeds then it also removes
6625 the account from the ModestAccountView: */
6626 gboolean is_default = FALSE;
6627 gchar *default_account_name = modest_account_mgr_get_default_account (account_mgr);
6628 if (default_account_name && (strcmp (default_account_name, account_name) == 0))
6630 g_free (default_account_name);
6632 removed = modest_account_mgr_remove_account (account_mgr, account_name);
6634 g_warning ("%s: modest_account_mgr_remove_account() failed.\n", __FUNCTION__);