1 /* Copyright (c) 2006, Nokia Corporation
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * * Neither the name of the Nokia Corporation nor the names of its
14 * contributors may be used to endorse or promote products derived from
15 * this software without specific prior written permission.
17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
18 * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
20 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
21 * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
22 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
23 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
24 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
25 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
26 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 #include <glib/gi18n.h>
32 #include <gdk/gdkkeysyms.h>
33 #include <tny-account-store-view.h>
34 #include <tny-gtk-account-list-model.h>
35 #include <tny-gtk-folder-store-tree-model.h>
36 #include <tny-gtk-header-list-model.h>
37 #include <tny-folder.h>
38 #include <tny-folder-store-observer.h>
39 #include <tny-account-store.h>
40 #include <tny-account.h>
41 #include <tny-folder.h>
42 #include <tny-camel-folder.h>
43 #include <tny-simple-list.h>
44 #include <modest-tny-folder.h>
45 #include <modest-tny-local-folders-account.h>
46 #include <modest-marshal.h>
47 #include <modest-icon-names.h>
48 #include <modest-tny-account-store.h>
49 #include <modest-text-utils.h>
50 #include <modest-runtime.h>
51 #include "modest-folder-view.h"
52 #include <modest-dnd.h>
53 #include <modest-platform.h>
54 #include <modest-account-mgr-helpers.h>
55 #include <modest-widget-memory.h>
56 #include <modest-ui-actions.h>
58 /* 'private'/'protected' functions */
59 static void modest_folder_view_class_init (ModestFolderViewClass *klass);
60 static void modest_folder_view_init (ModestFolderView *obj);
61 static void modest_folder_view_finalize (GObject *obj);
63 static void tny_account_store_view_init (gpointer g,
66 static void modest_folder_view_set_account_store (TnyAccountStoreView *self,
67 TnyAccountStore *account_store);
69 static gboolean update_model (ModestFolderView *self,
70 ModestTnyAccountStore *account_store);
72 static void on_selection_changed (GtkTreeSelection *sel, gpointer data);
74 static void on_account_update (TnyAccountStore *account_store,
78 static void on_accounts_reloaded (TnyAccountStore *store,
81 static gint cmp_rows (GtkTreeModel *tree_model,
86 static gboolean filter_row (GtkTreeModel *model,
90 static gboolean on_key_pressed (GtkWidget *self,
94 static void on_configuration_key_changed (ModestConf* conf,
96 ModestConfEvent event,
97 ModestFolderView *self);
100 static void on_drag_data_get (GtkWidget *widget,
101 GdkDragContext *context,
102 GtkSelectionData *selection_data,
107 static void on_drag_data_received (GtkWidget *widget,
108 GdkDragContext *context,
111 GtkSelectionData *selection_data,
116 static gboolean on_drag_motion (GtkWidget *widget,
117 GdkDragContext *context,
123 static gint expand_row_timeout (gpointer data);
125 static void setup_drag_and_drop (GtkTreeView *self);
128 FOLDER_SELECTION_CHANGED_SIGNAL,
129 FOLDER_DISPLAY_NAME_CHANGED_SIGNAL,
133 typedef struct _ModestFolderViewPrivate ModestFolderViewPrivate;
134 struct _ModestFolderViewPrivate {
135 TnyAccountStore *account_store;
136 TnyFolderStore *cur_folder_store;
138 gulong account_update_signal;
139 gulong changed_signal;
140 gulong accounts_reloaded_signal;
142 TnyFolderStoreQuery *query;
143 guint timer_expander;
145 gchar *local_account_name;
146 gchar *visible_account_id;
147 ModestFolderViewStyle style;
149 #define MODEST_FOLDER_VIEW_GET_PRIVATE(o) \
150 (G_TYPE_INSTANCE_GET_PRIVATE((o), \
151 MODEST_TYPE_FOLDER_VIEW, \
152 ModestFolderViewPrivate))
154 static GObjectClass *parent_class = NULL;
156 static guint signals[LAST_SIGNAL] = {0};
159 modest_folder_view_get_type (void)
161 static GType my_type = 0;
163 static const GTypeInfo my_info = {
164 sizeof(ModestFolderViewClass),
165 NULL, /* base init */
166 NULL, /* base finalize */
167 (GClassInitFunc) modest_folder_view_class_init,
168 NULL, /* class finalize */
169 NULL, /* class data */
170 sizeof(ModestFolderView),
172 (GInstanceInitFunc) modest_folder_view_init,
176 static const GInterfaceInfo tny_account_store_view_info = {
177 (GInterfaceInitFunc) tny_account_store_view_init, /* interface_init */
178 NULL, /* interface_finalize */
179 NULL /* interface_data */
183 my_type = g_type_register_static (GTK_TYPE_TREE_VIEW,
187 g_type_add_interface_static (my_type,
188 TNY_TYPE_ACCOUNT_STORE_VIEW,
189 &tny_account_store_view_info);
195 modest_folder_view_class_init (ModestFolderViewClass *klass)
197 GObjectClass *gobject_class;
198 gobject_class = (GObjectClass*) klass;
200 parent_class = g_type_class_peek_parent (klass);
201 gobject_class->finalize = modest_folder_view_finalize;
203 g_type_class_add_private (gobject_class,
204 sizeof(ModestFolderViewPrivate));
206 signals[FOLDER_SELECTION_CHANGED_SIGNAL] =
207 g_signal_new ("folder_selection_changed",
208 G_TYPE_FROM_CLASS (gobject_class),
210 G_STRUCT_OFFSET (ModestFolderViewClass,
211 folder_selection_changed),
213 modest_marshal_VOID__POINTER_BOOLEAN,
214 G_TYPE_NONE, 2, G_TYPE_POINTER, G_TYPE_BOOLEAN);
217 * This signal is emitted whenever the currently selected
218 * folder display name is computed. Note that the name could
219 * be different to the folder name, because we could append
220 * the unread messages count to the folder name to build the
221 * folder display name
223 signals[FOLDER_DISPLAY_NAME_CHANGED_SIGNAL] =
224 g_signal_new ("folder-display-name-changed",
225 G_TYPE_FROM_CLASS (gobject_class),
227 G_STRUCT_OFFSET (ModestFolderViewClass,
228 folder_display_name_changed),
230 g_cclosure_marshal_VOID__STRING,
231 G_TYPE_NONE, 1, G_TYPE_STRING);
235 text_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
236 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer data)
238 ModestFolderViewPrivate *priv;
243 GObject *instance = NULL;
245 g_return_if_fail (column);
246 g_return_if_fail (tree_model);
248 gtk_tree_model_get (tree_model, iter,
249 TNY_GTK_FOLDER_STORE_TREE_MODEL_NAME_COLUMN, &fname,
250 TNY_GTK_FOLDER_STORE_TREE_MODEL_ALL_COLUMN, &all,
251 TNY_GTK_FOLDER_STORE_TREE_MODEL_UNREAD_COLUMN, &unread,
252 TNY_GTK_FOLDER_STORE_TREE_MODEL_TYPE_COLUMN, &type,
253 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN, &instance,
255 rendobj = G_OBJECT(renderer);
266 priv = MODEST_FOLDER_VIEW_GET_PRIVATE (data);
268 gchar *item_name = NULL;
269 gint item_weight = 400;
271 if (type != TNY_FOLDER_TYPE_ROOT) {
274 if (modest_tny_folder_is_local_folder (TNY_FOLDER (instance))) {
276 type = modest_tny_folder_get_local_folder_type (TNY_FOLDER (instance));
277 if (type != TNY_FOLDER_TYPE_UNKNOWN) {
279 fname = g_strdup(modest_local_folder_info_get_type_display_name (type));
283 /* Select the number to show */
284 if ((type == TNY_FOLDER_TYPE_DRAFTS) || (type == TNY_FOLDER_TYPE_OUTBOX))
289 /* Use bold font style if there are unread messages */
291 item_name = g_strdup_printf ("%s (%d)", fname, unread);
294 item_name = g_strdup (fname);
298 } else if (TNY_IS_ACCOUNT (instance)) {
299 /* If it's a server account */
300 if (modest_tny_account_is_virtual_local_folders (
301 TNY_ACCOUNT (instance))) {
302 item_name = g_strdup (priv->local_account_name);
305 item_name = g_strdup (fname);
311 item_name = g_strdup ("unknown");
313 if (item_name && item_weight) {
314 /* Set the name in the treeview cell: */
315 g_object_set (rendobj,"text", item_name, "weight", item_weight, NULL);
317 /* Notify display name observers */
318 if (G_OBJECT (priv->cur_folder_store) == instance) {
319 g_signal_emit (G_OBJECT(data),
320 signals[FOLDER_DISPLAY_NAME_CHANGED_SIGNAL], 0,
328 g_object_unref (G_OBJECT (instance));
335 icon_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
336 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer data)
338 GObject *rendobj = NULL, *instance = NULL;
339 GdkPixbuf *pixbuf = NULL;
342 const gchar *account_id = NULL;
345 rendobj = G_OBJECT(renderer);
346 gtk_tree_model_get (tree_model, iter,
347 TNY_GTK_FOLDER_STORE_TREE_MODEL_TYPE_COLUMN, &type,
348 TNY_GTK_FOLDER_STORE_TREE_MODEL_NAME_COLUMN, &fname,
349 TNY_GTK_FOLDER_STORE_TREE_MODEL_UNREAD_COLUMN, &unread,
350 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN, &instance,
361 if (type == TNY_FOLDER_TYPE_NORMAL || type == TNY_FOLDER_TYPE_UNKNOWN) {
362 type = modest_tny_folder_guess_folder_type_from_name (fname);
366 case TNY_FOLDER_TYPE_ROOT:
367 if (TNY_IS_ACCOUNT (instance)) {
369 if (modest_tny_account_is_virtual_local_folders (
370 TNY_ACCOUNT (instance))) {
371 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_LOCAL_FOLDERS);
374 account_id = tny_account_get_id (TNY_ACCOUNT (instance));
376 if (!strcmp (account_id, MODEST_MMC_ACCOUNT_ID))
377 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_MMC);
379 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_ACCOUNT);
383 case TNY_FOLDER_TYPE_INBOX:
384 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_INBOX);
386 case TNY_FOLDER_TYPE_OUTBOX:
387 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_OUTBOX);
389 case TNY_FOLDER_TYPE_JUNK:
390 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_JUNK);
392 case TNY_FOLDER_TYPE_SENT:
393 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_SENT);
395 case TNY_FOLDER_TYPE_TRASH:
396 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_TRASH);
398 case TNY_FOLDER_TYPE_DRAFTS:
399 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_DRAFTS);
401 case TNY_FOLDER_TYPE_NORMAL:
403 pixbuf = modest_platform_get_icon (MODEST_FOLDER_ICON_NORMAL);
407 g_object_unref (G_OBJECT (instance));
411 g_object_set (rendobj, "pixbuf", pixbuf, NULL);
414 g_object_unref (pixbuf);
418 add_columns (GtkWidget *treeview)
420 GtkTreeViewColumn *column;
421 GtkCellRenderer *renderer;
422 GtkTreeSelection *sel;
425 column = gtk_tree_view_column_new ();
427 /* Set icon and text render function */
428 renderer = gtk_cell_renderer_pixbuf_new();
429 gtk_tree_view_column_pack_start (column, renderer, FALSE);
430 gtk_tree_view_column_set_cell_data_func(column, renderer,
431 icon_cell_data, treeview, NULL);
433 renderer = gtk_cell_renderer_text_new();
434 gtk_tree_view_column_pack_start (column, renderer, FALSE);
435 gtk_tree_view_column_set_cell_data_func(column, renderer,
436 text_cell_data, treeview, NULL);
438 /* Set selection mode */
439 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW(treeview));
440 gtk_tree_selection_set_mode (sel, GTK_SELECTION_SINGLE);
442 /* Set treeview appearance */
443 gtk_tree_view_column_set_spacing (column, 2);
444 gtk_tree_view_column_set_resizable (column, TRUE);
445 gtk_tree_view_column_set_fixed_width (column, TRUE);
446 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(treeview), FALSE);
447 gtk_tree_view_set_enable_search (GTK_TREE_VIEW(treeview), FALSE);
450 gtk_tree_view_append_column (GTK_TREE_VIEW(treeview),column);
454 modest_folder_view_init (ModestFolderView *obj)
456 ModestFolderViewPrivate *priv;
459 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(obj);
461 priv->timer_expander = 0;
462 priv->account_store = NULL;
464 priv->style = MODEST_FOLDER_VIEW_STYLE_SHOW_ALL;
465 priv->cur_folder_store = NULL;
466 priv->visible_account_id = NULL;
468 /* Initialize the local account name */
469 conf = modest_runtime_get_conf();
470 priv->local_account_name = modest_conf_get_string (conf, MODEST_CONF_DEVICE_NAME, NULL);
473 add_columns (GTK_WIDGET (obj));
475 /* Setup drag and drop */
476 setup_drag_and_drop (GTK_TREE_VIEW(obj));
478 /* Connect signals */
479 g_signal_connect (G_OBJECT (obj),
481 G_CALLBACK (on_key_pressed), NULL);
484 * Track changes in the local account name (in the device it
485 * will be the device name)
487 g_signal_connect (G_OBJECT(conf),
489 G_CALLBACK(on_configuration_key_changed), obj);
494 tny_account_store_view_init (gpointer g, gpointer iface_data)
496 TnyAccountStoreViewIface *klass = (TnyAccountStoreViewIface *)g;
498 klass->set_account_store_func = modest_folder_view_set_account_store;
504 modest_folder_view_finalize (GObject *obj)
506 ModestFolderViewPrivate *priv;
507 GtkTreeSelection *sel;
509 g_return_if_fail (obj);
511 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(obj);
513 if (priv->timer_expander != 0) {
514 g_source_remove (priv->timer_expander);
515 priv->timer_expander = 0;
518 if (priv->account_store) {
519 g_signal_handler_disconnect (G_OBJECT(priv->account_store),
520 priv->account_update_signal);
521 g_signal_handler_disconnect (G_OBJECT(priv->account_store),
522 priv->accounts_reloaded_signal);
523 g_object_unref (G_OBJECT(priv->account_store));
524 priv->account_store = NULL;
528 g_object_unref (G_OBJECT (priv->query));
532 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW(obj));
534 g_signal_handler_disconnect (G_OBJECT(sel), priv->changed_signal);
536 g_free (priv->local_account_name);
537 g_free (priv->visible_account_id);
539 G_OBJECT_CLASS(parent_class)->finalize (obj);
544 modest_folder_view_set_account_store (TnyAccountStoreView *self, TnyAccountStore *account_store)
546 ModestFolderViewPrivate *priv;
549 g_return_if_fail (MODEST_IS_FOLDER_VIEW (self));
550 g_return_if_fail (TNY_IS_ACCOUNT_STORE (account_store));
552 priv = MODEST_FOLDER_VIEW_GET_PRIVATE (self);
553 device = tny_account_store_get_device (account_store);
555 if (G_UNLIKELY (priv->account_store)) {
557 if (g_signal_handler_is_connected (G_OBJECT (priv->account_store),
558 priv->account_update_signal))
559 g_signal_handler_disconnect (G_OBJECT (priv->account_store),
560 priv->account_update_signal);
561 if (g_signal_handler_is_connected (G_OBJECT (priv->account_store),
562 priv->accounts_reloaded_signal))
563 g_signal_handler_disconnect (G_OBJECT (priv->account_store),
564 priv->accounts_reloaded_signal);
566 g_object_unref (G_OBJECT (priv->account_store));
569 priv->account_store = g_object_ref (G_OBJECT (account_store));
571 priv->account_update_signal =
572 g_signal_connect (G_OBJECT(account_store), "account_update",
573 G_CALLBACK (on_account_update), self);
575 priv->accounts_reloaded_signal =
576 g_signal_connect (G_OBJECT(account_store), "accounts_reloaded",
577 G_CALLBACK (on_accounts_reloaded), self);
579 g_object_unref (G_OBJECT (device));
583 on_account_update (TnyAccountStore *account_store, const gchar *account,
586 if (!update_model (MODEST_FOLDER_VIEW(user_data),
587 MODEST_TNY_ACCOUNT_STORE(account_store)))
588 g_printerr ("modest: failed to update model for changes in '%s'",
593 on_accounts_reloaded (TnyAccountStore *account_store,
596 ModestConf *conf = modest_runtime_get_conf ();
598 modest_widget_memory_save (conf, G_OBJECT (user_data), MODEST_CONF_FOLDER_VIEW_KEY);
599 update_model (MODEST_FOLDER_VIEW (user_data),
600 MODEST_TNY_ACCOUNT_STORE(account_store));
601 modest_widget_memory_restore (conf, G_OBJECT (user_data), MODEST_CONF_FOLDER_VIEW_KEY);
605 modest_folder_view_set_title (ModestFolderView *self, const gchar *title)
607 GtkTreeViewColumn *col;
609 g_return_if_fail (self);
611 col = gtk_tree_view_get_column (GTK_TREE_VIEW(self), 0);
613 g_printerr ("modest: failed get column for title\n");
617 gtk_tree_view_column_set_title (col, title);
618 gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(self),
623 modest_folder_view_new (TnyFolderStoreQuery *query)
626 ModestFolderViewPrivate *priv;
627 GtkTreeSelection *sel;
629 self = G_OBJECT (g_object_new (MODEST_TYPE_FOLDER_VIEW, NULL));
630 priv = MODEST_FOLDER_VIEW_GET_PRIVATE (self);
633 priv->query = g_object_ref (query);
635 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(self));
636 priv->changed_signal = g_signal_connect (sel, "changed",
637 G_CALLBACK (on_selection_changed), self);
639 return GTK_WIDGET(self);
642 /* this feels dirty; any other way to expand all the root items? */
644 expand_root_items (ModestFolderView *self)
647 path = gtk_tree_path_new_first ();
649 /* all folders should have child items, so.. */
650 while (gtk_tree_view_expand_row (GTK_TREE_VIEW(self), path, FALSE))
651 gtk_tree_path_next (path);
653 gtk_tree_path_free (path);
657 * We use this function to implement the
658 * MODEST_FOLDER_VIEW_STYLE_SHOW_ONE style. We only show the default
659 * account in this case, and the local folders.
662 filter_row (GtkTreeModel *model,
666 gboolean retval = TRUE;
668 GObject *instance = NULL;
670 gtk_tree_model_get (model, iter,
671 TNY_GTK_FOLDER_STORE_TREE_MODEL_TYPE_COLUMN, &type,
672 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN, &instance,
675 /* Do not show if there is no instance, this could indeed
676 happen when the model is being modified while it's being
677 drawn. This could occur for example when moving folders
682 if (type == TNY_FOLDER_TYPE_ROOT) {
683 /* TNY_FOLDER_TYPE_ROOT means that the instance is an account instead of a folder. */
684 if (TNY_IS_ACCOUNT (instance)) {
685 TnyAccount *acc = TNY_ACCOUNT (instance);
686 const gchar *account_id = tny_account_get_id (acc);
688 /* If it isn't a special folder,
689 * don't show it unless it is the visible account: */
690 if (!modest_tny_account_is_virtual_local_folders (acc) &&
691 strcmp (account_id, MODEST_MMC_ACCOUNT_ID)) {
692 /* Show only the visible account id */
693 ModestFolderViewPrivate *priv =
694 MODEST_FOLDER_VIEW_GET_PRIVATE (data);
695 if (priv->visible_account_id && strcmp (account_id, priv->visible_account_id))
701 /* The virtual local-folders folder store is also shown by default. */
703 g_object_unref (instance);
709 static void on_tnylist_accounts_debug_print(gpointer data, gpointer user_data)
711 TnyAccount* account = TNY_ACCOUNT(data);
712 const gchar *prefix = (const gchar*)(user_data);
714 printf("%s account id=%s\n", prefix, tny_account_get_id (account));
719 update_model (ModestFolderView *self, ModestTnyAccountStore *account_store)
721 ModestFolderViewPrivate *priv;
722 GtkTreeModel *model /* , *old_model */;
723 /* TnyAccount *local_account; */
724 TnyList *model_as_list;
726 g_return_val_if_fail (account_store, FALSE);
728 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
730 /* Notify that there is no folder selected */
731 g_signal_emit (G_OBJECT(self),
732 signals[FOLDER_SELECTION_CHANGED_SIGNAL], 0,
735 /* FIXME: the local accounts are not shown when the query
736 selects only the subscribed folders. */
737 /* model = tny_gtk_folder_store_tree_model_new (TRUE, priv->query); */
738 model = tny_gtk_folder_store_tree_model_new (TRUE, NULL);
740 /* Deal with the model via its TnyList Interface,
741 * filling the TnyList via a get_accounts() call: */
742 model_as_list = TNY_LIST(model);
744 /* Get the accounts: */
745 tny_account_store_get_accounts (TNY_ACCOUNT_STORE(account_store),
747 TNY_ACCOUNT_STORE_STORE_ACCOUNTS);
748 g_object_unref (model_as_list);
749 model_as_list = NULL;
751 GtkTreeModel *filter_model = NULL, *sortable = NULL;
753 sortable = gtk_tree_model_sort_new_with_model (model);
754 gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE(sortable),
755 TNY_GTK_FOLDER_STORE_TREE_MODEL_NAME_COLUMN,
757 gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (sortable),
758 TNY_GTK_FOLDER_STORE_TREE_MODEL_NAME_COLUMN,
759 cmp_rows, NULL, NULL);
761 /* Create filter model */
762 if (priv->style == MODEST_FOLDER_VIEW_STYLE_SHOW_ONE) {
763 filter_model = gtk_tree_model_filter_new (sortable, NULL);
764 gtk_tree_model_filter_set_visible_func (GTK_TREE_MODEL_FILTER (filter_model),
771 gtk_tree_view_set_model (GTK_TREE_VIEW(self),
772 (filter_model) ? filter_model : sortable);
773 expand_root_items (self); /* expand all account folders */
775 g_object_unref (model);
778 g_object_unref (filter_model);
780 g_object_unref (sortable);
787 on_selection_changed (GtkTreeSelection *sel, gpointer user_data)
790 TnyFolderStore *folder = NULL;
792 ModestFolderView *tree_view;
793 ModestFolderViewPrivate *priv;
796 g_return_if_fail (sel);
797 g_return_if_fail (user_data);
799 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(user_data);
801 /* folder was _un_selected if true */
802 if (!gtk_tree_selection_get_selected (sel, &model, &iter)) {
803 if (priv->cur_folder_store)
804 g_object_unref (priv->cur_folder_store);
805 priv->cur_folder_store = NULL;
807 /* Notify the display name observers */
808 g_signal_emit (G_OBJECT(user_data),
809 signals[FOLDER_DISPLAY_NAME_CHANGED_SIGNAL], 0,
814 tree_view = MODEST_FOLDER_VIEW (user_data);
816 gtk_tree_model_get (model, &iter,
817 TNY_GTK_FOLDER_STORE_TREE_MODEL_TYPE_COLUMN, &type,
818 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN, &folder,
821 /* If the folder is the same do not notify */
822 if (priv->cur_folder_store == folder) {
823 g_object_unref (folder);
827 /* Current folder was unselected */
828 if (priv->cur_folder_store) {
829 g_signal_emit (G_OBJECT(tree_view), signals[FOLDER_SELECTION_CHANGED_SIGNAL], 0,
830 priv->cur_folder_store, FALSE);
831 g_object_unref (priv->cur_folder_store);
834 /* New current references */
835 priv->cur_folder_store = folder;
837 /* New folder has been selected */
838 g_signal_emit (G_OBJECT(tree_view),
839 signals[FOLDER_SELECTION_CHANGED_SIGNAL],
844 modest_folder_view_get_selected (ModestFolderView *self)
846 ModestFolderViewPrivate *priv;
848 g_return_val_if_fail (self, NULL);
850 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
851 if (priv->cur_folder_store)
852 g_object_ref (priv->cur_folder_store);
854 return priv->cur_folder_store;
858 get_cmp_rows_type_pos (GObject *folder)
860 /* Remote accounts -> Local account -> MMC account .*/
863 if (TNY_IS_ACCOUNT (folder) &&
864 modest_tny_account_is_virtual_local_folders (
865 TNY_ACCOUNT (folder))) {
867 } else if (TNY_IS_ACCOUNT (folder)) {
868 TnyAccount *account = TNY_ACCOUNT (folder);
869 const gchar *account_id = tny_account_get_id (account);
870 if (!strcmp (account_id, MODEST_MMC_ACCOUNT_ID))
876 printf ("DEBUG: %s: unexpected type.\n", __FUNCTION__);
877 return -1; /* Should never happen */
882 * This function orders the mail accounts according to these rules:
883 * 1st - remote accounts
884 * 2nd - local account
888 cmp_rows (GtkTreeModel *tree_model, GtkTreeIter *iter1, GtkTreeIter *iter2,
892 gchar *name1, *name2;
894 GObject *folder1 = NULL;
895 GObject *folder2 = NULL;
897 gtk_tree_model_get (tree_model, iter1,
898 TNY_GTK_FOLDER_STORE_TREE_MODEL_NAME_COLUMN, &name1,
899 TNY_GTK_FOLDER_STORE_TREE_MODEL_TYPE_COLUMN, &type,
900 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN, &folder1,
902 gtk_tree_model_get (tree_model, iter2,
903 TNY_GTK_FOLDER_STORE_TREE_MODEL_NAME_COLUMN, &name2,
904 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN, &folder2,
907 if (type == TNY_FOLDER_TYPE_ROOT) {
908 /* Compare the types, so that
909 * Remote accounts -> Local account -> MMC account .*/
910 const gint pos1 = get_cmp_rows_type_pos (folder1);
911 const gint pos2 = get_cmp_rows_type_pos (folder2);
912 /* printf ("DEBUG: %s:\n type1=%s, pos1=%d\n type2=%s, pos2=%d\n",
913 __FUNCTION__, G_OBJECT_TYPE_NAME(folder1), pos1, G_OBJECT_TYPE_NAME(folder2), pos2); */
916 else if (pos1 > pos2)
919 /* Compare items of the same type: */
921 TnyAccount *account1 = NULL;
922 if (TNY_IS_ACCOUNT (folder1))
923 account1 = TNY_ACCOUNT (folder1);
925 TnyAccount *account2 = NULL;
926 if (TNY_IS_ACCOUNT (folder2))
927 account2 = TNY_ACCOUNT (folder2);
929 const gchar *account_id = account1 ? tny_account_get_id (account1) : NULL;
930 const gchar *account_id2 = account2 ? tny_account_get_id (account2) : NULL;
932 if (!account_id && !account_id2)
934 else if (!account_id)
936 else if (!account_id2)
938 else if (!strcmp (account_id, MODEST_MMC_ACCOUNT_ID))
941 cmp = modest_text_utils_utf8_strcmp (name1, name2, TRUE);
944 cmp = modest_text_utils_utf8_strcmp (name1, name2, TRUE);
948 g_object_unref(G_OBJECT(folder1));
950 g_object_unref(G_OBJECT(folder2));
958 /*****************************************************************************/
959 /* DRAG and DROP stuff */
960 /*****************************************************************************/
963 * This function fills the #GtkSelectionData with the row and the
964 * model that has been dragged. It's called when this widget is a
965 * source for dnd after the event drop happened
968 on_drag_data_get (GtkWidget *widget,
969 GdkDragContext *context,
970 GtkSelectionData *selection_data,
975 GtkTreeSelection *selection;
978 GtkTreePath *source_row;
980 selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (widget));
981 gtk_tree_selection_get_selected (selection, &model, &iter);
982 source_row = gtk_tree_model_get_path (model, &iter);
984 gtk_tree_set_row_drag_data (selection_data,
988 gtk_tree_path_free (source_row);
991 typedef struct _DndHelper {
992 gboolean delete_source;
993 GtkTreePath *source_row;
994 GdkDragContext *context;
1000 * This function is the callback of the
1001 * modest_mail_operation_xfer_msgs () and
1002 * modest_mail_operation_xfer_folder() calls. We check here if the
1003 * message/folder was correctly asynchronously transferred. The reason
1004 * to use the same callback is that the code is the same, it only has
1005 * to check that the operation went fine and then finalize the drag
1009 on_progress_changed (ModestMailOperation *mail_op,
1010 ModestMailOperationState *state,
1016 helper = (DndHelper *) user_data;
1018 if (!state->finished)
1021 if (state->status == MODEST_MAIL_OPERATION_STATUS_SUCCESS) {
1027 /* Notify the drag source. Never call delete, the monitor will
1028 do the job if needed */
1029 gtk_drag_finish (helper->context, success, FALSE, helper->time);
1031 /* Free the helper */
1032 gtk_tree_path_free (helper->source_row);
1033 g_slice_free (DndHelper, helper);
1037 * This function is used by drag_data_received_cb to manage drag and
1038 * drop of a header, i.e, and drag from the header view to the folder
1042 drag_and_drop_from_header_view (GtkTreeModel *source_model,
1043 GtkTreeModel *dest_model,
1044 GtkTreePath *dest_row,
1050 ModestMailOperation *mail_op;
1051 GtkTreeIter source_iter, dest_iter;
1054 gtk_tree_model_get_iter (source_model, &source_iter, helper->source_row);
1055 gtk_tree_model_get (source_model, &source_iter,
1056 TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
1060 gtk_tree_model_get_iter (dest_model, &dest_iter, dest_row);
1061 gtk_tree_model_get (dest_model, &dest_iter,
1062 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN,
1065 /* Transfer message */
1066 mail_op = modest_mail_operation_new (MODEST_MAIL_OPERATION_TYPE_RECEIVE, NULL);
1068 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
1070 g_signal_connect (G_OBJECT (mail_op), "progress-changed",
1071 G_CALLBACK (on_progress_changed), helper);
1073 headers = tny_simple_list_new ();
1074 tny_list_append (headers, G_OBJECT (header));
1075 modest_mail_operation_xfer_msgs (mail_op,
1078 helper->delete_source,
1082 g_object_unref (G_OBJECT (mail_op));
1083 g_object_unref (G_OBJECT (header));
1084 g_object_unref (G_OBJECT (folder));
1085 g_object_unref (headers);
1089 * This function is used by drag_data_received_cb to manage drag and
1090 * drop of a folder, i.e, and drag from the folder view to the same
1094 drag_and_drop_from_folder_view (GtkTreeModel *source_model,
1095 GtkTreeModel *dest_model,
1096 GtkTreePath *dest_row,
1097 GtkSelectionData *selection_data,
1100 ModestMailOperation *mail_op;
1101 GtkTreeIter parent_iter, iter;
1102 TnyFolderStore *parent_folder;
1105 /* Check if the drag is possible */
1106 /* if (!gtk_tree_path_compare (helper->source_row, dest_row) || */
1107 /* !gtk_tree_drag_dest_row_drop_possible (GTK_TREE_DRAG_DEST (dest_model), */
1109 /* selection_data)) { */
1110 if (!gtk_tree_path_compare (helper->source_row, dest_row)) {
1112 gtk_drag_finish (helper->context, FALSE, FALSE, helper->time);
1113 gtk_tree_path_free (helper->source_row);
1114 g_slice_free (DndHelper, helper);
1119 gtk_tree_model_get_iter (source_model, &parent_iter, dest_row);
1120 gtk_tree_model_get_iter (source_model, &iter, helper->source_row);
1121 gtk_tree_model_get (source_model, &parent_iter,
1122 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN,
1123 &parent_folder, -1);
1124 gtk_tree_model_get (source_model, &iter,
1125 TNY_GTK_FOLDER_STORE_TREE_MODEL_INSTANCE_COLUMN,
1128 /* Do the mail operation */
1129 mail_op = modest_mail_operation_new_with_error_handling (MODEST_MAIL_OPERATION_TYPE_RECEIVE,
1131 modest_ui_actions_move_folder_error_handler,
1133 modest_mail_operation_queue_add (modest_runtime_get_mail_operation_queue (),
1135 g_signal_connect (G_OBJECT (mail_op), "progress-changed",
1136 G_CALLBACK (on_progress_changed), helper);
1138 modest_mail_operation_xfer_folder (mail_op,
1141 helper->delete_source);
1144 g_object_unref (G_OBJECT (parent_folder));
1145 g_object_unref (G_OBJECT (folder));
1146 g_object_unref (G_OBJECT (mail_op));
1150 * This function receives the data set by the "drag-data-get" signal
1151 * handler. This information comes within the #GtkSelectionData. This
1152 * function will manage both the drags of folders of the treeview and
1153 * drags of headers of the header view widget.
1156 on_drag_data_received (GtkWidget *widget,
1157 GdkDragContext *context,
1160 GtkSelectionData *selection_data,
1165 GtkWidget *source_widget;
1166 GtkTreeModel *dest_model, *source_model;
1167 GtkTreePath *source_row, *dest_row;
1168 GtkTreeViewDropPosition pos;
1169 gboolean success = FALSE, delete_source = FALSE;
1170 DndHelper *helper = NULL;
1172 /* Do not allow further process */
1173 g_signal_stop_emission_by_name (widget, "drag-data-received");
1174 source_widget = gtk_drag_get_source_widget (context);
1176 /* Get the action */
1177 if (context->action == GDK_ACTION_MOVE) {
1178 delete_source = TRUE;
1180 /* Notify that there is no folder selected. We need to
1181 do this in order to update the headers view (and
1182 its monitors, because when moving, the old folder
1183 won't longer exist. We can not wait for the end of
1184 the operation, because the operation won't start if
1185 the folder is in use */
1186 if (source_widget == widget)
1187 g_signal_emit (G_OBJECT (widget),
1188 signals[FOLDER_SELECTION_CHANGED_SIGNAL], 0, NULL, TRUE);
1191 /* Check if the get_data failed */
1192 if (selection_data == NULL || selection_data->length < 0)
1193 gtk_drag_finish (context, success, FALSE, time);
1195 /* Get the models */
1196 gtk_tree_get_row_drag_data (selection_data,
1200 /* Select the destination model */
1201 if (source_widget == widget) {
1202 dest_model = source_model;
1204 dest_model = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
1207 /* Get the path to the destination row. Can not call
1208 gtk_tree_view_get_drag_dest_row() because the source row
1209 is not selected anymore */
1210 gtk_tree_view_get_dest_row_at_pos (GTK_TREE_VIEW (widget), x, y,
1213 /* Only allow drops IN other rows */
1214 if (!dest_row || pos == GTK_TREE_VIEW_DROP_BEFORE || pos == GTK_TREE_VIEW_DROP_AFTER)
1215 gtk_drag_finish (context, success, FALSE, time);
1217 /* Create the helper */
1218 helper = g_slice_new0 (DndHelper);
1219 helper->delete_source = delete_source;
1220 helper->source_row = gtk_tree_path_copy (source_row);
1221 helper->context = context;
1222 helper->time = time;
1224 /* Drags from the header view */
1225 if (source_widget != widget) {
1227 drag_and_drop_from_header_view (source_model,
1234 drag_and_drop_from_folder_view (source_model,
1242 gtk_tree_path_free (source_row);
1243 gtk_tree_path_free (dest_row);
1247 * We define a "drag-drop" signal handler because we do not want to
1248 * use the default one, because the default one always calls
1249 * gtk_drag_finish and we prefer to do it in the "drag-data-received"
1250 * signal handler, because there we have all the information available
1251 * to know if the dnd was a success or not.
1254 drag_drop_cb (GtkWidget *widget,
1255 GdkDragContext *context,
1263 if (!context->targets)
1266 /* Check if we're dragging a folder row */
1267 target = gtk_drag_dest_find_target (widget, context, NULL);
1269 /* Request the data from the source. */
1270 gtk_drag_get_data(widget, context, target, time);
1276 * This function expands a node of a tree view if it's not expanded
1277 * yet. Not sure why it needs the threads stuff, but gtk+`example code
1278 * does that, so that's why they're here.
1281 expand_row_timeout (gpointer data)
1283 GtkTreeView *tree_view = data;
1284 GtkTreePath *dest_path = NULL;
1285 GtkTreeViewDropPosition pos;
1286 gboolean result = FALSE;
1288 GDK_THREADS_ENTER ();
1290 gtk_tree_view_get_drag_dest_row (tree_view,
1295 (pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER ||
1296 pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE)) {
1297 gtk_tree_view_expand_row (tree_view, dest_path, FALSE);
1298 gtk_tree_path_free (dest_path);
1302 gtk_tree_path_free (dest_path);
1307 GDK_THREADS_LEAVE ();
1313 * This function is called whenever the pointer is moved over a widget
1314 * while dragging some data. It installs a timeout that will expand a
1315 * node of the treeview if not expanded yet. This function also calls
1316 * gdk_drag_status in order to set the suggested action that will be
1317 * used by the "drag-data-received" signal handler to know if we
1318 * should do a move or just a copy of the data.
1321 on_drag_motion (GtkWidget *widget,
1322 GdkDragContext *context,
1328 GtkTreeViewDropPosition pos;
1329 GtkTreePath *dest_row;
1330 ModestFolderViewPrivate *priv;
1331 GdkDragAction suggested_action;
1332 gboolean valid_location = FALSE;
1334 priv = MODEST_FOLDER_VIEW_GET_PRIVATE (widget);
1336 if (priv->timer_expander != 0) {
1337 g_source_remove (priv->timer_expander);
1338 priv->timer_expander = 0;
1341 gtk_tree_view_get_dest_row_at_pos (GTK_TREE_VIEW (widget),
1346 /* Do not allow drops between folders */
1348 pos == GTK_TREE_VIEW_DROP_BEFORE ||
1349 pos == GTK_TREE_VIEW_DROP_AFTER) {
1350 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW (widget), NULL, 0);
1351 gdk_drag_status(context, 0, time);
1352 valid_location = FALSE;
1355 valid_location = TRUE;
1358 /* Expand the selected row after 1/2 second */
1359 if (!gtk_tree_view_row_expanded (GTK_TREE_VIEW (widget), dest_row)) {
1360 gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (widget), dest_row, pos);
1361 priv->timer_expander = g_timeout_add (500, expand_row_timeout, widget);
1364 /* Select the desired action. By default we pick MOVE */
1365 suggested_action = GDK_ACTION_MOVE;
1367 if (context->actions == GDK_ACTION_COPY)
1368 gdk_drag_status(context, GDK_ACTION_COPY, time);
1369 else if (context->actions == GDK_ACTION_MOVE)
1370 gdk_drag_status(context, GDK_ACTION_MOVE, time);
1371 else if (context->actions & suggested_action)
1372 gdk_drag_status(context, suggested_action, time);
1374 gdk_drag_status(context, GDK_ACTION_DEFAULT, time);
1378 gtk_tree_path_free (dest_row);
1379 g_signal_stop_emission_by_name (widget, "drag-motion");
1380 return valid_location;
1384 /* Folder view drag types */
1385 const GtkTargetEntry folder_view_drag_types[] =
1387 { "GTK_TREE_MODEL_ROW", GTK_TARGET_SAME_WIDGET, MODEST_FOLDER_ROW },
1388 { "GTK_TREE_MODEL_ROW", GTK_TARGET_SAME_APP, MODEST_HEADER_ROW }
1392 * This function sets the treeview as a source and a target for dnd
1393 * events. It also connects all the requirede signals.
1396 setup_drag_and_drop (GtkTreeView *self)
1398 /* Set up the folder view as a dnd destination. Set only the
1399 highlight flag, otherwise gtk will have a different
1401 gtk_drag_dest_set (GTK_WIDGET (self),
1402 GTK_DEST_DEFAULT_HIGHLIGHT,
1403 folder_view_drag_types,
1404 G_N_ELEMENTS (folder_view_drag_types),
1405 GDK_ACTION_MOVE | GDK_ACTION_COPY);
1407 g_signal_connect (G_OBJECT (self),
1408 "drag_data_received",
1409 G_CALLBACK (on_drag_data_received),
1413 /* Set up the treeview as a dnd source */
1414 gtk_drag_source_set (GTK_WIDGET (self),
1416 folder_view_drag_types,
1417 G_N_ELEMENTS (folder_view_drag_types),
1418 GDK_ACTION_MOVE | GDK_ACTION_COPY);
1420 g_signal_connect (G_OBJECT (self),
1422 G_CALLBACK (on_drag_motion),
1425 g_signal_connect (G_OBJECT (self),
1427 G_CALLBACK (on_drag_data_get),
1430 g_signal_connect (G_OBJECT (self),
1432 G_CALLBACK (drag_drop_cb),
1437 * This function manages the navigation through the folders using the
1438 * keyboard or the hardware keys in the device
1441 on_key_pressed (GtkWidget *self,
1445 GtkTreeSelection *selection;
1447 GtkTreeModel *model;
1448 gboolean retval = FALSE;
1450 /* Up and Down are automatically managed by the treeview */
1451 if (event->keyval == GDK_Return) {
1452 /* Expand/Collapse the selected row */
1453 selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (self));
1454 if (gtk_tree_selection_get_selected (selection, &model, &iter)) {
1457 path = gtk_tree_model_get_path (model, &iter);
1459 if (gtk_tree_view_row_expanded (GTK_TREE_VIEW (self), path))
1460 gtk_tree_view_collapse_row (GTK_TREE_VIEW (self), path);
1462 gtk_tree_view_expand_row (GTK_TREE_VIEW (self), path, FALSE);
1463 gtk_tree_path_free (path);
1465 /* No further processing */
1473 * We listen to the changes in the local folder account name key,
1474 * because we want to show the right name in the view. The local
1475 * folder account name corresponds to the device name in the Maemo
1476 * version. We do this because we do not want to query gconf on each
1477 * tree view refresh. It's better to cache it and change whenever
1481 on_configuration_key_changed (ModestConf* conf,
1483 ModestConfEvent event,
1484 ModestFolderView *self)
1486 ModestFolderViewPrivate *priv;
1491 g_return_if_fail (MODEST_IS_FOLDER_VIEW (self));
1492 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
1494 if (!strcmp (key, MODEST_CONF_DEVICE_NAME)) {
1495 g_free (priv->local_account_name);
1497 if (event == MODEST_CONF_EVENT_KEY_UNSET)
1498 priv->local_account_name = g_strdup (MODEST_LOCAL_FOLDERS_DEFAULT_DISPLAY_NAME);
1500 priv->local_account_name = modest_conf_get_string (modest_runtime_get_conf(),
1501 MODEST_CONF_DEVICE_NAME, NULL);
1503 /* Force a redraw */
1504 #if GTK_CHECK_VERSION(2, 8, 0) /* gtk_tree_view_column_queue_resize is only available in GTK+ 2.8 */
1505 GtkTreeViewColumn * tree_column = gtk_tree_view_get_column (GTK_TREE_VIEW (self),
1506 TNY_GTK_FOLDER_STORE_TREE_MODEL_NAME_COLUMN);
1507 gtk_tree_view_column_queue_resize (tree_column);
1513 modest_folder_view_set_style (ModestFolderView *self,
1514 ModestFolderViewStyle style)
1516 ModestFolderViewPrivate *priv;
1518 g_return_if_fail (self);
1520 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
1522 priv->style = style;
1526 modest_folder_view_set_account_id_of_visible_server_account (ModestFolderView *self,
1527 const gchar *account_id)
1529 ModestFolderViewPrivate *priv;
1530 GtkTreeModel *model;
1532 g_return_if_fail (self);
1534 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
1536 /* This will be used by the filter_row callback,
1537 * to decided which rows to show: */
1538 if (priv->visible_account_id)
1539 g_free (priv->visible_account_id);
1540 priv->visible_account_id = g_strdup (account_id);
1543 model = gtk_tree_view_get_model (GTK_TREE_VIEW (self));
1544 if (GTK_IS_TREE_MODEL_FILTER (model))
1545 gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (model));
1549 modest_folder_view_get_account_id_of_visible_server_account (ModestFolderView *self)
1551 ModestFolderViewPrivate *priv;
1553 g_return_val_if_fail (self, NULL);
1555 priv = MODEST_FOLDER_VIEW_GET_PRIVATE(self);
1557 return (const gchar *) priv->visible_account_id;
1561 find_inbox_iter (GtkTreeModel *model, GtkTreeIter *iter, GtkTreeIter *inbox_iter)
1567 gtk_tree_model_get (model, iter,
1568 TNY_GTK_FOLDER_STORE_TREE_MODEL_TYPE_COLUMN,
1571 if (type == TNY_FOLDER_TYPE_INBOX) {
1572 *inbox_iter = *iter;
1576 if (gtk_tree_model_iter_children (model, &child, iter)) {
1577 if (find_inbox_iter (model, &child, inbox_iter))
1581 } while (gtk_tree_model_iter_next (model, iter));
1589 modest_folder_view_select_first_inbox_or_local (ModestFolderView *self)
1591 GtkTreeModel *model;
1592 GtkTreeIter iter, inbox_iter;
1593 GtkTreeSelection *sel;
1595 model = gtk_tree_view_get_model (GTK_TREE_VIEW (self));
1599 expand_root_items (self);
1600 sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (self));
1602 gtk_tree_model_get_iter_first (model, &iter);
1603 if (find_inbox_iter (model, &iter, &inbox_iter))
1604 gtk_tree_selection_select_iter (sel, &inbox_iter);
1606 gtk_tree_model_get_iter_first (model, &iter);
1607 gtk_tree_selection_select_iter (sel, &iter);