aboutsummaryrefslogblamecommitdiffstats
path: root/addressbook/gui/component/select-names/e-select-names.c
blob: 47172223250d2a59581be48d9934520105bd93b6 (plain) (tree)
1
2
3
4
5
6
7
8

                                                                           

                                          

                                                                
                                                                   
                                                        












                                                                    
                   
                 
                    
                                
 
                                       
                                        
                                     
 

                                  
                                                        
                                                                
                                           
                                                            
                                                  
 
                                  
                           
                                       
                                                             
                               
                                                   
                          
 
                                                                         
                                                                         
                                                     
                                                                           
 

                                                                                           

                                           



                           


                






                                                
                                           

                    
     

                              
                              

                    
                                                
                                                   







                                                                   

                  
                                                                                      







                                                    
                                   
 
                                              
 
                                                        
 
                                                       

 


                                                                                                          
                                                                          




                                          

                                                            


                                  
                                 


                                   

 



















                                                              
           
                                                                                             

                    
 
                            
 

                                     
 
                                                                                         

 
             
                                     
 
                           
                              
        
                            

                            
                                          
 





                                                                   
                                                           
                                                                                                                                        

 
           





                                                           
                          



                                                   


                                                                      













                                                                              
                                                     


                                           

                                                      
                        


                                                                                                   
                                                                             
        

                                                                          
 
                                                                  
                                                                  
         




                                                               
                                                    

                                                                              
                                                  












                                                                                  


















                                                                             


                                                            


                                                                    





















                                                            


                                                                                              
                             
                             
                        
                         
 

                                                                
 


                                       
 








                                                              



                                                                                                  
 


                                                               
 


                     
           
                                                                                        
 

                                                                                     


           
                                                              

                            
                                
                    

                         

                                       

                                                                                                                                                 
         

                                                                                       
         


                                         




                                                                                      




                                                                                    







                                                           


                                            


                                      
                       
                                


           
                                                                                    







                                                                                         
                                                                    




                                              


                                                                      
                                                                                                          












                                                                                                    
                                                                                                

                                                                                       
                                                                                              















                                                                 









                                                                            
                               
 
                                                      
                                                                      
                              
 
                           

 
           
                                                       
 
                                      



                           


                                                  



                                                  






















                                                                                                   


                                                                                                                  

                                  








                                                                                                 

                                                                             
                                   
 

                                                                     
 
                                                                                            


                                                                                                
                                                             
 


                                                                                             
                                             

                                                                                                           
                                                                                                                             
         
 



                                                                                                  
                                                                                     
                                                                                                                
                                                  
                                         
                                                                       
                                                                                 
                                                                                                                     
         
                                                     
 


                                                                                         
                                           
                                                                         
                                                                                   
                                                                          
                                                                           
                                                                                                                         
         
 
                                                            
                                             
                                                   
                                                                           
 
                                                                                             
                                                                    
                                                                                                 
                                                                         
                                                                                              













                                                                                             



                                                                                                        

                                                                      
                             
                                           
                                      
                    
                      

 
           
                                        
 
                                                              
 




                                                             









                                                                                              






















                                                                                                                   
 



                                            
 

                                                              


          
                         
 
                                     
        

                                                                  
                                           


           

                                                           
                                              

 
           
                                                                                          
 
                                                         

 
                        
                                 
                







                                                        
                                                                         
 
                                           



                        
                                                                                             
 
                                                
                                                                       
                                  
          
                                                            
 

                              
 
                                                                          

 
    


                                                             


                                 
                         
                             
                        
                         
                       

                             
 
                      




                                                                
                                                                                          


                                            
                                      
                                          
 
                                                                                              

                               
                                    
 
                                                  
 
                                                             
 


                                                            
 
                            

                                   
                                                          
                             
                               
 

                                                            
                                           
                                                            





                                                         
 


                                                                               






                                                                 
                                                                                     
                                                         

                                                                                                       

                                
                                                                                                 
 

                                                 

                                                              
 


                                                                 
        
                                                                
                                                            
 



                                                                                 
        
                                 
        
                                   




                                                                      

                                                                           

                                                              

                                      

 







                                                                                           





                                                                               






                                                                                           







                                                                                       

         
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* e-select-names.c
 * Copyright (C) 2000  Ximian, Inc.
 * Author: Chris Lahey <clahey@ximian.com>
 *
 * This library 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 library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 */

#include <config.h>
#include <string.h>
#include <glib.h>
#include <gtk/gtk.h>
#include <libgnome/gnome-i18n.h>

#include <gal/e-table/e-table-simple.h>
#include <gal/e-table/e-table-without.h>
#include <gal/widgets/e-popup-menu.h>

#include <libebook/e-book-async.h>
#include <libebook/e-contact.h>
#include <addressbook/gui/widgets/e-addressbook-model.h>
#include <addressbook/gui/widgets/e-addressbook-table-adapter.h>
#include <addressbook/util/eab-book-util.h>
#include <addressbook/gui/component/addressbook-component.h>
#include <addressbook/gui/component/addressbook.h>

#include "e-select-names-config.h"
#include "e-select-names.h"
#include "e-select-names-table-model.h"
#include <gal/widgets/e-categories-master-list-option-menu.h>
#include <gal/e-text/e-entry.h>
#include <e-util/e-categories-master-list-wombat.h>
#include "e-util/e-sexp.h"

static void e_select_names_init     (ESelectNames        *names);
static void e_select_names_class_init   (ESelectNamesClass   *klass);
static void e_select_names_dispose (GObject *object);
static void update_query (GtkWidget *widget, ESelectNames *e_select_names);

static void sync_table_and_models (ESelectNamesModel *triggering_model, ESelectNames *esl);

static GtkDialogClass *parent_class = NULL;
#define PARENT_TYPE gtk_dialog_get_type()

/* The arguments we take */
enum {
    ARG_0,
};

typedef struct {
    char                   *title;
    ESelectNamesModel      *source;
    ESelectNamesTableModel *table_model;
    ESelectNames           *names;
    GtkWidget              *label;
    GtkWidget              *button;
    GtkWidget              *recipient_table;
    gulong                  changed_id;
} ESelectNamesChild;

GType
e_select_names_get_type (void)
{
    static GType type = 0;

    if (!type) {
        static const GTypeInfo info =  {
            sizeof (ESelectNamesClass),
            NULL,           /* base_init */
            NULL,           /* base_finalize */
            (GClassInitFunc) e_select_names_class_init,
            NULL,           /* class_finalize */
            NULL,           /* class_data */
            sizeof (ESelectNames),
            0,             /* n_preallocs */
            (GInstanceInitFunc) e_select_names_init,
        };

        type = g_type_register_static (PARENT_TYPE, "ESelectNames", &info, 0);
    }

    return type;
}

static void
e_select_names_class_init (ESelectNamesClass *klass)
{
    GObjectClass *object_class;

    object_class = G_OBJECT_CLASS (klass);

    parent_class = g_type_class_peek_parent (klass);

    object_class->dispose = e_select_names_dispose;
}

GtkWidget *e_addressbook_create_ebook_table(char *name, char *string1, char *string2, int num1, int num2);

static void
search_result (EABModel *model, EBookViewStatus status, ESelectNames *esn)
{
    sync_table_and_models (NULL, esn);
}

static void
set_book(EBook *book, EBookStatus status, ESelectNames *esn)
{
    g_object_set(esn->model,
             "book", book,
             NULL);
    update_query (NULL, esn);
    g_object_unref(book);
    g_object_unref(esn->model);
    g_object_unref(esn);
}

static ESource *
find_first_source (ESourceList *source_list)
{
    GSList *groups, *sources, *l, *m;
            
    groups = e_source_list_peek_groups (source_list);
    for (l = groups; l; l = l->next) {
        ESourceGroup *group = l->data;
                
        sources = e_source_group_peek_sources (group);
        for (m = sources; m; m = m->next) {
            ESource *source = m->data;

            return source;
        }               
    }

    return NULL;
}

static void
addressbook_model_set_source (ESelectNames *e_select_names, EABModel *model, ESource *source)
{
    EBook *book;

    book = e_book_new();

    g_object_ref(e_select_names);
    g_object_ref(model);

    addressbook_load_source (book, source, (EBookCallback) set_book, e_select_names);
}

static void *
contact_key (const EContact *contact)
{
    EBook *book = NULL;
    const gchar *book_uri;
    
    if (contact == NULL)
        return NULL;

    g_assert (E_IS_CONTACT (contact));

#if notyet
    /* XXX we need a way to reproduce this here somehow.. or at
       least make sure we never collide between two contacts in
       different books. */
    book = e_contact_get_book (contact);
#endif
    book_uri = book ? e_book_get_uri (book) : "NoBook";
    return g_strdup_printf ("%s|%s", book_uri ? book_uri : "NoURI", (char*)e_contact_get_const ((EContact*)contact, E_CONTACT_UID));
}

static void
sync_one_model (gpointer k, gpointer val, gpointer closure)
{
    ETableWithout *etw = E_TABLE_WITHOUT (closure);
    ESelectNamesChild *child = val;
    ESelectNamesModel *model = child->source;
    gint i, count;
    EContact *contact;
    void *key;
    
    count = e_select_names_model_count (model);
    for (i = 0; i < count; ++i) {
        contact = e_select_names_model_get_contact (model, i);
        if (contact) {
            key = contact_key (contact);
            e_table_without_hide (etw, key);
            g_free (key);
        }
    }
}

static void
sync_table_and_models (ESelectNamesModel *triggering_model, ESelectNames *esl)
{
    e_table_without_show_all (E_TABLE_WITHOUT (esl->without));
    g_hash_table_foreach (esl->children, sync_one_model, esl->without);
}

static void
real_add_address_cb (int model_row, gpointer closure)
{
    ESelectNamesChild *child = closure;
    ESelectNames *names = child->names;
    const EContact *contact;
    EABDestination *dest = eab_destination_new ();
    gint mapped_row;

    mapped_row = e_table_subset_view_to_model_row (E_TABLE_SUBSET (names->without), model_row);

    contact = eab_model_contact_at (EAB_MODEL(names->model), mapped_row);
    
    if (contact != NULL) {
        eab_destination_set_contact (dest, (EContact*)contact, 0);

        e_select_names_model_append (child->source, dest);
        e_select_names_model_clean (child->source, FALSE);
    }
}

static void
real_add_address(ESelectNames *names, ESelectNamesChild *child)
{
    e_select_names_model_freeze (child->source);
    e_table_selected_row_foreach(e_table_scrolled_get_table(names->table),
                     real_add_address_cb, child);
    e_select_names_model_thaw (child->source);
}

static void
add_address(ETable *table, int row, int col, GdkEvent *event, ESelectNames *names)
{
    ESelectNamesChild *child;

    child = g_hash_table_lookup(names->children, names->def);
    if (child) {
        real_add_address(names, child);
    }
}

static void
sensitize_button (gpointer key, gpointer data, gpointer user_data)
{
    gboolean *sensitive = user_data;
    ESelectNamesChild *child = data;

    gtk_widget_set_sensitive (child->button, *sensitive);
}

static void
selection_change (ETable *table, ESelectNames *names)
{
    gboolean sensitive;

    sensitive = e_table_selected_count (table) > 0;

    g_hash_table_foreach (names->children, sensitize_button, &sensitive);
}

static void *
esn_get_key_fn (ETableModel *source, int row, void *closure)
{
    EABModel *model = EAB_MODEL (closure);
    const EContact *contact = eab_model_contact_at (model, row);
    void *key = contact_key (contact);
    return key;
}

static void *
esn_dup_key_fn (const void *key, void *closure)
{
    void *dup = (void *) g_strdup ((const gchar *) key);
    return dup;
}

static void
esn_free_gotten_key_fn (void *key, void *closure)
{
    g_free (key);
}

static void
esn_free_duped_key_fn (void *key, void *closure)
{
    g_free (key);
}

GtkWidget *
e_addressbook_create_ebook_table(char *name, char *string1, char *string2, int num1, int num2)
{
    ETableModel *adapter;
    ETableModel *without;
    EABModel *model;
    GtkWidget *table;

    model = eab_model_new ();
    adapter = E_TABLE_MODEL (eab_table_adapter_new (model));

    g_object_set(model,
             "editable", FALSE,
             NULL);

    without = e_table_without_new (adapter,
                       g_str_hash,
                       g_str_equal,
                       esn_get_key_fn,
                       esn_dup_key_fn,
                       esn_free_gotten_key_fn,
                       esn_free_duped_key_fn,
                       model);

    table = e_table_scrolled_new_from_spec_file (without,
                             NULL,
                             EVOLUTION_ETSPECDIR "/e-select-names.etspec",
                             NULL);

    g_object_set_data(G_OBJECT(table), "adapter", adapter);
    g_object_set_data(G_OBJECT(table), "without", without);
    g_object_set_data(G_OBJECT(table), "model", model);

    return table;
}

static void
source_selected (ESourceOptionMenu *menu, ESource *source, ESelectNames *e_select_names)
{
    addressbook_model_set_source (e_select_names, e_select_names->model, source);
    e_select_names_config_set_last_completion_book (e_source_peek_uid (source));
}

static void
update_query (GtkWidget *widget, ESelectNames *e_select_names)
{
    char *category = "";
    const char *search = "";
    char *query;
    char *q_array[4];
    int i;
    GString *s = g_string_new ("");

    if (e_select_names->categories) {
        category = e_categories_master_list_option_menu_get_category (E_CATEGORIES_MASTER_LIST_OPTION_MENU (e_select_names->categories));
    }
    if (e_select_names->select_entry) {
        search = gtk_entry_get_text (GTK_ENTRY (e_select_names->select_entry));
    }

    e_sexp_encode_string (s, search);

    i = 0;
    q_array[i++] = "(contains \"email\" \"\")";
    if (category && *category)
        q_array[i++] = g_strdup_printf ("(is \"category\" \"%s\")", category);
    if (search && *search)
        q_array[i++] = g_strdup_printf ("(or (beginswith \"email\" %s) "
                        "    (beginswith \"full_name\" %s) "
                        "    (beginswith \"nickname\" %s)"
                        "    (beginswith \"file_as\" %s))",
                        s->str, s->str, s->str, s->str);
    q_array[i++] = NULL;
    if (i > 2) {
        char *temp = g_strjoinv (" ", q_array);
        query = g_strdup_printf ("(and %s)", temp);
        g_free (temp);
    } else {
        query = g_strdup (q_array[0]);
    }
    g_object_set (e_select_names->model,
              "query", query,
              NULL);
    for (i = 1; q_array[i]; i++) {
        g_free (q_array[i]);
    }
    g_free (query);
    g_string_free (s, TRUE);
}

static void
status_message (EABModel *model, const gchar *message, ESelectNames *e_select_names)
{
    if (message == NULL)
        gtk_label_set_text (GTK_LABEL (e_select_names->status_message), "");
    else
        gtk_label_set_text (GTK_LABEL (e_select_names->status_message), message);
}

static void
categories_changed (GtkWidget *widget, ESelectNames *e_select_names)
{
    update_query (widget, e_select_names);
}

static void
select_entry_changed (GtkWidget *widget, ESelectNames *e_select_names)
{
    if (e_select_names->select_entry) {
        const char *select_string = gtk_entry_get_text (GTK_ENTRY (e_select_names->select_entry));
        char *select_strcoll_string = g_utf8_collate_key (select_string, -1);
        int count;
        ETable *table;
        int i;

        table = e_table_scrolled_get_table (e_select_names->table);

        count = e_table_model_row_count (e_select_names->without);

        for (i = 0; i < count; i++) {
            int model_row = e_table_view_to_model_row (table, i);
            char *row_strcoll_string =
                g_utf8_collate_key (e_table_model_value_at (e_select_names->without,
                                        E_CONTACT_FULL_NAME,
                                        model_row),
                            -1);
            if (g_utf8_collate (select_strcoll_string, row_strcoll_string) <= 0) {
                g_free (row_strcoll_string);
                break;
            }
            g_free (row_strcoll_string);
        }
        g_free (select_strcoll_string);
        if (i == count)
            i --;

        if (count > 0) {
            i = e_table_view_to_model_row (table, i);
            e_table_set_cursor_row (table, i);
        }
    }
}

GtkWidget *e_select_names_create_categories (gchar *name,
                         gchar *string1, gchar *string2,
                         gint int1, gint int2);

GtkWidget *
e_select_names_create_categories (gchar *name,
                  gchar *string1, gchar *string2,
                  gint int1, gint int2)
{
    ECategoriesMasterList *ecml;
    GtkWidget *option_menu;

    ecml = e_categories_master_list_wombat_new ();
    option_menu = e_categories_master_list_option_menu_new (ecml);
    g_object_unref (ecml);

    return option_menu;
}

static void
clear_widget (gpointer data, GObject *where_object_was)
{
    GtkWidget **widget_ref = data;
    *widget_ref = NULL;
}

static void
e_select_names_init (ESelectNames *e_select_names)
{
    GladeXML *gui;
    GtkWidget *widget, *button, *table, *esom;
    ESource *source = NULL;
    char *uid;
    
    gui = glade_xml_new (EVOLUTION_GLADEDIR "/select-names.glade", NULL, NULL);
    widget = glade_xml_get_widget (gui, "select-names-box");
    if (!widget) {
        g_object_unref (gui);
        return;
    }
    gtk_widget_ref (widget);
    gtk_container_remove (GTK_CONTAINER (widget->parent), widget);
    gtk_box_pack_start (GTK_DIALOG (e_select_names)->vbox, widget, TRUE, TRUE, 0);
    gtk_widget_unref (widget);

    gtk_dialog_add_buttons (GTK_DIALOG (e_select_names),
                GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
                GTK_STOCK_OK, GTK_RESPONSE_OK,
                NULL);

    gtk_window_set_modal (GTK_WINDOW (e_select_names), TRUE);
    gtk_window_set_default_size (GTK_WINDOW (e_select_names), 472, 512);
    gtk_window_set_title (GTK_WINDOW (e_select_names), _("Select Contents from Address Book"));
    gtk_window_set_resizable (GTK_WINDOW (e_select_names), TRUE);
    gtk_dialog_set_has_separator (GTK_DIALOG (e_select_names), FALSE);
    gtk_container_set_border_width (GTK_CONTAINER (e_select_names), 4);

    /* FIXME What to do on error/NULL ? */
    e_select_names->source_list = e_source_list_new_for_gconf_default ("/apps/evolution/addressbook/sources");
    
    e_select_names->gui = gui;

    /* Add the source menu */
    esom = e_source_option_menu_new (e_select_names->source_list);
    g_signal_connect (esom, "source_selected", G_CALLBACK (source_selected), e_select_names);
    gtk_widget_show (esom);

    table = glade_xml_get_widget (gui, "show_contacts_table");
    gtk_table_attach (GTK_TABLE (table), esom, 1, 2, 0, 1, GTK_FILL, GTK_FILL, 0, 0);

    /* Set up the rest of the widgets */
    e_select_names->children = g_hash_table_new(g_str_hash, g_str_equal);
    e_select_names->child_count = 0;
    e_select_names->def = NULL;

    gtk_dialog_set_default_response (GTK_DIALOG (e_select_names),
                     GTK_RESPONSE_OK);

    e_select_names->table = E_TABLE_SCROLLED(glade_xml_get_widget(gui, "table-source"));
    e_select_names->model = g_object_get_data(G_OBJECT(e_select_names->table), "model");
    e_select_names->adapter = g_object_get_data(G_OBJECT(e_select_names->table), "adapter");
    e_select_names->without = g_object_get_data(G_OBJECT(e_select_names->table), "without");
    gtk_widget_show (GTK_WIDGET (e_select_names->table));

    e_select_names->status_message = glade_xml_get_widget (gui, "status-message");
    if (e_select_names->status_message && !GTK_IS_LABEL (e_select_names->status_message))
        e_select_names->status_message = NULL;
    if (e_select_names->status_message) {
        e_select_names->status_id = g_signal_connect (e_select_names->model, "status_message",
                                  G_CALLBACK (status_message), e_select_names);
        g_object_weak_ref (G_OBJECT (e_select_names->status_message), clear_widget, &e_select_names->status_message);
    }

    e_select_names->search_id = g_signal_connect (e_select_names->model,
                              "search_result", G_CALLBACK (search_result),
                              e_select_names);

    e_select_names->categories = glade_xml_get_widget (gui, "custom-categories");
    if (e_select_names->categories && !E_IS_CATEGORIES_MASTER_LIST_OPTION_MENU (e_select_names->categories))
        e_select_names->categories = NULL;
    if (e_select_names->categories) {
        g_signal_connect(e_select_names->categories, "changed",
                 G_CALLBACK(categories_changed), e_select_names);
        g_object_weak_ref (G_OBJECT (e_select_names->categories), clear_widget, &e_select_names->categories);
    }
    gtk_widget_show (e_select_names->categories);

    e_select_names->select_entry = glade_xml_get_widget (gui, "entry-select");
    if (e_select_names->select_entry && !GTK_IS_ENTRY (e_select_names->select_entry))
        e_select_names->select_entry = NULL;
    if (e_select_names->select_entry) {
        g_signal_connect(e_select_names->select_entry, "changed",
                 G_CALLBACK(select_entry_changed), e_select_names);
        g_signal_connect(e_select_names->select_entry, "activate",
                 G_CALLBACK(update_query), e_select_names);
        g_object_weak_ref (G_OBJECT (e_select_names->select_entry), clear_widget, &e_select_names->select_entry);
    }

    button  = glade_xml_get_widget (gui, "button-find");
    if (button && GTK_IS_BUTTON (button))
        g_signal_connect(button, "clicked",
                 G_CALLBACK(update_query), e_select_names);

    g_signal_connect (e_table_scrolled_get_table (e_select_names->table), "double_click",
              G_CALLBACK (add_address), e_select_names);
    g_signal_connect (e_table_scrolled_get_table (e_select_names->table), "selection_change",
              G_CALLBACK (selection_change), e_select_names);
    selection_change (e_table_scrolled_get_table (e_select_names->table), e_select_names);

    /* Select a source for to display initially */
    uid = e_select_names_config_get_last_completion_book ();
    if (uid) {
        source = e_source_list_peek_source_by_uid (e_select_names->source_list, uid);
        g_free (uid);
    }
    
    if (!source)    
        source = find_first_source (e_select_names->source_list);

    /* FIXME What if we still can't find a source? */
    e_source_option_menu_select (E_SOURCE_OPTION_MENU (esom), source);

}

static void e_select_names_child_free(char *key, ESelectNamesChild *child, ESelectNames *e_select_names)
{
    g_signal_handler_disconnect(child->source, child->changed_id);

    g_free(child->title);
    g_object_unref(child->table_model);
    g_object_unref(child->source);
    g_free(key);
    g_free(child);
}

static void
e_select_names_dispose (GObject *object)
{
    ESelectNames *e_select_names = E_SELECT_NAMES(object);

    if (e_select_names->source_list) {
        g_object_unref (e_select_names->source_list);
        e_select_names->source_list = NULL;
    }
    
    if (e_select_names->status_id) {
        g_signal_handler_disconnect(e_select_names->model, e_select_names->status_id);
        e_select_names->status_id = 0;
    }

    if (e_select_names->search_id) {
        g_signal_handler_disconnect(e_select_names->model, e_select_names->search_id);
        e_select_names->search_id = 0;
    }

    if (e_select_names->gui) {
        g_object_unref(e_select_names->gui);
        e_select_names->gui = NULL;
    }

    if (e_select_names->children) {
        g_hash_table_foreach(e_select_names->children, (GHFunc) e_select_names_child_free, e_select_names);
        g_hash_table_destroy(e_select_names->children);
        e_select_names->children = NULL;
    }

    if (e_select_names->without) {
        g_object_unref(e_select_names->without);
        e_select_names->without = NULL;
    }
    if (e_select_names->adapter) {
        g_object_unref(e_select_names->adapter);
        e_select_names->adapter = NULL;
    }
    if (e_select_names->model) {
        g_object_unref(e_select_names->model);
        e_select_names->model = NULL;
    }

    if (e_select_names->def) {
        g_free(e_select_names->def);
        e_select_names->def = NULL;
    }

    if (G_OBJECT_CLASS(parent_class)->dispose)
        G_OBJECT_CLASS(parent_class)->dispose(object);
}

GtkWidget*
e_select_names_new (void)
{
    ESelectNames *e_select_names;
    
    e_select_names = g_object_new (E_TYPE_SELECT_NAMES, NULL);

    return GTK_WIDGET (e_select_names);
}

static void
button_clicked(GtkWidget *button, ESelectNamesChild *child)
{
    real_add_address(child->names, child);
}

static void
remove_address(ETable *table, int row, int col, GdkEvent *event, ESelectNamesChild *child)
{
    e_select_names_model_delete (child->source, row);
}

struct _RightClickData {
    ESelectNamesChild *child;
    int row;
};
typedef struct _RightClickData RightClickData;

static void
remove_cb (GtkWidget *widget, void *data)
{
    RightClickData *rcdata = (RightClickData *)data;

    e_select_names_model_delete (rcdata->child->source, rcdata->row);

    /* Free everything we've created */
    g_free (rcdata);
}

static void
section_right_click_cb (ETable *et, int row, int col, GdkEvent *ev, ESelectNamesChild *child)
{
    static EPopupMenu right_click_menu[] = {
        E_POPUP_ITEM (N_("Remove"), G_CALLBACK (remove_cb), 0),
        E_POPUP_TERMINATOR
    };
    RightClickData *rcdata = g_new0 (RightClickData, 1);

    rcdata->row = row;
    rcdata->child = child;

    e_popup_menu_run (right_click_menu, (GdkEvent *)ev, 0, 0, rcdata);
}

void
e_select_names_add_section (ESelectNames *e_select_names,
                const char *name, const char *id,
                ESelectNamesModel *source)
{
    ESelectNamesChild *child;
    GtkWidget *button;
    GtkWidget *label;
    GtkWidget *alignment;
    GtkTable *table;
    char *label_text;
    ETable *etable;
    ETableExtras *extras;
    ECell *string_cell;

    GtkWidget *sw;

    if (g_hash_table_lookup(e_select_names->children, id)) {
        return;
    }

    table = GTK_TABLE(glade_xml_get_widget (e_select_names->gui, "table-recipients"));

    child = g_new(ESelectNamesChild, 1);

    child->names = e_select_names;
    child->title = g_strdup (_(name));

    child->table_model = (ESelectNamesTableModel*)e_select_names_table_model_new (source);

    child->source = source;
    g_object_ref(child->source);

    alignment = gtk_alignment_new(0, 0, 1, 0);

    label_text = g_strconcat (child->title, " ->", NULL);

    label = gtk_label_new ("");

    gtk_label_set_markup (GTK_LABEL(label), label_text);

    g_free (label_text);

    button = gtk_button_new ();
    gtk_container_add (GTK_CONTAINER (button), label);
    child->label = label;
    child->button = button;

    gtk_container_add(GTK_CONTAINER(alignment), button);
    gtk_widget_show_all(alignment);
    g_signal_connect(button, "clicked",
             G_CALLBACK(button_clicked), child);
    gtk_table_attach(table, alignment,
             0, 1,
             e_select_names->child_count,
             e_select_names->child_count + 1,
             GTK_FILL, GTK_FILL,
             0, 0);

    etable = e_table_scrolled_get_table (e_select_names->table);
    gtk_widget_set_sensitive (button, e_table_selected_count (etable) > 0);

    extras = e_table_extras_new ();
    string_cell = e_table_extras_get_cell (extras, "string");

    g_object_set (string_cell,
              "underline_column", 2,
              NULL);
              
    sw = e_table_scrolled_new_from_spec_file (E_TABLE_MODEL (child->table_model),
                          extras,
                          EVOLUTION_ETSPECDIR "/e-select-names-section.etspec",
                          NULL);
    g_object_unref (extras);

    child->recipient_table = GTK_WIDGET (e_table_scrolled_get_table (E_TABLE_SCROLLED (sw)));

    g_signal_connect (child->recipient_table,
              "right_click",
              G_CALLBACK (section_right_click_cb),
              child);

    gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw),
                    GTK_POLICY_AUTOMATIC,
                    GTK_POLICY_AUTOMATIC);
    
    g_signal_connect(child->recipient_table, "double_click",
             G_CALLBACK(remove_address), child);

    child->changed_id = g_signal_connect (child->source,
                          "changed",
                          G_CALLBACK (sync_table_and_models),
                          e_select_names);
    
    gtk_widget_show_all (sw);
    
    gtk_table_attach(table, sw,
             1, 2,
             e_select_names->child_count,
             e_select_names->child_count + 1,
             GTK_FILL | GTK_EXPAND, GTK_FILL | GTK_EXPAND,
             0, 0);

    g_hash_table_insert(e_select_names->children, g_strdup(id), child);

    sync_table_and_models (child->source, e_select_names);

    e_select_names->child_count++;
}

void
e_select_names_set_default (ESelectNames *e_select_names,
                const char *id)
{
    ESelectNamesChild *child;

    if (e_select_names->def) {
        child = g_hash_table_lookup(e_select_names->children, e_select_names->def);
        if (child) {
            GtkWidget *label = child->label;

            /* set the previous default to non-bold */
            gtk_label_set_markup (GTK_LABEL (label), child->title);
        }
    }

    g_free(e_select_names->def);
    e_select_names->def = g_strdup(id);

    if (e_select_names->def) {
        child = g_hash_table_lookup(e_select_names->children, e_select_names->def);
        if (child) {
            GtkWidget *label = child->label;
            char *markup = g_strconcat ("<b>", child->title, "</b>", NULL);

            /* set the new default to bold */
            gtk_label_set_markup (GTK_LABEL (label), markup);
            g_free (markup);
        }
    }
}