aboutsummaryrefslogblamecommitdiffstats
path: root/mail/em-folder-tree-model.c
blob: 5dbd0a4007ef2074555f9dca49952c3b144eeede (plain) (tree)
1
2
3
4
5
6
7
8
9
10
  



                                                                
  



                                                                    
  
                                                                   
                                                                             
  
  



                                                        


   
                                 
 
                   
                  
                   



                      
                     
 

                       

                          
 




                                 
 



                                                                     



                                                

                                    
                                  




                                                                         

                                         
 
                                     
                                
                                
  
 
                   

                                












                                                               








                                                             

  
      
               
                       
                    


      
                    
                   


                   



                                                                     
                                                                


                                                                     
                                                                



                                                                       
                                                                

                                                                   
                                                                


                                                                     
                                                                


                                                                     
                                                                


                                                                   
                                                                
 
                                  
 

                                                                            
                  





































                                                                         


                                         
                              












                                                              
                                            
                                        
                                                                 

                                                      

                                                   
                                            
                                        
                                                                 

                                                      

                                                   
                                            
                                        
                                                                 

                                                      

                                                   
                                            
                                           
                                                                    

                                                      


                                                      
                                                    

                                                                            

                                                              

                                                              
                                                    

                                                                              

                                                              


                                                                



                                                                        

                                                              




                                                                        

                                                              





                                                                        


                  

                                  
 
                                      
 

                                                          
 





                                                       
 





                                                       
 





                                                       
 





                                                          
 


















                                                            
         













                                                               
 
 
                  

                                                               
 

                             

                                                            








































                                                                            

 
           


                                            
                                        
 
                                             
                             



                                
                                      
                                 
                     
 
                                                         
 

                            
                                               
                                                   
                                                
                                         
                    


                            
                                               
                                                   
                                                
                                         
                    
 








                                                                            
                                               
                                                           
                               
                                                                
                                
 
                
                                            
                                                                                  
                                
                                                                                       
                               
         
 



                                                           
                               
                                       
                                


                               
 

                       
 


                                    
                  

 
           
                                                                       
                                                































                                                                                

                                                                             




                           


                                                                        
 

                                                          


           


                                                                        
 

                                                          

 
           


                                                                         
 

                                                          

 
           


                                                                           
 
                                                    
                                                 
                                                      
                                                    

 





























                                                           
           







                                                                   



                                                    
 
                              




                                                              
 

                                                          


                                                              

         
                                                                       


           



                                                  
 
                              





                                                                    
 
                                  

                                            
                                                                  

                                                                
         
 
                                                                       

 
           



                                           
                                                         







                                                                          










                                                                 

         
                                                    
                                                                             


           
                                            
 

                                       
                                                         
 
                                                 
                                                
 
                                                     
                                                                              


           

                                               
                             










                                                                     



                                                              

          









                                                                 
                                                        

                                                            


           
                                                               


                                   




                                                                            
                                                          
                                                            
                                                                  


                                         
                             
                                     
                                  

                             
                                            

                                            

                                         
                               
                                     

                                    
                             

                                                
 




















                                                                      


           



                                                             
 




                                       
                      








                                                           
                                                                 




                                                              
                          
























                                                                         


                              


           
                                                    
 
                                
 
                                             

                                            
                                      
                                                     
 
                                                               
                                               

                                                      

 
                   
                               
 
                                                              
 
 
                   
                                       
 

                                                            

                                                                        
 
                                         

 
















                                                                     


                                                
















                                                                         

                                                           


                                                                     
                                    


    

                                                           


                                                           


                                            


                                                               

         

                                                      
 
                                       
 
                                                                   
                                                                  
                                                                     

                                                 
                                              
 
                                                                         

                                                                     
 


                                                                             
 



                                                                             


















                                                                          

                                          




                                                                        
                                                      

 

























































                                                                                
    

                                                               
                                                        

                                                          
 
                                      
                                 
                                      
                                  
                              
                     

                          
                            
                      
                                 
                         
                               
                                  
                                     
                                        



                                             
                                
                   
 




                                                           
                                                                 

                                      
                                                       

                                                                 
                       
         
 
                                            
 
                                                           
                                                                 
                                                         
 
                                                            
                                                                          
 
                          

                                                                              
 
                                                                      
                                                                             
                                  
 
                                                             
 

                                                                   
 


                              




                                                                          
                            
                                               
                                                    
                             

                                                                                
 
                                                           
                                   
                                     
 

                                                                                  
 

                                                       
 
                                                
                 
 
                                        
         
 
                          

                                        
                             
                                                            
                                                

                                                                      
                                                   

                                                                  

                                                                   

                                                                   

                                                                   

                                                                 

                                                                   
                                                 

                 
 




                                                         
                                                             

                                                       
                                                           

                                                                     
                                                                    

                                                        

                 
                                              
                                                     

                                                        

                                                                   
                 


                                           

         
                                                 


                                                                      
                                     
                                                              
                                             

                                                              


                                 
                                                      
                                              

                                                    



                                            
                                            
                                                    
                    
 


                     





                                                                 
 
                         


                                                                  
 
                   
                                                                     



                                                                 
                                                      




                                                     
                                           
                                                    
                                                 
                            
 
                                                                              

                                                                           

                       
 

                               
 
                    
                                                                       
 
                                       




                                                                      


                                                          
 
                                                              
                                                                      


                                      
 
                       
                                                                              

                                                                          
         

 
           

                                                         
                                                   
 

                                                     
                                          
                       
 

                                                                       

 
           

                                                         
                                                   
 

                                                     
                                          
                       
 
                                                                 


           


                                                           
                                                   
 
                                       
                            

                               
                          
 
                                                                  
                                                      
                       
 
                                                           

                                                             
                                  
 

                                                         
 


                                                                         
                       
 
                                            
                                  

                       
                                     
                                                               


                                                                        
 
                        
 
                                                      
                       
 
                                                           
                                                     
                                  
 
                                                                     

                                                                        

 

                                                          
                                                      
 



                                                    
 
                                                           
 

                                                                  
                                                                             

 


                                                            
                                                      

                                       
                            

                                 


                           
                                                       

                                                                 













                                                                         
                       

                                                           

                                                             



                                                                     
                               

                                                  
                       


                                                                       

                                                                      




                                                              
                                                        

                                       
                            

                          


                                                                       
                       

                                                           

                                                             

                                  

                                                         

 






















































































                                                                            
                                                  
 



                                                                      
                                                  

 
    
                                                         
                                                  
 

                                       

                               
                              
                                
                      
                                  
 

                                                           
 

                                            
                                        
                                                        

                                                                







                                                                       
                                                           
                       
 

                                                                 
                                                                 

                                      
 
                                        



                                                        
                                              


                                            
                    



                                                                              
                                           
                                                   


                                                         

                                                               
                    



                                                         
                                              


                                             
                                          
                                   


                                            
 


                                                          

                                                                   

                              

 


                                                            
 

                          
                      
 

                                                           
 
                                                                 
                       
                       
 




                                                                          
                                                            

                              
 
 


                                                           

                    

                                                                     








                                                                 

 
        


                                                             
 
                                       
                      
                          
 


                                                                      
 
                                                                 
                       
                             
 
                                                              
 


                                                       
 









                                                                              
 
                                                                             

 
       


                                                               
 

                                       
                         
                      
                           
 


                                                                     
 




                                                                 
 
                                                              



                                                        
 


                                                                      
 


                                                    
 



                                      


                    


















                                                                            
                      



                                                                     
                                                                 










                                                                             

                              


                         

                                                                  
                                                             




                                                        

                                 






                                                                  
                                                    
 




                                                              
 
















                                                                      
/*
 * 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 <config.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 <glib/gi18n.h>

#include <e-util/e-util.h>
#include <shell/e-shell.h>

#include "e-mail-account-store.h"
#include "e-mail-ui-session.h"
#include "em-utils.h"
#include "em-folder-utils.h"
#include "em-event.h"

#define EM_FOLDER_TREE_MODEL_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), EM_TYPE_FOLDER_TREE_MODEL, EMFolderTreeModelPrivate))

/* See GtkCellRendererSpinner:pulse property.
 * Animation cycles over 12 frames in 750 ms. */
#define SPINNER_PULSE_INTERVAL (750 / 12)

typedef struct _StoreInfo StoreInfo;

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 */

    EMailSession *session;
    EMailAccountStore *account_store;

    /* CamelStore -> StoreInfo */
    GHashTable *store_index;
    GMutex store_index_lock;
};

struct _StoreInfo {
    volatile gint ref_count;

    CamelStore *store;
    GtkTreeRowReference *row;

    /* CamelFolderInfo::full_name -> GtkTreeRowReference */
    GHashTable *full_hash;

    /* CamelStore signal handler IDs */
    gulong folder_created_handler_id;
    gulong folder_deleted_handler_id;
    gulong folder_renamed_handler_id;
    gulong folder_info_stale_handler_id;
    gulong folder_subscribed_handler_id;
    gulong folder_unsubscribed_handler_id;
    gulong connection_status_handler_id;
    gulong host_reachable_handler_id;

    /* For comparison with the current status. */
    CamelServiceConnectionStatus last_status;

    /* Spinner renderers have to be animated manually. */
    guint spinner_pulse_value;
    guint spinner_pulse_timeout_id;
};

enum {
    PROP_0,
    PROP_SELECTION,
    PROP_SESSION
};

enum {
    LOADING_ROW,
    LOADED_ROW,
    LAST_SIGNAL
};

/* Forward Declarations */
static void folder_tree_model_folder_created_cb
                        (CamelStore *store,
                         CamelFolderInfo *fi,
                         StoreInfo *si);
static void folder_tree_model_folder_deleted_cb
                        (CamelStore *store,
                         CamelFolderInfo *fi,
                         StoreInfo *si);
static void folder_tree_model_folder_renamed_cb
                        (CamelStore *store,
                         const gchar *old_name,
                         CamelFolderInfo *info,
                         StoreInfo *si);
static void folder_tree_model_folder_info_stale_cb
                        (CamelStore *store,
                         StoreInfo *si);
static void folder_tree_model_folder_subscribed_cb
                        (CamelStore *store,
                         CamelFolderInfo *fi,
                         StoreInfo *si);
static void folder_tree_model_folder_unsubscribed_cb
                        (CamelStore *store,
                         CamelFolderInfo *fi,
                         StoreInfo *si);
static void folder_tree_model_status_notify_cb
                        (CamelStore *store,
                         GParamSpec *pspec,
                         StoreInfo *si);

static guint signals[LAST_SIGNAL];

G_DEFINE_TYPE (EMFolderTreeModel, em_folder_tree_model, GTK_TYPE_TREE_STORE)

static StoreInfo *
store_info_ref (StoreInfo *si)
{
    g_return_val_if_fail (si != NULL, NULL);
    g_return_val_if_fail (si->ref_count > 0, NULL);

    g_atomic_int_inc (&si->ref_count);

    return si;
}

static void
store_info_unref (StoreInfo *si)
{
    g_return_if_fail (si != NULL);
    g_return_if_fail (si->ref_count > 0);

    if (g_atomic_int_dec_and_test (&si->ref_count)) {

        /* Check that we're fully disconnected. */
        g_warn_if_fail (si->folder_created_handler_id == 0);
        g_warn_if_fail (si->folder_deleted_handler_id == 0);
        g_warn_if_fail (si->folder_renamed_handler_id == 0);
        g_warn_if_fail (si->folder_info_stale_handler_id == 0);
        g_warn_if_fail (si->folder_subscribed_handler_id == 0);
        g_warn_if_fail (si->folder_unsubscribed_handler_id == 0);
        g_warn_if_fail (si->connection_status_handler_id == 0);
        g_warn_if_fail (si->host_reachable_handler_id == 0);
        g_warn_if_fail (si->spinner_pulse_timeout_id == 0);

        g_object_unref (si->store);
        gtk_tree_row_reference_free (si->row);
        g_hash_table_destroy (si->full_hash);

        g_slice_free (StoreInfo, si);
    }
}

static StoreInfo *
store_info_new (EMFolderTreeModel *model,
                CamelStore *store)
{
    CamelService *service;
    StoreInfo *si;
    gulong handler_id;

    si = g_slice_new0 (StoreInfo);
    si->ref_count = 1;
    si->store = g_object_ref (store);

    si->full_hash = g_hash_table_new_full (
        (GHashFunc) g_str_hash,
        (GEqualFunc) g_str_equal,
        (GDestroyNotify) g_free,
        (GDestroyNotify) gtk_tree_row_reference_free);

    handler_id = g_signal_connect_data (
        store, "folder-created",
        G_CALLBACK (folder_tree_model_folder_created_cb),
        store_info_ref (si),
        (GClosureNotify) store_info_unref, 0);
    si->folder_created_handler_id = handler_id;

    handler_id = g_signal_connect_data (
        store, "folder-deleted",
        G_CALLBACK (folder_tree_model_folder_deleted_cb),
        store_info_ref (si),
        (GClosureNotify) store_info_unref, 0);
    si->folder_deleted_handler_id = handler_id;

    handler_id = g_signal_connect_data (
        store, "folder-renamed",
        G_CALLBACK (folder_tree_model_folder_renamed_cb),
        store_info_ref (si),
        (GClosureNotify) store_info_unref, 0);
    si->folder_renamed_handler_id = handler_id;

    handler_id = g_signal_connect_data (
        store, "folder-info-stale",
        G_CALLBACK (folder_tree_model_folder_info_stale_cb),
        store_info_ref (si),
        (GClosureNotify) store_info_unref, 0);
    si->folder_info_stale_handler_id = handler_id;

    if (CAMEL_IS_SUBSCRIBABLE (store)) {
        handler_id = g_signal_connect_data (
            store, "folder-subscribed",
            G_CALLBACK (folder_tree_model_folder_subscribed_cb),
            store_info_ref (si),
            (GClosureNotify) store_info_unref, 0);
        si->folder_subscribed_handler_id = handler_id;

        handler_id = g_signal_connect_data (
            store, "folder-unsubscribed",
            G_CALLBACK (folder_tree_model_folder_unsubscribed_cb),
            store_info_ref (si),
            (GClosureNotify) store_info_unref, 0);
        si->folder_unsubscribed_handler_id = handler_id;
    }

    if (CAMEL_IS_NETWORK_SERVICE (store)) {
        handler_id = g_signal_connect_data (
            store, "notify::connection-status",
            G_CALLBACK (folder_tree_model_status_notify_cb),
            store_info_ref (si),
            (GClosureNotify) store_info_unref, 0);
        si->connection_status_handler_id = handler_id;

        handler_id = g_signal_connect_data (
            store, "notify::host-reachable",
            G_CALLBACK (folder_tree_model_status_notify_cb),
            store_info_ref (si),
            (GClosureNotify) store_info_unref, 0);
        si->host_reachable_handler_id = handler_id;
    }

    service = CAMEL_SERVICE (store);
    si->last_status = camel_service_get_connection_status (service);

    return si;
}

static void
store_info_dispose (StoreInfo *si)
{
    g_return_if_fail (si != NULL);

    /* Disconnect all signal handlers and whatever
     * else might be holding a StoreInfo reference. */

    if (si->folder_created_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->folder_created_handler_id);
        si->folder_created_handler_id = 0;
    }

    if (si->folder_deleted_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->folder_deleted_handler_id);
        si->folder_deleted_handler_id = 0;
    }

    if (si->folder_renamed_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->folder_renamed_handler_id);
        si->folder_renamed_handler_id = 0;
    }

    if (si->folder_info_stale_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->folder_info_stale_handler_id);
        si->folder_info_stale_handler_id = 0;
    }

    if (si->folder_subscribed_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->folder_subscribed_handler_id);
        si->folder_subscribed_handler_id = 0;
    }

    if (si->folder_unsubscribed_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->folder_unsubscribed_handler_id);
        si->folder_unsubscribed_handler_id = 0;
    }

    if (si->connection_status_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->connection_status_handler_id);
        si->connection_status_handler_id = 0;
    }

    if (si->host_reachable_handler_id > 0) {
        g_signal_handler_disconnect (
            si->store,
            si->host_reachable_handler_id);
        si->host_reachable_handler_id = 0;
    }

    if (si->spinner_pulse_timeout_id > 0) {
        g_source_remove (si->spinner_pulse_timeout_id);
        si->spinner_pulse_timeout_id = 0;
    }

    store_info_unref (si);
}

static StoreInfo *
folder_tree_model_store_index_lookup (EMFolderTreeModel *model,
                                      CamelStore *store)
{
    StoreInfo *si = NULL;

    g_return_val_if_fail (CAMEL_IS_STORE (store), NULL);

    g_mutex_lock (&model->priv->store_index_lock);

    si = g_hash_table_lookup (model->priv->store_index, store);
    if (si != NULL)
        store_info_ref (si);

    g_mutex_unlock (&model->priv->store_index_lock);

    return si;
}

static void
folder_tree_model_store_index_insert (EMFolderTreeModel *model,
                                      StoreInfo *si)
{
    g_return_if_fail (si != NULL);

    g_mutex_lock (&model->priv->store_index_lock);

    g_hash_table_insert (
        model->priv->store_index,
        si->store, store_info_ref (si));

    g_mutex_unlock (&model->priv->store_index_lock);
}

static gboolean
folder_tree_model_store_index_remove (EMFolderTreeModel *model,
                                      StoreInfo *si)
{
    gboolean removed;

    g_return_val_if_fail (si != NULL, FALSE);

    g_mutex_lock (&model->priv->store_index_lock);

    removed = g_hash_table_remove (model->priv->store_index, si->store);

    g_mutex_unlock (&model->priv->store_index_lock);

    return removed;
}

static gint
folder_tree_model_sort (GtkTreeModel *model,
                        GtkTreeIter *a,
                        GtkTreeIter *b,
                        gpointer unused)
{
    EMFolderTreeModel *folder_tree_model;
    gchar *aname, *bname;
    CamelService *service_a;
    CamelService *service_b;
    gboolean a_is_store;
    gboolean b_is_store;
    const gchar *store_uid = NULL;
    guint32 flags_a, flags_b;
    gint rv = -2;

    folder_tree_model = EM_FOLDER_TREE_MODEL (model);

    gtk_tree_model_get (
        model, a,
        COL_BOOL_IS_STORE, &a_is_store,
        COL_OBJECT_CAMEL_STORE, &service_a,
        COL_STRING_DISPLAY_NAME, &aname,
        COL_UINT_FLAGS, &flags_a,
        -1);

    gtk_tree_model_get (
        model, b,
        COL_BOOL_IS_STORE, &b_is_store,
        COL_OBJECT_CAMEL_STORE, &service_b,
        COL_STRING_DISPLAY_NAME, &bname,
        COL_UINT_FLAGS, &flags_b,
        -1);

    if (CAMEL_IS_SERVICE (service_a))
        store_uid = camel_service_get_uid (service_a);

    if (a_is_store && b_is_store) {
        rv = e_mail_account_store_compare_services (
            folder_tree_model->priv->account_store,
            service_a, service_b);

    } else if (g_strcmp0 (store_uid, E_MAIL_SESSION_VFOLDER_UID) == 0) {
        /* UNMATCHED is always last. */
        if (g_strcmp0 (aname, _("UNMATCHED")) == 0)
            rv = 1;
        else if (g_strcmp0 (bname, _("UNMATCHED")) == 0)
            rv = -1;

    } else {
        /* Inbox is always first. */
        if ((flags_a & CAMEL_FOLDER_TYPE_MASK) == CAMEL_FOLDER_TYPE_INBOX)
            rv = -1;
        else if ((flags_b & CAMEL_FOLDER_TYPE_MASK) == CAMEL_FOLDER_TYPE_INBOX)
            rv = 1;
    }

    if (rv == -2) {
        if (aname != NULL && bname != NULL)
            rv = g_utf8_collate (aname, bname);
        else if (aname == bname)
            rv = 0;
        else if (aname == NULL)
            rv = -1;
        else
            rv = 1;
    }

    g_free (aname);
    g_free (bname);

    g_clear_object (&service_a);
    g_clear_object (&service_b);

    return rv;
}

static void
folder_tree_model_remove_folders (EMFolderTreeModel *folder_tree_model,
                                  StoreInfo *si,
                                  GtkTreeIter *toplevel)
{
    GtkTreeModel *model;
    GtkTreeIter iter;
    gchar *full_name;
    gboolean is_store;
    gboolean iter_valid;

    model = GTK_TREE_MODEL (folder_tree_model);

    iter_valid = gtk_tree_model_iter_children (model, &iter, toplevel);

    while (iter_valid) {
        GtkTreeIter next = iter;

        /* Recursing will invalidate the current tree model
         * iterator, so we have to fetch the next one first. */
        iter_valid = gtk_tree_model_iter_next (model, &next);
        folder_tree_model_remove_folders (folder_tree_model, si, &iter);
        iter = next;
    }

    gtk_tree_model_get (
        model, toplevel,
        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);

    gtk_tree_store_remove (GTK_TREE_STORE (model), toplevel);

    if (is_store)
        folder_tree_model_store_index_remove (folder_tree_model, si);

    g_free (full_name);
}

static void
folder_tree_model_service_removed (EMailAccountStore *account_store,
                                   CamelService *service,
                                   EMFolderTreeModel *folder_tree_model)
{
    em_folder_tree_model_remove_store (
        folder_tree_model, CAMEL_STORE (service));
}

static void
folder_tree_model_service_enabled (EMailAccountStore *account_store,
                                   CamelService *service,
                                   EMFolderTreeModel *folder_tree_model)
{
    em_folder_tree_model_add_store (
        folder_tree_model, CAMEL_STORE (service));
}

static void
folder_tree_model_service_disabled (EMailAccountStore *account_store,
                                    CamelService *service,
                                    EMFolderTreeModel *folder_tree_model)
{
    em_folder_tree_model_remove_store (
        folder_tree_model, CAMEL_STORE (service));
}

static void
folder_tree_model_services_reordered (EMailAccountStore *account_store,
                                      gboolean default_restored,
                                      EMFolderTreeModel *folder_tree_model)
{
    /* This forces the tree store to re-sort. */
    gtk_tree_sortable_set_default_sort_func (
        GTK_TREE_SORTABLE (folder_tree_model),
        folder_tree_model_sort, NULL, NULL);
}

static gboolean
folder_tree_model_spinner_pulse_cb (gpointer user_data)
{
    GtkTreeModel *model;
    GtkTreePath *path;
    GtkTreeIter iter;
    StoreInfo *si;

    si = (StoreInfo *) user_data;

    if (!gtk_tree_row_reference_valid (si->row))
        return G_SOURCE_REMOVE;

    path = gtk_tree_row_reference_get_path (si->row);
    model = gtk_tree_row_reference_get_model (si->row);
    gtk_tree_model_get_iter (model, &iter, path);
    gtk_tree_path_free (path);

    gtk_tree_store_set (
        GTK_TREE_STORE (model), &iter,
        COL_STATUS_SPINNER_PULSE,
        si->spinner_pulse_value++,
        -1);

    if (si->spinner_pulse_value == G_MAXUINT)
        si->spinner_pulse_value = 0;

    return G_SOURCE_CONTINUE;
}

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;

        case PROP_SESSION:
            em_folder_tree_model_set_session (
                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;

        case PROP_SESSION:
            g_value_set_object (
                value,
                em_folder_tree_model_get_session (
                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;
    }

    if (priv->session != NULL) {
        g_object_unref (priv->session);
        priv->session = NULL;
    }

    if (priv->account_store != NULL) {
        g_signal_handlers_disconnect_matched (
            priv->account_store, G_SIGNAL_MATCH_DATA,
            0, 0, NULL, NULL, object);
        g_object_unref (priv->account_store);
        priv->account_store = NULL;
    }

    /* Chain up to parent's dispose() method. */
    G_OBJECT_CLASS (em_folder_tree_model_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->store_index);
    g_mutex_clear (&priv->store_index_lock);

    /* Chain up to parent's finalize() method. */
    G_OBJECT_CLASS (em_folder_tree_model_parent_class)->finalize (object);
}

static void
folder_tree_model_constructed (GObject *object)
{
    GType col_types[] = {
        G_TYPE_STRING,    /* display name */
        CAMEL_TYPE_STORE, /* CamelStore */
        G_TYPE_STRING,    /* full name */
        G_TYPE_STRING,    /* icon name */
        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 */
        G_TYPE_ICON,      /* status GIcon */
        G_TYPE_BOOLEAN,   /* status icon visible */
        G_TYPE_UINT,      /* status spinner pulse */
        G_TYPE_BOOLEAN,   /* status spinner visible */
    };

    gtk_tree_store_set_column_types (
        GTK_TREE_STORE (object), NUM_COLUMNS, col_types);
    gtk_tree_sortable_set_default_sort_func (
        GTK_TREE_SORTABLE (object),
        folder_tree_model_sort, NULL, NULL);
    gtk_tree_sortable_set_sort_column_id (
        GTK_TREE_SORTABLE (object),
        GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID,
        GTK_SORT_ASCENDING);

    /* Chain up to parent's constructed() method. */
    G_OBJECT_CLASS (em_folder_tree_model_parent_class)->
        constructed (object);
}

static void
em_folder_tree_model_class_init (EMFolderTreeModelClass *class)
{
    GObjectClass *object_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;
    object_class->constructed = folder_tree_model_constructed;

    g_object_class_install_property (
        object_class,
        PROP_SESSION,
        g_param_spec_object (
            "session",
            NULL,
            NULL,
            E_TYPE_MAIL_SESSION,
            G_PARAM_READWRITE));

    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);
}

static void
folder_tree_model_set_unread_count (EMFolderTreeModel *model,
                                    CamelStore *store,
                                    const gchar *full,
                                    gint unread)
{
    GtkTreeRowReference *reference;
    GtkTreeModel *tree_model;
    GtkTreePath *path;
    GtkTreeIter parent;
    GtkTreeIter iter;
    StoreInfo *si;
    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 = folder_tree_model_store_index_lookup (model, store);
    if (si == NULL)
        return;

    reference = g_hash_table_lookup (si->full_hash, full);
    if (!gtk_tree_row_reference_valid (reference))
        goto exit;

    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;
    }

exit:
    store_info_unref (si);
}

static void
em_folder_tree_model_init (EMFolderTreeModel *model)
{
    GHashTable *store_index;

    store_index = g_hash_table_new_full (
        (GHashFunc) g_direct_hash,
        (GEqualFunc) g_direct_equal,
        (GDestroyNotify) NULL,
        (GDestroyNotify) store_info_dispose);

    model->priv = EM_FOLDER_TREE_MODEL_GET_PRIVATE (model);
    model->priv->store_index = store_index;

    g_mutex_init (&model->priv->store_index_lock);
}

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 == selection)
        return;

    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");
}

EMailSession *
em_folder_tree_model_get_session (EMFolderTreeModel *model)
{
    g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), NULL);

    return model->priv->session;
}

void
em_folder_tree_model_set_session (EMFolderTreeModel *model,
                                  EMailSession *session)
{
    g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));

    if (model->priv->session == session)
        return;

    if (session != NULL) {
        g_return_if_fail (E_IS_MAIL_SESSION (session));
        g_object_ref (session);
    }

    if (model->priv->session != NULL)
        g_object_unref (model->priv->session);

    model->priv->session = session;

    /* FIXME Technically we should be disconnecting this signal
     *       when replacing an old session with a new session,
     *       but at present this function is only called once. */
    if (session != NULL) {
        EMailAccountStore *account_store;
        MailFolderCache *folder_cache;

        folder_cache = e_mail_session_get_folder_cache (session);
        account_store = e_mail_ui_session_get_account_store (
            E_MAIL_UI_SESSION (session));

        /* Keep our own reference since we connect to its signals. */
        g_warn_if_fail (model->priv->account_store == NULL);
        model->priv->account_store = g_object_ref (account_store);

        /* No need to connect to "service-added" emissions since it's
         * always immediately followed by either "service-enabled" or
         * "service-disabled". */

        g_signal_connect (
            account_store, "service-removed",
            G_CALLBACK (folder_tree_model_service_removed),
            model);

        g_signal_connect (
            account_store, "service-enabled",
            G_CALLBACK (folder_tree_model_service_enabled),
            model);

        g_signal_connect (
            account_store, "service-disabled",
            G_CALLBACK (folder_tree_model_service_disabled),
            model);

        g_signal_connect (
            account_store, "services-reordered",
            G_CALLBACK (folder_tree_model_services_reordered),
            model);

        g_signal_connect_swapped (
            folder_cache, "folder-unread-updated",
            G_CALLBACK (folder_tree_model_set_unread_count),
            model);
    }

    g_object_notify (G_OBJECT (model), "session");
}

/* Helper for em_folder_tree_model_set_folder_info() */
static void
folder_tree_model_get_drafts_folder_uri (ESourceRegistry *registry,
                                         CamelStore *store,
                                         gchar **drafts_folder_uri)
{
    ESource *source;
    const gchar *extension_name;

    /* In case we fail... */
    *drafts_folder_uri = NULL;

    source = em_utils_ref_mail_identity_for_store (registry, store);
    if (source == NULL)
        return;

    extension_name = E_SOURCE_EXTENSION_MAIL_COMPOSITION;
    if (e_source_has_extension (source, extension_name)) {
        ESourceMailComposition *extension;

        extension = e_source_get_extension (source, extension_name);

        *drafts_folder_uri =
            e_source_mail_composition_dup_drafts_folder (extension);
    }

    g_object_unref (source);
}

/* Helper for em_folder_tree_model_set_folder_info() */
static void
folder_tree_model_get_sent_folder_uri (ESourceRegistry *registry,
                                       CamelStore *store,
                                       gchar **sent_folder_uri)
{
    ESource *source;
    const gchar *extension_name;

    /* In case we fail... */
    *sent_folder_uri = NULL;

    source = em_utils_ref_mail_identity_for_store (registry, store);
    if (source == NULL)
        return;

    extension_name = E_SOURCE_EXTENSION_MAIL_SUBMISSION;
    if (e_source_has_extension (source, extension_name)) {
        ESourceMailSubmission *extension;

        extension = e_source_get_extension (source, extension_name);

        *sent_folder_uri =
            e_source_mail_submission_dup_sent_folder (extension);
    }

    g_object_unref (source);
}

void
em_folder_tree_model_set_folder_info (EMFolderTreeModel *model,
                                      GtkTreeIter *iter,
                                      CamelStore *store,
                                      CamelFolderInfo *fi,
                                      gint fully_loaded)
{
    GtkTreeRowReference *path_row;
    GtkTreeStore *tree_store;
    MailFolderCache *folder_cache;
    ESourceRegistry *registry;
    EMailSession *session;
    guint unread;
    GtkTreePath *path;
    GtkTreeIter sub;
    CamelFolder *folder;
    StoreInfo *si;
    gboolean emitted = FALSE;
    const gchar *uid;
    const gchar *icon_name;
    const gchar *display_name;
    guint32 flags, add_flags = 0;
    EMEventTargetCustomIcon *target;
    gboolean load = FALSE;
    gboolean folder_is_drafts = FALSE;
    gboolean folder_is_outbox = FALSE;
    gboolean folder_is_templates = FALSE;
    gboolean store_is_local;
    gchar *uri;

    g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
    g_return_if_fail (iter != NULL);
    g_return_if_fail (CAMEL_IS_STORE (store));
    g_return_if_fail (fi != NULL);

    si = folder_tree_model_store_index_lookup (model, store);
    g_return_if_fail (si != NULL);

    /* Make sure we don't already know about it. */
    if (g_hash_table_lookup (si->full_hash, fi->full_name)) {
        store_info_unref (si);
        return;
    }

    tree_store = GTK_TREE_STORE (model);

    session = em_folder_tree_model_get_session (model);
    folder_cache = e_mail_session_get_folder_cache (session);
    registry = e_mail_session_get_registry (session);

    uid = camel_service_get_uid (CAMEL_SERVICE (store));
    store_is_local = (g_strcmp0 (uid, E_MAIL_SESSION_LOCAL_UID) == 0);

    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);
    path_row = gtk_tree_row_reference_new (GTK_TREE_MODEL (model), path);
    gtk_tree_path_free (path);

    uri = e_mail_folder_uri_build (store, fi->full_name);

    g_hash_table_insert (
        si->full_hash, g_strdup (fi->full_name), path_row);

    store_info_unref (si);
    si = NULL;

    /* XXX If we have the folder, and its the Outbox folder, we need
     *     the total count, not unread.  We do the same for Drafts. */

    /* XXX This is duplicated in mail-folder-cache too, should perhaps
     *     be functionised. */
    unread = fi->unread;
    folder = mail_folder_cache_ref_folder (
        folder_cache, store, fi->full_name);
    if (folder != NULL) {
        folder_is_drafts = em_utils_folder_is_drafts (registry, folder);
        folder_is_outbox = em_utils_folder_is_outbox (registry, folder);

        if (folder_is_drafts || folder_is_outbox) {
            gint total;
            gint deleted;

            total = camel_folder_get_message_count (folder);
            deleted = camel_folder_get_deleted_message_count (folder);

            if (total > 0 && deleted != -1)
                total -= deleted;

            unread = MAX (total, 0);
        }

        g_object_unref (folder);
    }

    flags = fi->flags;
    display_name = fi->display_name;

    if (store_is_local) {
        if (strcmp (fi->full_name, "Drafts") == 0) {
            folder_is_drafts = TRUE;
            display_name = _("Drafts");
        } else if (strcmp (fi->full_name, "Templates") == 0) {
            folder_is_templates = TRUE;
            display_name = _("Templates");
        } else if (strcmp (fi->full_name, "Inbox") == 0) {
            flags = (flags & ~CAMEL_FOLDER_TYPE_MASK) |
                CAMEL_FOLDER_TYPE_INBOX;
            display_name = _("Inbox");
        } else if (strcmp (fi->full_name, "Outbox") == 0) {
            flags = (flags & ~CAMEL_FOLDER_TYPE_MASK) |
                CAMEL_FOLDER_TYPE_OUTBOX;
            display_name = _("Outbox");
        } else if (strcmp (fi->full_name, "Sent") == 0) {
            flags = (flags & ~CAMEL_FOLDER_TYPE_MASK) |
                CAMEL_FOLDER_TYPE_SENT;
            display_name = _("Sent");
        }
    }

    if ((flags & CAMEL_FOLDER_TYPE_MASK) == 0) {
        gchar *drafts_folder_uri;
        gchar *sent_folder_uri;

        folder_tree_model_get_drafts_folder_uri (
            registry, store, &drafts_folder_uri);

        folder_tree_model_get_sent_folder_uri (
            registry, store, &sent_folder_uri);

        if (!folder_is_drafts && drafts_folder_uri != NULL) {
            folder_is_drafts = e_mail_folder_uri_equal (
                CAMEL_SESSION (session),
                uri, drafts_folder_uri);
        }

        if (sent_folder_uri != NULL) {
            if (e_mail_folder_uri_equal (
                CAMEL_SESSION (session),
                uri, sent_folder_uri)) {
                add_flags = CAMEL_FOLDER_TYPE_SENT;
            }
        }

        g_free (drafts_folder_uri);
        g_free (sent_folder_uri);
    }

    /* 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 (folder_is_drafts)
            icon_name = "accessories-text-editor";
        else if (folder_is_templates)
            icon_name = "text-x-generic-template";
    }

    gtk_tree_store_set (
        tree_store, iter,
        COL_STRING_DISPLAY_NAME, display_name,
        COL_OBJECT_CAMEL_STORE, store,
        COL_STRING_FULL_NAME, fi->full_name,
        COL_STRING_ICON_NAME, icon_name,
        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, folder_is_drafts,
        -1);

    g_free (uri);
    uri = NULL;

    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_OBJECT_CAMEL_STORE, store,
            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_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, store, 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_tree_model_folder_created_cb (CamelStore *store,
                                     CamelFolderInfo *fi,
                                     StoreInfo *si)
{
    /* We only want created events to do more
     * work if we don't support subscriptions. */
    if (CAMEL_IS_SUBSCRIBABLE (store))
        return;

    if (g_hash_table_size (si->full_hash) > 0)
        folder_tree_model_folder_subscribed_cb (store, fi, si);
}

static void
folder_tree_model_folder_deleted_cb (CamelStore *store,
                                     CamelFolderInfo *fi,
                                     StoreInfo *si)
{
    /* We only want deleted events to do more
     * work if we don't support subscriptions. */
    if (CAMEL_IS_SUBSCRIBABLE (store))
        return;

    folder_tree_model_folder_unsubscribed_cb (store, fi, si);
}

static void
folder_tree_model_folder_renamed_cb (CamelStore *store,
                                     const gchar *old_name,
                                     CamelFolderInfo *info,
                                     StoreInfo *si)
{
    GtkTreeRowReference *reference;
    GtkTreeModel *model;
    GtkTreeIter root, iter;
    GtkTreePath *path;
    gchar *parent, *p;

    reference = g_hash_table_lookup (si->full_hash, old_name);
    if (!gtk_tree_row_reference_valid (reference))
        return;

    path = gtk_tree_row_reference_get_path (reference);
    model = gtk_tree_row_reference_get_model (reference);
    gtk_tree_model_get_iter (model, &iter, path);
    gtk_tree_path_free (path);

    folder_tree_model_remove_folders (
        EM_FOLDER_TREE_MODEL (model), si, &iter);

    /* Make sure we don't already have the new folder name. */
    reference = g_hash_table_lookup (si->full_hash, info->full_name);
    if (gtk_tree_row_reference_valid (reference))
        return;

    parent = g_strdup (info->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))
        return;

    path = gtk_tree_row_reference_get_path (reference);
    gtk_tree_model_get_iter (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 (
        EM_FOLDER_TREE_MODEL (model), &iter, store, info, TRUE);
}

static void
folder_tree_model_folder_info_stale_cb (CamelStore *store,
                                        StoreInfo *si)
{
    GtkTreeModel *model;

    if (!gtk_tree_row_reference_valid (si->row))
        return;

    model = gtk_tree_row_reference_get_model (si->row);

    /* Re-add the store.  The StoreInfo instance will be
     * discarded and the folder tree will be reconstructed. */
    em_folder_tree_model_add_store (EM_FOLDER_TREE_MODEL (model), store);
}

static void
folder_tree_model_folder_subscribed_cb (CamelStore *store,
                                        CamelFolderInfo *fi,
                                        StoreInfo *si)
{
    GtkTreeRowReference *reference;
    GtkTreeModel *model;
    GtkTreeIter parent, iter;
    GtkTreePath *path;
    gboolean load;
    gchar *dirname, *p;

    /* Make sure we don't already know about it? */
    if (g_hash_table_contains (si->full_hash, fi->full_name))
        return;

    /* Get our parent folder's path. */
    dirname = g_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))
        return;

    path = gtk_tree_row_reference_get_path (reference);
    model = gtk_tree_row_reference_get_model (reference);
    gtk_tree_model_get_iter (model, &parent, path);
    gtk_tree_path_free (path);

    /* Make sure parent's subfolders have already been loaded. */
    gtk_tree_model_get (
        model, &parent,
        COL_BOOL_LOAD_SUBDIRS, &load, -1);
    if (load)
        return;

    gtk_tree_store_append (GTK_TREE_STORE (model), &iter, &parent);

    em_folder_tree_model_set_folder_info (
        EM_FOLDER_TREE_MODEL (model), &iter, store, fi, TRUE);
}

static void
folder_tree_model_folder_unsubscribed_cb (CamelStore *store,
                                          CamelFolderInfo *fi,
                                          StoreInfo *si)
{
    GtkTreeRowReference *reference;
    GtkTreeModel *model;
    GtkTreePath *path;
    GtkTreeIter iter;

    reference = g_hash_table_lookup (si->full_hash, fi->full_name);
    if (!gtk_tree_row_reference_valid (reference))
        return;

    path = gtk_tree_row_reference_get_path (reference);
    model = gtk_tree_row_reference_get_model (reference);
    gtk_tree_model_get_iter (model, &iter, path);
    gtk_tree_path_free (path);

    folder_tree_model_remove_folders (
        EM_FOLDER_TREE_MODEL (model), si, &iter);
}

static void
folder_tree_model_update_status_icon (StoreInfo *si)
{
    CamelService *service;
    CamelServiceConnectionStatus status;
    GtkTreeModel *model;
    GtkTreePath *path;
    GtkTreeIter iter;
    GIcon *icon = NULL;
    const gchar *icon_name;
    gboolean was_connecting;
    gboolean host_reachable;

    g_return_if_fail (si != NULL);

    if (!gtk_tree_row_reference_valid (si->row))
        return;

    service = CAMEL_SERVICE (si->store);
    status = camel_service_get_connection_status (service);
    was_connecting = (si->last_status == CAMEL_SERVICE_CONNECTING);
    si->last_status = status;

    host_reachable = camel_network_service_get_host_reachable (
        CAMEL_NETWORK_SERVICE (service));

    switch (status) {
        case CAMEL_SERVICE_DISCONNECTED:
            if (!host_reachable)
                icon_name = "network-no-route-symbolic";
            else if (was_connecting)
                icon_name = "network-error-symbolic";
            else
                icon_name = "network-offline-symbolic";
            break;

        case CAMEL_SERVICE_CONNECTING:
            icon_name = NULL;
            break;

        case CAMEL_SERVICE_CONNECTED:
            icon_name = "network-idle-symbolic";
            break;

        case CAMEL_SERVICE_DISCONNECTING:
            icon_name = NULL;
            break;
    }

    if (icon_name == NULL && si->spinner_pulse_timeout_id == 0) {
        si->spinner_pulse_timeout_id = g_timeout_add_full (
            G_PRIORITY_DEFAULT,
            SPINNER_PULSE_INTERVAL,
            folder_tree_model_spinner_pulse_cb,
            store_info_ref (si),
            (GDestroyNotify) store_info_unref);
    }

    if (icon_name != NULL && si->spinner_pulse_timeout_id > 0) {
        g_source_remove (si->spinner_pulse_timeout_id);
        si->spinner_pulse_timeout_id = 0;
    }

    path = gtk_tree_row_reference_get_path (si->row);
    model = gtk_tree_row_reference_get_model (si->row);
    gtk_tree_model_get_iter (model, &iter, path);
    gtk_tree_path_free (path);

    if (icon_name != NULL) {
        /* Use fallbacks if symbolic icons are not available. */
        icon = g_themed_icon_new_with_default_fallbacks (icon_name);
    }

    gtk_tree_store_set (
        GTK_TREE_STORE (model), &iter,
        COL_STATUS_ICON, icon,
        COL_STATUS_ICON_VISIBLE, (icon_name != NULL),
        COL_STATUS_SPINNER_VISIBLE, (icon_name == NULL),
        -1);

    g_clear_object (&icon);

}

static void
folder_tree_model_status_notify_cb (CamelStore *store,
                                    GParamSpec *pspec,
                                    StoreInfo *si)
{
    /* Even though this is a GObject::notify signal, CamelService
     * always emits it from its GMainContext on the "main" thread,
     * so it's safe to modify the GtkTreeStore from here. */

    folder_tree_model_update_status_icon (si);
}

void
em_folder_tree_model_add_store (EMFolderTreeModel *model,
                                CamelStore *store)
{
    GtkTreeRowReference *reference;
    GtkTreeStore *tree_store;
    GtkTreeIter root, iter;
    GtkTreePath *path;
    CamelService *service;
    CamelProvider *provider;
    StoreInfo *si;
    const gchar *display_name;

    g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
    g_return_if_fail (CAMEL_IS_STORE (store));

    tree_store = GTK_TREE_STORE (model);

    service = CAMEL_SERVICE (store);
    provider = camel_service_get_provider (service);
    display_name = camel_service_get_display_name (service);

    /* Ignore stores that should not be added to the tree model. */

    if (provider == NULL)
        return;

    if ((provider->flags & CAMEL_PROVIDER_IS_STORAGE) == 0)
        return;

    if (em_utils_is_local_delivery_mbox_file (service))
        return;

    si = folder_tree_model_store_index_lookup (model, store);
    if (si != NULL) {
        em_folder_tree_model_remove_store (model, store);
        store_info_unref (si);
    }

    /* 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_OBJECT_CAMEL_STORE, store,
        COL_STRING_FULL_NAME, NULL,
        COL_BOOL_LOAD_SUBDIRS, TRUE,
        COL_BOOL_IS_STORE, TRUE,
        -1);

    path = gtk_tree_model_get_path (GTK_TREE_MODEL (model), &iter);
    reference = gtk_tree_row_reference_new (GTK_TREE_MODEL (model), path);

    si = store_info_new (model, store);
    si->row = reference;  /* takes ownership */

    folder_tree_model_store_index_insert (model, si);

    /* 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_OBJECT_CAMEL_STORE, store,
        COL_STRING_FULL_NAME, NULL,
        COL_BOOL_LOAD_SUBDIRS, FALSE,
        COL_BOOL_IS_STORE, FALSE,
        COL_BOOL_IS_FOLDER, FALSE,
        COL_UINT_UNREAD, 0,
        COL_UINT_UNREAD_LAST_SEL, 0,
        COL_BOOL_IS_DRAFT, FALSE,
        -1);

    if (CAMEL_IS_NETWORK_SERVICE (store))
        folder_tree_model_update_status_icon (si);

    g_signal_emit (model, signals[LOADED_ROW], 0, path, &root);
    gtk_tree_path_free (path);

    store_info_unref (si);
}

void
em_folder_tree_model_remove_store (EMFolderTreeModel *model,
                                   CamelStore *store)
{
    GtkTreePath *path;
    GtkTreeIter iter;
    StoreInfo *si;

    g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
    g_return_if_fail (CAMEL_IS_STORE (store));

    si = folder_tree_model_store_index_lookup (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 */
    folder_tree_model_remove_folders (model, si, &iter);

    store_info_unref (si);
}

GList *
em_folder_tree_model_list_stores (EMFolderTreeModel *model)
{
    GList *list;

    g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), NULL);

    g_mutex_lock (&model->priv->store_index_lock);

    list = g_hash_table_get_keys (model->priv->store_index);

    /* FIXME Listed CamelStores should be referenced here. */

    g_mutex_unlock (&model->priv->store_index_lock);

    return list;
}

gboolean
em_folder_tree_model_is_type_inbox (EMFolderTreeModel *model,
                                    CamelStore *store,
                                    const gchar *full)
{
    GtkTreeRowReference *reference;
    StoreInfo *si;
    guint32 flags = 0;

    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 = folder_tree_model_store_index_lookup (model, store);
    if (si == NULL)
        return FALSE;

    reference = g_hash_table_lookup (si->full_hash, full);

    if (gtk_tree_row_reference_valid (reference)) {
        GtkTreePath *path;
        GtkTreeIter iter;

        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);
    }

    store_info_unref (si);

    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)
{
    GtkTreeRowReference *reference;
    GtkTreePath *path;
    GtkTreeIter iter;
    StoreInfo *si;
    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 = folder_tree_model_store_index_lookup (model, store);
    if (si == NULL) {
        name = g_strdup (full);
        goto exit;
    }

    reference = g_hash_table_lookup (si->full_hash, full);
    if (!gtk_tree_row_reference_valid (reference)) {
        name = g_strdup (full);
        goto exit;
    }

    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);

exit:
    if (si != NULL)
        store_info_unref (si);

    return name;
}

/**
 * em_folder_tree_model_get_row_reference:
 * @model: an #EMFolderTreeModel
 * @store: a #CamelStore
 * @folder_name: a folder name, or %NULL
 *
 * Returns the #GtkTreeRowReference for the folder described by @store and
 * @folder_name.  If @folder_name is %NULL, returns the #GtkTreeRowReference
 * for the @store itself.  If no matching row is found, the function returns
 * %NULL.
 *
 * Returns: a valid #GtkTreeRowReference, or %NULL
 **/
GtkTreeRowReference *
em_folder_tree_model_get_row_reference (EMFolderTreeModel *model,
                                        CamelStore *store,
                                        const gchar *folder_name)
{
    GtkTreeRowReference *reference = NULL;
    StoreInfo *si;

    g_return_val_if_fail (EM_IS_FOLDER_TREE_MODEL (model), NULL);
    g_return_val_if_fail (CAMEL_IS_STORE (store), NULL);

    si = folder_tree_model_store_index_lookup (model, store);
    if (si == NULL)
        return NULL;

    if (folder_name != NULL)
        reference = g_hash_table_lookup (si->full_hash, folder_name);
    else
        reference = si->row;

    if (!gtk_tree_row_reference_valid (reference))
        reference = NULL;

    store_info_unref (si);

    return reference;
}

void
em_folder_tree_model_user_marked_unread (EMFolderTreeModel *model,
                                         CamelFolder *folder,
                                         guint n_marked)
{
    GtkTreeRowReference *reference;
    GtkTreePath *path;
    GtkTreeIter iter;
    CamelStore *parent_store;
    const gchar *folder_name;
    guint unread;

    /* The user marked messages in the given folder as unread.
     * Update our unread counts so we don't misinterpret this
     * event as new mail arriving. */

    g_return_if_fail (EM_IS_FOLDER_TREE_MODEL (model));
    g_return_if_fail (CAMEL_IS_FOLDER (folder));

    parent_store = camel_folder_get_parent_store (folder);
    folder_name = camel_folder_get_full_name (folder);

    reference = em_folder_tree_model_get_row_reference (
        model, parent_store, folder_name);

    g_return_if_fail (gtk_tree_row_reference_valid (reference));

    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_UNREAD, &unread, -1);

    unread += n_marked;

    gtk_tree_store_set (
        GTK_TREE_STORE (model), &iter,
        COL_UINT_UNREAD_LAST_SEL, unread,
        COL_UINT_UNREAD, unread, -1);
}