1253 lines
34 KiB
C
1253 lines
34 KiB
C
/*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) version 3.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with the program; if not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*
|
|
* Authors:
|
|
* Jeffrey Stedfast <fejj@ximian.com>
|
|
*
|
|
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
|
|
*
|
|
*/
|
|
|
|
#include "em-folder-tree-model.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <sys/stat.h>
|
|
|
|
#include "e-util/e-util.h"
|
|
#include "e-util/e-account-utils.h"
|
|
|
|
#include <glib/gi18n.h>
|
|
|
|
#include <camel/camel-file-utils.h>
|
|
|
|
#include "mail-config.h"
|
|
#include "mail-session.h"
|
|
#include "mail-tools.h"
|
|
#include "mail-mt.h"
|
|
|
|
/* sigh, these 2 only needed for outbox total count checking - a mess */
|
|
#include "mail-folder-cache.h"
|
|
|
|
#include "em-utils.h"
|
|
#include "em-folder-utils.h"
|
|
#include "em-event.h"
|
|
|
|
#include <camel/camel-folder.h>
|
|
#include <camel/camel-vee-store.h>
|
|
|
|
#include "e-mail-local.h"
|
|
|
|
#define d(x)
|
|
|
|
#define EM_FOLDER_TREE_MODEL_GET_PRIVATE(obj) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE \
|
|
((obj), EM_TYPE_FOLDER_TREE_MODEL, EMFolderTreeModelPrivate))
|
|
|
|
struct _EMFolderTreeModelPrivate {
|
|
/* This is set by EMailShellSidebar. It allows new EMFolderTree
|
|
* instances to initialize their selection and expanded states to
|
|
* mimic the sidebar. */
|
|
GtkTreeSelection *selection; /* weak reference */
|
|
|
|
EAccountList *accounts;
|
|
|
|
/* EAccount -> EMFolderTreeStoreInfo */
|
|
GHashTable *account_index;
|
|
|
|
/* CamelStore -> EMFolderTreeStoreInfo */
|
|
GHashTable *store_index;
|
|
|
|
/* URI -> GtkTreeRowReference */
|
|
GHashTable *uri_index;
|
|
|
|
gulong account_changed_id;
|
|
gulong account_removed_id;
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_SELECTION
|
|
};
|
|
|
|
enum {
|
|
LOADING_ROW,
|
|
LOADED_ROW,
|
|
FOLDER_ADDED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
extern CamelStore *vfolder_store;
|
|
|
|
static gpointer parent_class;
|
|
static guint signals[LAST_SIGNAL];
|
|
|
|
static void
|
|
store_info_free (EMFolderTreeModelStoreInfo *si)
|
|
{
|
|
camel_object_remove_event (si->store, si->created_id);
|
|
camel_object_remove_event (si->store, si->deleted_id);
|
|
camel_object_remove_event (si->store, si->renamed_id);
|
|
camel_object_remove_event (si->store, si->subscribed_id);
|
|
camel_object_remove_event (si->store, si->unsubscribed_id);
|
|
|
|
g_free (si->display_name);
|
|
camel_object_unref (si->store);
|
|
gtk_tree_row_reference_free (si->row);
|
|
g_hash_table_destroy (si->full_hash);
|
|
g_free (si);
|
|
}
|
|
|
|
static gint
|
|
folder_tree_model_sort (GtkTreeModel *model,
|
|
GtkTreeIter *a,
|
|
GtkTreeIter *b,
|
|
gpointer user_data)
|
|
{
|
|
gchar *aname, *bname;
|
|
CamelStore *store;
|
|
gboolean is_store;
|
|
guint32 aflags, bflags;
|
|
gint rv = -2;
|
|
|
|
gtk_tree_model_get (
|
|
model, a,
|
|
COL_BOOL_IS_STORE, &is_store,
|
|
COL_POINTER_CAMEL_STORE, &store,
|
|
COL_STRING_DISPLAY_NAME, &aname,
|
|
COL_UINT_FLAGS, &aflags, -1);
|
|
|
|
gtk_tree_model_get (
|
|
model, b,
|
|
COL_STRING_DISPLAY_NAME, &bname,
|
|
COL_UINT_FLAGS, &bflags, -1);
|
|
|
|
if (is_store) {
|
|
/* On This Computer is always first, and Search Folders
|
|
* is always last. */
|
|
if (!strcmp (aname, _("On This Computer")))
|
|
rv = -1;
|
|
else if (!strcmp (bname, _("On This Computer")))
|
|
rv = 1;
|
|
else if (!strcmp (aname, _("Search Folders")))
|
|
rv = 1;
|
|
else if (!strcmp (bname, _("Search Folders")))
|
|
rv = -1;
|
|
} else if (store == vfolder_store) {
|
|
/* UNMATCHED is always last. */
|
|
if (aname && !strcmp (aname, _("UNMATCHED")))
|
|
rv = 1;
|
|
else if (bname && !strcmp (bname, _("UNMATCHED")))
|
|
rv = -1;
|
|
} else {
|
|
/* Inbox is always first. */
|
|
if ((aflags & CAMEL_FOLDER_TYPE_MASK) == CAMEL_FOLDER_TYPE_INBOX)
|
|
rv = -1;
|
|
else if ((bflags & CAMEL_FOLDER_TYPE_MASK) == CAMEL_FOLDER_TYPE_INBOX)
|
|
rv = 1;
|
|
}
|
|
|
|
if (aname == NULL) {
|
|
if (bname == NULL)
|
|
rv = 0;
|
|
} else if (bname == NULL)
|
|
rv = 1;
|
|
|
|
if (rv == -2)
|
|
rv = g_utf8_collate (aname, bname);
|
|
|
|
g_free (aname);
|
|
g_free (bname);
|
|
|
|
return rv;
|
|
}
|
|
|
|
static void
|
|
account_changed_cb (EAccountList *accounts,
|
|
EAccount *account,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
CamelProvider *provider;
|
|
CamelStore *store;
|
|
CamelException ex;
|
|
gchar *uri;
|
|
|
|
si = g_hash_table_lookup (model->priv->account_index, account);
|
|
if (si == NULL)
|
|
return;
|
|
|
|
em_folder_tree_model_remove_store (model, si->store);
|
|
|
|
/* check if store needs to be added at all*/
|
|
if (!account->enabled ||!(uri = account->source->url))
|
|
return;
|
|
|
|
camel_exception_init (&ex);
|
|
if (!(provider = camel_provider_get(uri, &ex))) {
|
|
camel_exception_clear (&ex);
|
|
return;
|
|
}
|
|
|
|
/* make sure the new store belongs in the tree */
|
|
if (!(provider->flags & CAMEL_PROVIDER_IS_STORAGE))
|
|
return;
|
|
|
|
if (!(store = (CamelStore *) camel_session_get_service (session, uri, CAMEL_PROVIDER_STORE, &ex))) {
|
|
camel_exception_clear (&ex);
|
|
return;
|
|
}
|
|
|
|
em_folder_tree_model_add_store (model, store, account->name);
|
|
camel_object_unref (store);
|
|
}
|
|
|
|
static void
|
|
account_removed_cb (EAccountList *accounts,
|
|
EAccount *account,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
|
|
si = g_hash_table_lookup (model->priv->account_index, account);
|
|
if (si == NULL)
|
|
return;
|
|
|
|
em_folder_tree_model_remove_store (model, si->store);
|
|
}
|
|
|
|
static void
|
|
folder_tree_model_selection_finalized_cb (EMFolderTreeModel *model)
|
|
{
|
|
model->priv->selection = NULL;
|
|
|
|
g_object_notify (G_OBJECT (model), "selection");
|
|
}
|
|
|
|
static void
|
|
folder_tree_model_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_SELECTION:
|
|
em_folder_tree_model_set_selection (
|
|
EM_FOLDER_TREE_MODEL (object),
|
|
g_value_get_object (value));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
folder_tree_model_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_SELECTION:
|
|
g_value_set_object (
|
|
value,
|
|
em_folder_tree_model_get_selection (
|
|
EM_FOLDER_TREE_MODEL (object)));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
folder_tree_model_dispose (GObject *object)
|
|
{
|
|
EMFolderTreeModelPrivate *priv;
|
|
|
|
priv = EM_FOLDER_TREE_MODEL_GET_PRIVATE (object);
|
|
|
|
if (priv->selection != NULL) {
|
|
g_object_weak_unref (
|
|
G_OBJECT (priv->selection), (GWeakNotify)
|
|
folder_tree_model_selection_finalized_cb, object);
|
|
priv->selection = NULL;
|
|
}
|
|
|
|
/* Chain up to parent's dispose() method. */
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
folder_tree_model_finalize (GObject *object)
|
|
{
|
|
EMFolderTreeModelPrivate *priv;
|
|
|
|
priv = EM_FOLDER_TREE_MODEL_GET_PRIVATE (object);
|
|
|
|
g_hash_table_destroy (priv->account_index);
|
|
g_hash_table_destroy (priv->store_index);
|
|
g_hash_table_destroy (priv->uri_index);
|
|
|
|
g_signal_handler_disconnect (
|
|
priv->accounts, priv->account_changed_id);
|
|
g_signal_handler_disconnect (
|
|
priv->accounts, priv->account_removed_id);
|
|
|
|
/* Chain up to parent's finalize() method. */
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
folder_tree_model_class_init (EMFolderTreeModelClass *class)
|
|
{
|
|
GObjectClass *object_class;
|
|
|
|
parent_class = g_type_class_peek_parent (class);
|
|
g_type_class_add_private (class, sizeof (EMFolderTreeModelPrivate));
|
|
|
|
object_class = G_OBJECT_CLASS (class);
|
|
object_class->set_property = folder_tree_model_set_property;
|
|
object_class->get_property = folder_tree_model_get_property;
|
|
object_class->dispose = folder_tree_model_dispose;
|
|
object_class->finalize = folder_tree_model_finalize;
|
|
|
|
g_object_class_install_property (
|
|
object_class,
|
|
PROP_SELECTION,
|
|
g_param_spec_object (
|
|
"selection",
|
|
"Selection",
|
|
NULL,
|
|
GTK_TYPE_TREE_SELECTION,
|
|
G_PARAM_READWRITE));
|
|
|
|
signals[LOADING_ROW] = g_signal_new (
|
|
"loading-row",
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (EMFolderTreeModelClass, loading_row),
|
|
NULL, NULL,
|
|
e_marshal_VOID__POINTER_POINTER,
|
|
G_TYPE_NONE, 2,
|
|
G_TYPE_POINTER,
|
|
G_TYPE_POINTER);
|
|
|
|
signals[LOADED_ROW] = g_signal_new (
|
|
"loaded-row",
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (EMFolderTreeModelClass, loaded_row),
|
|
NULL, NULL,
|
|
e_marshal_VOID__POINTER_POINTER,
|
|
G_TYPE_NONE, 2,
|
|
G_TYPE_POINTER,
|
|
G_TYPE_POINTER);
|
|
|
|
signals[FOLDER_ADDED] = g_signal_new (
|
|
"folder-added",
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (EMFolderTreeModelClass, folder_added),
|
|
NULL, NULL,
|
|
e_marshal_VOID__STRING_STRING,
|
|
G_TYPE_NONE, 2,
|
|
G_TYPE_STRING,
|
|
G_TYPE_STRING);
|
|
}
|
|
|
|
static void
|
|
folder_tree_model_init (EMFolderTreeModel *model)
|
|
{
|
|
GHashTable *store_index;
|
|
GHashTable *uri_index;
|
|
|
|
GType col_types[] = {
|
|
G_TYPE_STRING, /* display name */
|
|
G_TYPE_POINTER, /* store object */
|
|
G_TYPE_STRING, /* full name */
|
|
G_TYPE_STRING, /* icon name */
|
|
G_TYPE_STRING, /* uri */
|
|
G_TYPE_UINT, /* unread count */
|
|
G_TYPE_UINT, /* flags */
|
|
G_TYPE_BOOLEAN, /* is a store node */
|
|
G_TYPE_BOOLEAN, /* is a folder node */
|
|
G_TYPE_BOOLEAN, /* has not-yet-loaded subfolders */
|
|
G_TYPE_UINT, /* last known unread count */
|
|
G_TYPE_BOOLEAN /* folder is a draft folder */
|
|
};
|
|
|
|
store_index = g_hash_table_new_full (
|
|
g_direct_hash, g_direct_equal,
|
|
(GDestroyNotify) NULL,
|
|
(GDestroyNotify) store_info_free);
|
|
|
|
uri_index = g_hash_table_new_full (
|
|
g_str_hash, g_str_equal,
|
|
(GDestroyNotify) g_free,
|
|
(GDestroyNotify) gtk_tree_row_reference_free);
|
|
|
|
model->priv = EM_FOLDER_TREE_MODEL_GET_PRIVATE (model);
|
|
model->priv->store_index = store_index;
|
|
model->priv->uri_index = uri_index;
|
|
|
|
gtk_tree_store_set_column_types (
|
|
GTK_TREE_STORE (model), NUM_COLUMNS, col_types);
|
|
gtk_tree_sortable_set_default_sort_func (
|
|
GTK_TREE_SORTABLE (model),
|
|
folder_tree_model_sort, NULL, NULL);
|
|
gtk_tree_sortable_set_sort_column_id (
|
|
GTK_TREE_SORTABLE (model),
|
|
GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID,
|
|
GTK_SORT_ASCENDING);
|
|
|
|
model->priv->accounts = e_get_account_list ();
|
|
model->priv->account_index =
|
|
g_hash_table_new (g_direct_hash, g_direct_equal);
|
|
model->priv->account_changed_id = g_signal_connect (
|
|
model->priv->accounts, "account-changed",
|
|
G_CALLBACK (account_changed_cb), model);
|
|
model->priv->account_removed_id = g_signal_connect (
|
|
model->priv->accounts, "account-removed",
|
|
G_CALLBACK (account_removed_cb), model);
|
|
}
|
|
|
|
GType
|
|
em_folder_tree_model_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
if (G_UNLIKELY (type == 0)) {
|
|
static const GTypeInfo type_info = {
|
|
sizeof (EMFolderTreeModelClass),
|
|
(GBaseInitFunc) NULL,
|
|
(GBaseFinalizeFunc) NULL,
|
|
(GClassInitFunc) folder_tree_model_class_init,
|
|
(GClassFinalizeFunc) NULL,
|
|
NULL, /* class_data */
|
|
sizeof (EMFolderTreeModel),
|
|
0, /* n_preallocs */
|
|
(GInstanceInitFunc) folder_tree_model_init,
|
|
NULL /* value_table */
|
|
};
|
|
|
|
type = g_type_register_static (
|
|
GTK_TYPE_TREE_STORE, "EMFolderTreeModel",
|
|
&type_info, 0);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
EMFolderTreeModel *
|
|
em_folder_tree_model_new (void)
|
|
{
|
|
return g_object_new (EM_TYPE_FOLDER_TREE_MODEL, NULL);
|
|
}
|
|
|
|
EMFolderTreeModel *
|
|
em_folder_tree_model_get_default (void)
|
|
{
|
|
static EMFolderTreeModel *default_folder_tree_model;
|
|
|
|
if (G_UNLIKELY (default_folder_tree_model == NULL))
|
|
default_folder_tree_model = em_folder_tree_model_new ();
|
|
|
|
return default_folder_tree_model;
|
|
}
|
|
|
|
GtkTreeSelection *
|
|
em_folder_tree_model_get_selection (EMFolderTreeModel *model)
|
|
{
|
|
g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), NULL);
|
|
|
|
return GTK_TREE_SELECTION (model->priv->selection);
|
|
}
|
|
|
|
void
|
|
em_folder_tree_model_set_selection (EMFolderTreeModel *model,
|
|
GtkTreeSelection *selection)
|
|
{
|
|
g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
|
|
|
|
if (selection != NULL)
|
|
g_return_if_fail (GTK_IS_TREE_SELECTION (selection));
|
|
|
|
if (model->priv->selection != NULL) {
|
|
g_object_weak_unref (
|
|
G_OBJECT (model->priv->selection), (GWeakNotify)
|
|
folder_tree_model_selection_finalized_cb, model);
|
|
model->priv->selection = NULL;
|
|
}
|
|
|
|
model->priv->selection = selection;
|
|
|
|
if (model->priv->selection != NULL)
|
|
g_object_weak_ref (
|
|
G_OBJECT (model->priv->selection), (GWeakNotify)
|
|
folder_tree_model_selection_finalized_cb, model);
|
|
|
|
g_object_notify (G_OBJECT (model), "selection");
|
|
}
|
|
|
|
void
|
|
em_folder_tree_model_set_folder_info (EMFolderTreeModel *model,
|
|
GtkTreeIter *iter,
|
|
EMFolderTreeModelStoreInfo *si,
|
|
CamelFolderInfo *fi,
|
|
gint fully_loaded)
|
|
{
|
|
GtkTreeRowReference *uri_row, *path_row;
|
|
GtkTreeStore *tree_store;
|
|
guint unread;
|
|
GtkTreePath *path;
|
|
GtkTreeIter sub;
|
|
gboolean load = FALSE;
|
|
gboolean is_drafts = FALSE;
|
|
gboolean is_templates = FALSE;
|
|
CamelFolder *folder;
|
|
gboolean emitted = FALSE;
|
|
const gchar *name;
|
|
const gchar *icon_name;
|
|
guint32 flags, add_flags = 0;
|
|
EMEventTargetCustomIcon *target;
|
|
|
|
/* make sure we don't already know about it? */
|
|
if (g_hash_table_lookup (si->full_hash, fi->full_name))
|
|
return;
|
|
|
|
tree_store = GTK_TREE_STORE (model);
|
|
|
|
if (!fully_loaded)
|
|
load = fi->child == NULL && !(fi->flags & (CAMEL_FOLDER_NOCHILDREN | CAMEL_FOLDER_NOINFERIORS));
|
|
|
|
path = gtk_tree_model_get_path (GTK_TREE_MODEL (model), iter);
|
|
uri_row = gtk_tree_row_reference_new (GTK_TREE_MODEL (model), path);
|
|
path_row = gtk_tree_row_reference_copy (uri_row);
|
|
gtk_tree_path_free (path);
|
|
|
|
g_hash_table_insert (
|
|
model->priv->uri_index, g_strdup (fi->uri), uri_row);
|
|
g_hash_table_insert (
|
|
si->full_hash, g_strdup (fi->full_name), path_row);
|
|
|
|
/* HACK: if we have the folder, and its the outbox folder, we need the total count, not unread */
|
|
/* HACK2: We do the same to the draft folder */
|
|
/* This is duplicated in mail-folder-cache too, should perhaps be functionised */
|
|
unread = fi->unread;
|
|
if (mail_note_get_folder_from_uri(fi->uri, &folder) && folder) {
|
|
is_drafts = em_utils_folder_is_drafts (folder, fi->uri);
|
|
|
|
if (is_drafts || em_utils_folder_is_outbox (folder, fi->uri)) {
|
|
gint total;
|
|
|
|
if ((total = camel_folder_get_message_count (folder)) > 0) {
|
|
gint deleted = camel_folder_get_deleted_message_count (folder);
|
|
|
|
if (deleted != -1)
|
|
total -= deleted;
|
|
}
|
|
|
|
unread = total > 0 ? total : 0;
|
|
}
|
|
|
|
camel_object_unref(folder);
|
|
}
|
|
|
|
/* TODO: maybe this should be handled by mail_get_folderinfo (except em-folder-tree doesn't use it, duh) */
|
|
flags = fi->flags;
|
|
name = fi->name;
|
|
if (si->store == e_mail_local_get_store ()) {
|
|
if (!strcmp(fi->full_name, "Drafts")) {
|
|
name = _("Drafts");
|
|
is_drafts = TRUE;
|
|
} else if (!strcmp(fi->full_name, "Templates")) {
|
|
name = _("Templates");
|
|
is_templates = TRUE;
|
|
} else if (!strcmp(fi->full_name, "Inbox")) {
|
|
flags = (flags & ~CAMEL_FOLDER_TYPE_MASK) | CAMEL_FOLDER_TYPE_INBOX;
|
|
name = _("Inbox");
|
|
} else if (!strcmp(fi->full_name, "Outbox")) {
|
|
flags = (flags & ~CAMEL_FOLDER_TYPE_MASK) | CAMEL_FOLDER_TYPE_OUTBOX;
|
|
name = _("Outbox");
|
|
} else if (!strcmp(fi->full_name, "Sent")) {
|
|
name = _("Sent");
|
|
flags = (flags & ~CAMEL_FOLDER_TYPE_MASK) | CAMEL_FOLDER_TYPE_SENT;
|
|
}
|
|
}
|
|
|
|
if (si->account && (flags & CAMEL_FOLDER_TYPE_MASK) == 0) {
|
|
if (!is_drafts && si->account->drafts_folder_uri) {
|
|
gchar *curi = em_uri_to_camel (si->account->drafts_folder_uri);
|
|
is_drafts = camel_store_folder_uri_equal (si->store, fi->uri, curi);
|
|
g_free (curi);
|
|
}
|
|
|
|
if (si->account->sent_folder_uri) {
|
|
gchar *curi = em_uri_to_camel (si->account->sent_folder_uri);
|
|
if (camel_store_folder_uri_equal (si->store, fi->uri, curi)) {
|
|
add_flags = CAMEL_FOLDER_TYPE_SENT;
|
|
}
|
|
g_free(curi);
|
|
}
|
|
}
|
|
|
|
/* Choose an icon name for the folder. */
|
|
icon_name = em_folder_utils_get_icon_name (flags | add_flags);
|
|
|
|
if (g_str_equal (icon_name, "folder")) {
|
|
if (is_drafts)
|
|
icon_name = "accessories-text-editor";
|
|
else if (is_templates)
|
|
icon_name = "text-x-generic-template";
|
|
}
|
|
|
|
gtk_tree_store_set (
|
|
tree_store, iter,
|
|
COL_STRING_DISPLAY_NAME, name,
|
|
COL_POINTER_CAMEL_STORE, si->store,
|
|
COL_STRING_FULL_NAME, fi->full_name,
|
|
COL_STRING_ICON_NAME, icon_name,
|
|
COL_STRING_URI, fi->uri,
|
|
COL_UINT_FLAGS, flags,
|
|
COL_BOOL_IS_STORE, FALSE,
|
|
COL_BOOL_IS_FOLDER, TRUE,
|
|
COL_BOOL_LOAD_SUBDIRS, load,
|
|
COL_UINT_UNREAD_LAST_SEL, 0,
|
|
COL_BOOL_IS_DRAFT, is_drafts,
|
|
-1);
|
|
|
|
target = em_event_target_new_custom_icon (em_event_peek(), tree_store, iter, fi->full_name, EM_EVENT_CUSTOM_ICON);
|
|
e_event_emit ((EEvent *)em_event_peek (), "folder.customicon", (EEventTarget *) target);
|
|
|
|
if (unread != ~0)
|
|
gtk_tree_store_set (
|
|
tree_store, iter, COL_UINT_UNREAD, unread,
|
|
COL_UINT_UNREAD_LAST_SEL, unread, -1);
|
|
|
|
if (load) {
|
|
/* create a placeholder node for our subfolders... */
|
|
gtk_tree_store_append (tree_store, &sub, iter);
|
|
gtk_tree_store_set (
|
|
tree_store, &sub,
|
|
COL_STRING_DISPLAY_NAME, _("Loading..."),
|
|
COL_POINTER_CAMEL_STORE, NULL,
|
|
COL_STRING_FULL_NAME, NULL,
|
|
COL_STRING_ICON_NAME, NULL,
|
|
COL_BOOL_LOAD_SUBDIRS, FALSE,
|
|
COL_BOOL_IS_STORE, FALSE,
|
|
COL_BOOL_IS_FOLDER, FALSE,
|
|
COL_STRING_URI, NULL,
|
|
COL_UINT_UNREAD, 0,
|
|
COL_UINT_UNREAD_LAST_SEL, 0,
|
|
COL_BOOL_IS_DRAFT, FALSE,
|
|
-1);
|
|
|
|
path = gtk_tree_model_get_path (GTK_TREE_MODEL (model), iter);
|
|
g_signal_emit (model, signals[LOADING_ROW], 0, path, iter);
|
|
gtk_tree_path_free (path);
|
|
return;
|
|
}
|
|
|
|
if (fi->child) {
|
|
fi = fi->child;
|
|
|
|
do {
|
|
gtk_tree_store_append (tree_store, &sub, iter);
|
|
|
|
if (!emitted) {
|
|
path = gtk_tree_model_get_path (GTK_TREE_MODEL (model), iter);
|
|
g_signal_emit (model, signals[LOADED_ROW], 0, path, iter);
|
|
gtk_tree_path_free (path);
|
|
emitted = TRUE;
|
|
}
|
|
|
|
em_folder_tree_model_set_folder_info (model, &sub, si, fi, fully_loaded);
|
|
fi = fi->next;
|
|
} while (fi);
|
|
}
|
|
|
|
if (!emitted) {
|
|
path = gtk_tree_model_get_path (GTK_TREE_MODEL (model), iter);
|
|
g_signal_emit (model, signals[LOADED_ROW], 0, path, iter);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
}
|
|
|
|
static void
|
|
folder_subscribed (CamelStore *store,
|
|
CamelFolderInfo *fi,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreeRowReference *reference;
|
|
GtkTreeIter parent, iter;
|
|
GtkTreePath *path;
|
|
gboolean load;
|
|
gchar *dirname, *p;
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
goto done;
|
|
|
|
/* make sure we don't already know about it? */
|
|
if (g_hash_table_lookup (si->full_hash, fi->full_name))
|
|
goto done;
|
|
|
|
/* get our parent folder's path */
|
|
dirname = alloca(strlen(fi->full_name)+1);
|
|
strcpy(dirname, fi->full_name);
|
|
p = strrchr(dirname, '/');
|
|
if (p == NULL) {
|
|
/* user subscribed to a toplevel folder */
|
|
reference = si->row;
|
|
} else {
|
|
*p = 0;
|
|
reference = g_hash_table_lookup (si->full_hash, dirname);
|
|
}
|
|
|
|
if (!gtk_tree_row_reference_valid (reference))
|
|
goto done;
|
|
|
|
path = gtk_tree_row_reference_get_path (reference);
|
|
gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &parent, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
/* make sure parent's subfolders have already been loaded */
|
|
gtk_tree_model_get (
|
|
GTK_TREE_MODEL (model), &parent,
|
|
COL_BOOL_LOAD_SUBDIRS, &load, -1);
|
|
if (load)
|
|
goto done;
|
|
|
|
/* append a new node */
|
|
gtk_tree_store_append (GTK_TREE_STORE (model), &iter, &parent);
|
|
|
|
em_folder_tree_model_set_folder_info (model, &iter, si, fi, TRUE);
|
|
|
|
g_signal_emit (model, signals[FOLDER_ADDED], 0, fi->full_name, fi->uri);
|
|
|
|
done:
|
|
camel_object_unref (store);
|
|
camel_folder_info_free (fi);
|
|
}
|
|
|
|
static void
|
|
folder_subscribed_cb (CamelStore *store,
|
|
gpointer event_data,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
CamelFolderInfo *fi;
|
|
|
|
camel_object_ref (store);
|
|
fi = camel_folder_info_clone (event_data);
|
|
|
|
mail_async_event_emit (
|
|
mail_async_event, MAIL_ASYNC_GUI, (MailAsyncFunc)
|
|
folder_subscribed, store, fi, model);
|
|
}
|
|
|
|
static void
|
|
folder_unsubscribed (CamelStore *store,
|
|
CamelFolderInfo *fi,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreeRowReference *reference;
|
|
GtkTreePath *path;
|
|
GtkTreeIter iter;
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
goto done;
|
|
|
|
reference = g_hash_table_lookup (si->full_hash, fi->full_name);
|
|
if (!gtk_tree_row_reference_valid (reference))
|
|
goto done;
|
|
|
|
path = gtk_tree_row_reference_get_path (reference);
|
|
gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &iter, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
em_folder_tree_model_remove_folders (model, si, &iter);
|
|
|
|
done:
|
|
camel_object_unref (store);
|
|
camel_folder_info_free (fi);
|
|
}
|
|
|
|
static void
|
|
folder_unsubscribed_cb (CamelStore *store,
|
|
gpointer event_data,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
CamelFolderInfo *fi;
|
|
|
|
camel_object_ref (store);
|
|
fi = camel_folder_info_clone (event_data);
|
|
|
|
mail_async_event_emit (
|
|
mail_async_event, MAIL_ASYNC_GUI, (MailAsyncFunc)
|
|
folder_unsubscribed, store, fi, model);
|
|
}
|
|
|
|
static void
|
|
folder_created_cb (CamelStore *store,
|
|
gpointer event_data,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
CamelFolderInfo *fi;
|
|
|
|
/* We only want created events to do more
|
|
* work if we don't support subscriptions. */
|
|
if (camel_store_supports_subscriptions (store))
|
|
return;
|
|
|
|
camel_object_ref (store);
|
|
fi = camel_folder_info_clone (event_data);
|
|
|
|
mail_async_event_emit (
|
|
mail_async_event, MAIL_ASYNC_GUI, (MailAsyncFunc)
|
|
folder_subscribed, store, fi, model);
|
|
}
|
|
|
|
static void
|
|
folder_deleted_cb (CamelStore *store,
|
|
gpointer event_data,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
CamelFolderInfo *fi;
|
|
|
|
/* We only want deleted events to do more
|
|
* work if we don't support subscriptions. */
|
|
if (camel_store_supports_subscriptions (store))
|
|
return;
|
|
|
|
camel_object_ref (store);
|
|
fi = camel_folder_info_clone (event_data);
|
|
|
|
mail_async_event_emit (
|
|
mail_async_event, MAIL_ASYNC_GUI, (MailAsyncFunc)
|
|
folder_unsubscribed_cb, store, fi, model);
|
|
}
|
|
|
|
static void
|
|
folder_renamed (CamelStore *store,
|
|
CamelRenameInfo *info,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreeRowReference *reference;
|
|
GtkTreeIter root, iter;
|
|
GtkTreePath *path;
|
|
gchar *parent, *p;
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
goto done;
|
|
|
|
reference = g_hash_table_lookup (si->full_hash, info->old_base);
|
|
if (!gtk_tree_row_reference_valid (reference))
|
|
goto done;
|
|
|
|
path = gtk_tree_row_reference_get_path (reference);
|
|
gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &iter, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
em_folder_tree_model_remove_folders (model, si, &iter);
|
|
|
|
parent = g_strdup(info->new->full_name);
|
|
p = strrchr(parent, '/');
|
|
if (p)
|
|
*p = 0;
|
|
if (p == NULL || parent == p)
|
|
/* renamed to a toplevel folder on the store */
|
|
reference = si->row;
|
|
else
|
|
reference = g_hash_table_lookup (si->full_hash, parent);
|
|
|
|
g_free (parent);
|
|
|
|
if (!gtk_tree_row_reference_valid (reference))
|
|
goto done;
|
|
|
|
path = gtk_tree_row_reference_get_path (reference);
|
|
gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &root, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
gtk_tree_store_append (GTK_TREE_STORE (model), &iter, &root);
|
|
em_folder_tree_model_set_folder_info (model, &iter, si, info->new, TRUE);
|
|
|
|
done:
|
|
camel_object_unref (store);
|
|
|
|
g_free (info->old_base);
|
|
camel_folder_info_free (info->new);
|
|
g_free (info);
|
|
}
|
|
|
|
static void
|
|
folder_renamed_cb (CamelStore *store,
|
|
gpointer event_data,
|
|
EMFolderTreeModel *model)
|
|
{
|
|
CamelRenameInfo *rinfo, *info = event_data;
|
|
|
|
camel_object_ref (store);
|
|
|
|
rinfo = g_new0 (CamelRenameInfo, 1);
|
|
rinfo->old_base = g_strdup (info->old_base);
|
|
rinfo->new = camel_folder_info_clone (info->new);
|
|
|
|
mail_async_event_emit (
|
|
mail_async_event, MAIL_ASYNC_GUI, (MailAsyncFunc)
|
|
folder_renamed, store, rinfo, model);
|
|
}
|
|
|
|
void
|
|
em_folder_tree_model_add_store (EMFolderTreeModel *model,
|
|
CamelStore *store,
|
|
const gchar *display_name)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreeRowReference *reference;
|
|
GtkTreeStore *tree_store;
|
|
GtkTreeIter root, iter;
|
|
GtkTreePath *path;
|
|
EAccount *account;
|
|
gchar *uri;
|
|
|
|
g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
|
|
g_return_if_fail (CAMEL_IS_STORE (store));
|
|
g_return_if_fail (display_name != NULL);
|
|
|
|
tree_store = GTK_TREE_STORE (model);
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si != NULL)
|
|
em_folder_tree_model_remove_store (model, store);
|
|
|
|
uri = camel_url_to_string (
|
|
((CamelService *) store)->url, CAMEL_URL_HIDE_ALL);
|
|
|
|
account = mail_config_get_account_by_source_url (uri);
|
|
|
|
/* add the store to the tree */
|
|
gtk_tree_store_append (tree_store, &iter, NULL);
|
|
gtk_tree_store_set (
|
|
tree_store, &iter,
|
|
COL_STRING_DISPLAY_NAME, display_name,
|
|
COL_POINTER_CAMEL_STORE, store,
|
|
COL_STRING_FULL_NAME, NULL,
|
|
COL_BOOL_LOAD_SUBDIRS, TRUE,
|
|
COL_BOOL_IS_STORE, TRUE,
|
|
COL_STRING_URI, uri, -1);
|
|
|
|
path = gtk_tree_model_get_path (GTK_TREE_MODEL (model), &iter);
|
|
reference = gtk_tree_row_reference_new (GTK_TREE_MODEL (model), path);
|
|
|
|
si = g_new (EMFolderTreeModelStoreInfo, 1);
|
|
si->display_name = g_strdup (display_name);
|
|
camel_object_ref (store);
|
|
si->store = store;
|
|
si->account = account;
|
|
si->row = gtk_tree_row_reference_copy (reference);
|
|
si->full_hash = g_hash_table_new_full (
|
|
g_str_hash, g_str_equal,
|
|
(GDestroyNotify) g_free,
|
|
(GDestroyNotify) gtk_tree_row_reference_free);
|
|
g_hash_table_insert (model->priv->store_index, store, si);
|
|
g_hash_table_insert (model->priv->account_index, account, si);
|
|
|
|
/* Transfer ownership of the URI and GtkTreeRowReference. */
|
|
g_hash_table_insert (model->priv->uri_index, uri, reference);
|
|
|
|
/* each store has folders... but we don't load them until the user demands them */
|
|
root = iter;
|
|
gtk_tree_store_append (tree_store, &iter, &root);
|
|
gtk_tree_store_set (
|
|
tree_store, &iter,
|
|
COL_STRING_DISPLAY_NAME, _("Loading..."),
|
|
COL_POINTER_CAMEL_STORE, NULL,
|
|
COL_STRING_FULL_NAME, NULL,
|
|
COL_BOOL_LOAD_SUBDIRS, FALSE,
|
|
COL_BOOL_IS_STORE, FALSE,
|
|
COL_BOOL_IS_FOLDER, FALSE,
|
|
COL_STRING_URI, NULL,
|
|
COL_UINT_UNREAD, 0,
|
|
COL_UINT_UNREAD_LAST_SEL, 0,
|
|
COL_BOOL_IS_DRAFT, FALSE,
|
|
-1);
|
|
|
|
/* listen to store events */
|
|
si->created_id = camel_object_hook_event (
|
|
store, "folder_created",
|
|
(CamelObjectEventHookFunc) folder_created_cb, model);
|
|
si->deleted_id = camel_object_hook_event (
|
|
store, "folder_deleted",
|
|
(CamelObjectEventHookFunc) folder_deleted_cb, model);
|
|
si->renamed_id = camel_object_hook_event (
|
|
store, "folder_renamed",
|
|
(CamelObjectEventHookFunc) folder_renamed_cb, model);
|
|
si->subscribed_id = camel_object_hook_event (
|
|
store, "folder_subscribed",
|
|
(CamelObjectEventHookFunc) folder_subscribed_cb, model);
|
|
si->unsubscribed_id = camel_object_hook_event (
|
|
store, "folder_unsubscribed",
|
|
(CamelObjectEventHookFunc) folder_unsubscribed_cb, model);
|
|
|
|
g_signal_emit (model, signals[LOADED_ROW], 0, path, &root);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
static void
|
|
em_folder_tree_model_remove_uri (EMFolderTreeModel *model, const gchar *uri)
|
|
{
|
|
g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
|
|
g_return_if_fail (uri != NULL);
|
|
|
|
g_hash_table_remove (model->priv->uri_index, uri);
|
|
}
|
|
|
|
static void
|
|
em_folder_tree_model_remove_store_info (EMFolderTreeModel *model,
|
|
CamelStore *store)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
|
|
g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
|
|
g_return_if_fail (CAMEL_IS_STORE (store));
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
return;
|
|
|
|
g_hash_table_remove (model->priv->account_index, si->account);
|
|
g_hash_table_remove (model->priv->store_index, si->store);
|
|
}
|
|
|
|
void
|
|
em_folder_tree_model_remove_folders (EMFolderTreeModel *model,
|
|
EMFolderTreeModelStoreInfo *si,
|
|
GtkTreeIter *toplevel)
|
|
{
|
|
gchar *uri, *full_name;
|
|
gboolean is_store, go;
|
|
GtkTreeIter iter;
|
|
|
|
if (gtk_tree_model_iter_children (GTK_TREE_MODEL (model), &iter, toplevel)) {
|
|
do {
|
|
GtkTreeIter next = iter;
|
|
|
|
go = gtk_tree_model_iter_next (GTK_TREE_MODEL (model), &next);
|
|
em_folder_tree_model_remove_folders (model, si, &iter);
|
|
iter = next;
|
|
} while (go);
|
|
}
|
|
|
|
gtk_tree_model_get (
|
|
GTK_TREE_MODEL (model), toplevel,
|
|
COL_STRING_URI, &uri,
|
|
COL_STRING_FULL_NAME, &full_name,
|
|
COL_BOOL_IS_STORE, &is_store, -1);
|
|
|
|
if (full_name != NULL)
|
|
g_hash_table_remove (si->full_hash, full_name);
|
|
|
|
if (uri != NULL)
|
|
em_folder_tree_model_remove_uri (model, uri);
|
|
|
|
gtk_tree_store_remove ((GtkTreeStore *) model, toplevel);
|
|
|
|
if (is_store)
|
|
em_folder_tree_model_remove_store_info (model, si->store);
|
|
|
|
g_free (full_name);
|
|
g_free (uri);
|
|
}
|
|
|
|
void
|
|
em_folder_tree_model_remove_store (EMFolderTreeModel *model,
|
|
CamelStore *store)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreePath *path;
|
|
GtkTreeIter iter;
|
|
|
|
g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
|
|
g_return_if_fail (CAMEL_IS_STORE (store));
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
return;
|
|
|
|
path = gtk_tree_row_reference_get_path (si->row);
|
|
gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &iter, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
/* recursively remove subfolders and finally the toplevel store */
|
|
em_folder_tree_model_remove_folders (model, si, &iter);
|
|
}
|
|
|
|
gboolean
|
|
em_folder_tree_model_is_type_inbox (EMFolderTreeModel *model,
|
|
CamelStore *store,
|
|
const gchar *full)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreeRowReference *reference;
|
|
GtkTreePath *path;
|
|
GtkTreeIter iter;
|
|
guint32 flags;
|
|
|
|
g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), FALSE);
|
|
g_return_val_if_fail (CAMEL_IS_STORE (store), FALSE);
|
|
g_return_val_if_fail (full != NULL, FALSE);
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
return FALSE;
|
|
|
|
reference = g_hash_table_lookup (si->full_hash, full);
|
|
if (!gtk_tree_row_reference_valid (reference))
|
|
return FALSE;
|
|
|
|
path = gtk_tree_row_reference_get_path (reference);
|
|
gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &iter, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
gtk_tree_model_get (
|
|
GTK_TREE_MODEL (model), &iter,
|
|
COL_UINT_FLAGS, &flags, -1);
|
|
|
|
return ((flags & CAMEL_FOLDER_TYPE_MASK) == CAMEL_FOLDER_TYPE_INBOX);
|
|
}
|
|
|
|
gchar *
|
|
em_folder_tree_model_get_folder_name (EMFolderTreeModel *model,
|
|
CamelStore *store,
|
|
const gchar *full)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreeRowReference *reference;
|
|
GtkTreePath *path;
|
|
GtkTreeIter iter;
|
|
gchar *name = NULL;
|
|
|
|
g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), NULL);
|
|
g_return_val_if_fail (CAMEL_IS_STORE (store), NULL);
|
|
g_return_val_if_fail (full != NULL, NULL);
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
return NULL;
|
|
|
|
reference = g_hash_table_lookup (si->full_hash, full);
|
|
if (!gtk_tree_row_reference_valid (reference))
|
|
return NULL;
|
|
|
|
path = gtk_tree_row_reference_get_path (reference);
|
|
gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &iter, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
gtk_tree_model_get (
|
|
GTK_TREE_MODEL (model), &iter,
|
|
COL_STRING_DISPLAY_NAME, &name, -1);
|
|
|
|
return name;
|
|
}
|
|
|
|
void
|
|
em_folder_tree_model_set_unread_count (EMFolderTreeModel *model,
|
|
CamelStore *store,
|
|
const gchar *full,
|
|
gint unread)
|
|
{
|
|
EMFolderTreeModelStoreInfo *si;
|
|
GtkTreeRowReference *reference;
|
|
GtkTreeModel *tree_model;
|
|
GtkTreePath *path;
|
|
GtkTreeIter parent;
|
|
GtkTreeIter iter;
|
|
guint old_unread = 0;
|
|
|
|
g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
|
|
g_return_if_fail (CAMEL_IS_STORE (store));
|
|
g_return_if_fail (full != NULL);
|
|
|
|
if (unread < 0)
|
|
return;
|
|
|
|
si = em_folder_tree_model_lookup_store_info (model, store);
|
|
if (si == NULL)
|
|
return;
|
|
|
|
reference = g_hash_table_lookup (si->full_hash, full);
|
|
if (!gtk_tree_row_reference_valid (reference))
|
|
return;
|
|
|
|
tree_model = GTK_TREE_MODEL (model);
|
|
|
|
path = gtk_tree_row_reference_get_path (reference);
|
|
gtk_tree_model_get_iter (tree_model, &iter, path);
|
|
gtk_tree_path_free (path);
|
|
|
|
gtk_tree_model_get (
|
|
tree_model, &iter,
|
|
COL_UINT_UNREAD_LAST_SEL, &old_unread, -1);
|
|
|
|
gtk_tree_store_set (
|
|
GTK_TREE_STORE (model), &iter,
|
|
COL_UINT_UNREAD, unread,
|
|
COL_UINT_UNREAD_LAST_SEL, MIN (old_unread, unread), -1);
|
|
|
|
/* Folders are displayed with a bold weight to indicate that
|
|
* they contain unread messages. We signal that parent rows
|
|
* have changed here to update them. */
|
|
while (gtk_tree_model_iter_parent (tree_model, &parent, &iter)) {
|
|
path = gtk_tree_model_get_path (tree_model, &parent);
|
|
gtk_tree_model_row_changed (tree_model, path, &parent);
|
|
gtk_tree_path_free (path);
|
|
iter = parent;
|
|
}
|
|
}
|
|
|
|
EMFolderTreeModelStoreInfo *
|
|
em_folder_tree_model_lookup_store_info (EMFolderTreeModel *model,
|
|
CamelStore *store)
|
|
{
|
|
g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), NULL);
|
|
g_return_val_if_fail (CAMEL_IS_STORE (store), NULL);
|
|
|
|
return g_hash_table_lookup (model->priv->store_index, store);
|
|
}
|
|
|
|
GtkTreeRowReference *
|
|
em_folder_tree_model_lookup_uri (EMFolderTreeModel *model,
|
|
const gchar *uri)
|
|
{
|
|
GtkTreeRowReference *reference;
|
|
|
|
g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), NULL);
|
|
g_return_val_if_fail (uri != NULL, NULL);
|
|
|
|
reference = g_hash_table_lookup (model->priv->uri_index, uri);
|
|
|
|
return gtk_tree_row_reference_valid (reference) ? reference : NULL;
|
|
}
|