aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--mail/ChangeLog7
-rw-r--r--mail/em-folder-tree.c164
2 files changed, 122 insertions, 49 deletions
diff --git a/mail/ChangeLog b/mail/ChangeLog
index 91613fcc5d..b8a952f7eb 100644
--- a/mail/ChangeLog
+++ b/mail/ChangeLog
@@ -1,3 +1,10 @@
+2003-11-18 Jeffrey Stedfast <fejj@ximian.com>
+
+ * em-folder-tree.c (tree_row_expanded): Queue the
+ camel_store_get_folder_info() call in another thread.
+ (em_folder_tree_get_folder_info__got): Moved all the logic of
+ tree_row_expanded here.
+
2003-11-17 Jeffrey Stedfast <fejj@ximian.com>
* em-folder-tree.c (tree_row_expanded): Fixed to check fi->child
diff --git a/mail/em-folder-tree.c b/mail/em-folder-tree.c
index 9aa51ce1fa..a6867272a7 100644
--- a/mail/em-folder-tree.c
+++ b/mail/em-folder-tree.c
@@ -355,6 +355,9 @@ em_folder_tree_destroy (GtkObject *obj)
priv->ddd = 0;
}
+ priv->treeview = NULL;
+ priv->model = NULL;
+
GTK_OBJECT_CLASS (parent_class)->destroy (obj);
}
@@ -379,8 +382,6 @@ folder_tree_new (EMFolderTreeModel *model)
renderer = gtk_cell_renderer_text_new ();
gtk_tree_view_column_pack_start (column, renderer, TRUE);
gtk_tree_view_column_set_cell_data_func (column, renderer, render_display_name, NULL, NULL);
- /*gtk_tree_view_insert_column_with_attributes ((GtkTreeView *) tree, -1, "",
- renderer, "text", 0, NULL);*/
selection = gtk_tree_view_get_selection ((GtkTreeView *) tree);
gtk_tree_selection_set_mode (selection, GTK_SELECTION_SINGLE);
@@ -869,7 +870,7 @@ tree_store_set_folder_info (GtkTreeStore *model, GtkTreeIter *iter,
GtkTreeIter sub;
gboolean load;
- load = (fi->flags & CAMEL_FOLDER_CHILDREN) && !(fi->flags & CAMEL_FOLDER_NOINFERIORS);
+ load = !fi->child && (fi->flags & CAMEL_FOLDER_CHILDREN) && !(fi->flags & CAMEL_FOLDER_NOINFERIORS);
path = gtk_tree_model_get_path ((GtkTreeModel *) model, iter);
uri_row = gtk_tree_row_reference_new ((GtkTreeModel *) model, path);
@@ -934,80 +935,145 @@ dump_fi (CamelFolderInfo *fi, int depth)
}
#endif
+struct _EMFolderTreeGetFolderInfo {
+ struct _mail_msg msg;
+
+ /* input data */
+ GtkTreeRowReference *root;
+ EMFolderTree *emft;
+ CamelStore *store;
+ char *top;
+
+ /* output data */
+ CamelFolderInfo *fi;
+};
+
static void
-tree_row_expanded (GtkTreeView *treeview, GtkTreeIter *root, GtkTreePath *tree_path, EMFolderTree *emft)
+em_folder_tree_get_folder_info__get (struct _mail_msg *mm)
{
- /* FIXME: might be best to call get_folder_info in another thread and add the nodes to the treeview in the callback? */
- struct _EMFolderTreePrivate *priv = emft->priv;
+ struct _EMFolderTreeGetFolderInfo *m = (struct _EMFolderTreeGetFolderInfo *) mm;
+ guint32 flags = 0;
+
+ if (camel_store_supports_subscriptions (m->store))
+ flags |= CAMEL_STORE_FOLDER_INFO_SUBSCRIBED;
+
+ m->fi = camel_store_get_folder_info (m->store, m->top, flags, &mm->ex);
+}
+
+static void
+em_folder_tree_get_folder_info__got (struct _mail_msg *mm)
+{
+ struct _EMFolderTreeGetFolderInfo *m = (struct _EMFolderTreeGetFolderInfo *) mm;
+ struct _EMFolderTreePrivate *priv = m->emft->priv;
struct _EMFolderTreeModelStoreInfo *si;
- CamelFolderInfo *fi, *child;
- CamelStore *store;
- CamelException ex;
+ GtkTreeIter root, iter;
+ CamelFolderInfo *fi;
GtkTreeStore *model;
- GtkTreeIter iter;
+ GtkTreePath *path;
gboolean load;
- char *path;
- char *top;
-
- model = (GtkTreeStore *) gtk_tree_view_get_model (treeview);
- gtk_tree_model_get ((GtkTreeModel *) model, root,
- COL_STRING_FOLDER_PATH, &path,
- COL_POINTER_CAMEL_STORE, &store,
- COL_BOOL_LOAD_SUBDIRS, &load,
- -1);
- if (!load)
+ /* check that we haven't been destroyed */
+ if (priv->treeview == NULL)
return;
- if (!(si = g_hash_table_lookup (priv->model->store_hash, store))) {
- g_assert_not_reached ();
+ if (!(si = g_hash_table_lookup (priv->model->store_hash, m->store))) {
+ /* store has been removed in the interim - do nothing */
return;
}
- /* get the first child (which will be a dummy if we haven't loaded the child folders yet) */
- gtk_tree_model_iter_children ((GtkTreeModel *) model, &iter, root);
+ model = (GtkTreeStore *) gtk_tree_view_get_model (priv->treeview);
- /* FIXME: this sucks ass, need to fix camel so that using path as @top will Just Work (tm) */
- /* NOTE: CamelImapStore will handle "" as toplevel, but CamelMboxStore wants NULL */
- if (!path || !strcmp (path, "/"))
- top = NULL;
- else
- top = path + 1;
+ path = gtk_tree_row_reference_get_path (m->root);
+ gtk_tree_model_get_iter ((GtkTreeModel *) model, &root, path);
+ gtk_tree_path_free (path);
- /* FIXME: are there any flags we want to pass when getting folder-info's? */
- camel_exception_init (&ex);
- if (!(fi = camel_store_get_folder_info (store, top, 0, &ex))) {
- /* FIXME: report error to user? or simply re-collapse node? or both? */
- g_warning ("can't get folder-info's for store '%s' at path='%s'", si->display_name, path);
- gtk_tree_store_remove (model, &iter);
- camel_exception_clear (&ex);
+ /* make sure we still need to load the tree subfolders... */
+ gtk_tree_model_get ((GtkTreeModel *) model, &root,
+ COL_BOOL_LOAD_SUBDIRS, &load,
+ -1);
+ if (!load)
return;
- }
+
+ /* get the first child (which will be a dummy node) */
+ gtk_tree_model_iter_children ((GtkTreeModel *) model, &iter, &root);
/* FIXME: camel is totally on crack here, @top's folder info
* should be @fi and fi->child should be what we want to fill
* our tree with... *sigh* */
- if (top && !strcmp (fi->full_name, top)) {
- if (!(child = fi->child))
- child = fi->sibling;
+ if (m->top && !strcmp (m->fi->full_name, m->top)) {
+ if (!(fi = m->fi->child))
+ fi = m->fi->sibling;
} else
- child = fi;
+ fi = m->fi;
- if (child == NULL) {
+ if (fi == NULL) {
/* no children afterall... remove the "Loading..." placeholder node */
gtk_tree_store_remove (model, &iter);
} else {
do {
- tree_store_set_folder_info (model, &iter, priv, si, child);
+ tree_store_set_folder_info (model, &iter, priv, si, fi);
- if ((child = child->sibling) != NULL)
- gtk_tree_store_append (model, &iter, root);
- } while (child != NULL);
+ if ((fi = fi->sibling) != NULL)
+ gtk_tree_store_append (model, &iter, &root);
+ } while (fi != NULL);
}
- gtk_tree_store_set (model, root, COL_BOOL_LOAD_SUBDIRS, FALSE, -1);
+ gtk_tree_store_set (model, &root, COL_BOOL_LOAD_SUBDIRS, FALSE, -1);
+}
+
+static void
+em_folder_tree_get_folder_info__free (struct _mail_msg *mm)
+{
+ struct _EMFolderTreeGetFolderInfo *m = (struct _EMFolderTreeGetFolderInfo *) mm;
- camel_store_free_folder_info (store, fi);
+ camel_store_free_folder_info (m->store, m->fi);
+
+ gtk_tree_row_reference_free (m->root);
+ /*g_object_unref (m->emft);*/
+ camel_object_unref (m->store);
+ g_free (m->top);
+}
+
+static struct _mail_msg_op get_folder_info_op = {
+ NULL,
+ em_folder_tree_get_folder_info__get,
+ em_folder_tree_get_folder_info__got,
+ em_folder_tree_get_folder_info__free,
+};
+
+static void
+tree_row_expanded (GtkTreeView *treeview, GtkTreeIter *root, GtkTreePath *tree_path, EMFolderTree *emft)
+{
+ struct _EMFolderTreeGetFolderInfo *m;
+ GtkTreeModel *model;
+ CamelStore *store;
+ const char *top;
+ gboolean load;
+ char *path;
+
+ model = gtk_tree_view_get_model (treeview);
+
+ gtk_tree_model_get (model, root,
+ COL_STRING_FOLDER_PATH, &path,
+ COL_POINTER_CAMEL_STORE, &store,
+ COL_BOOL_LOAD_SUBDIRS, &load,
+ -1);
+ if (!load)
+ return;
+
+ if (!path || !strcmp (path, "/"))
+ top = NULL;
+ else
+ top = path + 1;
+
+ m = mail_msg_new (&get_folder_info_op, NULL, sizeof (struct _EMFolderTreeGetFolderInfo));
+ m->root = gtk_tree_row_reference_new (model, tree_path);
+ camel_object_ref (store);
+ m->store = store;
+ m->emft = emft;
+ m->top = g_strdup (top);
+
+ e_thread_put (mail_thread_new, (EMsg *) m);
}