2da7e8ac28425c3acc043b1abd79d299b303c74e
[modest] / src / maemo / modest-msg-view-window.c
1 /* Copyright (c) 2006, Nokia Corporation
2  * All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
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.
16  *
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.
28  */
29 #include <glib/gi18n.h>
30 #include <string.h>
31 #include <tny-account-store.h>
32 #include <tny-simple-list.h>
33 #include <tny-header.h>
34 #include <tny-vfs-stream.h>
35 #include "modest-platform.h"
36 #include <modest-maemo-utils.h>
37 #include <modest-tny-msg.h>
38 #include <modest-msg-view-window.h>
39 #include <modest-attachments-view.h>
40 #include <modest-main-window-ui.h>
41 #include "modest-msg-view-window-ui-dimming.h"
42 #include <modest-widget-memory.h>
43 #include <modest-runtime.h>
44 #include <modest-window-priv.h>
45 #include <modest-tny-folder.h>
46 #include <modest-text-utils.h>
47 #include <modest-account-mgr-helpers.h>
48 #include "modest-progress-bar-widget.h"
49 #include "modest-defs.h"
50 #include "modest-hildon-includes.h"
51 #include <gtkhtml/gtkhtml-search.h>
52 #include "modest-ui-dimming-manager.h"
53 #include <gdk/gdkkeysyms.h>
54
55 #define DEFAULT_FOLDER "MyDocs/.documents"
56
57 static void  modest_msg_view_window_class_init   (ModestMsgViewWindowClass *klass);
58 static void  modest_msg_view_window_init         (ModestMsgViewWindow *obj);
59 static void  modest_msg_view_window_finalize     (GObject *obj);
60 static void  modest_msg_view_window_toggle_find_toolbar (GtkToggleAction *obj,
61                                                          gpointer data);
62 static void  modest_msg_view_window_find_toolbar_close (GtkWidget *widget,
63                                                         ModestMsgViewWindow *obj);
64 static void  modest_msg_view_window_find_toolbar_search (GtkWidget *widget,
65                                                         ModestMsgViewWindow *obj);
66
67 static void  modest_msg_view_window_set_zoom (ModestWindow *window,
68                                               gdouble zoom);
69 static gdouble modest_msg_view_window_get_zoom (ModestWindow *window);
70 static gboolean modest_msg_view_window_zoom_minus (ModestWindow *window);
71 static gboolean modest_msg_view_window_zoom_plus (ModestWindow *window);
72 static gboolean modest_msg_view_window_key_release_event (GtkWidget *window,
73                                                           GdkEventKey *event,
74                                                           gpointer userdata);
75 static gboolean modest_msg_view_window_window_state_event (GtkWidget *widget, 
76                                                            GdkEventWindowState *event, 
77                                                            gpointer userdata);
78 static void modest_msg_view_window_scroll_up (ModestWindow *window);
79 static void modest_msg_view_window_scroll_down (ModestWindow *window);
80 static void modest_msg_view_window_update_priority (ModestMsgViewWindow *window);
81
82 static void modest_msg_view_window_show_toolbar   (ModestWindow *window,
83                                                    gboolean show_toolbar);
84
85 static void modest_msg_view_window_clipboard_owner_change (GtkClipboard *clipboard,
86                                                            GdkEvent *event,
87                                                            ModestMsgViewWindow *window);
88
89 static void cancel_progressbar (GtkToolButton *toolbutton,
90                                 ModestMsgViewWindow *self);
91
92 static void         on_queue_changed                     (ModestMailOperationQueue *queue,
93                                                           ModestMailOperation *mail_op,
94                                                           ModestMailOperationQueueNotification type,
95                                                           ModestMsgViewWindow *self);
96
97 static void view_msg_cb (ModestMailOperation *mail_op, TnyHeader *header, TnyMsg *msg, gpointer user_data);
98
99 static void set_toolbar_mode (ModestMsgViewWindow *self, 
100                               ModestToolBarModes mode);
101
102 static gboolean set_toolbar_transfer_mode     (ModestMsgViewWindow *self); 
103
104
105 static void update_window_title (ModestMsgViewWindow *window);
106
107 static gchar * 
108 translate_func (const gchar *msgid,
109                 const gchar *domain_name);
110
111 /* list my signals */
112 enum {
113         /* MY_SIGNAL_1, */
114         /* MY_SIGNAL_2, */
115         LAST_SIGNAL
116 };
117
118 static const GtkToggleActionEntry msg_view_toggle_action_entries [] = {
119         { "FindInMessage",    GTK_STOCK_FIND,    N_("qgn_toolb_gene_find"), NULL, NULL, G_CALLBACK (modest_msg_view_window_toggle_find_toolbar), FALSE },
120         { "ToolsFindInMessage", NULL, N_("mcen_me_viewer_find"), NULL, NULL, G_CALLBACK (modest_msg_view_window_toggle_find_toolbar), FALSE },
121 };
122
123 static const GtkRadioActionEntry msg_view_zoom_action_entries [] = {
124         { "Zoom50", NULL, N_("mcen_me_viewer_50"), NULL, NULL, 50 },
125         { "Zoom80", NULL, N_("mcen_me_viewer_80"), NULL, NULL, 80 },
126         { "Zoom100", NULL, N_("mcen_me_viewer_100"), NULL, NULL, 100 },
127         { "Zoom120", NULL, N_("mcen_me_viewer_120"), NULL, NULL, 120 },
128         { "Zoom150", NULL, N_("mcen_me_viewer_150"), NULL, NULL, 150 },
129         { "Zoom200", NULL, N_("mcen_me_viewer_200"), NULL, NULL, 200 }
130 };
131
132 typedef struct _ModestMsgViewWindowPrivate ModestMsgViewWindowPrivate;
133 struct _ModestMsgViewWindowPrivate {
134
135         GtkWidget   *msg_view;
136         GtkWidget   *main_scroll;
137         GtkWidget   *find_toolbar;
138         gchar       *last_search;
139
140         /* Progress observers */
141         GtkWidget        *progress_bar;
142         GSList           *progress_widgets;
143
144         /* Tollbar items */
145         GtkWidget   *progress_toolitem;
146         GtkWidget   *cancel_toolitem;
147         GtkWidget   *prev_toolitem;
148         GtkWidget   *next_toolitem;
149         ModestToolBarModes current_toolbar_mode;
150
151         /* Optimized view enabled */
152         gboolean optimized_view;
153
154         GtkTreeModel *header_model;
155         GtkTreeRowReference *row_reference;
156
157         guint clipboard_change_handler;
158         guint queue_change_handler;
159
160         guint progress_bar_timeout;
161
162         gchar *msg_uid;
163 };
164
165 #define MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(o)      (G_TYPE_INSTANCE_GET_PRIVATE((o), \
166                                                     MODEST_TYPE_MSG_VIEW_WINDOW, \
167                                                     ModestMsgViewWindowPrivate))
168 /* globals */
169 static GtkWindowClass *parent_class = NULL;
170
171 /* uncomment the following if you have defined any signals */
172 /* static guint signals[LAST_SIGNAL] = {0}; */
173
174 GType
175 modest_msg_view_window_get_type (void)
176 {
177         static GType my_type = 0;
178         if (!my_type) {
179                 static const GTypeInfo my_info = {
180                         sizeof(ModestMsgViewWindowClass),
181                         NULL,           /* base init */
182                         NULL,           /* base finalize */
183                         (GClassInitFunc) modest_msg_view_window_class_init,
184                         NULL,           /* class finalize */
185                         NULL,           /* class data */
186                         sizeof(ModestMsgViewWindow),
187                         1,              /* n_preallocs */
188                         (GInstanceInitFunc) modest_msg_view_window_init,
189                         NULL
190                 };
191                 my_type = g_type_register_static (MODEST_TYPE_WINDOW,
192                                                   "ModestMsgViewWindow",
193                                                   &my_info, 0);
194         }
195         return my_type;
196 }
197
198 static void
199 save_state (ModestWindow *self)
200 {
201         modest_widget_memory_save (modest_runtime_get_conf (),
202                                    G_OBJECT(self), 
203                                    MODEST_CONF_MSG_VIEW_WINDOW_KEY);
204 }
205
206
207 static void
208 restore_settings (ModestMsgViewWindow *self)
209 {
210         modest_widget_memory_restore (modest_runtime_get_conf (),
211                                       G_OBJECT(self), 
212                                       MODEST_CONF_MSG_VIEW_WINDOW_KEY);
213 }
214
215 static void
216 modest_msg_view_window_class_init (ModestMsgViewWindowClass *klass)
217 {
218         GObjectClass *gobject_class;
219         ModestWindowClass *modest_window_class;
220         gobject_class = (GObjectClass*) klass;
221         modest_window_class = (ModestWindowClass *) klass;
222
223         parent_class            = g_type_class_peek_parent (klass);
224         gobject_class->finalize = modest_msg_view_window_finalize;
225
226         modest_window_class->set_zoom_func = modest_msg_view_window_set_zoom;
227         modest_window_class->get_zoom_func = modest_msg_view_window_get_zoom;
228         modest_window_class->zoom_minus_func = modest_msg_view_window_zoom_minus;
229         modest_window_class->zoom_plus_func = modest_msg_view_window_zoom_plus;
230         modest_window_class->show_toolbar_func = modest_msg_view_window_show_toolbar;
231
232         g_type_class_add_private (gobject_class, sizeof(ModestMsgViewWindowPrivate));
233
234         modest_window_class->save_state_func = save_state;
235 }
236
237 static void
238 modest_msg_view_window_init (ModestMsgViewWindow *obj)
239 {
240         ModestMsgViewWindowPrivate *priv;
241         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(obj);
242
243         priv->msg_view      = NULL;
244         priv->header_model  = NULL;
245         priv->clipboard_change_handler = 0;
246         priv->current_toolbar_mode = TOOLBAR_MODE_NORMAL;
247
248         priv->optimized_view  = FALSE;
249         priv->progress_bar_timeout = 0;
250         priv->msg_uid = NULL;
251 }
252
253
254 static gboolean
255 set_toolbar_transfer_mode (ModestMsgViewWindow *self)
256 {
257         ModestMsgViewWindowPrivate *priv = NULL;
258         
259         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (self), FALSE);
260
261         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(self);
262
263         set_toolbar_mode (self, TOOLBAR_MODE_TRANSFER);
264         
265         if (priv->progress_bar_timeout > 0) {
266                 g_source_remove (priv->progress_bar_timeout);
267                 priv->progress_bar_timeout = 0;
268         }
269         
270         return FALSE;
271 }
272
273 static void 
274 set_toolbar_mode (ModestMsgViewWindow *self, 
275                   ModestToolBarModes mode)
276 {
277         ModestWindowPrivate *parent_priv;
278         ModestMsgViewWindowPrivate *priv;
279         GtkAction *widget = NULL;
280
281         g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (self));
282
283         parent_priv = MODEST_WINDOW_GET_PRIVATE(self);
284         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(self);
285                         
286         /* Sets current toolbar mode */
287         priv->current_toolbar_mode = mode;
288
289         switch (mode) {
290         case TOOLBAR_MODE_NORMAL:
291                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarMessageNew");
292                 gtk_action_set_sensitive (widget, TRUE);
293                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarMessageReply");
294                 gtk_action_set_sensitive (widget, TRUE);
295                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarDeleteMessage");
296                 gtk_action_set_sensitive (widget, TRUE);
297                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarMessageMoveTo");
298                 gtk_action_set_sensitive (widget, TRUE);
299                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/FindInMessage");
300                 gtk_action_set_sensitive (widget, TRUE);
301
302                 if (priv->prev_toolitem)
303                         gtk_widget_show (priv->prev_toolitem);
304                 
305                 if (priv->next_toolitem)
306                         gtk_widget_show (priv->next_toolitem);
307                         
308                 if (priv->progress_toolitem)
309                         gtk_tool_item_set_expand (GTK_TOOL_ITEM (priv->progress_toolitem), FALSE);
310                 if (priv->progress_bar)
311                         gtk_widget_hide (priv->progress_bar);
312                         
313                 if (priv->cancel_toolitem)
314                         gtk_widget_hide (priv->cancel_toolitem);
315
316                 /* Hide toolbar if optimized view is enabled */
317                 if (priv->optimized_view) {
318                         gtk_widget_set_no_show_all (parent_priv->toolbar, TRUE);
319                         gtk_widget_hide (GTK_WIDGET(parent_priv->toolbar));
320                 }
321
322                 break;
323         case TOOLBAR_MODE_TRANSFER:
324                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarMessageNew");
325                 gtk_action_set_sensitive (widget, FALSE);
326                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarMessageReply");
327                 gtk_action_set_sensitive (widget, FALSE);
328                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarDeleteMessage");
329                 gtk_action_set_sensitive (widget, FALSE);
330                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/ToolbarMessageMoveTo");
331                 gtk_action_set_sensitive (widget, FALSE);
332                 widget = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/FindInMessage");
333                 gtk_action_set_sensitive (widget, FALSE);
334
335                 if (priv->prev_toolitem)
336                         gtk_widget_hide (priv->prev_toolitem);
337                 
338                 if (priv->next_toolitem)
339                         gtk_widget_hide (priv->next_toolitem);
340                 
341                 if (priv->progress_toolitem)
342                         gtk_tool_item_set_expand (GTK_TOOL_ITEM (priv->progress_toolitem), TRUE);
343                 if (priv->progress_bar)
344                         gtk_widget_show (priv->progress_bar);
345                         
346                 if (priv->cancel_toolitem)
347                         gtk_widget_show (priv->cancel_toolitem);
348
349                 /* Show toolbar if it's hiden (optimized view ) */
350                 if (priv->optimized_view) {
351                         gtk_widget_set_no_show_all (parent_priv->toolbar, FALSE);
352                         gtk_widget_show (GTK_WIDGET (parent_priv->toolbar));
353                 }
354
355                 break;
356         default:
357                 g_return_if_reached ();
358         }
359
360 }
361
362
363 static GtkWidget *
364 menubar_to_menu (GtkUIManager *ui_manager)
365 {
366         GtkWidget *main_menu;
367         GtkWidget *menubar;
368         GList *iter;
369
370         /* Create new main menu */
371         main_menu = gtk_menu_new();
372
373         /* Get the menubar from the UI manager */
374         menubar = gtk_ui_manager_get_widget (ui_manager, "/MenuBar");
375
376         iter = gtk_container_get_children (GTK_CONTAINER (menubar));
377         while (iter) {
378                 GtkWidget *menu;
379
380                 menu = GTK_WIDGET (iter->data);
381                 gtk_widget_reparent(menu, main_menu);
382
383                 iter = g_list_next (iter);
384         }
385         return main_menu;
386 }
387
388 static void
389 init_window (ModestMsgViewWindow *obj, TnyMsg *msg)
390 {
391         GtkWidget *main_vbox;
392         ModestMsgViewWindowPrivate *priv;
393         ModestWindowPrivate *parent_priv;
394         
395         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(obj);
396         parent_priv = MODEST_WINDOW_GET_PRIVATE(obj);
397
398         priv->msg_view = modest_msg_view_new (msg);
399         modest_msg_view_set_shadow_type (MODEST_MSG_VIEW (priv->msg_view), GTK_SHADOW_NONE);
400         main_vbox = gtk_vbox_new  (FALSE, 6);
401
402         /* Menubar */
403         parent_priv->menubar = menubar_to_menu (parent_priv->ui_manager);
404         gtk_widget_show_all (GTK_WIDGET(parent_priv->menubar));
405         hildon_window_set_menu    (HILDON_WINDOW(obj), GTK_MENU(parent_priv->menubar));
406
407         priv->main_scroll = gtk_scrolled_window_new (NULL, NULL);
408         gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (priv->main_scroll), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
409         gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (priv->main_scroll), GTK_SHADOW_NONE);
410
411         gtk_container_add (GTK_CONTAINER (priv->main_scroll), priv->msg_view);
412         gtk_box_pack_start (GTK_BOX(main_vbox), priv->main_scroll, TRUE, TRUE, 0);
413         gtk_container_add   (GTK_CONTAINER(obj), main_vbox);
414
415         priv->find_toolbar = hildon_find_toolbar_new (NULL);
416         gtk_widget_set_no_show_all (priv->find_toolbar, TRUE);
417         g_signal_connect (G_OBJECT (priv->find_toolbar), "close", G_CALLBACK (modest_msg_view_window_find_toolbar_close), obj);
418         g_signal_connect (G_OBJECT (priv->find_toolbar), "search", G_CALLBACK (modest_msg_view_window_find_toolbar_search), obj);
419         
420         priv->clipboard_change_handler = g_signal_connect (G_OBJECT (gtk_clipboard_get (GDK_SELECTION_PRIMARY)), "owner-change", G_CALLBACK (modest_msg_view_window_clipboard_owner_change), obj);
421         gtk_widget_show_all (GTK_WIDGET(main_vbox));
422         gtk_box_pack_end (GTK_BOX (main_vbox), priv->find_toolbar, FALSE, FALSE, 0);
423
424 }       
425
426
427 static void
428 modest_msg_view_window_finalize (GObject *obj)
429 {
430         ModestMsgViewWindowPrivate *priv;
431
432         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (obj);
433         if (priv->clipboard_change_handler > 0) {
434                 g_signal_handler_disconnect (gtk_clipboard_get (GDK_SELECTION_PRIMARY), priv->clipboard_change_handler);
435                 priv->clipboard_change_handler = 0;
436         }
437         if (priv->queue_change_handler > 0) {
438                 g_signal_handler_disconnect (G_OBJECT (modest_runtime_get_mail_operation_queue ()), priv->queue_change_handler);
439                 priv->queue_change_handler = 0;
440         }
441         if (priv->header_model != NULL) {
442                 g_object_unref (priv->header_model);
443                 priv->header_model = NULL;
444         }
445
446         /* disconnet operations queue observer */
447         
448         if (priv->progress_bar_timeout > 0) {
449                 g_source_remove (priv->progress_bar_timeout);
450                 priv->progress_bar_timeout = 0;
451         }
452
453         if (priv->row_reference) {
454                 gtk_tree_row_reference_free (priv->row_reference);
455                 priv->row_reference = NULL;
456         }
457
458         if (priv->msg_uid) {
459                 g_free (priv->msg_uid);
460                 priv->msg_uid = NULL;
461         }
462
463         G_OBJECT_CLASS(parent_class)->finalize (obj);
464 }
465
466
467
468 static gboolean
469 on_delete_event (GtkWidget *widget, GdkEvent *event, ModestMsgViewWindow *self)
470 {
471         modest_window_save_state (MODEST_WINDOW (self));
472         return FALSE;
473 }
474
475 ModestWindow *
476 modest_msg_view_window_new_with_header_model (TnyMsg *msg, 
477                                               const gchar *account_name,
478                                               const gchar *msg_uid,
479                                               GtkTreeModel *model, 
480                                               GtkTreeRowReference *row_reference)
481 {
482         ModestMsgViewWindow *window = NULL;
483         ModestMsgViewWindowPrivate *priv = NULL;
484
485         window = MODEST_MSG_VIEW_WINDOW(modest_msg_view_window_new (msg, account_name, msg_uid));
486         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window), NULL);
487
488         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
489
490         g_object_ref (model);
491         priv->header_model = model;
492         priv->row_reference = gtk_tree_row_reference_copy (row_reference);
493
494         modest_msg_view_window_update_priority (window);
495
496         /* Check toolbar dimming rules */
497         modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (window));
498
499         return MODEST_WINDOW(window);
500 }
501
502
503 ModestWindow *
504 modest_msg_view_window_new (TnyMsg *msg, 
505                             const gchar *account_name,
506                             const gchar *msg_uid)
507 {
508         ModestMsgViewWindow *self = NULL;
509         GObject *obj = NULL;
510         ModestMsgViewWindowPrivate *priv = NULL;
511         ModestWindowPrivate *parent_priv = NULL;
512         ModestDimmingRulesGroup *menu_rules_group = NULL;
513         ModestDimmingRulesGroup *toolbar_rules_group = NULL;
514         GtkActionGroup *action_group = NULL;
515         GError *error = NULL;
516         GdkPixbuf *window_icon = NULL;
517
518         g_return_val_if_fail (msg, NULL);
519         
520         obj = g_object_new(MODEST_TYPE_MSG_VIEW_WINDOW, NULL);
521         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(obj);
522         parent_priv = MODEST_WINDOW_GET_PRIVATE(obj);
523         self = MODEST_MSG_VIEW_WINDOW (obj);
524
525         priv->msg_uid = g_strdup (msg_uid);
526
527         parent_priv->ui_manager = gtk_ui_manager_new();
528         parent_priv->ui_dimming_manager = modest_ui_dimming_manager_new();
529
530         action_group = gtk_action_group_new ("ModestMsgViewWindowActions");
531         gtk_action_group_set_translation_domain (action_group, GETTEXT_PACKAGE);
532         gtk_action_group_set_translate_func (action_group, (GtkTranslateFunc) translate_func, GETTEXT_PACKAGE, g_free);
533
534         menu_rules_group = modest_dimming_rules_group_new ("ModestMenuDimmingRules");
535         toolbar_rules_group = modest_dimming_rules_group_new ("ModestToolbarDimmingRules");
536
537         /* Add common actions */
538         gtk_action_group_add_actions (action_group,
539                                       modest_action_entries,
540                                       G_N_ELEMENTS (modest_action_entries),
541                                       obj);
542         gtk_action_group_add_toggle_actions (action_group,
543                                              modest_toggle_action_entries,
544                                              G_N_ELEMENTS (modest_toggle_action_entries),
545                                              obj);
546         gtk_action_group_add_toggle_actions (action_group,
547                                              msg_view_toggle_action_entries,
548                                              G_N_ELEMENTS (msg_view_toggle_action_entries),
549                                              obj);
550         gtk_action_group_add_radio_actions (action_group,
551                                             msg_view_zoom_action_entries,
552                                             G_N_ELEMENTS (msg_view_zoom_action_entries),
553                                             100,
554                                             G_CALLBACK (modest_ui_actions_on_change_zoom),
555                                             obj);
556
557         gtk_ui_manager_insert_action_group (parent_priv->ui_manager, action_group, 0);
558         g_object_unref (action_group);
559
560         /* Load the UI definition */
561         gtk_ui_manager_add_ui_from_file (parent_priv->ui_manager, MODEST_UIDIR "modest-msg-view-window-ui.xml",
562                                          &error);
563         if (error) {
564                 g_printerr ("modest: could not merge modest-msg-view-window-ui.xml: %s\n", error->message);
565                 g_error_free (error);
566                 error = NULL;
567         }
568         /* ****** */
569
570         /* Add common dimming rules */
571         modest_dimming_rules_group_add_rules (menu_rules_group, 
572                                               modest_msg_view_menu_dimming_entries,
573                                               G_N_ELEMENTS (modest_msg_view_menu_dimming_entries),
574                                               self);
575         modest_dimming_rules_group_add_rules (toolbar_rules_group, 
576                                               modest_msg_view_toolbar_dimming_entries,
577                                               G_N_ELEMENTS (modest_msg_view_toolbar_dimming_entries),
578                                               self);
579
580         /* Insert dimming rules group for this window */
581         modest_ui_dimming_manager_insert_rules_group (parent_priv->ui_dimming_manager, menu_rules_group);
582         modest_ui_dimming_manager_insert_rules_group (parent_priv->ui_dimming_manager, toolbar_rules_group);
583         g_object_unref (menu_rules_group);
584         g_object_unref (toolbar_rules_group);
585
586         /* Add accelerators */
587         gtk_window_add_accel_group (GTK_WINDOW (obj), 
588                                     gtk_ui_manager_get_accel_group (parent_priv->ui_manager));
589         
590         /* Init window */
591         init_window (MODEST_MSG_VIEW_WINDOW(obj), msg);
592         restore_settings (MODEST_MSG_VIEW_WINDOW(obj));
593         
594         g_signal_connect (G_OBJECT(obj), "delete-event", G_CALLBACK(on_delete_event), obj);
595
596         g_signal_connect (G_OBJECT(priv->msg_view), "link_clicked",
597                           G_CALLBACK (modest_ui_actions_on_msg_link_clicked), obj);
598         g_signal_connect (G_OBJECT(priv->msg_view), "link_hover",
599                           G_CALLBACK (modest_ui_actions_on_msg_link_hover), obj);
600         g_signal_connect (G_OBJECT(priv->msg_view), "attachment_clicked",
601                           G_CALLBACK (modest_ui_actions_on_msg_attachment_clicked), obj);
602         g_signal_connect (G_OBJECT(priv->msg_view), "recpt_activated",
603                           G_CALLBACK (modest_ui_actions_on_msg_recpt_activated), obj);
604         g_signal_connect (G_OBJECT(priv->msg_view), "link_contextual",
605                           G_CALLBACK (modest_ui_actions_on_msg_link_contextual), obj);
606
607         g_signal_connect (G_OBJECT (obj), "key-release-event",
608                           G_CALLBACK (modest_msg_view_window_key_release_event),
609                           NULL);
610
611         g_signal_connect (G_OBJECT (obj), "window-state-event",
612                           G_CALLBACK (modest_msg_view_window_window_state_event),
613                           NULL);
614
615         /* Mail Operation Queue */
616         priv->queue_change_handler = g_signal_connect (G_OBJECT (modest_runtime_get_mail_operation_queue ()),
617                                                        "queue-changed",
618                                                        G_CALLBACK (on_queue_changed),
619                                                        obj);
620
621         modest_window_set_active_account (MODEST_WINDOW(obj), account_name);
622
623         priv->last_search = NULL;
624
625
626         /* Set window icon */
627         window_icon = modest_platform_get_icon (MODEST_APP_MSG_VIEW_ICON);
628         gtk_window_set_icon (GTK_WINDOW (obj), window_icon);
629
630         /* Init the clipboard actions dim status */
631         modest_msg_view_grab_focus(MODEST_MSG_VIEW (priv->msg_view));
632
633         update_window_title (MODEST_MSG_VIEW_WINDOW (obj));
634
635         return MODEST_WINDOW(obj);
636 }
637
638
639 gboolean 
640 modest_msg_view_window_toolbar_on_transfer_mode     (ModestMsgViewWindow *self)
641 {
642         ModestMsgViewWindowPrivate *priv= NULL; 
643
644         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (self), FALSE);
645         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (self);
646
647         return priv->current_toolbar_mode == TOOLBAR_MODE_TRANSFER;
648 }
649
650 TnyHeader*
651 modest_msg_view_window_get_header (ModestMsgViewWindow *self)
652 {
653         ModestMsgViewWindowPrivate *priv= NULL; 
654         TnyHeader *header = NULL;
655         GtkTreeIter iter;
656
657         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (self), NULL);
658         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (self);
659
660         /* Get current message iter */
661         gtk_tree_model_get_iter (priv->header_model, 
662                                  &iter, 
663                                  gtk_tree_row_reference_get_path (priv->row_reference));
664
665         /* Get current message header */
666         gtk_tree_model_get (priv->header_model, &iter, 
667                             TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN, 
668                             &header, -1);
669
670         return header;
671 }
672
673 TnyMsg*
674 modest_msg_view_window_get_message (ModestMsgViewWindow *self)
675 {
676         ModestMsgView *msg_view;
677         ModestMsgViewWindowPrivate *priv;
678
679         g_return_val_if_fail (self, NULL);
680
681         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(self);
682
683         msg_view = MODEST_MSG_VIEW (priv->msg_view);
684
685         return modest_msg_view_get_message (msg_view);
686 }
687
688 const gchar*
689 modest_msg_view_window_get_message_uid (ModestMsgViewWindow *self)
690 {
691         ModestMsgViewWindowPrivate *priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (self);
692         return (const gchar*) priv->msg_uid;
693 }
694
695 static void
696 toggle_action_set_active_block_notify (GtkToggleAction *action,
697                                        gboolean value)
698 {
699         GSList *proxies = NULL;
700
701         for (proxies = gtk_action_get_proxies (GTK_ACTION (action));
702              proxies != NULL; proxies = g_slist_next (proxies)) {
703                 GtkWidget *widget = (GtkWidget *) proxies->data;
704                 gtk_action_block_activate_from (GTK_ACTION (action), widget);
705         }
706
707         gtk_toggle_action_set_active (action, value);
708
709         for (proxies = gtk_action_get_proxies (GTK_ACTION (action));
710              proxies != NULL; proxies = g_slist_next (proxies)) {
711                 GtkWidget *widget = (GtkWidget *) proxies->data;
712                 gtk_action_unblock_activate_from (GTK_ACTION (action), widget);
713         }
714 }
715
716
717 static void 
718 modest_msg_view_window_toggle_find_toolbar (GtkToggleAction *toggle,
719                                             gpointer data)
720 {
721         ModestMsgViewWindow *window = MODEST_MSG_VIEW_WINDOW (data);
722         ModestMsgViewWindowPrivate *priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
723         ModestWindowPrivate *parent_priv = MODEST_WINDOW_GET_PRIVATE (window);
724         gboolean is_active;
725         GtkAction *action;
726
727         is_active = gtk_toggle_action_get_active (toggle);
728
729         if (is_active) {
730                 gtk_widget_show (priv->find_toolbar);
731                 hildon_find_toolbar_highlight_entry (HILDON_FIND_TOOLBAR (priv->find_toolbar), TRUE);
732         } else {
733                 gtk_widget_hide (priv->find_toolbar);
734         }
735
736         /* update the toggle buttons status */
737         action = gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/FindInMessage");
738         toggle_action_set_active_block_notify (GTK_TOGGLE_ACTION (action), is_active);
739         action = gtk_ui_manager_get_action (parent_priv->ui_manager, "/MenuBar/ToolsMenu/ToolsFindInMessageMenu");
740         toggle_action_set_active_block_notify (GTK_TOGGLE_ACTION (action), is_active);
741         
742 }
743
744 static void
745 modest_msg_view_window_find_toolbar_close (GtkWidget *widget,
746                                            ModestMsgViewWindow *obj)
747 {
748         GtkToggleAction *toggle;
749         ModestWindowPrivate *parent_priv;
750         parent_priv = MODEST_WINDOW_GET_PRIVATE (obj);
751         
752         toggle = GTK_TOGGLE_ACTION (gtk_ui_manager_get_action (parent_priv->ui_manager, "/ToolBar/FindInMessage"));
753         gtk_toggle_action_set_active (toggle, FALSE);
754 }
755
756 static void
757 modest_msg_view_window_find_toolbar_search (GtkWidget *widget,
758                                            ModestMsgViewWindow *obj)
759 {
760         gchar *current_search;
761         ModestMsgViewWindowPrivate *priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (obj);
762
763         g_object_get (G_OBJECT (widget), "prefix", &current_search, NULL);
764
765         if ((current_search == NULL) && (strcmp (current_search, "") == 0)) {
766                 g_free (current_search);
767                 return;
768         }
769
770         if ((priv->last_search == NULL) || (strcmp (priv->last_search, current_search) != 0)) {
771                 gboolean result;
772                 g_free (priv->last_search);
773                 priv->last_search = g_strdup (current_search);
774                 result = modest_msg_view_search (MODEST_MSG_VIEW (priv->msg_view),
775                                                  priv->last_search);
776         } else {
777                 modest_msg_view_search_next (MODEST_MSG_VIEW (priv->msg_view));
778         }
779         
780         g_free (current_search);
781                 
782 }
783
784 static void
785 modest_msg_view_window_set_zoom (ModestWindow *window,
786                                  gdouble zoom)
787 {
788         ModestMsgViewWindowPrivate *priv;
789      
790         g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window));
791
792         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
793         modest_msg_view_set_zoom (MODEST_MSG_VIEW (priv->msg_view), zoom);
794 }
795
796 static gdouble
797 modest_msg_view_window_get_zoom (ModestWindow *window)
798 {
799         ModestMsgViewWindowPrivate *priv;
800      
801         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window), 1.0);
802
803         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
804         return modest_msg_view_get_zoom (MODEST_MSG_VIEW (priv->msg_view));
805 }
806
807 static gboolean
808 modest_msg_view_window_zoom_plus (ModestWindow *window)
809 {
810         ModestWindowPrivate *parent_priv;
811         GtkRadioAction *zoom_radio_action;
812         GSList *group, *node;
813
814         parent_priv = MODEST_WINDOW_GET_PRIVATE (window);
815         zoom_radio_action = GTK_RADIO_ACTION (gtk_ui_manager_get_action (parent_priv->ui_manager, 
816                                                                          "/MenuBar/ViewMenu/ZoomMenu/Zoom50Menu"));
817
818         group = gtk_radio_action_get_group (zoom_radio_action);
819
820         if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (group->data))) {
821                 hildon_banner_show_information (NULL, NULL, _("mcen_ib_max_zoom_level"));
822                 return FALSE;
823         }
824
825         for (node = group; node != NULL; node = g_slist_next (node)) {
826                 if ((node->next != NULL) && gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (node->next->data))) {
827                         gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (node->data), TRUE);
828                         return TRUE;
829                 }
830         }
831         return FALSE;
832 }
833
834 static gboolean
835 modest_msg_view_window_zoom_minus (ModestWindow *window)
836 {
837         ModestWindowPrivate *parent_priv;
838         GtkRadioAction *zoom_radio_action;
839         GSList *group, *node;
840
841         parent_priv = MODEST_WINDOW_GET_PRIVATE (window);
842         zoom_radio_action = GTK_RADIO_ACTION (gtk_ui_manager_get_action (parent_priv->ui_manager, 
843                                                                          "/MenuBar/ViewMenu/ZoomMenu/Zoom50Menu"));
844
845         group = gtk_radio_action_get_group (zoom_radio_action);
846
847         for (node = group; node != NULL; node = g_slist_next (node)) {
848                 if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (node->data))) {
849                         if (node->next != NULL) {
850                                 gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (node->next->data), TRUE);
851                                 return TRUE;
852                         } else {
853                                 hildon_banner_show_information (NULL, NULL, _("mcen_ib_min_zoom_level"));
854                                 return FALSE;
855                         }
856                         break;
857                 }
858         }
859         return FALSE;
860 }
861
862 static gboolean
863 modest_msg_view_window_key_release_event (GtkWidget *window,
864                                           GdkEventKey *event,
865                                           gpointer userdata)
866 {
867         if (event->type == GDK_KEY_RELEASE) {
868                 switch (event->keyval) {
869                 case GDK_Up:
870                         modest_msg_view_window_scroll_up (MODEST_WINDOW (window));
871                         return TRUE;
872                         break;
873                 case GDK_Down:
874                         modest_msg_view_window_scroll_down (MODEST_WINDOW (window));
875                         return TRUE;
876                         break;
877                 default:
878                         return FALSE;
879                         break;
880                 };
881         } else {
882                 return FALSE;
883         }
884 }
885
886 static void
887 modest_msg_view_window_scroll_up (ModestWindow *window)
888 {
889         ModestMsgViewWindowPrivate *priv;
890
891         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
892         g_signal_emit_by_name (G_OBJECT (priv->main_scroll), "scroll-child", GTK_SCROLL_STEP_UP, FALSE);
893 }
894
895 static void
896 modest_msg_view_window_scroll_down (ModestWindow *window)
897 {
898         ModestMsgViewWindowPrivate *priv;
899
900         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
901         g_signal_emit_by_name (G_OBJECT (priv->main_scroll), "scroll-child", GTK_SCROLL_STEP_DOWN, FALSE);
902 }
903
904 gboolean
905 modest_msg_view_window_last_message_selected (ModestMsgViewWindow *window)
906 {
907         GtkTreePath *path;
908         ModestMsgViewWindowPrivate *priv;
909         GtkTreeIter tmp_iter;
910         gboolean has_next = FALSE;
911
912         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window), TRUE);
913         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
914
915         if (priv->header_model) {
916                 path = gtk_tree_row_reference_get_path (priv->row_reference);
917                 if (!path)
918                         return TRUE;
919                 while (!has_next) {
920                         TnyHeader *header;
921                         gtk_tree_path_next (path);
922                         if (!gtk_tree_model_get_iter (priv->header_model, &tmp_iter, path))
923                                 break;
924                         gtk_tree_model_get (priv->header_model, &tmp_iter, TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
925                                             &header, -1);
926                         if (!(tny_header_get_flags(header)&TNY_HEADER_FLAG_DELETED)) {
927                                 has_next = TRUE;
928                                 break;
929                         }
930                         
931                 }
932                 return !has_next;
933         } else {
934                 return TRUE;
935         }
936         
937 }
938
939 gboolean
940 modest_msg_view_window_has_headers_model (ModestMsgViewWindow *window)
941 {
942         ModestMsgViewWindowPrivate *priv;
943
944         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window), TRUE);
945         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
946
947         return priv->header_model != NULL;
948 }
949
950 gboolean
951 modest_msg_view_window_first_message_selected (ModestMsgViewWindow *window)
952 {
953         GtkTreePath *path;
954         ModestMsgViewWindowPrivate *priv;
955         gboolean result;
956         GtkTreeIter tmp_iter;
957
958         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window), TRUE);
959         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
960
961         if (priv->header_model) {
962                 gchar * path_string;
963                 path = gtk_tree_row_reference_get_path (priv->row_reference);
964                 if (!path)
965                         return TRUE;
966
967                 path_string = gtk_tree_path_to_string (path);
968                 result = (strcmp (path_string, "0")==0);
969                 if (result) {
970                         g_free (path_string);
971                         gtk_tree_path_free (path);
972                         return result;
973                 }
974
975                 while (result) {
976                         TnyHeader *header;
977
978                         gtk_tree_path_prev (path);
979                         
980                         if (!gtk_tree_model_get_iter (priv->header_model, &tmp_iter, path))
981                                 break;
982                         gtk_tree_model_get (priv->header_model, &tmp_iter, TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
983                                             &header, -1);
984                         if (!(tny_header_get_flags(header)&TNY_HEADER_FLAG_DELETED)) {
985                                 result = FALSE;
986                                 break;
987                         }
988
989                         path_string = gtk_tree_path_to_string (path);
990                         if (strcmp(path_string, "0")==0) {
991                                 g_free (path_string);
992                                 break;
993                         }
994                         g_free (path_string);
995                 }
996                 return result;
997         } else {
998                 return TRUE;
999         }
1000         
1001 }
1002
1003 gboolean        
1004 modest_msg_view_window_select_next_message (ModestMsgViewWindow *window)
1005 {
1006         TnyHeaderFlags flags;
1007         ModestMailOperation *mail_op = NULL;
1008         ModestMsgViewWindowPrivate *priv;
1009         GtkTreeIter tmp_iter;
1010
1011         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window), FALSE);
1012         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
1013
1014         if (priv->header_model) {
1015                 gtk_tree_model_get_iter (priv->header_model,
1016                                          &tmp_iter,
1017                                          gtk_tree_row_reference_get_path (priv->row_reference));
1018                 while (gtk_tree_model_iter_next (priv->header_model, &tmp_iter)) {
1019                         TnyHeader *header;
1020                         GtkTreePath *path;
1021
1022                         gtk_tree_model_get (priv->header_model, &tmp_iter, TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
1023                                             &header, -1);
1024                         if (!header)
1025                                 break;
1026                         if (tny_header_get_flags (header) & TNY_HEADER_FLAG_DELETED)
1027                                 continue;
1028
1029                         /* Update the row reference */
1030                         gtk_tree_row_reference_free (priv->row_reference);
1031                         path = gtk_tree_model_get_path (priv->header_model, &tmp_iter);
1032                         priv->row_reference = gtk_tree_row_reference_new (priv->header_model, path);
1033                         gtk_tree_path_free (path);
1034
1035                         /* Mark as read */
1036                         flags = tny_header_get_flags (header);
1037                         if (!(flags & TNY_HEADER_FLAG_SEEN))
1038                                 tny_header_set_flags (header, flags | TNY_HEADER_FLAG_SEEN);
1039
1040                         /* Msg download initied */
1041
1042                         /* New mail operation */
1043                         mail_op = modest_mail_operation_new (MODEST_MAIL_OPERATION_TYPE_RECEIVE, G_OBJECT(window));
1044                         modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
1045                         modest_mail_operation_get_msg (mail_op, header, view_msg_cb, NULL);
1046                         g_object_unref (mail_op);
1047
1048                         /* Update toolbar dimming rules */
1049                         modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (window));
1050
1051                         return TRUE;
1052                 }
1053         }
1054         return FALSE;           
1055 }
1056
1057 gboolean 
1058 modest_msg_view_window_select_first_message (ModestMsgViewWindow *self)
1059 {
1060         ModestMsgViewWindowPrivate *priv = NULL;
1061         ModestMailOperation *mail_op = NULL;
1062         TnyHeader *header = NULL;
1063         TnyHeaderFlags flags;
1064         GtkTreePath *path = NULL;
1065         GtkTreeIter iter;
1066
1067         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (self), FALSE);
1068         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (self);
1069
1070         /* Check that the model is not empty */
1071         if (!gtk_tree_model_get_iter_first (priv->header_model, &iter))
1072                 return FALSE;
1073
1074         /* Get the header */
1075         gtk_tree_model_get (priv->header_model, 
1076                             &iter, 
1077                             TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
1078                             &header, -1);
1079         
1080         g_return_val_if_fail (TNY_IS_HEADER (header), FALSE);
1081         if (tny_header_get_flags (header) & TNY_HEADER_FLAG_DELETED) {
1082                 g_object_unref (header);
1083                 return modest_msg_view_window_select_next_message (self);
1084         }
1085         
1086         /* Update the row reference */
1087         gtk_tree_row_reference_free (priv->row_reference);
1088         path = gtk_tree_path_new_first ();
1089         priv->row_reference = gtk_tree_row_reference_new (priv->header_model, path);
1090         gtk_tree_path_free (path);
1091
1092         /* Mark as read */
1093         flags = tny_header_get_flags (header);
1094         if (!(flags & TNY_HEADER_FLAG_SEEN))
1095                 tny_header_set_flags (header, flags | TNY_HEADER_FLAG_SEEN);
1096         
1097         /* New mail operation */
1098         mail_op = modest_mail_operation_new (MODEST_MAIL_OPERATION_TYPE_RECEIVE, G_OBJECT(self));
1099         modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
1100         modest_mail_operation_get_msg (mail_op, header, view_msg_cb, NULL);
1101         g_object_unref (mail_op);
1102
1103         /* Update toolbar dimming rules */
1104         modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (self));
1105         
1106         /* Free */
1107         g_object_unref (header);
1108
1109         return TRUE;
1110 }
1111  
1112 gboolean        
1113 modest_msg_view_window_select_previous_message (ModestMsgViewWindow *window)
1114 {
1115         TnyHeaderFlags flags;
1116         ModestMsgViewWindowPrivate *priv = NULL;
1117         GtkTreePath *path;
1118         ModestMailOperation *mail_op = NULL;
1119
1120         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window), FALSE);
1121         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
1122
1123         /* Return inmediatly if there is no header model */
1124         if (!priv->header_model)
1125                 return FALSE;
1126
1127         path = gtk_tree_row_reference_get_path (priv->row_reference);
1128         while (gtk_tree_path_prev (path)) {
1129                 TnyHeader *header;
1130                 GtkTreeIter iter;
1131
1132                 gtk_tree_model_get_iter (priv->header_model, &iter, path);
1133                 gtk_tree_model_get (priv->header_model, &iter, 
1134                                     TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
1135                                     &header, -1);
1136                 if (!header)
1137                         break;
1138                 if (tny_header_get_flags (header) & TNY_HEADER_FLAG_DELETED) {
1139                         g_object_unref (header);
1140                         continue;
1141                 }
1142
1143                 /* Update the row reference */
1144                 gtk_tree_row_reference_free (priv->row_reference);
1145                 priv->row_reference = gtk_tree_row_reference_new (priv->header_model, path);
1146                         
1147                 /* Mark as read */
1148                 flags = tny_header_get_flags (header);
1149                 if (!(flags & TNY_HEADER_FLAG_SEEN))
1150                         tny_header_set_flags (header, flags | TNY_HEADER_FLAG_SEEN);
1151
1152                 /* New mail operation */
1153                 mail_op = modest_mail_operation_new (MODEST_MAIL_OPERATION_TYPE_RECEIVE, G_OBJECT(window));
1154                 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (), mail_op);
1155                 modest_mail_operation_get_msg (mail_op, header, view_msg_cb, NULL);             
1156
1157                 /* Update toolbar dimming rules */
1158                 modest_ui_actions_check_toolbar_dimming_rules (MODEST_WINDOW (window));
1159
1160                 g_object_unref (header);
1161
1162                 return TRUE;
1163         }
1164
1165         return FALSE;
1166 }
1167
1168 static void
1169 view_msg_cb (ModestMailOperation *mail_op, 
1170              TnyHeader *header, 
1171              TnyMsg *msg, 
1172              gpointer user_data)
1173 {
1174         ModestMsgViewWindow *self = NULL;
1175         ModestMsgViewWindowPrivate *priv = NULL;
1176
1177         g_return_if_fail (TNY_IS_MSG (msg));
1178
1179         /* Get the window */ 
1180         self = (ModestMsgViewWindow *) modest_mail_operation_get_source (mail_op);
1181         g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (self));
1182
1183         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (self);
1184
1185         /* Set new message */
1186         modest_msg_view_set_message (MODEST_MSG_VIEW (priv->msg_view), msg);
1187         modest_msg_view_window_update_priority (self);
1188         update_window_title (MODEST_MSG_VIEW_WINDOW (self));
1189         modest_msg_view_grab_focus (MODEST_MSG_VIEW (priv->msg_view));
1190 }
1191
1192 TnyFolderType
1193 modest_msg_view_window_get_folder_type (ModestMsgViewWindow *window)
1194 {
1195         ModestMsgViewWindowPrivate *priv;
1196         TnyMsg *msg;
1197         TnyFolderType folder_type;
1198
1199         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
1200
1201         folder_type = TNY_FOLDER_TYPE_UNKNOWN;
1202
1203         msg = modest_msg_view_get_message (MODEST_MSG_VIEW (priv->msg_view));
1204         if (msg) {
1205                 TnyFolder *folder;
1206
1207                 folder = tny_msg_get_folder (msg);
1208                 
1209                 if (folder) {
1210                         folder_type = tny_folder_get_folder_type (folder);
1211                         
1212                         if (folder_type == TNY_FOLDER_TYPE_NORMAL || folder_type == TNY_FOLDER_TYPE_UNKNOWN) {
1213                                 const gchar *fname = tny_folder_get_name (folder);
1214                                 folder_type = modest_tny_folder_guess_folder_type_from_name (fname);
1215                         }
1216
1217                         g_object_unref (folder);
1218                 }
1219                 g_object_unref (msg);
1220         }
1221
1222         return folder_type;
1223 }
1224
1225
1226 static void
1227 modest_msg_view_window_update_priority (ModestMsgViewWindow *window)
1228 {
1229         ModestMsgViewWindowPrivate *priv;
1230         TnyHeaderFlags flags = 0;
1231
1232         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
1233
1234         if (priv->header_model) {
1235                 TnyHeader *header;
1236                 GtkTreeIter iter;
1237
1238                 gtk_tree_model_get_iter (priv->header_model, 
1239                                          &iter, 
1240                                          gtk_tree_row_reference_get_path (priv->row_reference));
1241
1242                 gtk_tree_model_get (priv->header_model, &iter, TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
1243                                     &header, -1);
1244                 flags = tny_header_get_flags (header);
1245         }
1246
1247         modest_msg_view_set_priority (MODEST_MSG_VIEW(priv->msg_view), flags);
1248
1249 }
1250
1251 static gboolean
1252 modest_msg_view_window_window_state_event (GtkWidget *widget, GdkEventWindowState *event, gpointer userdata)
1253 {
1254         if (event->changed_mask & GDK_WINDOW_STATE_FULLSCREEN) {
1255                 ModestWindowPrivate *parent_priv;
1256                 ModestWindowMgr *mgr;
1257                 gboolean is_fullscreen;
1258                 GtkAction *fs_toggle_action;
1259                 gboolean active;
1260
1261                 mgr = modest_runtime_get_window_mgr ();
1262                 is_fullscreen = (modest_window_mgr_get_fullscreen_mode (mgr))?1:0;
1263
1264                 parent_priv = MODEST_WINDOW_GET_PRIVATE (widget);
1265                 
1266                 fs_toggle_action = gtk_ui_manager_get_action (parent_priv->ui_manager, "/MenuBar/ViewMenu/ViewToggleFullscreenMenu");
1267                 active = (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (fs_toggle_action)))?1:0;
1268                 if (is_fullscreen != active) {
1269                         gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (fs_toggle_action), is_fullscreen);
1270                 }
1271         }
1272
1273         return FALSE;
1274
1275 }
1276
1277 void
1278 modest_msg_view_window_toggle_fullscreen (ModestMsgViewWindow *window)
1279 {
1280                 ModestWindowPrivate *parent_priv;
1281                 GtkAction *fs_toggle_action;
1282                 parent_priv = MODEST_WINDOW_GET_PRIVATE (window);
1283                 
1284                 fs_toggle_action = gtk_ui_manager_get_action (parent_priv->ui_manager, "/MenuBar/ViewMenu/ViewToggleFullscreenMenu");
1285                 gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (fs_toggle_action),
1286                                               !gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (fs_toggle_action)));
1287 }
1288
1289 static void
1290 set_homogeneous (GtkWidget *widget,
1291                  gpointer data)
1292 {
1293         if (GTK_IS_TOOL_ITEM (widget)) {
1294                 gtk_tool_item_set_expand (GTK_TOOL_ITEM (widget), TRUE);
1295                 gtk_tool_item_set_homogeneous (GTK_TOOL_ITEM (widget), TRUE);
1296         }
1297 }
1298
1299 static void
1300 modest_msg_view_window_show_toolbar (ModestWindow *self,
1301                                      gboolean show_toolbar)
1302 {
1303         ModestMsgViewWindowPrivate *priv = NULL;
1304         ModestWindowPrivate *parent_priv;
1305         GtkWidget *reply_button = NULL, *menu = NULL;
1306         GtkWidget *placeholder = NULL;
1307         gint insert_index;
1308         
1309         parent_priv = MODEST_WINDOW_GET_PRIVATE(self);
1310         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(self);
1311
1312         /* Set optimized view status */
1313         priv->optimized_view = !show_toolbar;
1314
1315         if (!parent_priv->toolbar) {
1316                 parent_priv->toolbar = gtk_ui_manager_get_widget (parent_priv->ui_manager, 
1317                                                                   "/ToolBar");
1318
1319                 /* Set homogeneous toolbar */
1320                 gtk_container_foreach (GTK_CONTAINER (parent_priv->toolbar), 
1321                                        set_homogeneous, NULL);
1322
1323                 priv->progress_toolitem = GTK_WIDGET (gtk_tool_item_new ());
1324                 priv->cancel_toolitem = gtk_ui_manager_get_widget (parent_priv->ui_manager, "/ToolBar/ToolbarCancel");
1325                 priv->next_toolitem = gtk_ui_manager_get_widget (parent_priv->ui_manager, "/ToolBar/ToolbarMessageNext");
1326                 priv->prev_toolitem = gtk_ui_manager_get_widget (parent_priv->ui_manager, "/ToolBar/ToolbarMessageBack");
1327                 gtk_tool_item_set_expand (GTK_TOOL_ITEM (priv->progress_toolitem), TRUE);
1328                 gtk_tool_item_set_homogeneous (GTK_TOOL_ITEM (priv->progress_toolitem), TRUE);
1329                 gtk_tool_item_set_homogeneous (GTK_TOOL_ITEM (priv->cancel_toolitem), FALSE);
1330                 gtk_tool_item_set_expand (GTK_TOOL_ITEM (priv->cancel_toolitem), FALSE);
1331
1332                 /* Add ProgressBar (Transfer toolbar) */ 
1333                 priv->progress_bar = modest_progress_bar_widget_new ();
1334                 gtk_widget_set_no_show_all (priv->progress_bar, TRUE);
1335                 placeholder = gtk_ui_manager_get_widget (parent_priv->ui_manager, "/ToolBar/ProgressbarView");
1336                 insert_index = gtk_toolbar_get_item_index(GTK_TOOLBAR (parent_priv->toolbar), GTK_TOOL_ITEM(placeholder));
1337                 gtk_container_add (GTK_CONTAINER (priv->progress_toolitem), priv->progress_bar);
1338                 gtk_toolbar_insert(GTK_TOOLBAR(parent_priv->toolbar), GTK_TOOL_ITEM (priv->progress_toolitem), insert_index);
1339                 
1340                 /* Connect cancel 'clicked' signal to abort progress mode */
1341                 g_signal_connect(priv->cancel_toolitem, "clicked",
1342                                  G_CALLBACK(cancel_progressbar),
1343                                  self);
1344                 
1345                 /* Add it to the observers list */
1346                 priv->progress_widgets = g_slist_prepend(priv->progress_widgets, priv->progress_bar);
1347
1348                 /* Add to window */
1349                 hildon_window_add_toolbar (HILDON_WINDOW (self), 
1350                                            GTK_TOOLBAR (parent_priv->toolbar));
1351
1352
1353                 /* Set reply button tap and hold menu */        
1354                 reply_button = gtk_ui_manager_get_widget (parent_priv->ui_manager, 
1355                                                           "/ToolBar/ToolbarMessageReply");
1356                 menu = gtk_ui_manager_get_widget (parent_priv->ui_manager, 
1357                                                   "/ToolbarReplyCSM");
1358                 gtk_widget_tap_and_hold_setup (GTK_WIDGET (reply_button), menu, NULL, 0);
1359         }
1360
1361         if (show_toolbar) {
1362                 /* Quick hack: this prevents toolbar icons "dance" when progress bar show status is changed */ 
1363                 /* TODO: resize mode migth be GTK_RESIZE_QUEUE, in order to avoid unneccesary shows */
1364                 gtk_container_set_resize_mode (GTK_CONTAINER(parent_priv->toolbar), GTK_RESIZE_IMMEDIATE);
1365                 
1366                 gtk_widget_show (GTK_WIDGET (parent_priv->toolbar));
1367                 set_toolbar_mode (MODEST_MSG_VIEW_WINDOW(self), TOOLBAR_MODE_NORMAL);                   
1368         } else {
1369                 gtk_widget_set_no_show_all (parent_priv->toolbar, TRUE);
1370                 gtk_widget_hide (GTK_WIDGET (parent_priv->toolbar));
1371         }
1372 }
1373
1374 static void 
1375 modest_msg_view_window_clipboard_owner_change (GtkClipboard *clipboard,
1376                                                GdkEvent *event,
1377                                                ModestMsgViewWindow *window)
1378 {
1379         ModestWindowPrivate *parent_priv;
1380 /*      GtkAction *action; */
1381         gboolean is_address;
1382         gchar *selection;
1383         GtkWidget *focused;
1384
1385         parent_priv = MODEST_WINDOW_GET_PRIVATE (window);
1386         selection = gtk_clipboard_wait_for_text (clipboard);
1387
1388         is_address = ((selection != NULL) && (modest_text_utils_validate_recipient (selection)));
1389         
1390 /*      action = gtk_ui_manager_get_action (parent_priv->ui_manager, "/MenuBar/ToolsMenu/ToolsAddToContactsMenu"); */
1391 /*      gtk_action_set_sensitive (action, is_address); */
1392
1393         focused = gtk_window_get_focus (GTK_WINDOW (window));
1394
1395 /*      action = gtk_ui_manager_get_action (parent_priv->ui_manager, "/MenuBar/EditMenu/EditCopyMenu"); */
1396 /*      gtk_action_set_sensitive (action, (selection != NULL) && (!MODEST_IS_ATTACHMENTS_VIEW (focused))); */
1397
1398 /*      action = gtk_ui_manager_get_action (parent_priv->ui_manager, "/MenuBar/EditMenu/EditCutMenu"); */
1399 /*      gtk_action_set_sensitive (action, (selection != NULL) && (!MODEST_IS_ATTACHMENTS_VIEW (focused))); */
1400
1401         g_free (selection);
1402 /*      modest_msg_view_window_update_dimmed (window); */
1403         
1404 }
1405
1406 static void
1407 cancel_progressbar (GtkToolButton *toolbutton,
1408                     ModestMsgViewWindow *self)
1409 {
1410         GSList *tmp;
1411         ModestMsgViewWindowPrivate *priv;
1412         
1413         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(self);
1414
1415         /* Get operation observers and cancel its current operation */
1416         tmp = priv->progress_widgets;
1417         while (tmp) {
1418                 modest_progress_object_cancel_current_operation (MODEST_PROGRESS_OBJECT(tmp->data));
1419                 tmp=g_slist_next(tmp);
1420         }
1421 }
1422 static gboolean
1423 observers_empty (ModestMsgViewWindow *self)
1424 {
1425         GSList *tmp = NULL;
1426         ModestMsgViewWindowPrivate *priv;
1427         gboolean is_empty = TRUE;
1428         guint pending_ops = 0;
1429  
1430         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(self);
1431         tmp = priv->progress_widgets;
1432
1433         /* Check all observers */
1434         while (tmp && is_empty)  {
1435                 pending_ops = modest_progress_object_num_pending_operations (MODEST_PROGRESS_OBJECT(tmp->data));
1436                 is_empty = pending_ops == 0;
1437                 
1438                 tmp = g_slist_next(tmp);
1439         }
1440         
1441         return is_empty;
1442 }
1443
1444
1445 static void
1446 on_queue_changed (ModestMailOperationQueue *queue,
1447                   ModestMailOperation *mail_op,
1448                   ModestMailOperationQueueNotification type,
1449                   ModestMsgViewWindow *self)
1450 {
1451         GSList *tmp;
1452         ModestMsgViewWindowPrivate *priv;
1453         ModestMailOperationTypeOperation op_type;
1454         ModestToolBarModes mode;
1455         
1456         g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (self));
1457         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE(self);
1458
1459         /* If this operations was created by another window, do nothing */
1460         if (!modest_mail_operation_is_mine (mail_op, G_OBJECT(self))) 
1461             return;
1462
1463         /* Get toolbar mode from operation id*/
1464         op_type = modest_mail_operation_get_type_operation (mail_op);
1465         switch (op_type) {
1466         case MODEST_MAIL_OPERATION_TYPE_SEND:
1467         case MODEST_MAIL_OPERATION_TYPE_RECEIVE:
1468                 mode = TOOLBAR_MODE_TRANSFER;
1469                 break;
1470         default:
1471                 mode = TOOLBAR_MODE_NORMAL;
1472                 
1473         }
1474                 
1475         /* Add operation observers and change toolbar if neccessary*/
1476         tmp = priv->progress_widgets;
1477         switch (type) {
1478         case MODEST_MAIL_OPERATION_QUEUE_OPERATION_ADDED:
1479                 if (mode == TOOLBAR_MODE_TRANSFER) {
1480                         /* Enable transfer toolbar mode */
1481                         set_toolbar_transfer_mode(self);
1482                         while (tmp) {
1483                                 modest_progress_object_add_operation (MODEST_PROGRESS_OBJECT (tmp->data),
1484                                                                       mail_op);
1485                                 tmp = g_slist_next (tmp);
1486                         }
1487                         
1488                 }
1489                 break;
1490         case MODEST_MAIL_OPERATION_QUEUE_OPERATION_REMOVED:
1491                 if (mode == TOOLBAR_MODE_TRANSFER) {
1492                         while (tmp) {
1493                                 modest_progress_object_remove_operation (MODEST_PROGRESS_OBJECT (tmp->data),
1494                                                                  mail_op);
1495                                 tmp = g_slist_next (tmp);
1496                                 
1497                         }
1498
1499                         /* If no more operations are being observed, NORMAL mode is enabled again */
1500                         if (observers_empty (self)) {
1501                                 set_toolbar_mode (self, TOOLBAR_MODE_NORMAL);
1502                         }
1503                 }
1504                 break;
1505         }
1506 }
1507
1508 GList *
1509 modest_msg_view_window_get_attachments (ModestMsgViewWindow *win) 
1510 {
1511         ModestMsgViewWindowPrivate *priv;
1512         GList *selected_attachments = NULL;
1513         
1514         g_return_val_if_fail (MODEST_IS_MSG_VIEW_WINDOW (win), NULL);
1515         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (win);
1516
1517         selected_attachments = modest_msg_view_get_selected_attachments (MODEST_MSG_VIEW (priv->msg_view));
1518         
1519         return selected_attachments;
1520 }
1521
1522 void
1523 modest_msg_view_window_view_attachment (ModestMsgViewWindow *window, TnyMimePart *mime_part)
1524 {
1525         ModestMsgViewWindowPrivate *priv;
1526         g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window));
1527         g_return_if_fail (TNY_IS_MIME_PART (mime_part) || (mime_part == NULL));
1528
1529         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
1530
1531         if (mime_part == NULL) {
1532                 gboolean error = FALSE;
1533                 GList *selected_attachments = modest_msg_view_get_selected_attachments (MODEST_MSG_VIEW (priv->msg_view));
1534                 if (selected_attachments == NULL) {
1535                         error = TRUE;
1536                 } else if (g_list_length (selected_attachments) > 1) {
1537                         hildon_banner_show_information (NULL, NULL, _("mcen_ib_unable_to_display_more"));
1538                         error = TRUE;
1539                 } else {
1540                         mime_part = (TnyMimePart *) selected_attachments->data;
1541                         g_object_ref (mime_part);
1542                 }
1543                 g_list_foreach (selected_attachments, (GFunc) g_object_unref, NULL);
1544                 g_list_free (selected_attachments);
1545
1546                 if (error)
1547                         return;
1548         } else {
1549                 g_object_ref (mime_part);
1550         }
1551
1552         if (!TNY_IS_MSG (mime_part)) {
1553                 gchar *filepath = NULL;
1554                 const gchar *att_filename = tny_mime_part_get_filename (mime_part);
1555                 gchar *extension = NULL;
1556                 TnyFsStream *temp_stream = NULL;
1557
1558                 if (att_filename) {
1559                         extension = g_strrstr (att_filename, ".");
1560                         if (extension != NULL)
1561                                 extension++;
1562                 }
1563
1564                 temp_stream = modest_maemo_utils_create_temp_stream (extension, &filepath);
1565
1566                 if (temp_stream) {
1567                         const gchar *content_type;
1568                         content_type = tny_mime_part_get_content_type (mime_part);
1569                         tny_mime_part_decode_to_stream (mime_part, TNY_STREAM (temp_stream));
1570                         
1571                         modest_platform_activate_file (filepath, content_type);
1572                         g_object_unref (temp_stream);
1573                         g_free (filepath);
1574                         /* TODO: delete temporary file */
1575                 }
1576         } else {
1577                 /* message attachment */
1578                 TnyHeader *header = NULL;
1579                 ModestWindowMgr *mgr;
1580                 ModestWindow *msg_win = NULL;
1581
1582                 header = tny_msg_get_header (TNY_MSG (mime_part));
1583                 mgr = modest_runtime_get_window_mgr ();
1584                 msg_win = modest_window_mgr_find_window_by_header (mgr, header);
1585
1586                 if (!msg_win) {
1587                         gchar *account = g_strdup (modest_window_get_active_account (MODEST_WINDOW (window)));
1588                         if (!account)
1589                                 account = modest_account_mgr_get_default_account (modest_runtime_get_account_mgr ());
1590                         msg_win = modest_msg_view_window_new (TNY_MSG (mime_part), account, NULL);
1591                         modest_window_mgr_register_window (mgr, msg_win);
1592                         gtk_window_set_transient_for (GTK_WINDOW (msg_win), GTK_WINDOW (window));
1593                 }
1594
1595                 gtk_widget_show_all (GTK_WIDGET (msg_win));
1596         }
1597         g_object_unref (mime_part);
1598 }
1599
1600 static gboolean
1601 save_mime_part_to_file (const gchar *filename, TnyMimePart *mime_part)
1602 {
1603         GnomeVFSResult result;
1604         GnomeVFSHandle *handle;
1605         TnyStream *stream;
1606         
1607         result = gnome_vfs_create (&handle, filename, GNOME_VFS_OPEN_WRITE, FALSE, 0777);
1608         if (result != GNOME_VFS_OK) {
1609                 hildon_banner_show_information (NULL, NULL, _("mail_ib_file_operation_failed"));
1610                 return FALSE;
1611         }
1612         stream = tny_vfs_stream_new (handle);
1613         tny_mime_part_decode_to_stream (mime_part, stream);
1614         g_object_unref (G_OBJECT (stream));
1615         return TRUE;
1616 }
1617
1618 static gboolean
1619 save_mime_part_to_file_with_checks (GtkWindow *parent, const gchar *filename, TnyMimePart *mime_part)
1620 {
1621         if (modest_maemo_utils_file_exists (filename)) {
1622                 GtkWidget *confirm_overwrite_dialog;
1623                 confirm_overwrite_dialog = hildon_note_new_confirmation (GTK_WINDOW (parent),
1624                                                                          _("TODO: confirm overwrite"));
1625                 if (gtk_dialog_run (GTK_DIALOG (confirm_overwrite_dialog)) != GTK_RESPONSE_OK) {
1626                         gtk_widget_destroy (confirm_overwrite_dialog);
1627                         return FALSE;
1628                 }
1629                 gtk_widget_destroy (confirm_overwrite_dialog);
1630         }
1631
1632         return save_mime_part_to_file (filename, mime_part);
1633 }
1634
1635 void
1636 modest_msg_view_window_save_attachments (ModestMsgViewWindow *window, GList *mime_parts)
1637 {
1638         gboolean clean_list = FALSE;
1639         ModestMsgViewWindowPrivate *priv;
1640         g_return_if_fail (MODEST_IS_MSG_VIEW_WINDOW (window));
1641
1642         priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
1643
1644         if (mime_parts == NULL) {
1645                 mime_parts = modest_msg_view_get_selected_attachments (MODEST_MSG_VIEW (priv->msg_view));
1646                 if (mime_parts == NULL)
1647                         return;
1648                 clean_list = TRUE;
1649         }
1650
1651         if (mime_parts->next == NULL) {
1652                 /* only one attachment selected */
1653                 GtkWidget *save_dialog = NULL;
1654                 TnyMimePart *mime_part = (TnyMimePart *) mime_parts->data;
1655                 if (!TNY_IS_MSG (mime_part) && tny_mime_part_is_attachment (mime_part)) {
1656                         const gchar *filename;
1657                         gchar *folder;
1658                         save_dialog = hildon_file_chooser_dialog_new (GTK_WINDOW (window), GTK_FILE_CHOOSER_ACTION_SAVE);
1659                         folder = g_build_filename (g_get_home_dir (), DEFAULT_FOLDER, NULL);
1660                         gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (save_dialog), folder);
1661                         g_free (folder);
1662                         filename = tny_mime_part_get_filename (mime_part);
1663                         if (filename != NULL)
1664                                 gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (save_dialog), filename);
1665                         while (gtk_dialog_run (GTK_DIALOG (save_dialog)) == GTK_RESPONSE_OK) {
1666                                 gchar *filename_tmp = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (save_dialog));
1667                                 gboolean save_result;
1668                                 if (!modest_maemo_utils_folder_writable (filename_tmp)) {
1669                                         g_free (filename_tmp);
1670                                         hildon_banner_show_information (NULL, NULL, _("TODO: read only location"));
1671                                         continue;
1672                                 }
1673                                 save_result = save_mime_part_to_file_with_checks (GTK_WINDOW (save_dialog), 
1674                                                                                   filename_tmp, mime_part);
1675                                 g_free (filename_tmp);
1676                                 if (save_result)
1677                                         break;
1678                                 else
1679                                         continue;
1680                         }
1681                         gtk_widget_destroy (save_dialog);
1682                 } else {
1683                         g_warning ("Tried to save a non-file attachment");
1684                 }
1685         } else {
1686                 GtkWidget *save_dialog = NULL;
1687                 gchar *folder;
1688                 gchar *save_multiple_str = g_strdup_printf (_("FIXME: %d attachments"), 
1689                                                             g_list_length (mime_parts));
1690                 save_dialog = hildon_file_chooser_dialog_new (GTK_WINDOW (window), GTK_FILE_CHOOSER_ACTION_SAVE);
1691                 g_object_set (G_OBJECT (save_dialog), "save-multiple", save_multiple_str, NULL);
1692                 
1693                 folder = g_build_filename (g_get_home_dir (), DEFAULT_FOLDER, NULL);
1694                 gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (save_dialog), folder);
1695                 g_free (folder);
1696                 if (gtk_dialog_run (GTK_DIALOG (save_dialog)) == GTK_RESPONSE_OK) {
1697                         gchar *foldername = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (save_dialog));
1698                         GList *node = NULL;
1699                         gboolean attachment_found = FALSE;
1700                         if (!modest_maemo_utils_folder_writable (foldername)) {
1701                                 g_free (foldername);
1702                                 hildon_banner_show_information (NULL, NULL, _("TODO: read only location"));
1703                         }
1704                         for (node = mime_parts; node != NULL; node = g_list_next (node)) {
1705                                 TnyMimePart *mime_part = (TnyMimePart *) node->data;
1706                                 if (tny_mime_part_is_attachment (mime_part)) {
1707                                         const gchar *att_filename = tny_mime_part_get_filename (mime_part);
1708                                         if (att_filename != NULL) {
1709                                                 gchar *full_filename;
1710                                                 gboolean save_result;
1711                                                 full_filename = g_build_filename (foldername, att_filename, NULL);
1712                                                 attachment_found = TRUE;
1713                                                 
1714                                                 save_result = save_mime_part_to_file_with_checks (GTK_WINDOW (save_dialog), 
1715                                                                                                   full_filename, mime_part);
1716                                                 g_free (full_filename);
1717                                                 if (!save_result)
1718                                                         break;
1719                                         }
1720                                 }
1721                         }
1722                         gtk_widget_destroy (save_dialog);
1723                 } else {
1724                         g_warning ("Tried to save a non-file attachment");
1725                 }
1726                 /* more than one attachment selected */
1727         }
1728         if (clean_list) {
1729                 g_list_foreach (mime_parts, (GFunc) g_object_unref, NULL);
1730                 g_list_free (mime_parts);
1731         }
1732 }
1733
1734 void
1735 modest_msg_view_window_remove_attachments (ModestMsgViewWindow *window, GList *mime_parts)
1736 {
1737 /*      g_message ("not implemented %s", __FUNCTION__); */
1738 }
1739
1740
1741 static void
1742 update_window_title (ModestMsgViewWindow *window)
1743 {
1744         ModestMsgViewWindowPrivate *priv = MODEST_MSG_VIEW_WINDOW_GET_PRIVATE (window);
1745         TnyMsg *msg = NULL;
1746         TnyHeader *header = NULL;
1747         const gchar *subject = NULL;
1748
1749         msg = modest_msg_view_get_message (MODEST_MSG_VIEW (priv->msg_view));
1750         if (msg != NULL) {
1751                 header = tny_msg_get_header (msg);
1752                 subject = tny_header_get_subject (header);
1753         }
1754
1755         if ((subject == NULL)||(subject[0] == '\0'))
1756                 subject = _("mail_va_no_subject");
1757
1758         gtk_window_set_title (GTK_WINDOW (window), subject);
1759 }
1760
1761 static gchar * 
1762 translate_func (const gchar *msgid,
1763                 const gchar *domain_name)
1764 {
1765         return _(msgid);
1766 }