aboutsummaryrefslogblamecommitdiffstats
path: root/mail/subscribe-dialog.c
blob: 30327b0fef8fbf174f7b6c766f5ace1c66aabeb3 (plain) (tree)
1
2
3
4
5
                                                                           
                                          
  
                                                
  














                                                                        
  
   
 

                   
                             
                                
                        

                            
                                  


                                         






                                          
                                 
 

                       
                         

                        
                        

                       
 
                                             



                                             
 
                                                                        
                                                                                                                                                                         

                                                                                                                                                           

                                                                         

                                                                         

                       
                                                                       
                                                                                                                                                       
                                                                         
                                                                         

                                                                         


                       


                              

  


                       

  
                                                     
 

                                                                
           
                                         










                                                                                           
                                                                






                                  
                                             
 


                                                                                              





                                                           
                                                                      
                                                         

                                            

                                
                                                                              





                                                          
                                                                               
                                            
                                                                    





                                            
                          
 

















                                                                                                    
 


























                                                                                   
 
 

































                                                                                                             






















                                                                             
 
                    

 



























                                                                               
           
                                                                               
 

                                                                               
 


                                                     
 


                                                       
 






















                                                                               
                                                                               









                                                                                   

 







                                                        
           

                                                                                             
 


























                                                                                   
                   



























                                                                                   
 













































































                                                                                               


                                                          
 

                                                                                   
 







                                                                                     


           
                                        

                                                   
                                                          




                                     
                                             

                                                        



                                                                        

 
           

                                                              
 



                                                                        

 


                                                          
                                                         

                                                                                    
 

                                                       

 
           
                                                         
 
                                                           
 
                                                                                 
                                                                    

 


                                                            
                                                         

                                                                                    
 

                                                         


 
           
                                                           
 
                                                           
 
                                                                                 


                                                                      
           

                                                              








                                                                     

 
           

                                                        
                                                           

                                                                            






                                                
 

                                           

 
 
     































































                                                                              
      




                                       
                                                      
 
                               


            
                                                                                      




                              
                                                                           




                     
                                                                




                             
                                                                                     




                                      
                                                                                      




                             
                                                                           




                                               
                                                                                     
 


                                                                         
                                     
                                  



                                                                  
                                                              
                                                               
                                                                      
                    
                                                                  
         


           
                                                                                                          




                     
                                                                                        





                     



















                                                                               
                                                                    



















                                                                                           
           


















                                                                                     
 


















                                                                                             
           
                                                   

                                                   



                                                                          

                                                            

                          
                                                                      
                                                                                                         
















                                                                            










                                                                               










                                                                                    
                                             
                                                         
            
                                                       

                                                           



 




                                                                                                      





                                                                      
                                                                                  

                                   

                                                                         






                                                                      
           
                                                                
 
                   
                       

                                                         

                                                                        



                                                          
 
 



                                                                        















                                                                     
                                               
 
                             
                    
                              
                                     
                                     
                                                
                                          
                             
 
                                   
 
                               
                                                                              

                                        

                                                                      

                                     


                                                                                   



                                                     
                                                     
 
                                                            


                                                         
                                   
 
                                                   
 


                                                
 



                                                                                


                                                                                   
                                        
                                  
     



                                                             
                                                                       



                                      
      
 












                                                                            
                                       
 

                                                                                  
 
                                              





















                                                                                    
                                                                   
 
                                                                             

                                                                            

                                       
                                                       
 


                                                                                               
 





                                                                           



                                                                                    


                                     


                                                                                     
                          
                                                         



                                      






                                                                               
     
                                          

      
                                            
                                    
                                           





                                                   

                                 


           
                                            
 



                                       




                                                                          
 


                                                                       
 



                                                            



                                        
                                                        


           
                                                          
 
                                                         
 
                                                                     


           
                                         



           
                                                                     
 
                                                        

          

                            
                          
                         
                           

                               
                              
 
                                       


           
                                            
 
                                          
 
                                                                       
 
                                                                          
 
                                                  

 
                                                                                                                                    
 
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* subscribe-dialog.c: Subscribe dialog */
/*
 *  Authors: Chris Toshok <toshok@helixcode.com>
 *
 *  Copyright 2000 Helix Code, Inc. (www.helixcode.com)
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  (at your option) any later version.
 *
 *  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 Street #330, Boston, MA 02111-1307, USA.
 *
 */

#include <config.h>
#include <gnome.h>
#include "subscribe-dialog.h"
#include "e-util/e-html-utils.h"
#include "e-title-bar.h"
#include <gtkhtml/gtkhtml.h>
#include <gal/util/e-util.h>
#include <gal/widgets/e-unicode.h>
#include <gal/e-table/e-cell-toggle.h>
#include <gal/e-table/e-table-scrolled.h>
#include <gal/e-table/e-tree-simple.h>
#include <gal/e-paned/e-hpaned.h>
#include <bonobo/bonobo-main.h>
#include <bonobo/bonobo-object.h>
#include <bonobo/bonobo-generic-factory.h>
#include <bonobo/bonobo-control.h> 
#include <bonobo/bonobo-ui-component.h>
#include <bonobo/bonobo-ui-util.h>
#include <bonobo/bonobo-widget.h>

#include "mail.h"
#include "mail-tools.h"
#include "mail-threads.h"
#include "camel/camel.h"

#include "art/empty.xpm"
#include "art/mark.xpm"


#define DEFAULT_STORE_TABLE_WIDTH         200
#define DEFAULT_WIDTH                     500
#define DEFAULT_HEIGHT                    300

#define PARENT_TYPE (gtk_object_get_type ())

#define FOLDER_ETABLE_SPEC "<ETableSpecification cursor-mode=\"line\"> \
        <ETableColumn model_col=\"0\" pixbuf=\"subscribed-image\" expansion=\"0.0\" minimum_width=\"16\" resizable=\"false\" cell=\"cell_toggle\" compare=\"integer\"/> \
        <ETableColumn model_col=\"1\" _title=\"Folder\" expansion=\"1.0\" minimum_width=\"20\" resizable=\"true\" cell=\"cell_tree\" compare=\"string\"/> \
    <ETableState>                                       \
        <column source=\"0\"/>                                  \
        <column source=\"1\"/>                                  \
            <grouping></grouping>                                   \
    </ETableState>                                      \
</ETableSpecification>"

#define STORE_ETABLE_SPEC "<ETableSpecification cursor-mode=\"line\"> \
        <ETableColumn model_col=\"0\" _title=\"Store\" expansion=\"1.0\" minimum_width=\"20\" resizable=\"true\" cell=\"string\" compare=\"string\"/> \
    <ETableState>                                       \
        <column source=\"0\"/>                                  \
            <grouping></grouping>                                   \
    </ETableState>                                      \
</ETableSpecification>"

enum {
    FOLDER_COL_SUBSCRIBED,
    FOLDER_COL_NAME,
    FOLDER_COL_LAST
};

enum {
    STORE_COL_NAME,
    STORE_COL_LAST
};

static GtkObjectClass *subscribe_dialog_parent_class;

static void build_tree (SubscribeDialog *sc, CamelStore *store);

static void
set_pixmap (BonoboUIComponent *component,
        const char        *xml_path,
        const char        *icon)
{
    char *path;
    GdkPixbuf *pixbuf;

    path = g_concat_dir_and_file (EVOLUTION_DATADIR "/images/evolution/buttons", icon);

    pixbuf = gdk_pixbuf_new_from_file (path);
    g_return_if_fail (pixbuf != NULL);

    bonobo_ui_util_set_pixbuf (component, xml_path, pixbuf);

    gdk_pixbuf_unref (pixbuf);

    g_free (path);
}

static void
update_pixmaps (BonoboUIComponent *component)
{
    set_pixmap (component, "/Toolbar/SubscribeFolder", "fetch-mail.png"); /* XXX */
    set_pixmap (component, "/Toolbar/UnsubscribeFolder", "compose-message.png"); /* XXX */
    set_pixmap (component, "/Toolbar/RefreshList", "forward.png"); /* XXX */
}

static GtkWidget*
make_folder_search_widget (GtkSignalFunc start_search_func,
               gpointer user_data_for_search)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (user_data_for_search);
    GtkWidget *search_hbox = gtk_hbox_new (FALSE, 0);

    sc->search_entry = gtk_entry_new ();

    if (start_search_func) {
        gtk_signal_connect (GTK_OBJECT (sc->search_entry), "activate",
                    start_search_func,
                    user_data_for_search);
    }
    
    /* add the search entry to the our search_vbox */
    gtk_box_pack_start (GTK_BOX (search_hbox),
                gtk_label_new(_("Display folders starting with:")),
                FALSE, TRUE, 0);
    gtk_box_pack_start (GTK_BOX (search_hbox), sc->search_entry,
                FALSE, TRUE, 3);

    return search_hbox;
}


/* Our async operations */

typedef void (*SubscribeGetStoreCallback)(SubscribeDialog *sc, CamelStore *store, gpointer cb_data);
typedef void (*SubscribeFolderCallback)(SubscribeDialog *sc, gboolean success, gpointer cb_data);

/* ** GET STORE ******************************************************* */

typedef struct get_store_input_s {
    SubscribeDialog *sc;
    gchar *url;
    SubscribeGetStoreCallback cb;
    gpointer cb_data;
} get_store_input_t;

typedef struct get_store_data_s {
    CamelStore *store;
} get_store_data_t;

static gchar *
describe_get_store (gpointer in_data, gboolean gerund)
{
    get_store_input_t *input = (get_store_input_t *) in_data;

    if (gerund) {
        return g_strdup_printf (_("Getting store for \"%s\""), input->url);
    }
    else {
        return g_strdup_printf (_("Get store for \"%s\""), input->url);
    }
}

static void
setup_get_store (gpointer in_data, gpointer op_data, CamelException *ex)
{
}

static void
do_get_store (gpointer in_data, gpointer op_data, CamelException *ex)
{
    get_store_input_t *input = (get_store_input_t *)in_data;
    get_store_data_t *data = (get_store_data_t*)op_data;

    mail_tool_camel_lock_up ();
    data->store = camel_session_get_store (session, input->url, ex);
    mail_tool_camel_lock_down ();

    if (camel_exception_is_set (ex))
        data->store = NULL;
}

static void
cleanup_get_store (gpointer in_data, gpointer op_data, CamelException *ex)
{
    get_store_input_t *input = (get_store_input_t *)in_data;
    get_store_data_t *data = (get_store_data_t*)op_data;

    input->cb (input->sc, data->store, input->cb_data);

    g_free (input->url);
}

static const mail_operation_spec op_get_store = {
        describe_get_store,
        sizeof (get_store_data_t),
        setup_get_store,
        do_get_store,
        cleanup_get_store
};

static void
subscribe_do_get_store (SubscribeDialog *sc, const char *url, SubscribeGetStoreCallback cb, gpointer cb_data)
{
    get_store_input_t *input;

        input = g_new (get_store_input_t, 1);
    input->sc = sc;
    input->url = g_strdup (url);
    input->cb = cb;
    input->cb_data = cb_data;

        mail_operation_queue (&op_get_store, input, TRUE);
}

/* ** SUBSCRIBE FOLDER ******************************************************* */
/* Given a CamelFolderInfo, construct the corresponding
 * EvolutionStorage path to it.
 */
static char *
storage_tree_path (CamelFolderInfo *info)
{
    int len;
    CamelFolderInfo *i;
    char *path, *p;

    for (len = 0, i = info; i; i = i->parent)
        len += strlen (i->name) + 1;

    /* We do this backwards because that's the way the pointers point. */
    path = g_malloc (len + 1);
    p = path + len;
    *p = '\0';
    for (i = info; i; i = i->parent) {
        len = strlen (i->name);
        p -= len;
        memcpy (p, i->name, len);
        *--p = '/';
    }

    return path;
}

typedef struct subscribe_folder_input_s {
    SubscribeDialog *sc;
    CamelStore *store;
    CamelFolderInfo *info;
    SubscribeFolderCallback cb;
    gpointer cb_data;
} subscribe_folder_input_t;

typedef struct subscribe_folder_data_s {
    char *path;
    char *name;
    char *url;
} subscribe_folder_data_t;

static gchar *
describe_subscribe_folder (gpointer in_data, gboolean gerund)
{
    subscribe_folder_input_t *input = (subscribe_folder_input_t *) in_data;

    if (gerund)
        return g_strdup_printf
            (_("Subscribing to folder \"%s\""),
             input->info->name);
    else
        return g_strdup_printf (_("Subscribe to folder \"%s\""),
                    input->info->name);
}

static void
setup_subscribe_folder (gpointer in_data, gpointer op_data, CamelException *ex)
{
    subscribe_folder_input_t *input = (subscribe_folder_input_t *) in_data;
    subscribe_folder_data_t *data = (subscribe_folder_data_t *) op_data;

    data->path = storage_tree_path (input->info);
    data->name = g_strdup (input->info->name);
    data->url = g_strdup (input->info->url);

    camel_object_ref (CAMEL_OBJECT (input->store));
    gtk_object_ref (GTK_OBJECT (input->sc));
}

static void
do_subscribe_folder (gpointer in_data, gpointer op_data, CamelException *ex)
{
    subscribe_folder_input_t *input = (subscribe_folder_input_t *) in_data;
    subscribe_folder_data_t *data = (subscribe_folder_data_t *) op_data;

    mail_tool_camel_lock_up ();
    camel_store_subscribe_folder (input->store, data->name, ex);
    mail_tool_camel_lock_down ();
}

static void
cleanup_subscribe_folder (gpointer in_data, gpointer op_data,
              CamelException *ex)
{
    subscribe_folder_input_t *input = (subscribe_folder_input_t *) in_data;
    subscribe_folder_data_t *data = (subscribe_folder_data_t *) op_data;

    if (!camel_exception_is_set (ex))
        evolution_storage_new_folder (input->sc->storage,
                          data->path,
                          data->name, "mail",
                          data->url,
                          _("(No description)") /* XXX */);

    if (input->cb)
        input->cb (input->sc, !camel_exception_is_set(ex), input->cb_data);

    g_free (data->path);
    g_free (data->name);
    g_free (data->url);

    camel_object_unref (CAMEL_OBJECT (input->store));
    gtk_object_unref (GTK_OBJECT (input->sc));
}

static const mail_operation_spec op_subscribe_folder = {
    describe_subscribe_folder,
    sizeof (subscribe_folder_data_t),
    setup_subscribe_folder,
    do_subscribe_folder,
    cleanup_subscribe_folder
};

static void
subscribe_do_subscribe_folder (SubscribeDialog *sc, CamelStore *store, CamelFolderInfo *info,
                   SubscribeFolderCallback cb, gpointer cb_data)
{
    subscribe_folder_input_t *input;

    g_return_if_fail (CAMEL_IS_STORE (store));
    g_return_if_fail (info);

    input = g_new (subscribe_folder_input_t, 1);
    input->sc = sc;
    input->store = store;
    input->info = info;
    input->cb = cb;
    input->cb_data = cb_data;

    mail_operation_queue (&op_subscribe_folder, input, TRUE);
}

/* ** UNSUBSCRIBE FOLDER ******************************************************* */

typedef struct unsubscribe_folder_input_s {
    SubscribeDialog *sc;
    CamelStore *store;
    CamelFolderInfo *info;
    SubscribeFolderCallback cb;
    gpointer cb_data;
} unsubscribe_folder_input_t;

typedef struct unsubscribe_folder_data_s {
    char *name;
    char *path;
} unsubscribe_folder_data_t;

static gchar *
describe_unsubscribe_folder (gpointer in_data, gboolean gerund)
{
    unsubscribe_folder_input_t *input = (unsubscribe_folder_input_t *) in_data;

    if (gerund)
        return g_strdup_printf
            (_("Unsubscribing to folder \"%s\""),
             input->info->name);
    else
        return g_strdup_printf (_("Unsubscribe to folder \"%s\""),
                    input->info->name);
}

static void
setup_unsubscribe_folder (gpointer in_data, gpointer op_data, CamelException *ex)
{
    unsubscribe_folder_input_t *input = (unsubscribe_folder_input_t *) in_data;
    unsubscribe_folder_data_t *data = (unsubscribe_folder_data_t *) op_data;

    data->name = g_strdup (input->info->name);
    data->path = storage_tree_path (input->info);

    camel_object_ref (CAMEL_OBJECT (input->store));
    gtk_object_ref (GTK_OBJECT (input->sc));
}

static void
do_unsubscribe_folder (gpointer in_data, gpointer op_data, CamelException *ex)
{
    unsubscribe_folder_input_t *input = (unsubscribe_folder_input_t *) in_data;
    unsubscribe_folder_data_t *data = (unsubscribe_folder_data_t *) op_data;

    mail_tool_camel_lock_up ();
    camel_store_unsubscribe_folder (input->store, data->name, ex);
    mail_tool_camel_lock_down ();
}

static void
cleanup_unsubscribe_folder (gpointer in_data, gpointer op_data,
                CamelException *ex)
{
    unsubscribe_folder_input_t *input = (unsubscribe_folder_input_t *) in_data;
    unsubscribe_folder_data_t *data = (unsubscribe_folder_data_t *) op_data;

    if (!camel_exception_is_set (ex))
        evolution_storage_removed_folder (input->sc->storage, data->path);

    if (input->cb)
        input->cb (input->sc, !camel_exception_is_set(ex), input->cb_data);

    g_free (data->path);
    g_free (data->name);

    camel_object_unref (CAMEL_OBJECT (input->store));
    gtk_object_unref (GTK_OBJECT (input->sc));
}

static const mail_operation_spec op_unsubscribe_folder = {
    describe_unsubscribe_folder,
    sizeof (unsubscribe_folder_data_t),
    setup_unsubscribe_folder,
    do_unsubscribe_folder,
    cleanup_unsubscribe_folder
};

static void
subscribe_do_unsubscribe_folder (SubscribeDialog *sc, CamelStore *store, CamelFolderInfo *info,
                 SubscribeFolderCallback cb, gpointer cb_data)
{
    unsubscribe_folder_input_t *input;

    g_return_if_fail (CAMEL_IS_STORE (store));
    g_return_if_fail (info);

    input = g_new (unsubscribe_folder_input_t, 1);
    input->sc = sc;
    input->store = store;
    input->info = info;
    input->cb = cb;
    input->cb_data = cb_data;

    mail_operation_queue (&op_unsubscribe_folder, input, TRUE);
}



static gboolean
folder_info_subscribed (SubscribeDialog *sc, CamelFolderInfo *info)
{
    return camel_store_folder_subscribed (sc->store, info->full_name);
}

static void
node_changed_cb (SubscribeDialog *sc, gboolean changed, gpointer data)
{
    ETreePath *node = data;

    if (changed)
        e_tree_model_node_changed (sc->folder_model, node);
}

static void
subscribe_folder_info (SubscribeDialog *sc, CamelFolderInfo *info, ETreePath *node)
{
    /* folders without urls cannot be subscribed to */
    if (info->url == NULL)
        return;

    subscribe_do_subscribe_folder (sc, sc->store, info, node_changed_cb, node);
}

static void
unsubscribe_folder_info (SubscribeDialog *sc, CamelFolderInfo *info, ETreePath *node)
{
    /* folders without urls cannot be subscribed to */
    if (info->url == NULL)
        return;

    subscribe_do_unsubscribe_folder (sc, sc->store, info, node_changed_cb, node);
}

static void
subscribe_close (BonoboUIComponent *uic,
         void *user_data, const char *path)
{
    SubscribeDialog *sc = (SubscribeDialog*)user_data;

    gtk_widget_destroy (sc->app);
}

static void
subscribe_select_all (BonoboUIComponent *uic,
              void *user_data, const char *path)
{
    SubscribeDialog *sc = (SubscribeDialog*)user_data;
    ETableScrolled *scrolled = E_TABLE_SCROLLED (sc->folder_etable);
    
    e_table_select_all (scrolled->table);
}

static void
subscribe_invert_selection (BonoboUIComponent *uic,
                void *user_data, const char *path)
{
    SubscribeDialog *sc = (SubscribeDialog*)user_data;
    ETableScrolled *scrolled = E_TABLE_SCROLLED (sc->folder_etable);
    
    e_table_invert_selection (scrolled->table);
}

static void
subscribe_folder_foreach (int model_row, gpointer closure)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (closure);
    ETreePath *node = e_tree_model_node_at_row (sc->folder_model, model_row);
    CamelFolderInfo *info = e_tree_model_node_get_data (sc->folder_model, node);

    if (!folder_info_subscribed (sc, info))
        subscribe_folder_info (sc, info, node);
}

static void
subscribe_folders (GtkWidget *widget, gpointer user_data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (user_data);

    e_table_selected_row_foreach (E_TABLE_SCROLLED(sc->folder_etable)->table,
                      subscribe_folder_foreach, sc);
}

static void
unsubscribe_folder_foreach (int model_row, gpointer closure)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (closure);
    ETreePath *node = e_tree_model_node_at_row (sc->folder_model, model_row);
    CamelFolderInfo *info = e_tree_model_node_get_data (sc->folder_model, node);

    if (folder_info_subscribed(sc, info))
        unsubscribe_folder_info (sc, info, node);
}


static void
unsubscribe_folders (GtkWidget *widget, gpointer user_data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (user_data);

    e_table_selected_row_foreach (E_TABLE_SCROLLED(sc->folder_etable)->table,
                      unsubscribe_folder_foreach, sc);
}

static void
subscribe_refresh_list (GtkWidget *widget, gpointer user_data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (user_data);

    e_utf8_gtk_entry_set_text (GTK_ENTRY (sc->search_entry), "");
    if (sc->search_top) {
        g_free (sc->search_top);
        sc->search_top = NULL;
    }
    if (sc->store)
        build_tree (sc, sc->store);
}

static void
subscribe_search (GtkWidget *widget, gpointer user_data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (user_data);
    char* search_pattern = e_utf8_gtk_entry_get_text(GTK_ENTRY(widget));

    if (sc->search_top) {
        g_free (sc->search_top);
        sc->search_top = NULL;
    }

    if (search_pattern && *search_pattern)
        sc->search_top = search_pattern;

    if (sc->store)
        build_tree (sc, sc->store);
}


#if 0
/* HTML Helpers */
static void
html_size_req (GtkWidget *widget, GtkRequisition *requisition)
{
    if (GTK_LAYOUT (widget)->height > 90)
        requisition->height = 90;
    else
        requisition->height = GTK_LAYOUT (widget)->height;
}

/* Returns a GtkHTML which is already inside a GtkScrolledWindow. If
 * @white is TRUE, the GtkScrolledWindow will be inside a GtkFrame.
 */
static GtkWidget *
html_new (gboolean white)
{
    GtkWidget *html, *scrolled, *frame;
    GtkStyle *style;
    
    html = gtk_html_new ();
    GTK_LAYOUT (html)->height = 0;
    gtk_signal_connect (GTK_OBJECT (html), "size_request",
                GTK_SIGNAL_FUNC (html_size_req), NULL);
    gtk_html_set_editable (GTK_HTML (html), FALSE);
    style = gtk_rc_get_style (html);
    if (style) {
        gtk_html_set_default_background_color (GTK_HTML (html),
                               white ? &style->white :
                               &style->bg[0]);
    }
    gtk_widget_set_sensitive (html, FALSE);
    scrolled = gtk_scrolled_window_new (NULL, NULL);
    gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled),
                    GTK_POLICY_NEVER,
                    GTK_POLICY_NEVER);
    gtk_container_add (GTK_CONTAINER (scrolled), html);
    
    if (white) {
        frame = gtk_frame_new (NULL);
        gtk_frame_set_shadow_type (GTK_FRAME (frame),
                       GTK_SHADOW_ETCHED_IN);
        gtk_container_add (GTK_CONTAINER (frame), scrolled);
        gtk_widget_show_all (frame);
    } else
        gtk_widget_show_all (scrolled);
    
    return html;
}

static void
put_html (GtkHTML *html, char *text)
{
    GtkHTMLStream *handle;
    
    text = e_text_to_html (text, (E_TEXT_TO_HTML_CONVERT_NL |
                      E_TEXT_TO_HTML_CONVERT_SPACES |
                      E_TEXT_TO_HTML_CONVERT_URLS));
    handle = gtk_html_begin (html);
    gtk_html_write (html, handle, "<HTML><BODY>", 12);
    gtk_html_write (html, handle, text, strlen (text));
    gtk_html_write (html, handle, "</BODY></HTML>", 14);
    g_free (text);
    gtk_html_end (html, handle, GTK_HTML_STREAM_OK);
}
#endif


/* etable stuff for the subscribe ui */

static int
folder_etable_col_count (ETableModel *etm, void *data)
{
    return FOLDER_COL_LAST;
}

static void*
folder_etable_duplicate_value (ETableModel *etm, int col, const void *val, void *data)
{
    return g_strdup (val);
}

static void
folder_etable_free_value (ETableModel *etm, int col, void *val, void *data)
{
    g_free (val);
}

static void*
folder_etable_init_value (ETableModel *etm, int col, void *data)
{
    return g_strdup ("");
}

static gboolean
folder_etable_value_is_empty (ETableModel *etm, int col, const void *val, void *data)
{
    return !(val && *(char *)val);
}

static char*
folder_etable_value_to_string (ETableModel *etm, int col, const void *val, void *data)
{
    return g_strdup(val);
}

static GdkPixbuf*
folder_etree_icon_at (ETreeModel *etree, ETreePath *path, void *model_data)
{
    return NULL; /* XXX no icons for now */
}

static void*
folder_etree_value_at (ETreeModel *etree, ETreePath *path, int col, void *model_data)
{
    SubscribeDialog *dialog = SUBSCRIBE_DIALOG (model_data);
    CamelFolderInfo *info = e_tree_model_node_get_data (etree, path);

    if (col == FOLDER_COL_NAME) {
        return info->name;
    }
    else /* FOLDER_COL_SUBSCRIBED */ {
        /* folders without urls cannot be subscribed to */
        if (info->url == NULL)
            return GINT_TO_POINTER(0); /* empty */
        else if (!folder_info_subscribed(dialog, info))
            return GINT_TO_POINTER(0); /* XXX unchecked */
        else
            return GUINT_TO_POINTER (1); /* checked */
    }
}

static void
folder_etree_set_value_at (ETreeModel *etree, ETreePath *path, int col, const void *val, void *model_data)
{
    /* nothing */
}

static gboolean
folder_etree_is_editable (ETreeModel *etree, ETreePath *path, int col, void *model_data)
{
    return FALSE;
}



static int
store_etable_col_count (ETableModel *etm, void *data)
{
    return STORE_COL_LAST;
}

static int
store_etable_row_count (ETableModel *etm, void *data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (data);

    return g_list_length (sc->store_list);
}

static void*
store_etable_value_at (ETableModel *etm, int col, int row, void *data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (data);
    CamelStore *store = (CamelStore*)g_list_nth_data (sc->store_list, row);

    return camel_service_get_name (CAMEL_SERVICE (store), TRUE);
}

static void
store_etable_set_value_at (ETableModel *etm, int col, int row, const void *val, void *data)
{
    /* nada */
}

static gboolean
store_etable_is_editable (ETableModel *etm, int col, int row, void *data)
{
    return FALSE;
}

static void*
store_etable_duplicate_value (ETableModel *etm, int col, const void *val, void *data)
{
    return g_strdup (val);
}

static void
store_etable_free_value (ETableModel *etm, int col, void *val, void *data)
{
    g_free (val);
}

static void*
store_etable_initialize_value (ETableModel *etm, int col, void *data)
{
    return g_strdup ("");
}

static gboolean
store_etable_value_is_empty (ETableModel *etm, int col, const void *val, void *data)
{
    return !(val && *(char *)val);
}

static char*
store_etable_value_to_string (ETableModel *etm, int col, const void *val, void *data)
{
    return g_strdup(val);
}



static void
build_etree_from_folder_info (SubscribeDialog *sc, ETreePath *parent, CamelFolderInfo *info)
{
    CamelFolderInfo *i;

    if (info == NULL)
        return;

    for (i = info; i; i = i->sibling) {
        ETreePath *node = e_tree_model_node_insert (sc->folder_model, parent, -1, i);
        build_etree_from_folder_info (sc, node, i->child);
    }
}

static void
build_tree (SubscribeDialog *sc, CamelStore *store)
{
    CamelException *ex = camel_exception_new();

    /* free up the existing CamelFolderInfo* if there is any */
    if (sc->folder_info)
        camel_store_free_folder_info (sc->store, sc->folder_info);
    if (sc->storage)
        gtk_object_unref (GTK_OBJECT (sc->storage));

    sc->store = store;
    sc->storage = mail_lookup_storage (CAMEL_SERVICE (sc->store));
    sc->folder_info = camel_store_get_folder_info (sc->store, sc->search_top, TRUE, TRUE, FALSE, ex);

    if (camel_exception_is_set (ex)) {
        printf ("camel_store_get_folder_info failed\n");
        camel_exception_free (ex);
        return;
    }

    e_tree_model_node_remove (sc->folder_model, sc->folder_root);
    sc->folder_root = e_tree_model_node_insert (sc->folder_model, NULL,
                            0, NULL);


    build_etree_from_folder_info (sc, sc->folder_root, sc->folder_info);

    camel_exception_free (ex);
}

static void
storage_selected_cb (ETable *table,
             int row,
             gpointer data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (data);
    CamelStore *store = (CamelStore*)g_list_nth_data (sc->store_list, row);

    build_tree (sc, store);
}



static void
folder_toggle_cb (ETable *table,
          int row,
          gpointer data)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (data);
    ETreePath *node = e_tree_model_node_at_row (sc->folder_model, row);
    CamelFolderInfo *info = e_tree_model_node_get_data (sc->folder_model, node);

    if (folder_info_subscribed(sc, info))
        unsubscribe_folder_info (sc, info, node);
    else
        subscribe_folder_info (sc, info, node);

    e_tree_model_node_changed (sc->folder_model, node);
}



#define EXAMPLE_DESCR "And the beast shall come forth surrounded by a roiling cloud of vengeance.\n" \
"  The house of the unbelievers shall be razed and they shall be scorched to the\n" \
"          earth. Their tags shall blink until the end of days. \n" \
"                 from The Book of Mozilla, 12:10"

static BonoboUIVerb verbs [] = {
    /* File Menu */
    BONOBO_UI_UNSAFE_VERB ("FileCloseWin", subscribe_close),

    /* Edit Menu */
    BONOBO_UI_UNSAFE_VERB ("EditSelectAll", subscribe_select_all),
    BONOBO_UI_UNSAFE_VERB ("EditInvertSelection", subscribe_invert_selection),
    
    /* Folder Menu / Toolbar */
    BONOBO_UI_UNSAFE_VERB ("SubscribeFolder", subscribe_folders),
    BONOBO_UI_UNSAFE_VERB ("UnsubscribeFolder", unsubscribe_folders),

    /* Toolbar Specific */
    BONOBO_UI_UNSAFE_VERB ("RefreshList", subscribe_refresh_list),

    BONOBO_UI_VERB_END
};

static void
store_cb (SubscribeDialog *sc, CamelStore *store, gpointer data)
{
    if (!store)
        return;

    if (camel_store_supports_subscriptions (store)) {
        sc->store_list = g_list_prepend (sc->store_list, store);
        e_table_model_row_inserted (sc->store_model, 0);
    }
    else {
        camel_object_unref (CAMEL_OBJECT (store));
    }
}

static void
populate_store_foreach (MailConfigService *service, SubscribeDialog *sc)
{
    subscribe_do_get_store (sc, service->url, store_cb, NULL);
}

static void
populate_store_list (SubscribeDialog *sc)
{
    GSList *sources;

    sources = mail_config_get_sources ();
    g_slist_foreach (sources, (GFunc)populate_store_foreach, sc);
    sources = mail_config_get_news ();
    g_slist_foreach (sources, (GFunc)populate_store_foreach, sc);

    e_table_model_changed (sc->store_model);
}

static void
subscribe_dialog_gui_init (SubscribeDialog *sc)
{
    ETableExtras *extras;
    ECell *cell;
    GdkPixbuf *toggles[2];
    BonoboUIComponent *component;
    BonoboUIContainer *container;
    GtkWidget         *folder_search_widget;
    BonoboControl     *search_control;
    CORBA_Environment ev;

    CORBA_exception_init (&ev);

    /* Construct the app */
    sc->app = bonobo_win_new ("subscribe-dialog", "Manage Subscriptions");

    /* Build the menu and toolbar */
    container = bonobo_ui_container_new ();
    bonobo_ui_container_set_win (container, BONOBO_WIN (sc->app));

    /* set up the bonobo stuff */
    component = bonobo_ui_component_new_default ();
    bonobo_ui_component_set_container (
        component, bonobo_object_corba_objref (BONOBO_OBJECT (container)));
    
    bonobo_ui_component_add_verb_list_with_data (
        component, verbs, sc);

    bonobo_ui_component_freeze (component, NULL);

    bonobo_ui_util_set_ui (component, EVOLUTION_DATADIR,
                   "evolution-subscribe.xml",
                   "evolution-subscribe");

    update_pixmaps (component);

    bonobo_ui_component_thaw (component, NULL);

    sc->table = gtk_table_new (1, 2, FALSE);

    sc->hpaned = e_hpaned_new ();

    folder_search_widget = make_folder_search_widget (subscribe_search, sc);
    gtk_widget_show_all (folder_search_widget);
    search_control = bonobo_control_new (folder_search_widget);

    bonobo_ui_component_object_set (
        component, "/Toolbar/FolderSearch",
        bonobo_object_corba_objref (BONOBO_OBJECT (search_control)), NULL);
                    
    /* set our our contents */
#if 0
    sc->description = html_new (TRUE);
    put_html (GTK_HTML (sc->description), EXAMPLE_DESCR);

    gtk_table_attach (
        GTK_TABLE (sc->table), sc->description->parent->parent,
        0, 1, 0, 1,
        GTK_FILL | GTK_EXPAND,
        0,
        0, 0);
#endif

    /* set up the store etable */
    sc->store_model = e_table_simple_new (store_etable_col_count,
                          store_etable_row_count,
                          store_etable_value_at,
                          store_etable_set_value_at,
                          store_etable_is_editable,
                          store_etable_duplicate_value,
                          store_etable_free_value,
                          store_etable_initialize_value,
                          store_etable_value_is_empty,
                          store_etable_value_to_string,
                          sc);

    extras = e_table_extras_new ();

    sc->store_etable = e_table_scrolled_new (E_TABLE_MODEL(sc->store_model),
                         extras, STORE_ETABLE_SPEC, NULL);

    gtk_object_sink (GTK_OBJECT (extras));

    gtk_signal_connect (GTK_OBJECT (E_TABLE_SCROLLED (sc->store_etable)->table),
                "cursor_change", GTK_SIGNAL_FUNC (storage_selected_cb),
                sc);

    /* set up the folder etable */
    sc->folder_model = e_tree_simple_new (folder_etable_col_count,
                          folder_etable_duplicate_value,
                          folder_etable_free_value,
                          folder_etable_init_value,
                          folder_etable_value_is_empty,
                          folder_etable_value_to_string,
                          folder_etree_icon_at,
                          folder_etree_value_at,
                          folder_etree_set_value_at,
                          folder_etree_is_editable,
                          sc);

    sc->folder_root = e_tree_model_node_insert (sc->folder_model, NULL,
                            0, NULL);

    e_tree_model_root_node_set_visible (sc->folder_model, FALSE);
    e_tree_model_set_expanded_default (sc->folder_model, TRUE);

    toggles[0] = gdk_pixbuf_new_from_xpm_data ((const char **)empty_xpm);
    toggles[1] = gdk_pixbuf_new_from_xpm_data ((const char **)mark_xpm);

    extras = e_table_extras_new ();

    cell = e_cell_text_new(NULL, GTK_JUSTIFY_LEFT);

    e_table_extras_add_cell (extras, "cell_text", cell);
    e_table_extras_add_cell (extras, "cell_toggle", e_cell_toggle_new (0, 2, toggles));
    e_table_extras_add_cell (extras, "cell_tree", e_cell_tree_new(NULL, NULL, TRUE, cell));

    gtk_object_set (GTK_OBJECT (cell),
            "bold_column", FOLDER_COL_SUBSCRIBED,
            NULL);

    e_table_extras_add_pixbuf (extras, "subscribed-image", toggles[1]);

    sc->folder_etable = e_table_scrolled_new (E_TABLE_MODEL(sc->folder_model),
                          extras, FOLDER_ETABLE_SPEC, NULL);

    gtk_object_sink (GTK_OBJECT (extras));
    gdk_pixbuf_unref(toggles[0]);
    gdk_pixbuf_unref(toggles[1]);

    gtk_signal_connect (GTK_OBJECT (E_TABLE_SCROLLED (sc->folder_etable)->table),
                "double_click", GTK_SIGNAL_FUNC (folder_toggle_cb),
                sc);
    gtk_table_attach (
        GTK_TABLE (sc->table), sc->folder_etable,
        0, 1, 1, 3,
        GTK_FILL | GTK_EXPAND,
        GTK_FILL | GTK_EXPAND,
        0, 0);

    e_paned_add1 (E_PANED (sc->hpaned), sc->store_etable);
    e_paned_add2 (E_PANED (sc->hpaned), sc->table);
    e_paned_set_position (E_PANED (sc->hpaned), DEFAULT_STORE_TABLE_WIDTH);

    bonobo_win_set_contents (BONOBO_WIN (sc->app), sc->hpaned);

#if 0
    gtk_widget_show (sc->description);
#endif

    gtk_widget_show (sc->folder_etable);
    gtk_widget_show (sc->table);
    gtk_widget_show (sc->store_etable);
    gtk_widget_show (sc->hpaned);

    /* FIXME: Session management and stuff?  */
    gtk_window_set_default_size (
        GTK_WINDOW (sc->app),
        DEFAULT_WIDTH, DEFAULT_HEIGHT);

    populate_store_list (sc);
}

static void
subscribe_dialog_destroy (GtkObject *object)
{
    SubscribeDialog *sc;

    sc = SUBSCRIBE_DIALOG (object);

    /* free our folder information */
    e_tree_model_node_remove (sc->folder_model, sc->folder_root);
    gtk_object_unref (GTK_OBJECT (sc->folder_model));
    if (sc->folder_info)
        camel_store_free_folder_info (sc->store, sc->folder_info);

    /* free our store information */
    gtk_object_unref (GTK_OBJECT (sc->store_model));
    g_list_foreach (sc->store_list, (GFunc)gtk_object_unref, NULL);

    /* free our storage */
    if (sc->storage)
        gtk_object_unref (GTK_OBJECT (sc->storage));

    /* free our search */
    if (sc->search_top)
        g_free (sc->search_top);

    subscribe_dialog_parent_class->destroy (object);
}

static void
subscribe_dialog_class_init (GtkObjectClass *object_class)
{
    object_class->destroy = subscribe_dialog_destroy;

    subscribe_dialog_parent_class = gtk_type_class (PARENT_TYPE);
}

static void
subscribe_dialog_init (GtkObject *object)
{
}

static void
subscribe_dialog_construct (GtkObject *object, Evolution_Shell shell)
{
    SubscribeDialog *sc = SUBSCRIBE_DIALOG (object);

    /*
     * Our instance data
     */
    sc->shell = shell;
    sc->store = NULL;
    sc->storage = NULL;
    sc->folder_info = NULL;
    sc->store_list = NULL;
    sc->search_top = NULL;

    subscribe_dialog_gui_init (sc);
}

GtkWidget *
subscribe_dialog_new (Evolution_Shell shell)
{
    SubscribeDialog *subscribe_dialog;

    subscribe_dialog = gtk_type_new (subscribe_dialog_get_type ());

    subscribe_dialog_construct (GTK_OBJECT (subscribe_dialog), shell);

    return GTK_WIDGET (subscribe_dialog->app);
}

E_MAKE_TYPE (subscribe_dialog, "SubscribeDialog", SubscribeDialog, subscribe_dialog_class_init, subscribe_dialog_init, PARENT_TYPE);