f2c346665ee8206f82b90fabe889470bfd3de417
[modest] / src / widgets / modest-folder-view.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
30 #include <glib/gi18n.h>
31 #include <string.h>
32
33 #include <tny-account-tree-model.h>
34 #include <tny-account-store-iface.h>
35 #include <tny-account-iface.h>
36 #include <tny-msg-folder-iface.h>
37 #include <tny-summary-window-iface.h>
38 #include <modest-icon-names.h>
39 #include <modest-icon-factory.h>
40 #include <modest-tny-account-store.h>
41
42 #include "modest-folder-view.h"
43
44
45 /* 'private'/'protected' functions */
46 static void modest_folder_view_class_init  (ModestFolderViewClass *klass);
47 static void modest_folder_view_init        (ModestFolderView *obj);
48 static void modest_folder_view_finalize    (GObject *obj);
49
50 static gboolean update_model (ModestFolderView *self,
51                               ModestTnyAccountStore *account_store);
52 static gboolean update_model_empty (ModestFolderView *self);
53 static void on_selection_changed (GtkTreeSelection *sel, gpointer data);
54 static gboolean modest_folder_view_update_model (ModestFolderView *self,
55                                                  TnyAccountStoreIface *account_store);
56
57 enum {
58         FOLDER_SELECTED_SIGNAL,
59         LAST_SIGNAL
60 };
61
62 typedef struct _ModestFolderViewPrivate ModestFolderViewPrivate;
63 struct _ModestFolderViewPrivate {
64
65         TnyAccountStoreIface *account_store;
66         TnyMsgFolderIface    *cur_folder;
67         gboolean             view_is_empty;
68
69         gulong               sig1, sig2;
70         GMutex               *lock;
71 };
72 #define MODEST_FOLDER_VIEW_GET_PRIVATE(o)                               \
73         (G_TYPE_INSTANCE_GET_PRIVATE((o),                               \
74                                      MODEST_TYPE_FOLDER_VIEW,           \
75                                      ModestFolderViewPrivate))
76 /* globals */
77 static GObjectClass *parent_class = NULL;
78
79 static guint signals[LAST_SIGNAL] = {0}; 
80
81 GType
82 modest_folder_view_get_type (void)
83 {
84         static GType my_type = 0;
85         if (!my_type) {
86                 static const GTypeInfo my_info = {
87                         sizeof(ModestFolderViewClass),
88                         NULL,           /* base init */
89                         NULL,           /* base finalize */
90                         (GClassInitFunc) modest_folder_view_class_init,
91                         NULL,           /* class finalize */
92                         NULL,           /* class data */
93                         sizeof(ModestFolderView),
94                         1,              /* n_preallocs */
95                         (GInstanceInitFunc) modest_folder_view_init,
96                 };
97                                 
98                 my_type = g_type_register_static (GTK_TYPE_TREE_VIEW,
99                                                   "ModestFolderView",
100                                                   &my_info, 0);         
101         }
102         return my_type;
103 }
104
105 static void
106 modest_folder_view_class_init (ModestFolderViewClass *klass)
107 {
108         GObjectClass *gobject_class;
109         gobject_class = (GObjectClass*) klass;
110
111         parent_class            = g_type_class_peek_parent (klass);
112         gobject_class->finalize = modest_folder_view_finalize;
113         
114         klass->update_model = modest_folder_view_update_model;
115
116         g_type_class_add_private (gobject_class,
117                                   sizeof(ModestFolderViewPrivate));
118         
119         signals[FOLDER_SELECTED_SIGNAL] = 
120                 g_signal_new ("folder_selected",
121                               G_TYPE_FROM_CLASS (gobject_class),
122                               G_SIGNAL_RUN_FIRST,
123                               G_STRUCT_OFFSET (ModestFolderViewClass,
124                                                folder_selected),
125                               NULL, NULL,
126                               g_cclosure_marshal_VOID__POINTER,
127                               G_TYPE_NONE, 1, G_TYPE_POINTER); 
128 }
129
130
131
132 static void
133 text_cell_data  (GtkTreeViewColumn *column,  GtkCellRenderer *renderer,
134                    GtkTreeModel *tree_model,  GtkTreeIter *iter,  gpointer data)
135 {
136         GObject *rendobj;
137         gchar *fname;
138         gint unread;
139         TnyMsgFolderType type;
140         
141         gtk_tree_model_get (tree_model, iter,
142                             TNY_ACCOUNT_TREE_MODEL_NAME_COLUMN, &fname,
143                             TNY_ACCOUNT_TREE_MODEL_TYPE_COLUMN, &type,
144                             TNY_ACCOUNT_TREE_MODEL_UNREAD_COLUMN, &unread, -1);
145         rendobj = G_OBJECT(renderer);
146
147         if (unread > 0) {
148                 gchar *folder_title = g_strdup_printf ("%s (%d)", fname, unread);
149                 g_object_set (rendobj,"text", folder_title,  "weight", 800, NULL);
150                 g_free (folder_title);
151         } else 
152                 g_object_set (rendobj,"text", fname, "weight", 400, NULL);
153                 
154         g_free (fname);
155 }
156
157 /* FIXME: move these to TnyMail */
158 enum {
159
160         TNY_MSG_FOLDER_TYPE_NOTES = TNY_MSG_FOLDER_TYPE_SENT + 1, /* urgh */
161         TNY_MSG_FOLDER_TYPE_DRAFTS,
162         TNY_MSG_FOLDER_TYPE_CONTACTS,
163         TNY_MSG_FOLDER_TYPE_CALENDAR
164 };
165         
166 static TnyMsgFolderType
167 guess_folder_type (const gchar* name)
168 {
169         TnyMsgFolderType type;
170         gchar *folder;
171
172         g_return_val_if_fail (name, TNY_MSG_FOLDER_TYPE_NORMAL);
173         
174         type = TNY_MSG_FOLDER_TYPE_NORMAL;
175         folder = g_utf8_strdown (name, strlen(name));
176
177         if (strcmp (folder, "inbox") == 0 ||
178             strcmp (folder, _("inbox")) == 0)
179                 type = TNY_MSG_FOLDER_TYPE_INBOX;
180         else if (strcmp (folder, "outbox") == 0 ||
181                  strcmp (folder, _("outbox")) == 0)
182                 type = TNY_MSG_FOLDER_TYPE_OUTBOX;
183         else if (g_str_has_prefix(folder, "junk") ||
184                  g_str_has_prefix(folder, _("junk")))
185                 type = TNY_MSG_FOLDER_TYPE_JUNK;
186         else if (g_str_has_prefix(folder, "trash") ||
187                  g_str_has_prefix(folder, _("trash")))
188                 type = TNY_MSG_FOLDER_TYPE_JUNK;
189         else if (g_str_has_prefix(folder, "sent") ||
190                  g_str_has_prefix(folder, _("sent")))
191                 type = TNY_MSG_FOLDER_TYPE_SENT;
192
193         /* these are not *really* TNY_ types */
194         else if (g_str_has_prefix(folder, "draft") ||
195                  g_str_has_prefix(folder, _("draft")))
196                 type = TNY_MSG_FOLDER_TYPE_DRAFTS;
197         else if (g_str_has_prefix(folder, "notes") ||
198                  g_str_has_prefix(folder, _("notes")))
199                 type = TNY_MSG_FOLDER_TYPE_NOTES;
200         else if (g_str_has_prefix(folder, "contacts") ||
201                  g_str_has_prefix(folder, _("contacts")))
202                 type = TNY_MSG_FOLDER_TYPE_CONTACTS;
203         else if (g_str_has_prefix(folder, "calendar") ||
204                  g_str_has_prefix(folder, _("calendar")))
205                 type = TNY_MSG_FOLDER_TYPE_CALENDAR;
206         
207         g_free (folder);
208         return type;
209 }
210
211
212 static void
213 icon_cell_data  (GtkTreeViewColumn *column,  GtkCellRenderer *renderer,
214                  GtkTreeModel *tree_model,  GtkTreeIter *iter, gpointer data)
215 {
216         GObject *rendobj;
217         GdkPixbuf *pixbuf;
218         TnyMsgFolderType type;
219         gchar *fname = NULL;
220         gint unread;
221         
222         rendobj = G_OBJECT(renderer);
223         gtk_tree_model_get (tree_model, iter,
224                             TNY_ACCOUNT_TREE_MODEL_TYPE_COLUMN, &type,
225                             TNY_ACCOUNT_TREE_MODEL_NAME_COLUMN, &fname,
226                             TNY_ACCOUNT_TREE_MODEL_UNREAD_COLUMN, &unread, -1);
227         rendobj = G_OBJECT(renderer);
228         
229         if (type == TNY_MSG_FOLDER_TYPE_NORMAL)
230                 type = guess_folder_type (fname);
231         
232         if (fname);
233                 g_free (fname);
234
235         switch (type) {
236         case TNY_MSG_FOLDER_TYPE_INBOX:
237                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_INBOX);
238                 break;
239         case TNY_MSG_FOLDER_TYPE_OUTBOX:
240                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_OUTBOX);
241                 break;
242         case TNY_MSG_FOLDER_TYPE_JUNK:
243                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_JUNK);
244                 break;
245         case TNY_MSG_FOLDER_TYPE_SENT:
246                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_SENT);
247                 break;
248         case TNY_MSG_FOLDER_TYPE_DRAFTS:
249                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_DRAFTS);
250                 break;
251         case TNY_MSG_FOLDER_TYPE_NOTES:
252                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_NOTES);
253                 break;
254         case TNY_MSG_FOLDER_TYPE_CALENDAR:
255                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_CALENDAR);
256                 break;
257         case TNY_MSG_FOLDER_TYPE_CONTACTS:
258                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_CONTACTS);
259                 break;
260         case TNY_MSG_FOLDER_TYPE_NORMAL:
261         default:
262                 pixbuf = modest_icon_factory_get_icon (MODEST_FOLDER_ICON_NORMAL);
263                 break;
264         }
265
266         g_object_set (rendobj,
267                       "pixbuf-expander-open",
268                       modest_icon_factory_get_icon (MODEST_FOLDER_ICON_OPEN),
269                       "pixbuf-expander-closed",
270                       modest_icon_factory_get_icon (MODEST_FOLDER_ICON_CLOSED),
271                       "pixbuf", pixbuf,
272                       NULL);
273 }
274
275 static void
276 modest_folder_view_init (ModestFolderView *obj)
277 {
278         ModestFolderViewPrivate *priv;
279         GtkTreeViewColumn *column;
280         GtkCellRenderer *renderer;
281         GtkTreeSelection *sel;
282         
283         priv =  MODEST_FOLDER_VIEW_GET_PRIVATE(obj);
284         
285         priv->view_is_empty     = TRUE;
286         priv->account_store = NULL;
287         priv->cur_folder = NULL;
288
289         priv->lock = g_mutex_new ();
290         
291         column = gtk_tree_view_column_new ();
292         gtk_tree_view_column_set_title (column,
293                                         _("All Mail Folders"));
294         
295         gtk_tree_view_append_column (GTK_TREE_VIEW(obj),
296                                      column);
297         
298         renderer = gtk_cell_renderer_pixbuf_new();
299         gtk_tree_view_column_pack_start (column, renderer, FALSE);
300         gtk_tree_view_column_set_cell_data_func(column, renderer,
301                                                 icon_cell_data, NULL, NULL);
302         
303         renderer = gtk_cell_renderer_text_new();
304         gtk_tree_view_column_pack_start (column, renderer, FALSE);
305         gtk_tree_view_column_set_cell_data_func(column, renderer,
306                                                 text_cell_data, NULL, NULL);
307         
308         sel = gtk_tree_view_get_selection (GTK_TREE_VIEW(obj));
309         gtk_tree_selection_set_mode (sel, GTK_SELECTION_SINGLE);
310
311         gtk_tree_view_column_set_spacing (column, 2);
312         gtk_tree_view_column_set_resizable (column, TRUE);
313         gtk_tree_view_column_set_fixed_width (column, TRUE);            
314         gtk_tree_view_set_headers_visible   (GTK_TREE_VIEW(obj), TRUE);
315         gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(obj), FALSE);
316
317 }
318
319
320 static void
321 modest_folder_view_finalize (GObject *obj)
322 {
323         ModestFolderViewPrivate *priv;
324         GtkTreeSelection    *sel;
325         
326         g_return_if_fail (obj);
327         
328         priv =  MODEST_FOLDER_VIEW_GET_PRIVATE(obj);
329         if (priv->account_store) {
330                 g_signal_handler_disconnect (G_OBJECT(priv->account_store),
331                                              priv->sig1);
332                 g_object_unref (G_OBJECT(priv->account_store));
333                 priv->account_store = NULL;
334         }
335
336         if (priv->lock) {
337                 g_mutex_free (priv->lock);
338                 priv->lock = NULL;
339         }
340
341         sel = gtk_tree_view_get_selection (GTK_TREE_VIEW(obj));
342         if (sel)
343                 g_signal_handler_disconnect (G_OBJECT(sel), priv->sig2);
344         
345         G_OBJECT_CLASS(parent_class)->finalize (obj);
346 }
347
348
349 static void
350 on_account_update (TnyAccountStoreIface *account_store, const gchar *account,
351                    gpointer user_data)
352 {
353         update_model_empty (MODEST_FOLDER_VIEW(user_data));
354         
355         if (!update_model (MODEST_FOLDER_VIEW(user_data), account_store))
356                 g_printerr ("modest: failed to update model for changes in '%s'",
357                             account);
358 }
359
360
361 GtkWidget*
362 modest_folder_view_new (ModestTnyAccountStore *account_store)
363 {
364         GObject *self;
365         ModestFolderViewPrivate *priv;
366         GtkTreeSelection *sel;
367         
368         g_return_val_if_fail (account_store, NULL);
369         
370         self = G_OBJECT(g_object_new(MODEST_TYPE_FOLDER_VIEW, NULL));
371         priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
372         
373         if (!update_model (MODEST_FOLDER_VIEW(self), TNY_ACCOUNT_STORE_IFACE(account_store)))
374                 g_printerr ("modest: failed to update model");
375         
376         priv->sig1 = g_signal_connect (G_OBJECT(account_store), "account_update",
377                                        G_CALLBACK (on_account_update), self);   
378         
379         sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(self));
380         priv->sig2 = g_signal_connect (sel, "changed",
381                                        G_CALLBACK(on_selection_changed), self);
382                                         
383         return GTK_WIDGET(self);
384 }
385
386
387
388
389 static gboolean
390 update_model_empty (ModestFolderView *self)
391 {
392         GtkTreeIter  iter;
393         GtkTreeStore *store;
394         ModestFolderViewPrivate *priv;
395         
396         g_return_val_if_fail (self, FALSE);
397         
398         store = gtk_tree_store_new (1, G_TYPE_STRING);
399         gtk_tree_store_append (store, &iter, NULL);
400
401         gtk_tree_store_set (store, &iter, 0,
402                             _("(empty)"), -1);
403
404         gtk_tree_view_set_model (GTK_TREE_VIEW(self),
405                                  GTK_TREE_MODEL(store));
406         g_object_unref (store);
407
408         priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
409         priv->view_is_empty = TRUE;
410
411         g_signal_emit (G_OBJECT(self), signals[FOLDER_SELECTED_SIGNAL], 0,
412                        NULL);
413         
414         return TRUE;
415 }
416
417
418 static gboolean
419 update_model (ModestFolderView *self, ModestTnyAccountStore *account_store)
420 {
421         ModestFolderViewPrivate *priv;
422         TnyListIface     *account_list;
423         GtkTreeModel     *model, *sortable;
424         
425         g_return_val_if_fail (account_store, FALSE);
426         
427         priv =  MODEST_FOLDER_VIEW_GET_PRIVATE(self);
428
429         model        = GTK_TREE_MODEL(tny_account_tree_model_new ());
430         account_list = TNY_LIST_IFACE(model);
431
432         update_model_empty (self); /* cleanup */
433         priv->view_is_empty = TRUE;
434         
435         tny_account_store_iface_get_accounts (TNY_ACCOUNT_STORE_IFACE(account_store),
436                                               account_list,
437                                               TNY_ACCOUNT_STORE_IFACE_STORE_ACCOUNTS);
438         if (!account_list) /* no store accounts found */ 
439                 return TRUE;
440         
441         sortable = gtk_tree_model_sort_new_with_model (model);
442         gtk_tree_view_set_model (GTK_TREE_VIEW(self), sortable);
443
444         priv->view_is_empty = FALSE;    
445         g_object_unref (model);
446
447         return TRUE;
448
449
450
451 static void
452 on_selection_changed (GtkTreeSelection *sel, gpointer user_data)
453 {
454         GtkTreeModel            *model;
455         TnyMsgFolderIface       *folder = NULL;
456         GtkTreeIter             iter;
457         ModestFolderView *tree_view;
458         ModestFolderViewPrivate *priv;
459
460         g_return_if_fail (sel);
461         g_return_if_fail (user_data);
462
463         priv = MODEST_FOLDER_VIEW_GET_PRIVATE(user_data);
464
465         /* is_empty means that there is only the 'empty' item */
466         if (priv->view_is_empty)
467                 return;
468         
469         /* folder was _un_selected if true */
470         if (!gtk_tree_selection_get_selected (sel, &model, &iter)) {
471                 priv->cur_folder = NULL;
472                 return; 
473         }
474
475         tree_view = MODEST_FOLDER_VIEW (user_data);
476
477         gtk_tree_model_get (model, &iter,
478                             TNY_ACCOUNT_TREE_MODEL_INSTANCE_COLUMN,
479                             &folder, -1);
480
481         if (priv->cur_folder) 
482                 tny_msg_folder_iface_expunge (priv->cur_folder);
483         priv->cur_folder = folder;
484
485         /* folder will not be defined if you click eg. on the root node */
486         if (folder)
487                 g_signal_emit (G_OBJECT(tree_view), signals[FOLDER_SELECTED_SIGNAL], 0,
488                                folder);
489 }
490
491
492 static gboolean
493 modest_folder_view_update_model (ModestFolderView *self, TnyAccountStoreIface *account_store)
494 {
495         gboolean retval;
496         
497         g_return_val_if_fail (MODEST_IS_FOLDER_VIEW (self), FALSE);
498         retval = update_model (self, MODEST_TNY_ACCOUNT_STORE(account_store)); /* ugly */
499
500         g_signal_emit (G_OBJECT(self), signals[FOLDER_SELECTED_SIGNAL],
501                        0, NULL);
502
503         return retval;
504 }