aboutsummaryrefslogblamecommitdiffstats
path: root/shell/e-shell-view.c
blob: 7d369a9037554598ddeb6a4fdbfba3405dc21c20 (plain) (tree)
1
2
3
4
5
6
7
8

                                                                           
  
                                        

                                                                
                                                                   
                                                        




                                                                    
  





                                                               


                                         
   

                    
                   

      


                      
                  
 
                 
                                
                                
                                    
                                         
                                 


                                  
 
                                 


                                    


                                         
                                         
 

                               

                                 
                       
                      

                                     
                          

                             
                               
                        
 


                              
 
                                              
 


                           
                        


                          


                           
 



                                                                
                                          
                                        
                                        
 


                                                   

                                       
 

                                                                            

                                



                                        


                              
                            
                          

                             
                                    
                               

                            
                                  
                                
                                         

                                        

                                       
                              

                                         
                                   
                            
 




                                                                             
                                


                                                                                      

                                          
 


                                                                       
 

                                          


      

                                        
                     
                   
  
 

                                          
                                      
 
                                      
                                      
 
                         
                          
 

                            












                                                                           
 
 

                                                                                   
                                                                                               

 






















                                      

                         




                                  
                                                                 



                                
 






















                                                                                                















                                                                                                  




















                                                                         
































                                                                                                       



































                                                                                                              
 


















                                                              







                                                                               
                                                            
 













                                                                                                             





                                                                                                                                         





                                                                           


 

                                  








                                                                      
 
           
                                                                        
 

                                

                                

                                           


                                                                                                  
                                                                                              



                                                      



                                                                                             







                                                     
 
                                              
                                               

                                                     

                                                                     
 
                                                                                                             



                                                                                             

 
          




                                                                           
                                

                                         

                                

                                                                       
                             

                                                  

                    


           

                                                            






                                         
                                                  
 
                                                                     
 
                                                        
                                                                                                             


           

                                                 


                                
                             



                                         

                                        





                                                          
                                                               
                                                                                                     


                       





                                                                                                      
                              
 








                                                                                                                   


           


                                          

                           


                                

                                            


                                                                               
                                                                   


                                                                                       
                                                                                  









                                                                                             

                                                                                            




                                                                                      
 

                                                     
                                                        


                                                                                              

 
 
 

                                            















                                                                    













                                                   








                                                    
                                              
                                               

                                                     
                                                           
                                                            



                             

                                                              
 
                                      
                                    
 
                                               
 
                                             
                                                           
                             


                                                                             





                                                                                                      


 

                 

















                                                                  


                                                                 
                                                               

                                                                   



                 
                                                                     
           


                                                    
                                             
                                 
 
                               
 
                                         
 


                                     
                                                                                                      

                                                                 
                
                                                           
         

 








                                                                      
                                                           

 



                                                                   
                               

                               

                                         
 
                                                             
                                                       

 






                                                                  
                   

                                         
 

                                                           
 
                      

 






















                                                                                                   


                                                          

 
                                                                          
           

                                                         




                                         

                                                                 

 
                                                            
           

                                                      





                                         


                       
                                                       


                                               






















                                                                 
 
                                   







































                                                                

                    
           
                                                    
 

                                    
                        
                                
 

                                

                                                                                         

                                                                              

                                                                               



                                                                                              
 
                                                       




                                                                      
 
                                                                           

                                       
 
                                                                     
 
                                                                                          
                                                                         
 

                                                                                              
 
                                           
                                                      
                                       


                                                  

                                                                               

 
           


                                             


















                                                                                     



                                                  






                                                                                              
                                              


                                













                                                                                              






                                                                                       


           
                                          
 
                                
 

                                

                                                   
 
                                          
                                           

                                    
 




                                                    
                                              







                                         










                                                                
 




                                                
 
 



                                              
                                  




                                            


                                                                             
                                                                                       

                                                                                          

 
 
           
                                      
 
                                
                                 
                            
 

                                


                                                                                        


                                                                                 


                                                                              


                                                                                    

                                    
                                                 
 




                                                                            




                                                                                                       

                                                                       

                                                  

                                                                                 
                                                                                



                                                                                    

                                            
                                                                                              
                                                                                 
 

                                                                             
                                                                                  

                                                                                         
 
                                       

                                                                                     
                                                                             
                                                                                  
 












                                                                                       
 

                          
                                               
                                             
                                                 
                                         


                                            
                                                 
                                           
 

                                   

                                         
 
                                                   
                                                                                             

 


                         
           


                                       
 


                              
 
                                           
 
                            
 
 
           
                           
 

                                
                 
 

                                           
 

                                                       


                                                              


                                                                  


                                                                            


                                                            









                                                                                                                     
                                                                                  
                                                 
 
                                                                 
 
                           
 




                                                              

                                         
                      
 
                                                              

 




                                   
 
                                     

                                                
 
                                        
 
                                                           
 

                                                                    

                                                     
                                                                                                       



                                                        

                                                                  

                                                     
                                                                                                     



                                                        











                                                                                    
                                                                          

                       
 
 



                                
 
                                            
 
                                             
                                             
                                             
                                                                                      
                                             
                                             
 

                                                            




                                             
 



                                             
 
                                             

                                             
                                             
                                             
 



                                              
 

                                          
 
                                                                                   
 
                                             
 


                                             
                                

 
 


                                              
                                                               



                                                         

                                                                        


           
                                                                 

                                                  

                                                                        

 
           
                                                                       












                                                             
                                                          

                                             









                                                        
           


                                                                               









                                                  
                                                                                                         
                                                             


           








                                                                         
                                                                         

 
           
                                            
                                    
                              


                                
                              



                                         
                                                              
                                                       

                       



                                                                  
 












                                                          








                                   
            
                                               
                                          
 
                                
                         
 



                                                                  
 

                                
                                                                                                                    





                                                           
                            
                                                        
 

                                                                     
 


                                                                                           
 

                                                                                     

                                                                                
 
                                                                   
                                                              
                                                                                                            
 
                                                              
 
                                                                                                    
 
                                   
 

                                                                                                   
                                             
 
                                                         
 
                                                            
 
                                                                                  


                                                                                                   


                                                                                                          
 

                                                                                                                  
 
                    
 
 



                                                                             
                                               
            
                                
 
                       
 

                                                        
 
                                                      
 
                                                                  

 












                                                                                  
 




                                               


                            









                                                              
                                                                      





                            
                                           
                                     

                                




                                


                                 
                                                          
 

                                                                                                       


                                         
                                                                            


                                
                                                                             
                
                                                                                     




                                   
                                                

                                           



                                                  


                                

                             


                                                                                      
                                                                                                          

         
                                
                                                                                                                   






                                                                                                          
         


           



                                               
                         
                         

                                

                                
                         


                                






                                                                  

                                                         
                           
                    
                         
 
                           
                              



                                                                                                
                                                                 
                                                                 
                                                                          



                                                                                                              

                                         
                                              
                                                                                   



                                                                                  
 






                                                                                  
                                   
                                                                                                 
            
                                                                                                                   
 

                                                                                             


                                                          
 



                                   


                                                                               



                                                                                                          




                                                                                 















                                                        
                                                                         





                                                                         
                                                                                        




                                             
                                                                          



                                                                        
                       






                                                                                          

                                                                                         


                                                  
 
                                
                              
                           

                                                     
                         
 

                                                 
 
                                                                

                                     
 
                                                      
                                                                             

                                                                                              
         
 
                                                                                                               





                                                                 





                                                                                      

 
           

                                              
 
                                
                                          

                                            

                                           
                                

                                                                                  
                                                      
 

                                                                                         
                                                                             

                                                                                           
                                                                             

                                                                                                 
                                                                             

                                                                                    
                                                                             


                                                                                               

                                                                   



                                                                      

 


                                   





                                                           
                   
                        
                                    
                                

                                





                                                                                                         

                                                            





                                                                                        

                                           
                                                           
                            
 













                                                                                                      
 
                                                                                        








                                                                


                                                            
 
                                                                                    
 
                                                                                                  
                               
                                                                                


 












                                                                

                            












                                                                         





                                                              

                            
 

                                                                          
 


                                                 
                                                                                                


                                         
 
                                
                             
                                                      
                                                  
                                               
                                     
                           
                          




                                     
                                  










                                 
                                                             






                                                                                         
 
                                                                                                      
 
                                                                                                         
                                               

                            
                                                                              

                                   
 
                                                                                                            
 

                                              
                            
         
 

                                   


                                              
                                                                           
                                                                                     










                                                                                                          
 
                                                    
 
                                       

 


                                           
                               


                                
 

                                
                                                                                             

                                       


                                   







                                                              
 
                   




                                

                                                  
                             
 


                                   
                                        
 
                                                                                      
 
                                                                                        


                                                         
                                                                 

                    

 



                                     
 
                                
                   
                        
 

                                





                                                                             
                                                          
 

                                                                            
                                                                                                                


                                                          

                                   
 

                              

         
                                                                             

                               

         



                                                                 
                                                       
                                                         


                                                                                                                        
                                                                   
                                                                                

                               

         


                      


                                                                    

                                               







                                                              
                                            
 
                                                        
 





                                                                             
                      
 
 









                                                                   
 
    

                                                       







                                                        
                                                

                       
                   

                                                                  
                                                                                             

                 
                                                                
                                                             
                                                                                    

                                                               
                 
                                                                 
         
 
                                           
 

                                                                                           


    

                                                     







                                                        
                                              

                       


                                                                                               
 


                                                                                       

                                                                                                                

                                                                      
                                                             
                                                                                              
                                                                     
                 
 

                                                                      


                                                                                       

                                                                                                                
         
 
                                         
 

                                                                                         

 

                                                        
 

                                                                   
 
                                                    

 

                                                      
 

                                                                   
 
                                                  

 
 








                                                                  








                                                                  

                                                             



                                                                  
                                              

 








                                                                  








                                                                  







                                                      








                                                                  

























                                                                                   

































                                                                                  
















                                                                                                            
 


                              

            
                                                                      




                                                   
                                         
 
                                 
                                
                                   
                        
                           
                       

                       
                         


                                                                   
 
                                







                                                                







                                                                                           




                                                                          
 

                                                                                           
                     
 

                                                                                             
                     
 
                                                           



                                                                                             


                                                               



                                                                                                  

                     
                                                         
                                                        
                        
                                                              
            
                                                                                   


                                                                           
 







                                                                              
 
                        
 






                                                                                             




                                                                                  
                          
 





                              

            
                                                                      




                                                   
                                         
 
                                 
                                

                                   
                           
                                                  


                                                                   
 
                                
                                                           
 
                                                 
 
                                                             
 

                                                                









                                                                             

                                                                     
                                                                       
                     
 
                                                           
                                                     
                                                       
                     
 
                                                             
                                                     
                                                         
                     
 

                                                           


                                                                   
                     
 

                                                               


                                                                        
                     
 

                                                             



                                                                                        
                                                                                
 
                           
                     
 
                                                                           
 











                                                                              




                                                                                  

                                                                    
                          
 


                    



































                                                                                                            
 
 









                                                                     
                                                                                                     


 
                                                                                          
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* e-shell-view.c
 *
 * Copyright (C) 2000, 2001 Ximian, Inc.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 *
 * 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
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public
 * License along with this program; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 *
 * Authors:
 *   Ettore Perazzoli <ettore@ximian.com>
 *   Miguel de Icaza <miguel@ximian.com>
 *   Matt Loper <matt@ximian.com>
 */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <ctype.h>

#include <glib.h>
#include <libgnome/gnome-defs.h>
#include <libgnome/gnome-i18n.h>
#include <libgnomeui/gnome-window.h>
#include <libgnomeui/gnome-window-icon.h>
#include <libgnomeui/gnome-app.h>
#include <bonobo/bonobo-socket.h>
#include <bonobo/bonobo-ui-util.h>
#include <bonobo/bonobo-widget.h>

#include <gal/e-paned/e-hpaned.h>
#include <gal/util/e-util.h>
#include <gal/widgets/e-gui-utils.h>
#include <gal/widgets/e-unicode.h>
#include <gal/widgets/e-scroll-frame.h>

#include "widgets/misc/e-clipped-label.h"
#include "widgets/misc/e-bonobo-widget.h"

#include "e-util/e-gtk-utils.h"

#include "evolution-shell-view.h"

#include "e-gray-bar.h"
#include "e-history.h"
#include "e-shell-constants.h"
#include "e-shell-folder-title-bar.h"
#include "e-shell-utils.h"
#include "e-shell.h"
#include "e-shortcuts-view.h"
#include "e-storage-set-view.h"
#include "e-title-bar.h"

#include "e-shell-view.h"
#include "e-shell-view-menu.h"


static BonoboWindowClass *parent_class = NULL;

struct _View {
    char *uri;
    GtkWidget *control;
    EFolder *folder;
};
typedef struct _View View;

struct _EShellViewPrivate {
    /* The shell.  */
    EShell *shell;

    /* EvolutionShellView Bonobo object for implementing the
           Evolution::ShellView interface.  */
    EvolutionShellView *corba_interface;

    /* The UI handler & container.  */
    BonoboUIComponent *ui_component;
    BonoboUIContainer *ui_container;

    /* History of visited (evolution:) URIs. */
    EHistory *history;

    /* Currently displayed URI.  */
    char *uri;

    /* Delayed selection, used when a path doesn't exist in an EStorage.
           Cleared when we're signaled with "folder_selected".  */
    char *delayed_selection;

    /* uri to go to at timeout */
    unsigned int set_folder_timeout;
    char        *set_folder_uri;

    /* Tooltips.  */
    GtkTooltips *tooltips;

    /* The widgetry.  */
    GtkWidget *appbar;
    GtkWidget *hpaned;
    GtkWidget *view_vbox;
    GtkWidget *folder_title_bar;
    GtkWidget *view_hpaned;
    GtkWidget *contents;
    GtkWidget *notebook;
    GtkWidget *shortcut_frame;
    GtkWidget *shortcut_bar;
    GtkWidget *storage_set_title_bar;
    GtkWidget *storage_set_view;
    GtkWidget *storage_set_view_box;

    /* The status bar widgetry.  */
    GtkWidget *status_bar;
    GtkWidget *offline_toggle;
    GtkWidget *offline_toggle_pixmap;
    GtkWidget *menu_hint_label;
    GtkWidget *task_bar;

    /* The pop-up window for the folder-tree (i.e. the one we create when
       the user clicks on the folder title.  */
    GtkWidget *folder_bar_popup;

    /* The views we have already open.  */
    GHashTable *uri_to_view;

    /* Position of the handles in the paneds, to be restored when we show elements
           after hiding them.  */
    unsigned int hpaned_position;
    unsigned int view_hpaned_position;

    /* Whether the shortcut and folder bars are visible or not.  */
    unsigned int shortcut_bar_shown : 1;
    unsigned int folder_bar_shown : 1;

    /* List of sockets we created.  */
    GList *sockets;
};

enum {
    SHORTCUT_BAR_VISIBILITY_CHANGED,
    FOLDER_BAR_VISIBILITY_CHANGED,
    VIEW_CHANGED,
    LAST_SIGNAL
};

static guint signals[LAST_SIGNAL] = { 0 };

#define DEFAULT_SHORTCUT_BAR_WIDTH 100

#define DEFAULT_TREE_WIDTH         130
#define MIN_POPUP_TREE_WIDTH       130

#define DEFAULT_WIDTH 705
#define DEFAULT_HEIGHT 550

#define SET_FOLDER_DELAY 250


/* The icons for the offline/online status.  */

static GdkPixmap *offline_pixmap = NULL;
static GdkBitmap *offline_mask = NULL;

static GdkPixmap *online_pixmap = NULL;
static GdkBitmap *online_mask = NULL;


static void        update_for_current_uri         (EShellView *shell_view);
static void        update_offline_toggle_status   (EShellView *shell_view);
static const char *get_storage_set_path_from_uri  (const char *uri);


/* Boo.  */
static void new_folder_cb (EStorageSet *storage_set, const char *path, void *data);
static gboolean display_uri (EShellView *shell_view, const char *uri, gboolean add_to_history);


/* View handling.  */

static View *
view_new (const char *uri,
      GtkWidget *control)
{
    View *new;

    new = g_new (View, 1);
    new->uri     = g_strdup (uri);
    new->control = control;

    return new;
}

static void
view_destroy (View *view)
{
    g_free (view->uri);
    g_free (view);
}


/* Utility functions.  */

static GtkWidget *
create_label_for_empty_page (void)
{
    GtkWidget *label;

    label = e_clipped_label_new (_("(No folder displayed)"));
    gtk_widget_show (label);

    return label;
}

/* Initialize the icons.  */
static void
load_images (void)
{
    GdkPixbuf *pixbuf;

    pixbuf = gdk_pixbuf_new_from_file (EVOLUTION_IMAGES "/offline.png");
    if (pixbuf == NULL) {
        g_warning ("Cannot load `%s'", EVOLUTION_IMAGES "/offline.png");
    } else {
        gdk_pixbuf_render_pixmap_and_mask (pixbuf, &offline_pixmap, &offline_mask, 128);
        gdk_pixbuf_unref (pixbuf);
    }

    pixbuf = gdk_pixbuf_new_from_file (EVOLUTION_IMAGES "/online.png");
    if (pixbuf == NULL) {
        g_warning ("Cannot load `%s'", EVOLUTION_IMAGES "/online.png");
    } else {
        gdk_pixbuf_render_pixmap_and_mask (pixbuf, &online_pixmap, &online_mask, 128);
        gdk_pixbuf_unref (pixbuf);
    }
}

static void
cleanup_delayed_selection (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;

    if (priv->delayed_selection != NULL) {
        g_free (priv->delayed_selection);
        priv->delayed_selection = NULL;
        gtk_signal_disconnect_by_func (GTK_OBJECT (e_shell_get_storage_set (priv->shell)),
                           GTK_SIGNAL_FUNC (new_folder_cb),
                           shell_view);
    }
}

static GtkWidget *
find_socket (GtkContainer *container)
{
    GList *children, *tmp;

    children = gtk_container_children (container);
    while (children) {
        if (BONOBO_IS_SOCKET (children->data))
            return children->data;
        else if (GTK_IS_CONTAINER (children->data)) {
            GtkWidget *socket = find_socket (children->data);
            if (socket)
                return socket;
        }
        tmp = children->next;
        g_list_free_1 (children);
        children = tmp;
    }
    return NULL;
}

static void
setup_verb_sensitivity_for_folder (EShellView *shell_view,
                   const char *path)
{
    EShellViewPrivate *priv;
    BonoboUIComponent *ui_component;
    const char *prop;

    priv = shell_view->priv;

    /* Adjust sensitivity for menu options depending on whether the folder
           selected is a stock folder.  */

    if (path == NULL) {
        prop = "0";
    } else {
        EFolder *folder;

        folder = e_storage_set_get_folder (e_shell_get_storage_set (priv->shell), path);
        if (folder != NULL && ! e_folder_get_is_stock (folder))
            prop = "1";
        else
            prop = "0";
    }

    ui_component = e_shell_view_get_bonobo_ui_component (shell_view);

    bonobo_ui_component_set_prop (ui_component, "/commands/MoveFolder", "sensitive", prop, NULL);
    bonobo_ui_component_set_prop (ui_component, "/commands/CopyFolder", "sensitive", prop, NULL);
    bonobo_ui_component_set_prop (ui_component, "/commands/DeleteFolder", "sensitive", prop, NULL);
    bonobo_ui_component_set_prop (ui_component, "/commands/RenameFolder", "sensitive", prop, NULL);
}


static void
update_navigation_buttons (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;

    e_shell_folder_title_bar_update_navigation_buttons (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar),
                                e_history_has_prev (priv->history),
                                e_history_has_next (priv->history));
}

static int
history_uri_matching_func (const void *a,
               const void *b)
{
    const char *s1, *s2;

    s1 = (const char *) a;
    s2 = (const char *) b;

    return strcmp (s1, s2);
}

static void
remove_uri_from_history (EShellView *shell_view,
             const char *uri)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;
    
    e_history_remove_matching (priv->history, uri, history_uri_matching_func);
}


/* Callbacks for the EStorageSet.  */

static void
storage_set_removed_folder_callback (EStorageSet *storage_set,
                     const char *path,
                     void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    GtkWidget *socket;
    View *view;
    int destroy_connection_id;
    int page_num;
    char *uri;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    uri = g_strconcat (E_SHELL_URI_PREFIX, path, NULL);

    remove_uri_from_history (shell_view, uri);
    update_navigation_buttons (shell_view);

    /* (Note that at this point the current URI in the history might have
       been changed and not match the current view.  But we catch this case
       when checking if this was the current view, below.)  */

    view = g_hash_table_lookup (priv->uri_to_view, uri);

    g_free (uri);

    if (view == NULL)
        return;

    socket = find_socket (GTK_CONTAINER (view->control));
    priv->sockets = g_list_remove (priv->sockets, socket);

    destroy_connection_id = GPOINTER_TO_INT (gtk_object_get_data (GTK_OBJECT (socket),
                                      "e_shell_view_destroy_connection_id"));
    gtk_signal_disconnect (GTK_OBJECT (socket), destroy_connection_id);

    page_num = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), view->control);

    if (strncmp (priv->uri, E_SHELL_URI_PREFIX, E_SHELL_URI_PREFIX_LEN) == 0
        && strcmp (priv->uri + E_SHELL_URI_PREFIX_LEN, path) == 0) {
        e_shell_view_display_uri (shell_view, "evolution:/local/Inbox");
    }

    bonobo_control_frame_control_deactivate (BONOBO_CONTROL_FRAME (bonobo_widget_get_control_frame (BONOBO_WIDGET (view->control))));
    gtk_widget_destroy (view->control);

    g_hash_table_remove (priv->uri_to_view, view->uri);
    view_destroy (view);

    gtk_notebook_remove_page (GTK_NOTEBOOK (priv->notebook), page_num);
}


/* Folder bar pop-up handling.  */

static void
reparent (GtkWidget *widget,
      GtkContainer *new_container)
{
    gtk_widget_ref (widget);
    gtk_container_remove (GTK_CONTAINER (widget->parent), widget);
    gtk_container_add (GTK_CONTAINER (new_container), widget);
    gtk_widget_unref (widget);
}

static void
reparent_storage_set_view_box_and_destroy_popup (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;

    if (priv->folder_bar_popup == NULL)
        return;

    gtk_widget_ref (priv->storage_set_view_box);
    gtk_container_remove (GTK_CONTAINER (priv->folder_bar_popup), priv->storage_set_view_box);
    e_paned_pack1 (E_PANED (priv->view_hpaned), priv->storage_set_view_box, FALSE, FALSE);
    gtk_widget_unref (priv->storage_set_view_box);

    gtk_widget_destroy (priv->folder_bar_popup);
    priv->folder_bar_popup = NULL;

    /* Re-enable DnD on the StorageSetView (it got disabled when displaying
       the pop-up).  */
    e_storage_set_view_set_allow_dnd (E_STORAGE_SET_VIEW (priv->storage_set_view), TRUE);
}

static void
popdown_transient_folder_bar (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;

    gdk_pointer_ungrab (GDK_CURRENT_TIME);
    gdk_keyboard_ungrab (GDK_CURRENT_TIME);
    gtk_grab_remove (priv->storage_set_view_box);

    reparent_storage_set_view_box_and_destroy_popup (shell_view);
    gtk_widget_hide (priv->storage_set_view_box);

    e_shell_folder_title_bar_set_toggle_state (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), FALSE);

    /* Re-enable DnD on the StorageSetView (it got disabled when displaying
       the pop-up).  */
    e_storage_set_view_set_allow_dnd (E_STORAGE_SET_VIEW (priv->storage_set_view), TRUE);
}

static int
storage_set_view_box_button_release_event_cb (GtkWidget *widget,
                          GdkEventButton *button_event,
                          void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    if (button_event->window == E_PANED (priv->view_hpaned)->handle
        || button_event->button != 1)
        return FALSE;

    popdown_transient_folder_bar (shell_view);

    return TRUE;
}

static void
popup_storage_set_view_button_clicked (ETitleBar *title_bar,
                       void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    g_assert (priv->folder_bar_popup != NULL);

    reparent_storage_set_view_box_and_destroy_popup (shell_view);

    e_shell_view_show_folder_bar (shell_view, TRUE);
    e_shell_folder_title_bar_set_toggle_state (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), FALSE);
}

static void
folder_bar_popup_map_callback (GtkWidget *widget,
                   void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    guint32 current_time;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    current_time = GDK_CURRENT_TIME;

    if (gdk_pointer_grab (widget->window, TRUE,
                  (GDK_BUTTON_PRESS_MASK
                   | GDK_BUTTON_RELEASE_MASK
                   | GDK_ENTER_NOTIFY_MASK
                   | GDK_LEAVE_NOTIFY_MASK
                   | GDK_POINTER_MOTION_MASK),
                  NULL, NULL, current_time) != 0) {
        g_warning ("e-shell-view.c:folder_bar_popup_map_callback() -- pointer grab failed.");
        return;
    }

    if (gdk_keyboard_grab (widget->window, TRUE, 0) != 0) {
        g_warning ("e-shell-view.c:folder_bar_popup_map_callback() -- keyboard grab failed.");
        gdk_pointer_ungrab (current_time);
        return;
    }

    gtk_grab_add (widget);

    gtk_signal_connect_while_alive (GTK_OBJECT (widget), "button_release_event",
                    GTK_SIGNAL_FUNC (storage_set_view_box_button_release_event_cb), shell_view,
                    GTK_OBJECT (priv->folder_bar_popup));
    gtk_signal_connect_while_alive (GTK_OBJECT (priv->storage_set_view), "button_release_event",
                    GTK_SIGNAL_FUNC (storage_set_view_box_button_release_event_cb), shell_view,
                    GTK_OBJECT (priv->folder_bar_popup));
    gtk_signal_connect_while_alive (GTK_OBJECT (priv->storage_set_title_bar), "button_clicked",
                    GTK_SIGNAL_FUNC (popup_storage_set_view_button_clicked), shell_view,
                    GTK_OBJECT (priv->folder_bar_popup));
}

static void
pop_up_folder_bar (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    int x, y;
    int orig_x, orig_y;

    priv = shell_view->priv;

    g_assert (! priv->folder_bar_shown);

    e_title_bar_set_button_mode (E_TITLE_BAR (priv->storage_set_title_bar),
                     E_TITLE_BAR_BUTTON_MODE_PIN);

    priv->folder_bar_popup = gtk_window_new (GTK_WINDOW_POPUP);

    /* We need to show the storage set view box and do a pointer grab to catch the
           mouse clicks.  But until the box is shown, we cannot grab.  So we connect to
           the "map" signal; `storage_set_view_box_map_cb()' will do the grab.  */
    gtk_signal_connect (GTK_OBJECT (priv->folder_bar_popup), "map",
                GTK_SIGNAL_FUNC (folder_bar_popup_map_callback), shell_view);

    x = priv->folder_title_bar->allocation.x;
    y = priv->folder_title_bar->allocation.y + priv->folder_title_bar->allocation.height;

    gdk_window_get_origin (priv->folder_title_bar->window, &orig_x, &orig_y);
    x += orig_x;
    y += orig_y + 2;

    priv->view_hpaned_position = MAX (priv->view_hpaned_position, MIN_POPUP_TREE_WIDTH);

    gtk_window_set_default_size (GTK_WINDOW (priv->folder_bar_popup),
                     priv->view_hpaned_position,
                     priv->view_hpaned->allocation.height);

    reparent (priv->storage_set_view_box, GTK_CONTAINER (priv->folder_bar_popup));

    gtk_widget_show (priv->storage_set_view_box);

    gtk_widget_popup (priv->folder_bar_popup, x, y);

    /* Disable DnD or "interesting" things will happen.  */
    e_storage_set_view_set_allow_dnd (E_STORAGE_SET_VIEW (priv->storage_set_view), FALSE);
}



/* Switching views on a tree view click.  */

static int
set_folder_timeout (gpointer data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    /* set to 0 so we don't remove it in _display_uri() */
    priv->set_folder_timeout = 0;
    e_shell_view_display_uri (shell_view, priv->set_folder_uri);

    return FALSE;
}

static int
popdown_transient_folder_bar_idle (void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    popdown_transient_folder_bar (shell_view);

    gtk_object_unref (GTK_OBJECT (shell_view));

    return FALSE;
}

static void
switch_on_folder_tree_click (EShellView *shell_view,
                 const char *path)
{
    EShellViewPrivate *priv;
    char *uri;

    priv = shell_view->priv;

    gdk_pointer_ungrab (GDK_CURRENT_TIME);
    gdk_keyboard_ungrab (GDK_CURRENT_TIME);
    gtk_grab_remove (priv->storage_set_view_box);

    uri = g_strconcat (E_SHELL_URI_PREFIX, path, NULL);
    if (priv->uri != NULL && !strcmp (uri, priv->uri)) {
        g_free (uri);
        return;
    }

    if (priv->set_folder_timeout != 0)
        gtk_timeout_remove (priv->set_folder_timeout);

    g_free (priv->set_folder_uri);
    priv->set_folder_uri = NULL;

    cleanup_delayed_selection (shell_view);

    if (priv->folder_bar_popup != NULL) {
        e_shell_view_display_uri (shell_view, uri);
        g_free (uri);

        gtk_object_ref (GTK_OBJECT (shell_view));
        gtk_idle_add (popdown_transient_folder_bar_idle, shell_view);
        return;
    }

    priv->set_folder_uri = uri;

    priv->set_folder_timeout = gtk_timeout_add (SET_FOLDER_DELAY, set_folder_timeout, shell_view);
}


/* Callbacks.  */

/* Callback when a new folder is added.  removed when we clear the
   delayed_selection */
static void
new_folder_cb (EStorageSet *storage_set,
           const char *path,
           void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    char *delayed_path;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    delayed_path = strchr (priv->delayed_selection, ':');
    if (delayed_path) {
        delayed_path ++;
        if (!strcmp(path, delayed_path)) {
            char *uri;

            uri = g_strdup (priv->delayed_selection);
            cleanup_delayed_selection (shell_view);
            e_shell_view_display_uri (shell_view, uri);
            g_free (uri);
        }
    }
}

/* Callback called when an icon on the shortcut bar gets clicked.  */
static void
activate_shortcut_cb (EShortcutsView *shortcut_view,
              EShortcuts *shortcuts,
              const char *uri,
              gboolean in_new_window,
              void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    if (in_new_window) {
        EShellView *new_view;

        new_view = e_shell_create_view (e_shell_view_get_shell (shell_view), uri, shell_view);
        e_shell_view_show_shortcut_bar (new_view, FALSE);
        e_shell_view_show_folder_bar (new_view, FALSE);
    } else {
        e_shell_view_display_uri (shell_view, uri);
    }
}

/* Callback when user chooses "Hide shortcut bar" via a right click */
static void
hide_requested_cb (EShortcutsView *shortcut_view,
           void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    e_shell_view_show_shortcut_bar (shell_view, FALSE);
}

/* Callback called when a folder on the tree view gets clicked.  */
static void
folder_selected_cb (EStorageSetView *storage_set_view,
            const char *path,
            void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    setup_verb_sensitivity_for_folder (shell_view, path);
    switch_on_folder_tree_click (shell_view, path);
}

/* Callback called when a storage in the tree view is clicked.  */
static void
storage_selected_cb (EStorageSetView *storage_set_view,
             const char *name,
             void *data)
{
    EShellView *shell_view;
    char *path;

    shell_view = E_SHELL_VIEW (data);

    path = g_strconcat (G_DIR_SEPARATOR_S, name, NULL);
    switch_on_folder_tree_click (shell_view, path);

    g_free (path);
}

/* Callbacks for the folder context menu in the folder bar.  */

static void
folder_context_menu_popping_up_cb (EStorageSetView *storage_set_view,
                   const char *path,
                   void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    setup_verb_sensitivity_for_folder (shell_view, path);
}

static void
folder_context_menu_popped_down_cb (EStorageSetView *storage_set_view,
                    void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    setup_verb_sensitivity_for_folder (shell_view, e_shell_view_get_current_path (shell_view));

    if (shell_view->priv->folder_bar_popup != NULL)
        popdown_transient_folder_bar (shell_view);
}

/* Callback called when the button on the tree's title bar is clicked.  */
static void
storage_set_view_button_clicked_cb (ETitleBar *title_bar,
                    void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    if (shell_view->priv->folder_bar_popup == NULL)
        e_shell_view_show_folder_bar (shell_view, FALSE);
}

/* Callback called when the title bar button is clicked.  */
static void
title_bar_toggled_cb (EShellFolderTitleBar *title_bar,
              gboolean state,
              void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);

    if (! state)
        return;

    if (shell_view->priv->folder_bar_popup == NULL)
        pop_up_folder_bar (shell_view);
}

/* Callback called when the offline toggle button is clicked.  */
static void
offline_toggle_clicked_cb (GtkButton *button,
               void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    switch (e_shell_get_line_status (priv->shell)) {
    case E_SHELL_LINE_STATUS_ONLINE:
        e_shell_go_offline (priv->shell, shell_view);
        break;
    case E_SHELL_LINE_STATUS_OFFLINE:
        e_shell_go_online (priv->shell, shell_view);
        break;
    default:
        g_assert_not_reached ();
    }
}


/* Navigation button callbacks.  */

static void
back_clicked_callback (EShellFolderTitleBar *title_bar,
               void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    const char *new_uri;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    if (! e_history_has_prev (priv->history))
        return;

    new_uri = (const char *) e_history_prev (priv->history);

    display_uri (shell_view, new_uri, FALSE);
}

static void
forward_clicked_callback (EShellFolderTitleBar *title_bar,
              void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    const char *new_uri;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    if (! e_history_has_next (priv->history))
        return;

    new_uri = (const char *) e_history_next (priv->history);

    display_uri (shell_view, new_uri, FALSE);
}


/* Widget setup.  */

static void
setup_storage_set_subwindow (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    GtkWidget *storage_set_view;
    GtkWidget *vbox;
    GtkWidget *scroll_frame;

    priv = shell_view->priv;

    storage_set_view = e_storage_set_view_new (e_shell_get_storage_set (priv->shell),
                           priv->ui_container);
    gtk_signal_connect (GTK_OBJECT (storage_set_view), "folder_selected",
                GTK_SIGNAL_FUNC (folder_selected_cb), shell_view);
    gtk_signal_connect (GTK_OBJECT (storage_set_view), "storage_selected",
                GTK_SIGNAL_FUNC (storage_selected_cb), shell_view);
    gtk_signal_connect (GTK_OBJECT (storage_set_view), "folder_context_menu_popping_up",
                GTK_SIGNAL_FUNC (folder_context_menu_popping_up_cb), shell_view);
    gtk_signal_connect (GTK_OBJECT (storage_set_view), "folder_context_menu_popped_down",
                GTK_SIGNAL_FUNC (folder_context_menu_popped_down_cb), shell_view);

    scroll_frame = e_scroll_frame_new (NULL, NULL);
    e_scroll_frame_set_policy (E_SCROLL_FRAME (scroll_frame),
                   GTK_POLICY_AUTOMATIC,
                   GTK_POLICY_AUTOMATIC);
    e_scroll_frame_set_shadow_type (E_SCROLL_FRAME (scroll_frame),
                    GTK_SHADOW_IN);

    gtk_container_add (GTK_CONTAINER (scroll_frame), storage_set_view);

    vbox = gtk_vbox_new (FALSE, 0);

    priv->storage_set_title_bar = e_title_bar_new (_("Folders"));

    gtk_box_pack_start (GTK_BOX (vbox), priv->storage_set_title_bar, FALSE, FALSE, 0);
    gtk_box_pack_start (GTK_BOX (vbox), scroll_frame, TRUE, TRUE, 0);

    gtk_signal_connect (GTK_OBJECT (priv->storage_set_title_bar), "button_clicked",
                GTK_SIGNAL_FUNC (storage_set_view_button_clicked_cb), shell_view);

    gtk_widget_show (storage_set_view);
    gtk_widget_show (priv->storage_set_title_bar);
    gtk_widget_show (scroll_frame);

    priv->storage_set_view_box = vbox;
    priv->storage_set_view = storage_set_view;

    /* Notice we don't show the vbox here yet.  By default it's hidden.  */
}

static void
setup_offline_toggle (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    GtkWidget *toggle;
    GtkWidget *pixmap;

    priv = shell_view->priv;

    toggle = gtk_button_new ();
    GTK_WIDGET_UNSET_FLAGS (toggle, GTK_CAN_FOCUS);
    gtk_button_set_relief (GTK_BUTTON (toggle), GTK_RELIEF_NONE);

    gtk_signal_connect (GTK_OBJECT (toggle), "clicked",
                GTK_SIGNAL_FUNC (offline_toggle_clicked_cb), shell_view);

    pixmap = gtk_pixmap_new (offline_pixmap, offline_mask);

    gtk_container_add (GTK_CONTAINER (toggle), pixmap);

    gtk_widget_show (toggle);
    gtk_widget_show (pixmap);

    priv->offline_toggle        = toggle;
    priv->offline_toggle_pixmap = pixmap;

    update_offline_toggle_status (shell_view);

    g_assert (priv->status_bar != NULL);

    gtk_box_pack_start (GTK_BOX (priv->status_bar), priv->offline_toggle, FALSE, TRUE, 0);
}

static void
setup_menu_hint_label (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;

    priv->menu_hint_label = gtk_label_new ("");
    gtk_misc_set_alignment (GTK_MISC (priv->menu_hint_label), 0.0, 0.5);

    gtk_box_pack_start (GTK_BOX (priv->status_bar), priv->menu_hint_label, TRUE, TRUE, 0);
}

static void
setup_task_bar (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;

    priv->task_bar = e_task_bar_new ();

    g_assert (priv->status_bar != NULL);

    gtk_box_pack_start (GTK_BOX (priv->status_bar), priv->task_bar, TRUE, TRUE, 0);
    gtk_widget_show (priv->task_bar);
}

static void
create_status_bar (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = shell_view->priv;

    priv->status_bar = gtk_hbox_new (FALSE, 2);
    gtk_widget_show (priv->status_bar);

    setup_offline_toggle (shell_view);
    setup_menu_hint_label (shell_view);
    setup_task_bar (shell_view);
}


/* Menu hints for the status bar.  */

static void
ui_engine_add_hint_callback (BonoboUIEngine *engine,
                 const char *hint,
                 void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    gtk_label_set (GTK_LABEL (priv->menu_hint_label), hint);
    gtk_widget_show (priv->menu_hint_label);
    gtk_widget_hide (priv->task_bar);
}

static void
ui_engine_remove_hint_callback (BonoboUIEngine *engine,
                void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    gtk_widget_hide (priv->menu_hint_label);
    gtk_widget_show (priv->task_bar);
}

static void
setup_statusbar_hints (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    BonoboUIEngine *ui_engine;

    priv = shell_view->priv;

    g_assert (priv->status_bar != NULL);

    ui_engine = bonobo_window_get_ui_engine (BONOBO_WINDOW (shell_view));
 
    gtk_signal_connect (GTK_OBJECT (ui_engine), "add_hint",
                GTK_SIGNAL_FUNC (ui_engine_add_hint_callback), shell_view);
    gtk_signal_connect (GTK_OBJECT (ui_engine), "remove_hint",
                GTK_SIGNAL_FUNC (ui_engine_remove_hint_callback), shell_view);
}


static void
setup_widgets (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    GtkWidget *contents_vbox;
    GtkWidget *gray_bar;

    priv = shell_view->priv;

    /* The shortcut bar.  */

    priv->shortcut_bar = e_shortcuts_new_view (e_shell_get_shortcuts (priv->shell));
    gtk_signal_connect (GTK_OBJECT (priv->shortcut_bar), "activate_shortcut",
                GTK_SIGNAL_FUNC (activate_shortcut_cb), shell_view);

    gtk_signal_connect (GTK_OBJECT (priv->shortcut_bar), "hide_requested",
                GTK_SIGNAL_FUNC (hide_requested_cb), shell_view);

    priv->shortcut_frame = gtk_frame_new (NULL);
    gtk_frame_set_shadow_type (GTK_FRAME (priv->shortcut_frame), GTK_SHADOW_IN);

    /* The storage set view.  */

    setup_storage_set_subwindow (shell_view);

    /* The tabless notebook which we used to contain the views.  */

    priv->notebook = gtk_notebook_new ();
    gtk_notebook_set_show_border (GTK_NOTEBOOK (priv->notebook), FALSE);
    gtk_notebook_set_show_tabs (GTK_NOTEBOOK (priv->notebook), FALSE);

    /* Page for "No URL displayed" message.  */

    gtk_notebook_append_page (GTK_NOTEBOOK (priv->notebook), create_label_for_empty_page (), NULL);

    /* Put things into a paned and the paned into the GnomeApp.  */

    priv->view_vbox = gtk_vbox_new (FALSE, 0);

    priv->folder_title_bar = e_shell_folder_title_bar_new ();
    gtk_signal_connect (GTK_OBJECT (priv->folder_title_bar), "title_toggled",
                GTK_SIGNAL_FUNC (title_bar_toggled_cb), shell_view);
    gtk_signal_connect (GTK_OBJECT (priv->folder_title_bar), "back_clicked",
                GTK_SIGNAL_FUNC (back_clicked_callback), shell_view);
    gtk_signal_connect (GTK_OBJECT (priv->folder_title_bar), "forward_clicked",
                GTK_SIGNAL_FUNC (forward_clicked_callback), shell_view);

    priv->view_hpaned = e_hpaned_new ();
    e_paned_pack1 (E_PANED (priv->view_hpaned), priv->storage_set_view_box, FALSE, FALSE);
    e_paned_pack2 (E_PANED (priv->view_hpaned), priv->notebook, TRUE, FALSE);

    gray_bar = e_gray_bar_new ();
    gtk_container_add (GTK_CONTAINER (gray_bar), priv->folder_title_bar);
    gtk_box_pack_start (GTK_BOX (priv->view_vbox), gray_bar, FALSE, FALSE, 2);

    gtk_box_pack_start (GTK_BOX (priv->view_vbox), priv->view_hpaned, TRUE, TRUE, 0);

    priv->hpaned = e_hpaned_new ();
    gtk_container_add (GTK_CONTAINER (priv->shortcut_frame), priv->shortcut_bar);
    e_paned_pack1 (E_PANED (priv->hpaned), priv->shortcut_frame, FALSE, FALSE);
    e_paned_pack2 (E_PANED (priv->hpaned), priv->view_vbox, TRUE, FALSE);
    e_paned_set_position (E_PANED (priv->hpaned), DEFAULT_SHORTCUT_BAR_WIDTH);

    /* The status bar.  */

    create_status_bar (shell_view);
    setup_statusbar_hints (shell_view);

    /* The contents.  */

    contents_vbox = gtk_vbox_new (FALSE, 0);
    gtk_box_pack_start (GTK_BOX (contents_vbox), priv->hpaned, TRUE, TRUE, 0);
    gtk_box_pack_start (GTK_BOX (contents_vbox), priv->status_bar, FALSE, TRUE, 0);
    gtk_widget_show (contents_vbox);

    bonobo_window_set_contents (BONOBO_WINDOW (shell_view), contents_vbox);

    /* Show stuff.  */

    gtk_widget_show (priv->shortcut_frame);
    gtk_widget_show (priv->shortcut_bar);
    gtk_widget_show (priv->storage_set_view);
    gtk_widget_show (priv->notebook);
    gtk_widget_show (priv->hpaned);
    gtk_widget_show (priv->view_hpaned);
    gtk_widget_show (priv->view_vbox);
    gtk_widget_show (priv->folder_title_bar);
    gtk_widget_show (priv->status_bar);

    gtk_widget_show (gray_bar);

    priv->shortcut_bar_shown = TRUE;
    priv->folder_bar_shown   = FALSE;

    /* FIXME: Session management and stuff?  */
    gtk_window_set_default_size (GTK_WINDOW (shell_view), DEFAULT_WIDTH, DEFAULT_HEIGHT);
}


/* GtkObject methods.  */

static void
hash_foreach_destroy_view (void *name,
               void *value,
               void *data)
{
    View *view;

    view = (View *) value;

    gtk_widget_destroy (view->control);

    view_destroy (view);
}

static void
destroy (GtkObject *object)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    GList *p;

    shell_view = E_SHELL_VIEW (object);
    priv = shell_view->priv;

    gtk_object_unref (GTK_OBJECT (priv->tooltips));

    if (priv->history != NULL)
        gtk_object_unref (GTK_OBJECT (priv->history));

    if (priv->shell != NULL)
        bonobo_object_unref (BONOBO_OBJECT (priv->shell));

    if (priv->corba_interface != NULL)
        bonobo_object_unref (BONOBO_OBJECT (priv->corba_interface));

    if (priv->folder_bar_popup != NULL)
        gtk_widget_destroy (priv->folder_bar_popup);

    for (p = priv->sockets; p != NULL; p = p->next) {
        GtkWidget *socket_widget;
        int destroy_connection_id;

        socket_widget = GTK_WIDGET (p->data);
        destroy_connection_id = GPOINTER_TO_INT (gtk_object_get_data (GTK_OBJECT (socket_widget),
                                          "e_shell_view_destroy_connection_id"));
        gtk_signal_disconnect (GTK_OBJECT (socket_widget), destroy_connection_id);
    }

    g_hash_table_foreach (priv->uri_to_view, hash_foreach_destroy_view, NULL);
    g_hash_table_destroy (priv->uri_to_view);

    bonobo_object_unref (BONOBO_OBJECT (priv->ui_component));

    g_free (priv->uri);

    if (priv->set_folder_timeout != 0)
        gtk_timeout_remove (priv->set_folder_timeout);

    g_free (priv->set_folder_uri);

    g_free (priv->delayed_selection);

    g_free (priv);

    (* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
}


/* Initialization.  */

static void
class_init (EShellViewClass *klass)
{
    GtkObjectClass *object_class;

    object_class = (GtkObjectClass *) klass;

    object_class->destroy = destroy;

    parent_class = gtk_type_class (BONOBO_TYPE_WINDOW);

    signals[SHORTCUT_BAR_VISIBILITY_CHANGED]
        = gtk_signal_new ("shortcut_bar_visibility_changed",
                  GTK_RUN_FIRST,
                  object_class->type,
                  GTK_SIGNAL_OFFSET (EShellViewClass, shortcut_bar_visibility_changed),
                  gtk_marshal_NONE__INT,
                  GTK_TYPE_NONE, 1,
                  GTK_TYPE_INT);

    signals[FOLDER_BAR_VISIBILITY_CHANGED]
        = gtk_signal_new ("folder_bar_visibility_changed",
                  GTK_RUN_FIRST,
                  object_class->type,
                  GTK_SIGNAL_OFFSET (EShellViewClass, folder_bar_visibility_changed),
                  gtk_marshal_NONE__INT,
                  GTK_TYPE_NONE, 1,
                  GTK_TYPE_INT);

    signals[VIEW_CHANGED]
        = gtk_signal_new ("view_changed",
                  GTK_RUN_FIRST,
                  object_class->type,
                  GTK_SIGNAL_OFFSET (EShellViewClass, view_changed),
                  e_marshal_NONE__POINTER_POINTER_POINTER_POINTER,
                  GTK_TYPE_NONE, 4,
                  GTK_TYPE_STRING,
                  GTK_TYPE_STRING,
                  GTK_TYPE_STRING,
                  GTK_TYPE_STRING);

    gtk_object_class_add_signals (object_class, signals, LAST_SIGNAL);

    load_images ();
}

static void
init (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    priv = g_new (EShellViewPrivate, 1);

    priv->shell                   = NULL;
    priv->corba_interface         = NULL;
    priv->ui_component            = NULL;
    priv->history                 = e_history_new ((EHistoryItemFreeFunc) g_free);
    priv->uri                     = NULL;
    priv->delayed_selection       = NULL;

    priv->tooltips                = gtk_tooltips_new ();

    priv->appbar                  = NULL;
    priv->hpaned                  = NULL;
    priv->view_hpaned             = NULL;
    priv->contents                = NULL;
    priv->notebook                = NULL;

    priv->storage_set_title_bar   = NULL;
    priv->storage_set_view        = NULL;
    priv->storage_set_view_box    = NULL;
    priv->shortcut_bar            = NULL;

    priv->status_bar              = NULL;
    priv->offline_toggle          = NULL;
    priv->offline_toggle_pixmap   = NULL;
    priv->menu_hint_label         = NULL;
    priv->task_bar                = NULL;

    priv->folder_bar_popup        = NULL;

    priv->shortcut_bar_shown      = FALSE;
    priv->folder_bar_shown        = FALSE;

    priv->hpaned_position         = 0;
    priv->view_hpaned_position    = 0;

    priv->uri_to_view             = g_hash_table_new (g_str_hash, g_str_equal);

    priv->sockets             = NULL;

    priv->set_folder_timeout      = 0;
    priv->set_folder_uri          = NULL;

    shell_view->priv = priv;
}


/* EvolutionShellView interface callbacks.  */

static void
corba_interface_set_message_cb (EvolutionShellView *shell_view,
                     const char *message,
                     gboolean busy,
                     void *data)
{
    /* Don't do anything here anymore.  The interface is going to be
       deprecated soon.  */
}

static void
corba_interface_unset_message_cb (EvolutionShellView *shell_view,
                       void *data)
{
    /* Don't do anything here anymore.  The interface is going to be
       deprecated soon.  */
}

static void
corba_interface_change_current_view_cb (EvolutionShellView *shell_view,
                         const char *uri,
                         void *data)
{
    EShellView *view;

    view = E_SHELL_VIEW (data);

    g_return_if_fail (view != NULL);

    e_shell_view_display_uri (view, uri);
}

static void
corba_interface_set_title (EvolutionShellView *shell_view,
               const char *title,
               void *data)
{
    EShellView *view;

    view = E_SHELL_VIEW (data);
    
    g_return_if_fail (view != NULL);

    gtk_window_set_title (GTK_WINDOW (view), title);
}

static void
corba_interface_set_folder_bar_label (EvolutionShellView *evolution_shell_view,
                      const char *text,
                      void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;

    g_return_if_fail (data != NULL);
    g_return_if_fail (E_IS_SHELL_VIEW (data));

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    e_shell_folder_title_bar_set_folder_bar_label (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar),
                               text);
}

static void
unmerge_on_error (BonoboObject *object,
          CORBA_Object  cobject,
          CORBA_Environment *ev)
{
    BonoboWindow *win;

    win = bonobo_ui_container_get_win (BONOBO_UI_CONTAINER (object));

    if (win)
        bonobo_window_deregister_component_by_ref (win, cobject);
}

static void
updated_folder_cb (EStorageSet *storage_set,
           const char *path,
           void *data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    const char *view_path;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    view_path = get_storage_set_path_from_uri (priv->uri);
    if (view_path && strcmp (path, view_path) != 0)
        return;

    /* Update the folder title bar and the window title bar */
    update_for_current_uri (shell_view);
}


/* Shell callbacks.  */

static void
shell_line_status_changed_cb (EShell *shell,
                  EShellLineStatus new_status,
                  void *data)
{
    EShellView *shell_view;

    shell_view = E_SHELL_VIEW (data);
    update_offline_toggle_status (shell_view);
}

static int
delete_event_cb (GtkWidget *widget,
         GdkEventAny *ev,
         void *data)
{
    return FALSE;
}


EShellView *
e_shell_view_construct (EShellView *shell_view,
            EShell     *shell)
{
    EShellViewPrivate *priv;
    EShellView *view;

    g_return_val_if_fail (shell != NULL, NULL);
    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL (shell), NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    priv = shell_view->priv;

    view = E_SHELL_VIEW (bonobo_window_construct (BONOBO_WINDOW (shell_view), "evolution", "Ximian Evolution"));

    if (!view) {
        gtk_object_unref (GTK_OBJECT (shell_view));
        return NULL;
    }       

    priv->shell = shell;
    bonobo_object_ref (BONOBO_OBJECT (priv->shell));

    gtk_signal_connect (GTK_OBJECT (view), "delete_event",
                GTK_SIGNAL_FUNC (delete_event_cb), NULL);

    gtk_signal_connect_while_alive (GTK_OBJECT (e_shell_get_storage_set (priv->shell)),
                    "updated_folder", updated_folder_cb, shell_view,
                    GTK_OBJECT (shell_view));

    priv->ui_container = bonobo_ui_container_new ();
    bonobo_ui_container_set_win (priv->ui_container, BONOBO_WINDOW (shell_view));
    gtk_signal_connect (GTK_OBJECT (priv->ui_container), "system_exception",
                GTK_SIGNAL_FUNC (unmerge_on_error), NULL);

    priv->ui_component = bonobo_ui_component_new ("evolution");
    bonobo_ui_component_set_container (priv->ui_component,
                       bonobo_object_corba_objref (BONOBO_OBJECT (priv->ui_container)));

    bonobo_ui_component_freeze (priv->ui_component, NULL);

    bonobo_ui_util_set_ui (priv->ui_component, EVOLUTION_DATADIR, "evolution.xml", "evolution");

    setup_widgets (shell_view);

    bonobo_ui_engine_config_set_path (bonobo_window_get_ui_engine (BONOBO_WINDOW (shell_view)),
                      "/evolution/UIConf/kvps");
    e_shell_view_menu_setup (shell_view);

    e_shell_view_show_folder_bar (shell_view, FALSE);

    bonobo_ui_component_thaw (priv->ui_component, NULL);

    gtk_signal_connect_while_alive (GTK_OBJECT (shell), "line_status_changed",
                    GTK_SIGNAL_FUNC (shell_line_status_changed_cb), shell_view,
                    GTK_OBJECT (shell_view));

    gtk_signal_connect_while_alive (GTK_OBJECT (e_shell_get_storage_set (shell)), "removed_folder",
                    GTK_SIGNAL_FUNC (storage_set_removed_folder_callback), shell_view,
                    GTK_OBJECT (shell_view));

    e_shell_user_creatable_items_handler_attach_menus (e_shell_get_user_creatable_items_handler (priv->shell),
                               shell_view);

    return view;
}

/* WARNING: Don't use `e_shell_view_new()' to create new views for the shell
   unless you know what you are doing; this is just the standard GTK+
   constructor thing and it won't allow the shell to do the required
   bookkeeping for the created views.  Instead, the right way to create a new
   view is calling `e_shell_create_view()'.  */
EShellView *
e_shell_view_new (EShell *shell)
{
    GtkWidget *new;

    g_return_val_if_fail (shell != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL (shell), NULL);

    new = gtk_type_new (e_shell_view_get_type ());

    return e_shell_view_construct (E_SHELL_VIEW (new), shell);
}

const GNOME_Evolution_ShellView
e_shell_view_get_corba_interface (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    g_return_val_if_fail (shell_view != NULL, CORBA_OBJECT_NIL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), CORBA_OBJECT_NIL);

    priv = shell_view->priv;

    return bonobo_object_corba_objref (BONOBO_OBJECT (priv->corba_interface));
}


static const char *
get_storage_set_path_from_uri (const char *uri)
{
    const char *colon;

    if (uri == NULL)
        return NULL;

    if (g_path_is_absolute (uri))
        return NULL;

    colon = strchr (uri, ':');
    if (colon == NULL || colon == uri || colon[1] == '\0')
        return NULL;

    if (! g_path_is_absolute (colon + 1))
        return NULL;

    if (g_strncasecmp (uri, E_SHELL_URI_PREFIX, colon - uri) != 0)
        return NULL;

    return colon + 1;
}

static void
update_window_icon (EShellView *shell_view,
            const char *type)
{
    EShellViewPrivate *priv;
    const char *icon_name;
    char *icon_path;

    priv = shell_view->priv;

    if (type == NULL) {
        icon_path = NULL;
    } else {
        EFolderTypeRegistry *folder_type_registry;

        folder_type_registry = e_shell_get_folder_type_registry (priv->shell);
        icon_name = e_folder_type_registry_get_icon_name_for_type (folder_type_registry, type);
        if (icon_name == NULL)
            icon_path = NULL;
        else
            icon_path = e_shell_get_icon_path (icon_name, TRUE);
    }

    if (icon_path == NULL) {
        gnome_window_icon_set_from_default (GTK_WINDOW (shell_view));
    } else {
        gnome_window_icon_set_from_file (GTK_WINDOW (shell_view), icon_path);
        g_free (icon_path);
    }
}

static void
update_folder_title_bar (EShellView *shell_view,
             const char *title,
             const char *type)
{
    EShellViewPrivate *priv;
    EFolderTypeRegistry *folder_type_registry;
    GdkPixbuf *folder_icon;

    priv = shell_view->priv;

    if (type == NULL) {
        title = NULL;
        folder_icon = NULL;
    } else {
        folder_type_registry = e_shell_get_folder_type_registry (priv->shell);
        folder_icon = e_folder_type_registry_get_icon_for_type (folder_type_registry, type, TRUE);
    }

    if (folder_icon != NULL)
        e_shell_folder_title_bar_set_icon (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), folder_icon);

    if (title != NULL) {
        char *s;

        s = e_utf8_to_gtk_string (GTK_WIDGET (priv->folder_title_bar), title);
        e_shell_folder_title_bar_set_title (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), s);
        g_free (s);
    }
}

static void
update_for_current_uri (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    EFolder *folder;
    const char *path;
    const char *type;
    const char *folder_name;
    char *title;
    char *utf8_window_title;
    char *gtk_window_title;
    int unread_count;

    priv = shell_view->priv;

    /* if we update when there is a timeout set, the selection
     * will jump around against the user's wishes.  so we just
     * return.
     */     
    if (priv->set_folder_timeout != 0)
        return;

    path = get_storage_set_path_from_uri (priv->uri);

    folder_name = NULL;
    type = NULL;
    unread_count = 0;

    if (path == NULL) {
        folder = NULL;
    } else {
        folder = e_storage_set_get_folder (e_shell_get_storage_set (priv->shell), path);

        if (folder != NULL) {
            folder_name = e_folder_get_name (folder);
            type = e_folder_get_type_string (folder);
            unread_count = e_folder_get_unread_count (folder);
        } else if (path != NULL) {
            EStorage *storage;

            storage = e_storage_set_get_storage (e_shell_get_storage_set (priv->shell), path + 1);
            unread_count = 0;

            if (storage != NULL) {
                folder_name = e_storage_get_display_name (storage);
                type = e_storage_get_toplevel_node_type (storage);
            }
        } 
    }

    if (unread_count > 0)
        title = g_strdup_printf (_("%s (%d)"), folder_name, unread_count);
    else if (folder_name == NULL)
        title = g_strdup (_("(None)"));
    else
        title = g_strdup (folder_name);

    if (SUB_VERSION[0] == '\0')
        utf8_window_title = g_strdup_printf ("%s - Ximian Evolution %s", title, VERSION);
    else
        utf8_window_title = g_strdup_printf ("%s - Ximian Evolution %s [%s]", title, VERSION, SUB_VERSION);

    gtk_window_title = e_utf8_to_gtk_string (GTK_WIDGET (shell_view), utf8_window_title);
    gtk_window_set_title (GTK_WINDOW (shell_view), gtk_window_title);

    update_folder_title_bar (shell_view, title, type);
    update_window_icon (shell_view, type);

    g_free (gtk_window_title);
    g_free (utf8_window_title);
    g_free (title);

    gtk_signal_handler_block_by_func (GTK_OBJECT (priv->storage_set_view),
                      GTK_SIGNAL_FUNC (folder_selected_cb),
                      shell_view);

    if (path != NULL)
        e_storage_set_view_set_current_folder (E_STORAGE_SET_VIEW (priv->storage_set_view), path);

    gtk_signal_handler_unblock_by_func (GTK_OBJECT (priv->storage_set_view),
                        GTK_SIGNAL_FUNC (folder_selected_cb),
                        shell_view);
}

static void
update_offline_toggle_status (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    GdkPixmap *icon_pixmap;
    GdkBitmap *icon_mask;
    const char *tooltip;
    gboolean sensitive;

    priv = shell_view->priv;

    switch (e_shell_get_line_status (priv->shell)) {
    case E_SHELL_LINE_STATUS_ONLINE:
        icon_pixmap = online_pixmap;
        icon_mask   = online_mask;
        sensitive   = TRUE;
        tooltip     = _("Ximian Evolution is currently online.  "
                "Click on this button to work offline.");
        break;
    case E_SHELL_LINE_STATUS_GOING_OFFLINE:
        icon_pixmap = online_pixmap;
        icon_mask   = online_mask;
        sensitive   = FALSE;
        tooltip     = _("Ximian Evolution is in the process of going offline.");
        break;
    case E_SHELL_LINE_STATUS_OFFLINE:
        icon_pixmap = offline_pixmap;
        icon_mask   = offline_mask;
        sensitive   = TRUE;
        tooltip     = _("Ximian Evolution is currently offline.  "
                "Click on this button to work online.");
        break;
    default:
        g_assert_not_reached ();
        return;
    }

    gtk_pixmap_set (GTK_PIXMAP (priv->offline_toggle_pixmap), icon_pixmap, icon_mask);
    gtk_widget_set_sensitive (priv->offline_toggle, sensitive);
    gtk_tooltips_set_tip (priv->tooltips, priv->offline_toggle, tooltip, NULL);
}

/* This displays the specified page, doing the appropriate Bonobo activation/deactivation
   magic to make sure things work nicely.  FIXME: Crappy way to solve the issue.  */
static void
set_current_notebook_page (EShellView *shell_view,
               int page_num)
{
    EShellViewPrivate *priv;
    GtkNotebook *notebook;
    GtkWidget *current;
    BonoboControlFrame *old_control_frame = NULL;
    BonoboControlFrame *new_control_frame;
    int current_page;

    priv = shell_view->priv;
    notebook = GTK_NOTEBOOK (priv->notebook);

    current_page = gtk_notebook_get_current_page (notebook);
    if (current_page == page_num)
        return;

    if (current_page != -1 && current_page != 0) {
        current = gtk_notebook_get_nth_page (notebook, current_page);
        old_control_frame = bonobo_widget_get_control_frame (BONOBO_WIDGET (current));
        bonobo_control_frame_set_autoactivate (old_control_frame, FALSE);
    }

    e_shell_folder_title_bar_set_folder_bar_label  (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), "");
    gtk_notebook_set_page (notebook, page_num);

    if (page_num == -1 || page_num == 0)
        return;

    current = gtk_notebook_get_nth_page (notebook, page_num);
    new_control_frame = bonobo_widget_get_control_frame (BONOBO_WIDGET (current));
    bonobo_control_frame_set_autoactivate (new_control_frame, FALSE);
 
    bonobo_control_frame_control_activate (new_control_frame);
    if (old_control_frame)
        bonobo_control_frame_control_deactivate (old_control_frame);
}

static void
setup_corba_interface (EShellView *shell_view,
               GtkWidget *control)
{
    EShellViewPrivate *priv;
    BonoboControlFrame *control_frame;
    EvolutionShellView *corba_interface;

    g_return_if_fail (control != NULL);

    priv = shell_view->priv;

    control_frame = bonobo_widget_get_control_frame (BONOBO_WIDGET (control));
    corba_interface = evolution_shell_view_new ();

    gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "set_message",
                    GTK_SIGNAL_FUNC (corba_interface_set_message_cb),
                    shell_view, GTK_OBJECT (shell_view));
    gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "unset_message",
                    GTK_SIGNAL_FUNC (corba_interface_unset_message_cb),
                    shell_view, GTK_OBJECT (shell_view));
    gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "change_current_view",
                    GTK_SIGNAL_FUNC (corba_interface_change_current_view_cb),
                    shell_view, GTK_OBJECT (shell_view));
    gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "set_title",
                    GTK_SIGNAL_FUNC (corba_interface_set_title),
                    shell_view, GTK_OBJECT (shell_view));
    gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "set_folder_bar_label",
                    GTK_SIGNAL_FUNC (corba_interface_set_folder_bar_label),
                    shell_view, GTK_OBJECT (shell_view));

    bonobo_object_add_interface (BONOBO_OBJECT (control_frame),
                     BONOBO_OBJECT (corba_interface));

    bonobo_object_ref (BONOBO_OBJECT (corba_interface));
    priv->corba_interface = corba_interface;
}


/* Socket destruction handling.  */

static void
socket_destroy_cb (GtkWidget *socket_widget, gpointer data)
{
    EShellView *shell_view;
    EShellViewPrivate *priv;
    EFolder *folder;
    View *view;
    const char *uri;
    gboolean viewing_closed_uri;
    const char *current_uri;
    const char *path;
    const char *folder_type;

    shell_view = E_SHELL_VIEW (data);
    priv = shell_view->priv;

    uri = (const char *) gtk_object_get_data (GTK_OBJECT (socket_widget), "e_shell_view_folder_uri");

    view = g_hash_table_lookup (priv->uri_to_view, uri);
    if (view == NULL) {
        g_warning ("What?! Destroyed socket for non-existing URI?  -- %s", uri);
        return;
    }

    priv->sockets = g_list_remove (priv->sockets, socket_widget);

    gtk_widget_destroy (view->control);

    g_hash_table_remove (priv->uri_to_view, view->uri);
    view_destroy (view);

    path = get_storage_set_path_from_uri (uri);
    folder = e_storage_set_get_folder (e_shell_get_storage_set (priv->shell), path);

    if (folder != NULL) {
        folder_type = e_folder_get_type_string (folder);
    } else {
        EStorage *storage;

        storage = e_storage_set_get_storage (e_shell_get_storage_set (priv->shell), path + 1);
        if (storage == NULL)
            folder_type = NULL;
        else
            folder_type = e_storage_get_toplevel_node_type (storage);
    }

    /* See if we were actively viewing the uri for the socket that's being closed */
    current_uri = e_shell_view_get_current_uri (shell_view);
    if (current_uri == NULL) {
        viewing_closed_uri = FALSE;
    } else {
        if (strcmp (uri, current_uri) == 0)
            viewing_closed_uri = TRUE;
        else
            viewing_closed_uri = FALSE;
    }

    if (viewing_closed_uri)
        e_shell_view_display_uri (shell_view, NULL);

    e_shell_component_maybe_crashed (priv->shell, uri, folder_type, shell_view);

    /* We were actively viewing the component that just crashed, so flip to the default URI */
    if (viewing_closed_uri)
        e_shell_view_display_uri (shell_view, E_SHELL_VIEW_DEFAULT_URI);
}


static const char *
get_type_for_storage (EShellView *shell_view,
              const char *name,
              const char **physical_uri_return)
{
    EShellViewPrivate *priv;
    EStorageSet *storage_set;
    EStorage *storage;

    priv = shell_view->priv;

    storage_set = e_shell_get_storage_set (priv->shell);
    storage = e_storage_set_get_storage (storage_set, name);
    if (!storage)
        return NULL;

    *physical_uri_return = e_storage_get_toplevel_node_uri (storage);

    return e_storage_get_toplevel_node_type (storage);
}

static const char *
get_type_for_folder (EShellView *shell_view,
             const char *path,
             const char **physical_uri_return)
{
    EShellViewPrivate *priv;
    EStorageSet *storage_set;
    EFolder *folder;

    priv = shell_view->priv;

    storage_set = e_shell_get_storage_set (priv->shell);
    folder = e_storage_set_get_folder (storage_set, path);
    if (!folder)
        return NULL;

    if (physical_uri_return != NULL)
        *physical_uri_return = e_folder_get_physical_uri (folder);

    return e_folder_get_type_string (folder);
}

/* Create a new view for @uri with @control.  It assumes a view for @uri does not exist yet.  */
static View *
get_view_for_uri (EShellView *shell_view,
          const char *uri)
{
    EShellViewPrivate *priv;
    CORBA_Environment ev;
    EvolutionShellComponentClient *handler_client;
    EFolderTypeRegistry *folder_type_registry;
    GNOME_Evolution_ShellComponent handler;
    Bonobo_UIContainer container;
    GtkWidget *control;
    GtkWidget *socket;
    Bonobo_Control corba_control;
    const char *path;
    const char *slash;
    const char *physical_uri;
    const char *folder_type;
    int destroy_connection_id;

    priv = shell_view->priv;

    path = strchr (uri, ':');
    if (path == NULL)
        return NULL;

    path++;
    if (*path == '\0')
        return NULL;

    /* FIXME: This code needs to be made more robust.  */
    slash = strchr (path + 1, G_DIR_SEPARATOR);
    if (slash == NULL || slash[1] == '\0')
        folder_type = get_type_for_storage (shell_view, path + 1, &physical_uri);
    else
        folder_type = get_type_for_folder (shell_view, path, &physical_uri);
    if (folder_type == NULL)
        return NULL;

    folder_type_registry = e_shell_get_folder_type_registry (e_shell_view_get_shell (shell_view));

    handler_client = e_folder_type_registry_get_handler_for_type (folder_type_registry, folder_type);
    if (handler_client == CORBA_OBJECT_NIL)
        return NULL;

    handler = bonobo_object_corba_objref (BONOBO_OBJECT (handler_client));

    CORBA_exception_init (&ev);

    corba_control = GNOME_Evolution_ShellComponent_createView (handler, physical_uri, folder_type, &ev);

    if (ev._major != CORBA_NO_EXCEPTION) {
        CORBA_exception_free (&ev);
        return NULL;
    }

    CORBA_exception_free (&ev);

    if (corba_control == CORBA_OBJECT_NIL)
        return NULL;

    container = bonobo_ui_component_get_container (priv->ui_component);
    control = e_bonobo_widget_new_control_from_objref (corba_control, container);

    socket = find_socket (GTK_CONTAINER (control));
    destroy_connection_id = gtk_signal_connect (GTK_OBJECT (socket), "destroy",
                            GTK_SIGNAL_FUNC (socket_destroy_cb),
                            shell_view);
    gtk_object_set_data (GTK_OBJECT (socket),
                 "e_shell_view_destroy_connection_id",
                 GINT_TO_POINTER (destroy_connection_id));
    gtk_object_set_data_full (GTK_OBJECT (socket), "e_shell_view_folder_uri", g_strdup (uri), g_free);

    priv->sockets = g_list_prepend (priv->sockets, socket);

    setup_corba_interface (shell_view, control);

    return view_new (uri, control);
}

static gboolean
show_existing_view (EShellView *shell_view,
            const char *uri,
            View *view)
{
    EShellViewPrivate *priv;
    int notebook_page;

    priv = shell_view->priv;

    notebook_page = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), view->control);
    g_assert (notebook_page != -1);

    g_free (priv->uri);
    priv->uri = g_strdup (uri);

    set_current_notebook_page (shell_view, notebook_page);

    return TRUE;
}

static gboolean
create_new_view_for_uri (EShellView *shell_view,
             const char *uri)
{
    View *view;
    EShellViewPrivate *priv;
    int page_num;

    priv = shell_view->priv;

    view = get_view_for_uri (shell_view, uri);
    if (view == NULL)
        return FALSE;

    g_free (priv->uri);
    priv->uri = g_strdup (uri);

    gtk_widget_show (view->control);

    gtk_notebook_append_page (GTK_NOTEBOOK (priv->notebook), view->control, NULL);

    page_num = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), view->control);
    g_assert (page_num != -1);
    set_current_notebook_page (shell_view, page_num);

    g_hash_table_insert (priv->uri_to_view, view->uri, view);

    return TRUE;
}

static gboolean
display_uri (EShellView *shell_view,
         const char *uri,
         gboolean add_to_history)
{
    EShellViewPrivate *priv;
    View *view;
    gboolean retval;

    priv = shell_view->priv;

    if (uri == NULL && priv->uri == NULL)
        return TRUE;

    if (priv->uri != NULL && uri != NULL && strcmp (priv->uri, uri) == 0)
        return TRUE;

    bonobo_window_freeze (BONOBO_WINDOW (shell_view));

    if (uri == NULL) {
        gtk_notebook_remove_page (GTK_NOTEBOOK (priv->notebook), 0);
        gtk_notebook_prepend_page (GTK_NOTEBOOK (priv->notebook), create_label_for_empty_page (), NULL);

        set_current_notebook_page (shell_view, 0);

        g_free (priv->uri);
        priv->uri = NULL;

        retval = TRUE;
        goto end;
    }

    if (strncmp (uri, E_SHELL_URI_PREFIX, E_SHELL_URI_PREFIX_LEN) != 0) {
        retval = FALSE;
        goto end;
    }

    view = g_hash_table_lookup (priv->uri_to_view, uri);
    if (view != NULL) {
        show_existing_view (shell_view, uri, view);
    } else if (! create_new_view_for_uri (shell_view, uri)) {
        cleanup_delayed_selection (shell_view);
        priv->delayed_selection = g_strdup (uri);
        e_gtk_signal_connect_full_while_alive (GTK_OBJECT (e_shell_get_storage_set (priv->shell)), "new_folder",
                               GTK_SIGNAL_FUNC (new_folder_cb), NULL,
                               shell_view, NULL,
                               FALSE, TRUE,
                               GTK_OBJECT (shell_view));
        retval = FALSE;
        goto end;
    }

    retval = TRUE;

 end:
    if (add_to_history && retval == TRUE && priv->uri != NULL)
        e_history_add (priv->history, g_strdup (priv->uri));

    update_navigation_buttons (shell_view);

    g_free (priv->set_folder_uri);
    priv->set_folder_uri = NULL;

    if (priv->set_folder_timeout != 0) {
        gtk_timeout_remove (priv->set_folder_timeout);
        priv->set_folder_timeout = 0;
    }

    update_for_current_uri (shell_view);

    bonobo_window_thaw (BONOBO_WINDOW (shell_view));

    gtk_signal_emit (GTK_OBJECT (shell_view), signals[VIEW_CHANGED],
             e_shell_view_get_current_path (shell_view),
             e_shell_view_get_current_uri (shell_view),
             e_shell_view_get_current_folder_type (shell_view),
             e_shell_view_get_current_component_id (shell_view));

    return retval;
}

gboolean
e_shell_view_display_uri (EShellView *shell_view,
              const char *uri)
{
    g_return_val_if_fail (shell_view != NULL, FALSE);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE);

    return display_uri (shell_view, uri, TRUE);
}


void
e_shell_view_show_shortcut_bar (EShellView *shell_view,
                gboolean show)
{
    EShellViewPrivate *priv;

    g_return_if_fail (shell_view != NULL);
    g_return_if_fail (E_IS_SHELL_VIEW (shell_view));

    priv = shell_view->priv;

    if (!! show == priv->shortcut_bar_shown)
        return;

    if (show) {
        if (! GTK_WIDGET_VISIBLE (priv->shortcut_frame)) {
            gtk_widget_show (priv->shortcut_frame);
            e_paned_set_position (E_PANED (priv->hpaned), priv->hpaned_position);
        }
    } else {
        if (GTK_WIDGET_VISIBLE (priv->shortcut_frame)) {
            /* FIXME this is a private field!  */
            priv->hpaned_position = E_PANED (priv->hpaned)->child1_size;

            gtk_widget_hide (priv->shortcut_frame);
        }
        e_paned_set_position (E_PANED (priv->hpaned), 0);
    }

    priv->shortcut_bar_shown = !! show;

    gtk_signal_emit (GTK_OBJECT (shell_view), signals[SHORTCUT_BAR_VISIBILITY_CHANGED],
             priv->shortcut_bar_shown);
}

void
e_shell_view_show_folder_bar (EShellView *shell_view,
                  gboolean show)
{
    EShellViewPrivate *priv;

    g_return_if_fail (shell_view != NULL);
    g_return_if_fail (E_IS_SHELL_VIEW (shell_view));

    priv = shell_view->priv;

    if (!! show == priv->folder_bar_shown)
        return;

    if (show) {
        gtk_widget_show (priv->storage_set_view_box);
        e_paned_set_position (E_PANED (priv->view_hpaned), priv->view_hpaned_position);

        e_title_bar_set_button_mode (E_TITLE_BAR (priv->storage_set_title_bar),
                         E_TITLE_BAR_BUTTON_MODE_CLOSE);

        e_shell_folder_title_bar_set_title_clickable (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar),
                                  FALSE);
    } else {
        if (GTK_WIDGET_VISIBLE (priv->storage_set_view_box)) {
            /* FIXME this is a private field!  */
            priv->view_hpaned_position = E_PANED (priv->view_hpaned)->child1_size;
            gtk_widget_hide (priv->storage_set_view_box);
        }

        e_paned_set_position (E_PANED (priv->view_hpaned), 0);

        e_title_bar_set_button_mode (E_TITLE_BAR (priv->storage_set_title_bar),
                         E_TITLE_BAR_BUTTON_MODE_PIN);

        e_shell_folder_title_bar_set_title_clickable (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar),
                                  TRUE);
    }

        priv->folder_bar_shown = !! show;

    gtk_signal_emit (GTK_OBJECT (shell_view), signals[FOLDER_BAR_VISIBILITY_CHANGED],
             priv->folder_bar_shown);
}

gboolean
e_shell_view_shortcut_bar_shown (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, FALSE);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE);

    return shell_view->priv->shortcut_bar_shown;
}

gboolean
e_shell_view_folder_bar_shown (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, FALSE);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE);

    return shell_view->priv->folder_bar_shown;
}


ETaskBar *
e_shell_view_get_task_bar (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return E_TASK_BAR (shell_view->priv->task_bar);
}

EShell *
e_shell_view_get_shell (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return shell_view->priv->shell;
}

BonoboUIComponent *
e_shell_view_get_bonobo_ui_component (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return shell_view->priv->ui_component;
}

BonoboUIContainer *
e_shell_view_get_bonobo_ui_container (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return shell_view->priv->ui_container;
}

GtkWidget *
e_shell_view_get_appbar (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return shell_view->priv->appbar;
}

/**
 * e_shell_view_get_current_uri:
 * @shell_view: A pointer to an EShellView object
 * 
 * Get the URI currently displayed by this shell view.
 * 
 * Return value: 
 **/
const char *
e_shell_view_get_current_uri (EShellView *shell_view)
{
    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return shell_view->priv->uri;
}

/**
 * e_shell_view_get_current_path:
 * @shell_view: A pointer to an EShellView object
 * 
 * Get the path of the current displayed folder.
 * 
 * Return value: 
 **/
const char *
e_shell_view_get_current_path (EShellView *shell_view)
{
    const char *current_uri;
    const char *current_path;

    current_uri = e_shell_view_get_current_uri (shell_view);
    if (current_uri == NULL)
        return NULL;

    if (strncmp (current_uri, E_SHELL_URI_PREFIX, E_SHELL_URI_PREFIX_LEN) == 0)
        current_path = current_uri + E_SHELL_URI_PREFIX_LEN;
    else
        current_path = NULL;

    return current_path;
}

const char *
e_shell_view_get_current_physical_uri (EShellView *shell_view)
{
    const char *current_path;
    const char *physical_uri;

    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    current_path = e_shell_view_get_current_path (shell_view);
    if (current_path == NULL)
        return NULL;

    if (get_type_for_folder (shell_view, current_path, &physical_uri) == NULL)
        return NULL;
    else
        return physical_uri;
}

const char *
e_shell_view_get_current_folder_type (EShellView *shell_view)
{
    const char *current_path;

    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    current_path = e_shell_view_get_current_path (shell_view);
    if (current_path == NULL)
        return NULL;

    return get_type_for_folder (shell_view, current_path, NULL);
}

const char *
e_shell_view_get_current_component_id (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    EFolderTypeRegistry *type_registry;
    EvolutionShellComponentClient *component_client;
    const char *current_folder_type;

    priv = shell_view->priv;

    type_registry = e_shell_get_folder_type_registry (priv->shell);
    current_folder_type = e_shell_view_get_current_folder_type (shell_view);
    component_client = e_folder_type_registry_get_handler_for_type (type_registry, current_folder_type);

    return evolution_shell_component_client_get_id (component_client);
}


/**
 * e_shell_view_save_settings:
 * @shell_view: 
 * @prefix: 
 * 
 * Save settings for @shell_view at the specified gnome config @prefix
 * 
 * Return value: TRUE if successful, FALSE if not.
 **/
gboolean
e_shell_view_save_settings (EShellView *shell_view,
                int view_num)
{
    Bonobo_ConfigDatabase db;
    EShellViewPrivate *priv;
    EShortcutBar *shortcut_bar;
    const char *uri;
    char *prefix, *key;
    char *filename;
    int num_groups;
    int group;
    struct stat temp;

    g_return_val_if_fail (shell_view != NULL, FALSE);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE);

    priv = shell_view->priv;
    shortcut_bar = E_SHORTCUT_BAR (priv->shortcut_bar);

    db = e_shell_get_config_db (priv->shell);

    g_return_val_if_fail (db != CORBA_OBJECT_NIL, FALSE);

    prefix = g_strdup_printf ("/Shell/Views/%d/", view_num);

    key = g_strconcat (prefix, "Width", NULL);
    bonobo_config_set_long (db, key, GTK_WIDGET (shell_view)->allocation.width, NULL);
    g_free (key);

    key = g_strconcat (prefix, "Height", NULL);
    bonobo_config_set_long (db, key, GTK_WIDGET (shell_view)->allocation.height, NULL);
    g_free (key);

    key = g_strconcat (prefix, "CurrentShortcutsGroupNum", NULL);
    bonobo_config_set_long (db, key, 
            e_shell_view_get_current_shortcuts_group_num (shell_view),
        NULL);
    g_free (key);

    key = g_strconcat (prefix, "FolderBarShown", NULL);
    bonobo_config_set_long (db, key, e_shell_view_folder_bar_shown (shell_view), NULL);
    g_free (key);

    key = g_strconcat (prefix, "ShortcutBarShown", NULL);
    bonobo_config_set_long (db, key, e_shell_view_shortcut_bar_shown (shell_view), NULL);
    g_free (key);

    key = g_strconcat (prefix, "HPanedPosition", NULL);
    if (GTK_WIDGET_VISIBLE (priv->shortcut_frame))
        bonobo_config_set_long (db, key, E_PANED (priv->hpaned)->child1_size, NULL); 
    else
        bonobo_config_set_long (db, key, priv->hpaned_position, NULL);
    g_free (key);

    key = g_strconcat (prefix, "ViewHPanedPosition", NULL);
    if (GTK_WIDGET_VISIBLE (priv->storage_set_view_box))
        bonobo_config_set_long (db, key, E_PANED (priv->view_hpaned)->child1_size, NULL); 
    else
        bonobo_config_set_long (db, key, priv->view_hpaned_position, NULL);
    g_free (key);

    key = g_strconcat (prefix, "DisplayedURI", NULL);
    uri = e_shell_view_get_current_uri (shell_view);
    if (uri != NULL)
        bonobo_config_set_string (db, key, uri, NULL);
    else
        bonobo_config_set_string (db, key, E_SHELL_VIEW_DEFAULT_URI, NULL);
    g_free (key);

    num_groups = e_shortcut_model_get_num_groups (shortcut_bar->model);

    for (group = 0; group < num_groups; group++) {
        key = g_strdup_printf ("%sShortcutBarGroup%dIconMode", prefix,
                       group);
        bonobo_config_set_long (db, key,
                e_shortcut_bar_get_view_type (shortcut_bar, group),
            NULL);
        g_free (key);
    }

    g_free (prefix);

    /* If ~/evolution/config/ doesn't exist yet, make it */
    filename = g_strdup_printf ("%s/config/", e_shell_get_local_directory (priv->shell));
    if (stat (filename, &temp) != 0)
        mkdir (filename, S_IRWXU);
    g_free (filename);

    /* Save the expanded state for this ShellView's StorageSetView */
    filename = g_strdup_printf ("%s/config/storage-set-view-expanded:view_%d",
                    e_shell_get_local_directory (priv->shell),
                    view_num);
    e_tree_save_expanded_state (E_TREE (priv->storage_set_view),
                    filename);
    g_free (filename);

    return TRUE;
}

/**
 * e_shell_view_load_settings:
 * @shell_view: 
 * @prefix: 
 * 
 * Load settings for @shell_view at the specified gnome config @prefix
 * 
 * Return value: 
 **/
gboolean
e_shell_view_load_settings (EShellView *shell_view,
                int view_num)
{
    Bonobo_ConfigDatabase db;
    EShellViewPrivate *priv;
    EShortcutBar *shortcut_bar;
    int num_groups, group, val;
    long width, height;
    char *stringval, *prefix, *filename, *key;

    g_return_val_if_fail (shell_view != NULL, FALSE);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE);

    priv = shell_view->priv;
    shortcut_bar = E_SHORTCUT_BAR (priv->shortcut_bar);

    db = e_shell_get_config_db (priv->shell);

    g_return_val_if_fail (db != CORBA_OBJECT_NIL, FALSE);

    prefix = g_strdup_printf ("/Shell/Views/%d/", view_num);

    key = g_strconcat (prefix, "Width", NULL);
    width = bonobo_config_get_long (db, key, NULL);
    g_free (key);

    key = g_strconcat (prefix, "Height", NULL);
    height = bonobo_config_get_long (db, key, NULL);
    g_free (key);

    gtk_window_set_default_size (GTK_WINDOW (shell_view), width, height);

    key = g_strconcat (prefix, "CurrentShortcutsGroupNum", NULL);
    val = bonobo_config_get_long (db, key, NULL);
    e_shell_view_set_current_shortcuts_group_num (shell_view, val);
    g_free (key);

    key = g_strconcat (prefix, "FolderBarShown", NULL);
    val = bonobo_config_get_long (db, key, NULL);
    e_shell_view_show_folder_bar (shell_view, val);
    g_free (key);

    key = g_strconcat (prefix, "ShortcutBarShown", NULL);
    val = bonobo_config_get_long (db, key, NULL);
    e_shell_view_show_shortcut_bar (shell_view, val);
    g_free (key);

    key = g_strconcat (prefix, "HPanedPosition", NULL);
    val = bonobo_config_get_long (db, key, NULL);
    if (priv->shortcut_bar_shown)
        e_paned_set_position (E_PANED (priv->hpaned), val);
    priv->hpaned_position = val;
    g_free (key);

    key = g_strconcat (prefix, "ViewHPanedPosition", NULL);
    val = bonobo_config_get_long (db, key, NULL);
    if (priv->folder_bar_shown)
        e_paned_set_position (E_PANED (priv->view_hpaned), val);
    priv->view_hpaned_position = val;
    g_free (key);

    key = g_strconcat (prefix, "DisplayedURI", NULL);
    stringval = bonobo_config_get_string (db, key, NULL);
    if (stringval) {
        if (! e_shell_view_display_uri (shell_view, stringval))
            e_shell_view_display_uri (shell_view, E_SHELL_VIEW_DEFAULT_URI);
    } else
        e_shell_view_display_uri (shell_view, E_SHELL_VIEW_DEFAULT_URI);

    g_free (stringval);
    g_free (key);

    num_groups = e_shortcut_model_get_num_groups (shortcut_bar->model);

    for (group = 0; group < num_groups; group++) {
        int iconmode;

        key = g_strdup_printf ("%sShortcutBarGroup%dIconMode", prefix,
                       group);
        iconmode = bonobo_config_get_long (db, key, NULL);
        g_free (key);

        e_shortcut_bar_set_view_type (shortcut_bar, group, iconmode);
    }

    g_free (prefix);

    /* Load the expanded state for the ShellView's StorageSetView */
    filename = g_strdup_printf ("%s/config/storage-set-view-expanded:view_%d",
                    e_shell_get_local_directory (priv->shell),
                    view_num);
    e_tree_load_expanded_state (E_TREE (priv->storage_set_view),
                    filename);
    g_free (filename);

    return TRUE;
}


/* FIXME: This function could become static */
void
e_shell_view_set_current_shortcuts_group_num (EShellView *shell_view, int group_num)
{
    EShellViewPrivate *priv;
    EShortcutsView *shortcuts_view;

    g_return_if_fail (shell_view != NULL);
    g_return_if_fail (E_IS_SHELL_VIEW (shell_view));

    priv = shell_view->priv;

    shortcuts_view = E_SHORTCUTS_VIEW (priv->shortcut_bar);

    e_group_bar_set_current_group_num (E_GROUP_BAR (E_SHORTCUT_BAR (shortcuts_view)), group_num, FALSE);
}

int
e_shell_view_get_current_shortcuts_group_num (EShellView *shell_view)
{
    EShellViewPrivate *priv;
    EShortcutsView *shortcuts_view;
    int group;

    g_return_val_if_fail (shell_view != NULL, -1);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), -1);

    priv = shell_view->priv;

    shortcuts_view = E_SHORTCUTS_VIEW (priv->shortcut_bar);

    group = e_group_bar_get_current_group_num (E_GROUP_BAR (E_SHORTCUT_BAR (shortcuts_view)));

    return group;
}


const char *
e_shell_view_get_folder_bar_right_click_path (EShellView *shell_view)
{
    EShellViewPrivate *priv;

    g_return_val_if_fail (shell_view != NULL, NULL);
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    priv = shell_view->priv;

    return e_storage_set_view_get_right_click_path (E_STORAGE_SET_VIEW (priv->storage_set_view));
}


E_MAKE_TYPE (e_shell_view, "EShellView", EShellView, class_init, init, BONOBO_TYPE_WINDOW)