sync with latest 0.9.0pre1
[claws.git] / src / addressbook.c
index 00323e0c0d68956890ddf4a3f3d58ac723e1fbdd..8d5bdc2c35ea3ba68789276436eee9f1d7490567 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
- * Copyright (C) 1999-2002 Hiroyuki Yamamoto
+ * Copyright (C) 1999-2003 Hiroyuki Yamamoto
  *
  * 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
@@ -54,7 +54,7 @@
 #include "menu.h"
 #include "stock_pixmap.h"
 #include "xml.h"
-#include "prefs.h"
+#include "prefs_gtk.h"
 #include "procmime.h"
 #include "utils.h"
 #include "gtkutils.h"
 #include "editgroup.h"
 #include "editaddress.h"
 #include "editbook.h"
-#include "ldif.h"
 #include "importldif.h"
-#include "mutt.h"
 #include "importmutt.h"
-#include "pine.h"
 #include "importpine.h"
 
 #ifdef USE_JPILOT
@@ -88,7 +85,7 @@
 
 #ifdef USE_LDAP
 #include <pthread.h>
-#include "syldap.h"
+#include "ldapserver.h"
 #include "editldap.h"
 
 #define ADDRESSBOOK_LDAP_BUSYMSG "Busy"
@@ -97,6 +94,8 @@
 #include "addrselect.h"
 #include "addrclip.h"
 #include "addrgather.h"
+#include "adbookbase.h"
+#include "exphtmldlg.h"
 
 typedef enum
 {
@@ -293,11 +292,6 @@ static void addressbook_move_nodes_up              (GtkCTree       *ctree,
                                                GtkCTreeNode    *node);
 static GtkCTreeNode *addressbook_find_group_node (GtkCTreeNode *parent,
                                                   ItemGroup    *group);
-static GtkCTreeNode *addressbook_find_folder_node( GtkCTreeNode        *parent,
-                                                  ItemFolder   *folder );
-
-static void addressbook_delete_object          (AddressObject  *obj);
-
 static void key_pressed                                (GtkWidget      *widget,
                                                 GdkEventKey    *event,
                                                 gpointer        data);
@@ -307,9 +301,19 @@ static gint addressbook_treenode_compare_func      (GtkCList       *clist,
 static gint addressbook_list_compare_func      (GtkCList       *clist,
                                                 gconstpointer   ptr1,
                                                 gconstpointer   ptr2);
+static void addressbook_folder_load_one_person (GtkCTree *clist, 
+                                                ItemPerson *person,  
+                                                AddressTypeControlItem *atci, 
+                                                AddressTypeControlItem *atciMail);
+static void addressbook_folder_refresh_one_person(GtkCTree *clist, 
+                                                 ItemPerson *person);
+static void addressbook_folder_remove_one_person(GtkCTree *clist, 
+                                                ItemPerson *person);
+static void addressbook_folder_remove_node     (GtkCTree *clist, 
+                                                GtkCTreeNode *node);
 
 #ifdef USE_LDAP
-static void addressbook_ldap_show_message      (SyldapServer *server);
+static void addressbook_ldap_show_message      ( LdapServer *server );
 #endif
 
 /* LUT's and IF stuff */
@@ -335,6 +339,7 @@ static void addressbook_list_select_remove  ( AddrItemObject    *aio );
 static void addressbook_import_ldif_cb         ( void );
 static void addressbook_import_mutt_cb         ( void );
 static void addressbook_import_pine_cb         ( void );
+static void addressbook_export_html_cb         ( void );
 static void addressbook_clip_cut_cb            ( void );
 static void addressbook_clip_copy_cb           ( void );
 static void addressbook_clip_paste_cb          ( void );
@@ -374,23 +379,15 @@ static GtkItemFactoryEntry addressbook_entries[] =
        {N_("/_Address/_Edit"),         "<alt>Return",  addressbook_edit_address_cb,    0, NULL},
        {N_("/_Address/_Delete"),       NULL,           addressbook_delete_address_cb,  0, NULL},
        {N_("/_Tools/---"),             NULL,           NULL, 0, "<Separator>"},
-       {N_("/_Tools/Import _LDIF file"), NULL,         addressbook_import_ldif_cb,     0, NULL},
-       {N_("/_Tools/Import M_utt file"), NULL,         addressbook_import_mutt_cb,     0, NULL},
-       {N_("/_Tools/Import _Pine file"), NULL,         addressbook_import_pine_cb,     0, NULL},
+       {N_("/_Tools/Import _LDIF file..."), NULL,      addressbook_import_ldif_cb,     0, NULL},
+       {N_("/_Tools/Import M_utt file..."), NULL,      addressbook_import_mutt_cb,     0, NULL},
+       {N_("/_Tools/Import _Pine file..."), NULL,      addressbook_import_pine_cb,     0, NULL},
+       {N_("/_Tools/---"),             NULL,           NULL, 0, "<Separator>"},
+       {N_("/_Tools/Export _HTML..."), NULL,           addressbook_export_html_cb,     0, NULL},
        {N_("/_Help"),                  NULL,           NULL, 0, "<LastBranch>"},
        {N_("/_Help/_About"),           NULL,           about_show, 0, NULL}
 };
 
-/* New options to be added. */
-/*
-       {N_("/_Tools"),                 NULL,           NULL, 0, "<Branch>"},
-       {N_("/_Tools/Import _Mozilla"), NULL,           NULL,                           0, NULL},
-       {N_("/_Tools/Import _vCard"),   NULL,           NULL,                           0, NULL},
-       {N_("/_Tools/---"),             NULL,           NULL, 0, "<Separator>"},
-       {N_("/_Tools/Export _LDIF file"), NULL,         NULL,                           0, NULL},
-       {N_("/_Tools/Export v_Card"),   NULL,           NULL,                           0, NULL},
-*/
-
 static GtkItemFactoryEntry addressbook_tree_popup_entries[] =
 {
        {N_("/New _Address"),   NULL, addressbook_new_address_cb, 0, NULL},
@@ -421,6 +418,79 @@ static GtkItemFactoryEntry addressbook_list_popup_entries[] =
        {N_("/Pa_ste Address"), NULL, addressbook_clip_paste_address_cb,     0, NULL}
 };
 
+/**
+ * Structure of error message table.
+ */
+typedef struct _ErrMsgTableEntry ErrMsgTableEntry;
+struct _ErrMsgTableEntry {
+       gint    code;
+       gchar   *description;
+};
+
+static gchar *_errMsgUnknown_ = N_( "Unknown" );
+
+/**
+ * Lookup table of error messages for general errors. Note that a NULL
+ * description signifies the end of the table.
+ */
+static ErrMsgTableEntry _lutErrorsGeneral_[] = {
+       { MGU_SUCCESS,          N_("Success") },
+       { MGU_BAD_ARGS,         N_("Bad arguments") },
+       { MGU_NO_FILE,          N_("File not specified") },
+       { MGU_OPEN_FILE,        N_("Error opening file") },
+       { MGU_ERROR_READ,       N_("Error reading file") },
+       { MGU_EOF,              N_("End of file encountered") },
+       { MGU_OO_MEMORY,        N_("Error allocating memory") },
+       { MGU_BAD_FORMAT,       N_("Bad file format") },
+       { MGU_ERROR_WRITE,      N_("Error writing to file") },
+       { MGU_OPEN_DIRECTORY,   N_("Error opening directory") },
+       { MGU_NO_PATH,          N_("No path specified") },
+       { 0,                    NULL }
+};
+
+#ifdef USE_LDAP
+/**
+ * Lookup table of error messages for LDAP errors.
+ */
+static ErrMsgTableEntry _lutErrorsLDAP_[] = {
+       { MGU_SUCCESS,          N_("Success") },
+       { MGU_LDAP_CONNECT,     N_("Error connecting to LDAP server") },
+       { MGU_LDAP_INIT,        N_("Error initializing LDAP") },
+       { MGU_LDAP_BIND,        N_("Error binding to LDAP server") },
+       { MGU_LDAP_SEARCH,      N_("Error searching LDAP database") },
+       { MGU_LDAP_TIMEOUT,     N_("Timeout performing LDAP operation") },
+       { MGU_LDAP_CRITERIA,    N_("Error in LDAP search criteria") },
+       { MGU_LDAP_CRITERIA,    N_("Error in LDAP search criteria") },
+       { MGU_LDAP_NOENTRIES,   N_("No LDAP entries found for search criteria") },
+       { 0,                    NULL }
+};
+#endif
+
+/**
+ * Lookup message for specified error code.
+ * \param lut  Lookup table.
+ * \param code Code to lookup.
+ * \return Description associated to code.
+ */
+static gchar *addressbook_err2string( ErrMsgTableEntry lut[], gint code ) {
+        gchar *desc = NULL;
+        ErrMsgTableEntry entry;
+        gint i;
+
+        for( i = 0; ; i++ ) {
+                entry = lut[ i ];
+                if( entry.description == NULL ) break;
+                if( entry.code == code ) {
+                        desc = entry.description;
+                        break;
+                }
+        }
+        if( ! desc ) {
+               desc = _errMsgUnknown_;
+        }
+        return desc;
+}
+
 void addressbook_open(Compose *target)
 {
        /* Initialize all static members */
@@ -439,11 +509,12 @@ void addressbook_open(Compose *target)
                addressbook_load_tree();
                gtk_ctree_select(GTK_CTREE(addrbook.ctree),
                                 GTK_CTREE_NODE(GTK_CLIST(addrbook.ctree)->row_list));
-       } else
+       }
+       else {
                gtk_widget_hide(addrbook.window);
+       }
 
        gtk_widget_show_all(addrbook.window);
-
        addressbook_set_target_compose(target);
 }
 
@@ -466,7 +537,6 @@ void addressbook_destroy() {
 void addressbook_set_target_compose(Compose *target)
 {
        addrbook.target_compose = target;
-
        addressbook_button_set_sensitive();
 }
 
@@ -791,8 +861,7 @@ static void addressbook_create(void)
 
 }
 
-static gint addressbook_close(void)
-{
+static gint addressbook_close( void ) {
        gtk_widget_hide(addrbook.window);
        addressbook_export_to_file();
        return TRUE;
@@ -817,10 +886,12 @@ static void addressbook_ds_status_message( AddressDataSource *ds, gchar *msg ) {
                gchar *name;
 
                name = addrindex_ds_get_name( ds );
-               sprintf( addressbook_msgbuf, "%s: %s", name, msg );
+               g_snprintf( addressbook_msgbuf, sizeof(addressbook_msgbuf),
+                           "%s: %s", name, msg );
        }
        else {
-               sprintf( addressbook_msgbuf, "%s", msg );
+               g_snprintf( addressbook_msgbuf, sizeof(addressbook_msgbuf),
+                           "%s", msg );
        }
        addressbook_status_show( addressbook_msgbuf );
 }
@@ -828,16 +899,19 @@ static void addressbook_ds_status_message( AddressDataSource *ds, gchar *msg ) {
 static void addressbook_ds_show_message( AddressDataSource *ds ) {
        gint retVal;
        gchar *name;
+       gchar *desc;
        *addressbook_msgbuf = '\0';
        if( ds ) {
                name = addrindex_ds_get_name( ds );
                retVal = addrindex_ds_get_status_code( ds );
                if( retVal == MGU_SUCCESS ) {
-                       sprintf( addressbook_msgbuf, "%s", name );
+                       g_snprintf( addressbook_msgbuf,
+                                   sizeof(addressbook_msgbuf), "%s", name );
                }
                else {
-                       sprintf( addressbook_msgbuf, "%s: %s", name,
-                               mgu_error2string( retVal ) );
+                       desc = addressbook_err2string( _lutErrorsGeneral_, retVal );
+                       g_snprintf( addressbook_msgbuf, 
+                           sizeof(addressbook_msgbuf), "%s: %s", name, desc );
                }
        }
        addressbook_status_show( addressbook_msgbuf );
@@ -883,7 +957,8 @@ static void addressbook_del_clicked(GtkButton *button, gpointer data)
        AddrItemObject *aio;
        AddrSelectItem *item;
        GList *list, *node;
-
+       gboolean refreshList = FALSE;
+       
        pobj = gtk_ctree_node_get_row_data(ctree, addrbook.opened );
        g_return_if_fail(pobj != NULL);
 
@@ -897,7 +972,7 @@ static void addressbook_del_clicked(GtkButton *button, gpointer data)
        /* Test for read only */
        iface = ds->interface;
        if( iface->readOnly ) {
-               aval = alertpanel( _("Delete address(es)"),
+               alertpanel( _("Delete address(es)"),
                        _("This address data is readonly and cannot be deleted."),
                        _("Close"), NULL, NULL );
                return;
@@ -945,9 +1020,11 @@ static void addressbook_del_clicked(GtkButton *button, gpointer data)
                                }
                                /* Remove group from parent node */
                                gtk_ctree_remove_node( ctree, nd );
+                               refreshList = TRUE;
                        }
                        else if( aio->type == ADDR_ITEM_PERSON ) {
                                ItemPerson *item = ( ItemPerson * ) aio;
+                               addressbook_folder_remove_one_person( clist, item );
                                item = addrbook_remove_person( abf, item );
                                if( item ) {
                                        addritem_free_item_person( item );
@@ -960,11 +1037,12 @@ static void addressbook_del_clicked(GtkButton *button, gpointer data)
                                if( item ) {
                                        addritem_free_item_email( item );
                                }
+                               addressbook_folder_refresh_one_person( clist, person );
                        }
                }
                g_list_free( list );
                addressbook_list_select_clear();
-               gtk_ctree_select( ctree, addrbook.opened);
+               if( refreshList ) gtk_ctree_select( ctree, addrbook.opened);
                return;
        }
        else if( pobj->type == ADDR_ITEM_GROUP ) {
@@ -1038,7 +1116,7 @@ gchar *addressbook_format_address( AddrItemObject * aio ) {
        }
        if( address ) {
                if( name && name[0] != '\0' ) {
-                       if( strchr_with_skip_quote( name, '"', ',' ) )
+                       if( name[0] != '"' && strpbrk( name, ",.[]<>" ) != NULL )
                                buf = g_strdup_printf( "\"%s\" <%s>", name, address );
                        else
                                buf = g_strdup_printf( "%s <%s>", name, address );
@@ -1062,6 +1140,14 @@ static void addressbook_to_clicked(GtkButton *button, gpointer data)
        compose = addrbook.target_compose;
        if( ! compose ) return;
 
+       /* Nothing selected, but maybe there is something in text entry */
+       addr = gtk_entry_get_text( GTK_ENTRY( addrbook.entry) );
+       if ( addr ) {
+               compose_entry_append(
+                       compose, addr, (ComposeEntryType)data );
+       }
+
+       /* Select from address list */
        list = addrselect_get_list( _addressSelect_ );
        node = list;
        while( node ) {
@@ -1114,6 +1200,7 @@ static void addressbook_menuitem_set_sensitive( AddressObject *obj, GtkCTreeNode
        gboolean canAdd = FALSE;
        gboolean canEditTr = TRUE;
        gboolean editAddress = FALSE;
+       gboolean canExport = TRUE;
        AddressTypeControlItem *atci = NULL;
        AddressDataSource *ds = NULL;
        AddressInterface *iface = NULL;
@@ -1129,7 +1216,7 @@ static void addressbook_menuitem_set_sensitive( AddressObject *obj, GtkCTreeNode
                                menu_set_sensitive( addrbook.menu_factory, atci->menuCommand, TRUE );
                        }
                }
-               canEditTr = FALSE;
+               canEditTr = canExport = FALSE;
        }
        else if( obj->type == ADDR_DATASOURCE ) {
                AdapterDSource *ads = ADAPTER_DSOURCE(obj);
@@ -1139,7 +1226,7 @@ static void addressbook_menuitem_set_sensitive( AddressObject *obj, GtkCTreeNode
                        canAdd = canEdit = editAddress = TRUE;
                }
                if( ! iface->haveLibrary ) {
-                       canAdd = canEdit = editAddress = FALSE;
+                       canAdd = canEdit = editAddress = canExport = FALSE;
                }
        }
        else if( obj->type == ADDR_ITEM_FOLDER ) {
@@ -1179,6 +1266,9 @@ static void addressbook_menuitem_set_sensitive( AddressObject *obj, GtkCTreeNode
 
        menu_set_sensitive( addrbook.menu_factory, "/File/Edit",      canEditTr );
        menu_set_sensitive( addrbook.menu_factory, "/File/Delete",    canEditTr );
+
+       /* Export data */
+       menu_set_sensitive( addrbook.menu_factory, "/Tools/Export HTML...", canExport );
 }
 
 static void addressbook_tree_selected(GtkCTree *ctree, GtkCTreeNode *node,
@@ -2024,7 +2114,7 @@ static void addressbook_treenode_edit_cb(gpointer data, guint action,
        if( name && parentNode ) {
                /* Update node in tree view */
                addressbook_change_node_name( node, name );
-               gtk_ctree_sort_node(ctree, parentNode);
+               gtk_sctree_sort_node(ctree, parentNode);
                gtk_ctree_expand( ctree, node );
                gtk_ctree_select( ctree, node );
        }
@@ -2156,10 +2246,9 @@ static void addressbook_new_address_cb( gpointer data, guint action, GtkWidget *
                if( ADAPTER_DSOURCE(pobj)->subType == ADDR_BOOK ) {
                        /* New address */
                        ItemPerson *person = addressbook_edit_person( abf, NULL, NULL, FALSE );
-                       if( person ) {
-                               if( addrbook.treeSelected == addrbook.opened ) {
-                                       gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
-                               }
+                       if( person && addrbook.treeSelected == addrbook.opened ) {
+                               gtk_clist_unselect_all( GTK_CLIST(addrbook.clist) );
+                               addressbook_folder_refresh_one_person( GTK_CTREE(addrbook.clist), person );
                        }
                }
        }
@@ -2212,33 +2301,6 @@ static GtkCTreeNode *addressbook_find_group_node( GtkCTreeNode *parent, ItemGrou
        return NULL;
 }
 
-/*
-* Search for specified child folder node in address index tree.
-* Enter: parent Parent node.
-*        folder Folder to find.
-*/
-static GtkCTreeNode *addressbook_find_folder_node( GtkCTreeNode *parent, ItemFolder *folder ) {
-       GtkCTreeNode *node = NULL;
-       GtkCTreeRow *currRow;
-
-       currRow = GTK_CTREE_ROW( parent );
-       if( currRow ) {
-               node = currRow->children;
-               while( node ) {
-                       AddressObject *obj;
-
-                       obj = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), node );
-                       if( obj->type == ADDR_ITEM_FOLDER ) {
-                               ItemFolder *f = ADAPTER_FOLDER(obj)->itemFolder;
-                               if( f == folder ) return node;
-                       }
-                       currRow = GTK_CTREE_ROW(node);
-                       node = currRow->sibling;
-               }
-       }
-       return NULL;
-}
-
 static AddressBookFile *addressbook_get_book_file() {
        AddressBookFile *abf = NULL;
        AddressDataSource *ds = NULL;
@@ -2271,7 +2333,7 @@ static void addressbook_move_nodes_up( GtkCTree *ctree, GtkCTreeNode *node ) {
                while( (child = currRow->children) ) {
                        gtk_ctree_move( ctree, child, parent, node );
                }
-               gtk_ctree_sort_node( ctree, parent );
+               gtk_sctree_sort_node( ctree, parent );
        }
 }
 
@@ -2314,7 +2376,7 @@ static void addressbook_edit_address_cb( gpointer data, guint action, GtkWidget
                        ItemPerson *person;
                        person = ( ItemPerson * ) ADDRITEM_PARENT(email);
                        if( addressbook_edit_person( abf, NULL, person, TRUE ) == NULL ) return;
-                       gtk_ctree_select( ctree, addrbook.opened );
+                       addressbook_folder_refresh_one_person( clist, person );
                        invalidate_address_completion();
                        return;
                }
@@ -2323,8 +2385,8 @@ static void addressbook_edit_address_cb( gpointer data, guint action, GtkWidget
                /* Edit person - basic page */
                ItemPerson *person = ( ItemPerson * ) obj;
                if( addressbook_edit_person( abf, NULL, person, FALSE ) == NULL ) return;
-               gtk_ctree_select( ctree, addrbook.opened);
                invalidate_address_completion();
+               addressbook_folder_refresh_one_person( clist, person );
                return;
        }
        else if( obj->type == ADDR_ITEM_GROUP ) {
@@ -2341,8 +2403,8 @@ static void addressbook_edit_address_cb( gpointer data, guint action, GtkWidget
        /* Update tree node with node name */
        if( node == NULL ) return;
        addressbook_change_node_name( node, name );
-       gtk_ctree_sort_node( ctree, parentNode );
-       gtk_ctree_select( ctree, addrbook.opened );
+       gtk_sctree_sort_node( ctree, parentNode );
+       gtk_ctree_select( ctree, addrbook.opened ); 
 }
 
 static void addressbook_delete_address_cb(gpointer data, guint action,
@@ -2422,91 +2484,141 @@ static void addressbook_load_group( GtkCTree *clist, ItemGroup *itemGroup ) {
        }
 }
 
-static void addressbook_folder_load_person( GtkCTree *clist, ItemFolder *itemFolder ) {
-       GList *items;
-       AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_PERSON );
-       AddressTypeControlItem *atciMail = addrbookctl_lookup( ADDR_ITEM_EMAIL );
-
-       if( atci == NULL ) return;
-       if( atciMail == NULL ) return;
-
-       /* Load email addresses */
-       items = addritem_folder_get_person_list( itemFolder );
-       for( ; items != NULL; items = g_list_next( items ) ) {
-               GtkCTreeNode *nodePerson = NULL;
-               GtkCTreeNode *nodeEMail = NULL;
-               gchar *text[N_COLS];
-               gboolean flgFirst = TRUE, haveAddr = FALSE;
-               ItemPerson *person;
-               GList *node;
+static void addressbook_folder_load_one_person(
+               GtkCTree *clist, ItemPerson *person,
+               AddressTypeControlItem *atci,
+               AddressTypeControlItem *atciMail )
+{
+       GtkCTreeNode *nodePerson = NULL;
+       GtkCTreeNode *nodeEMail = NULL;
+       gchar *text[N_COLS];
+       gboolean flgFirst = TRUE, haveAddr = FALSE;
+       GList *node;
 
-               person = ( ItemPerson * ) items->data;
-               if( person == NULL ) continue;
+       if( person == NULL ) return;
 
-               text[COL_NAME] = NULL;
-               node = person->listEMail;
-               while( node ) {
-                       ItemEMail *email = node->data;
-                       gchar *eMailAddr = NULL;
-                       node = g_list_next( node );
+       text[COL_NAME] = NULL;
+       node = person->listEMail;
+       while( node ) {
+               ItemEMail *email = node->data;
+               gchar *eMailAddr = NULL;
+               node = g_list_next( node );
 
-                       text[COL_ADDRESS] = email->address;
-                       text[COL_REMARKS] = email->remarks;
-                       eMailAddr = ADDRITEM_NAME(email);
-                       if( eMailAddr && *eMailAddr == '\0' ) eMailAddr = NULL;
-                       if( flgFirst ) {
-                               /* First email belongs with person */
-                               gchar *str = addressbook_format_item_clist( person, email );
-                               if( str ) {
-                                       text[COL_NAME] = str;
-                               }
-                               else {
-                                       text[COL_NAME] = ADDRITEM_NAME(person);
-                               }
-                               nodePerson = gtk_ctree_insert_node(
-                                               clist, NULL, NULL,
-                                               text, FOLDER_SPACING,
-                                               atci->iconXpm, atci->maskXpm,
-                                               atci->iconXpmOpen, atci->maskXpmOpen,
-                                               FALSE, person->isOpened );
-                               g_free( str );
-                               str = NULL;
-                               gtk_ctree_node_set_row_data(clist, nodePerson, person );
+               text[COL_ADDRESS] = email->address;
+               text[COL_REMARKS] = email->remarks;
+               eMailAddr = ADDRITEM_NAME(email);
+               if( eMailAddr && *eMailAddr == '\0' ) eMailAddr = NULL;
+               if( flgFirst ) {
+                       /* First email belongs with person */
+                       gchar *str = addressbook_format_item_clist( person, email );
+                       if( str ) {
+                               text[COL_NAME] = str;
                        }
                        else {
-                               /* Subsequent email is a child node of person */
-                               text[COL_NAME] = ADDRITEM_NAME(email);
-                               nodeEMail = gtk_ctree_insert_node(
-                                               clist, nodePerson, NULL,
-                                               text, FOLDER_SPACING,
-                                               atciMail->iconXpm, atciMail->maskXpm,
-                                               atciMail->iconXpmOpen, atciMail->maskXpmOpen,
-                                               FALSE, TRUE );
-                               gtk_ctree_node_set_row_data(clist, nodeEMail, email );
+                               text[COL_NAME] = ADDRITEM_NAME(person);
                        }
-                       flgFirst = FALSE;
-                       haveAddr = TRUE;
-               }
-               if( ! haveAddr ) {
-                       /* Have name without EMail */
-                       text[COL_NAME] = ADDRITEM_NAME(person);
-                       text[COL_ADDRESS] = NULL;
-                       text[COL_REMARKS] = NULL;
                        nodePerson = gtk_ctree_insert_node(
                                        clist, NULL, NULL,
                                        text, FOLDER_SPACING,
                                        atci->iconXpm, atci->maskXpm,
                                        atci->iconXpmOpen, atci->maskXpmOpen,
                                        FALSE, person->isOpened );
+                       g_free( str );
+                       str = NULL;
                        gtk_ctree_node_set_row_data(clist, nodePerson, person );
                }
-               gtk_ctree_sort_node(GTK_CTREE(clist), NULL);
+               else {
+                       /* Subsequent email is a child node of person */
+                       text[COL_NAME] = ADDRITEM_NAME(email);
+                       nodeEMail = gtk_ctree_insert_node(
+                                       clist, nodePerson, NULL,
+                                       text, FOLDER_SPACING,
+                                       atciMail->iconXpm, atciMail->maskXpm,
+                                       atciMail->iconXpmOpen, atciMail->maskXpmOpen,
+                                       FALSE, TRUE );
+                       gtk_ctree_node_set_row_data(clist, nodeEMail, email );
+               }
+               flgFirst = FALSE;
+               haveAddr = TRUE;
+       }
+       if( ! haveAddr ) {
+               /* Have name without EMail */
+               text[COL_NAME] = ADDRITEM_NAME(person);
+               text[COL_ADDRESS] = NULL;
+               text[COL_REMARKS] = NULL;
+               nodePerson = gtk_ctree_insert_node(
+                               clist, NULL, NULL,
+                               text, FOLDER_SPACING,
+                               atci->iconXpm, atci->maskXpm,
+                               atci->iconXpmOpen, atci->maskXpmOpen,
+                               FALSE, person->isOpened );
+               gtk_ctree_node_set_row_data(clist, nodePerson, person );
+       }
+       gtk_sctree_sort_node(GTK_CTREE(clist), NULL);
+       return;
+}
+
+static void addressbook_folder_load_person( GtkCTree *clist, ItemFolder *itemFolder ) {
+       GList *items;
+       AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_PERSON );
+       AddressTypeControlItem *atciMail = addrbookctl_lookup( ADDR_ITEM_EMAIL );
+
+       if( atci == NULL ) return;
+       if( atciMail == NULL ) return;
+
+       /* Load email addresses */
+       items = addritem_folder_get_person_list( itemFolder );
+       for( ; items != NULL; items = g_list_next( items ) ) {
+               addressbook_folder_load_one_person( clist, items->data, atci, atciMail );
        }
        /* Free up the list */
        mgu_clear_list( items );
        g_list_free( items );
 }
 
+static void addressbook_folder_remove_node( GtkCTree *clist, GtkCTreeNode *node ) { 
+       addrbook.listSelected = NULL;
+       gtk_ctree_remove_node( clist, node );
+       addressbook_menubar_set_sensitive( FALSE );
+       addressbook_menuitem_set_sensitive(
+               gtk_ctree_node_get_row_data(
+                       GTK_CTREE(clist), addrbook.treeSelected ),
+               addrbook.treeSelected );
+}
+
+static void addressbook_folder_refresh_one_person( GtkCTree *clist, ItemPerson *person ) {
+       AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_PERSON );
+       AddressTypeControlItem *atciMail = addrbookctl_lookup( ADDR_ITEM_EMAIL );
+       GtkCTreeNode *node;
+       if( atci == NULL ) return;
+       if( atciMail == NULL ) return;
+       if( person == NULL ) return;
+       /* unload the person */
+       
+       node = gtk_ctree_find_by_row_data( clist, NULL, person );
+       if( node )
+               addressbook_folder_remove_node( clist, node );
+       addressbook_folder_load_one_person( clist, person, atci, atciMail );
+       node = gtk_ctree_find_by_row_data( clist, NULL, person );
+       if( node ) {
+               gtk_ctree_select( clist, node );
+               if (!gtk_ctree_node_is_visible( clist, node ) ) 
+                       gtk_ctree_node_moveto( clist, node, 0, 0, 0 );
+       }
+}
+
+static void addressbook_folder_remove_one_person( GtkCTree *clist, ItemPerson *person ) {
+       GtkCTreeNode *node;
+       gint row;
+       
+       if( person == NULL ) return;
+       node = gtk_ctree_find_by_row_data( clist, NULL, person );
+       row  = gtk_clist_find_row_from_data( GTK_CLIST(clist), person );
+       if( node ) {
+               addressbook_folder_remove_node( clist, node );
+       }
+}
+
 static void addressbook_folder_load_group( GtkCTree *clist, ItemFolder *itemFolder ) {
        GList *items;
        AddressTypeControlItem *atci =  addrbookctl_lookup( ADDR_ITEM_GROUP );
@@ -2528,59 +2640,13 @@ static void addressbook_folder_load_group( GtkCTree *clist, ItemFolder *itemFold
                                      atci->iconXpmOpen, atci->maskXpmOpen,
                                      FALSE, FALSE);
                gtk_ctree_node_set_row_data(clist, nodeGroup, group );
-               gtk_ctree_sort_node(clist, NULL);
+               gtk_sctree_sort_node(clist, NULL);
        }
        /* Free up the list */
        mgu_clear_list( items );
        g_list_free( items );
 }
 
-/*
- * Load data sources into list.
- */
-static void addressbook_node_load_datasource( GtkCTree *clist, AddressObject *obj ) {
-       AdapterInterface *adapter;
-       AddressInterface *iface;
-       AddressTypeControlItem *atci = NULL;
-       GtkCTreeNode *newNode, *node;
-       GtkCTreeRow *row;
-       GtkCell *cell = NULL;
-       gchar *text[N_COLS];
-
-       adapter = ADAPTER_INTERFACE(obj);
-       if( adapter == NULL ) return;
-       iface = adapter->interface;
-       atci = adapter->atci;
-       if( atci == NULL ) return;
-
-       /* Create nodes in list copying values for data sources in tree */
-       row = GTK_CTREE_ROW( adapter->treeNode );
-       if( row ) {
-               node = row->children;
-               while( node ) {
-                       gpointer data;
-
-                       data = gtk_ctree_node_get_row_data( clist, node );
-                       row = GTK_CTREE_ROW( node );
-                       cell = ( ( GtkCListRow * )row )->cell;
-                       text[COL_NAME] = cell->u.text;
-                       text[COL_ADDRESS] = NULL;
-                       text[COL_REMARKS] = NULL;
-                       newNode = gtk_ctree_insert_node( clist, NULL, NULL,
-                                     text, FOLDER_SPACING,
-                                     atci->iconXpm, atci->maskXpm,
-                                     atci->iconXpmOpen, atci->maskXpmOpen,
-                                     FALSE, FALSE);
-                       /*
-                       gtk_ctree_node_set_row_data( clist, newNode, data );
-                       gtk_ctree_node_set_row_data_full( clist, newNode, NULL, NULL );
-                       */
-                       node = row->sibling;
-               }
-       }
-       gtk_ctree_sort_node( clist, NULL );
-}
-
 /*
 * Search ctree widget callback function.
 * Enter: pA Pointer to node.
@@ -2727,8 +2793,10 @@ static void addressbook_set_clist( AddressObject *obj ) {
                        /* Load root folder */
                        ItemFolder *rootFolder = NULL;
                        rootFolder = addrindex_ds_get_root_folder( ds );
-                       addressbook_folder_load_person( ctreelist, addrindex_ds_get_root_folder( ds ) );
-                       addressbook_folder_load_group( ctreelist, addrindex_ds_get_root_folder( ds ) );
+                       addressbook_folder_load_person(
+                               ctreelist, addrindex_ds_get_root_folder( ds ) );
+                       addressbook_folder_load_group(
+                               ctreelist, addrindex_ds_get_root_folder( ds ) );
                }
        }
        else {
@@ -2793,7 +2861,8 @@ AdapterDSource *addressbook_create_ds_adapter( AddressDataSource *ds,
 }
 
 void addressbook_ads_set_name( AdapterDSource *adapter, gchar *value ) {
-       ADDRESS_OBJECT_NAME(adapter) = mgu_replace_string( ADDRESS_OBJECT_NAME(adapter), value );
+       ADDRESS_OBJECT_NAME(adapter) =
+               mgu_replace_string( ADDRESS_OBJECT_NAME(adapter), value );
 }
 
 /*
@@ -2824,8 +2893,10 @@ static void addressbook_load_tree( void ) {
                                        ds = nodeDS->data;
                                        newNode = NULL;
                                        name = addrindex_ds_get_name( ds );
-                                       ads = addressbook_create_ds_adapter( ds, atci->objectType, name );
-                                       newNode = addressbook_add_object( node, ADDRESS_OBJECT(ads) );
+                                       ads = addressbook_create_ds_adapter(
+                                                       ds, atci->objectType, name );
+                                       newNode = addressbook_add_object(
+                                                       node, ADDRESS_OBJECT(ads) );
                                        nodeDS = g_list_next( nodeDS );
                                }
                                gtk_ctree_expand( ctree, node );
@@ -3001,7 +3072,7 @@ static GtkCTreeNode *addressbook_add_object(GtkCTreeNode *node,
                }
        }
 
-       gtk_ctree_sort_node(ctree, node);
+       gtk_sctree_sort_node(ctree, node);
 
        return added;
 }
@@ -3037,7 +3108,7 @@ static GtkCTreeNode *addressbook_node_add_group( GtkCTreeNode *node, AddressData
                        atci->treeLeaf, atci->treeExpand );
        gtk_ctree_node_set_row_data_full( ctree, newNode, adapter,
                addressbook_free_treenode );
-       gtk_ctree_sort_node( ctree, node );
+       gtk_sctree_sort_node( ctree, node );
        return newNode;
 }
 
@@ -3050,7 +3121,8 @@ static GtkCTreeNode *addressbook_node_add_group( GtkCTreeNode *node, AddressData
 * Return: Inserted node.
 */
 static GtkCTreeNode *addressbook_node_add_folder(
-               GtkCTreeNode *node, AddressDataSource *ds, ItemFolder *itemFolder, AddressObjectType otype )
+               GtkCTreeNode *node, AddressDataSource *ds,
+               ItemFolder *itemFolder, AddressObjectType otype )
 {
        GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
        GtkCTreeNode *newNode = NULL;
@@ -3098,31 +3170,10 @@ static GtkCTreeNode *addressbook_node_add_folder(
                addressbook_node_add_group( newNode, ds, item );
                listItems = g_list_next( listItems );
        }
-       gtk_ctree_sort_node( ctree, node );
+       gtk_sctree_sort_node( ctree, node );
        return newNode;
 }
 
-static void addressbook_delete_object(AddressObject *obj) {
-       AdapterDSource *ads = NULL;
-       AddressDataSource *ds = NULL;
-       if (!obj) return;
-
-       /* Remove data source. */
-       /* printf( "Delete obj type : %d\n", obj->type ); */
-
-       ads = ADAPTER_DSOURCE(obj);
-       if( ads == NULL ) return;
-       ds = ads->dataSource;
-       if( ds == NULL ) return;
-
-       /* Remove data source */
-       if( addrindex_index_remove_datasource( _addressIndex_, ds ) ) {
-               addrindex_free_datasource( ds );
-       }
-       /* Free up Adapter object */
-       g_free( ads );
-}
-
 void addressbook_export_to_file( void ) {
        if( _addressIndex_ ) {
                /* Save all new address book data */
@@ -3272,34 +3323,36 @@ static void addressbook_new_ldap_cb( gpointer data, guint action, GtkWidget *wid
        }
 }
 
-static void addressbook_ldap_show_message( SyldapServer *svr ) {
+static void addressbook_ldap_show_message( LdapServer *svr ) {
        gchar *name;
+       gchar *desc;
        *addressbook_msgbuf = '\0';
        if( svr ) {
-               name = syldap_get_name( svr );
-               if( svr->busyFlag ) {
-                       sprintf( addressbook_msgbuf, "%s: %s", name, ADDRESSBOOK_LDAP_BUSYMSG );
+               name = ldapsvr_get_name( svr );
+               if( svr->retVal == MGU_SUCCESS ) {
+                       g_snprintf( addressbook_msgbuf,
+                                   sizeof(addressbook_msgbuf), "%s",
+                                   name );
                }
                else {
-                       if( svr->retVal == MGU_SUCCESS ) {
-                               sprintf( addressbook_msgbuf, "%s", name );
-                       }
-                       else {
-                               sprintf( addressbook_msgbuf, "%s: %s", name, mgu_error2string( svr->retVal ) );
-                       }
+                       desc = addressbook_err2string(
+                                       _lutErrorsLDAP_, svr->retVal );
+                       g_snprintf( addressbook_msgbuf,
+                                   sizeof(addressbook_msgbuf),
+                                   "%s: %s", name, desc );
                }
        }
        addressbook_status_show( addressbook_msgbuf );
 }
 
-static void ldapsearch_callback( SyldapServer *sls ) {
+static void addressbook_ldap_show_results( LdapServer *server ) {
        GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
        AddressObject *obj;
        AdapterDSource *ads = NULL;
        AddressDataSource *ds = NULL;
        AddressInterface *iface = NULL;
 
-       if( sls == NULL ) return;
+       if( server == NULL ) return;
        if( ! addrbook.treeSelected ) return;
        if( GTK_CTREE_ROW( addrbook.treeSelected )->level == 1 ) return;
 
@@ -3308,24 +3361,78 @@ static void ldapsearch_callback( SyldapServer *sls ) {
        if( obj->type == ADDR_DATASOURCE ) {
                ads = ADAPTER_DSOURCE(obj);
                if( ads->subType == ADDR_LDAP ) {
-                       SyldapServer *server;
+                       LdapServer *ldapSvr;
 
                        ds = ads->dataSource;
                        if( ds == NULL ) return;
                        iface = ds->interface;
                        if( ! iface->haveLibrary ) return;
                        server = ds->rawDataSource;
-                       if( server == sls ) {
+                       if( ldapSvr == server ) {
                                /* Read from cache */
                                gtk_widget_show_all(addrbook.window);
                                addressbook_set_clist( obj );
-                               addressbook_ldap_show_message( sls );
+                               addressbook_ldap_show_message( server );
                                gtk_widget_show_all(addrbook.window);
                                gtk_entry_set_text( GTK_ENTRY(addrbook.entry), "" );
                        }
                }
        }
 }
+
+static gint _idleID_ = 0;
+static gchar *_tempMessage_ = "Busy searching LDAP...";
+
+/*
+ * LDAP idle function. This function is called during UI idle time while
+ * an LDAP search is in progress.
+ * Enter: data Reference to LDAP server object.
+ */
+static void addressbook_ldap_idle( gpointer data ) {
+}
+
+/*
+ * LDAP search completion function.
+ */
+static void addressbook_ldap_idle_end( void ) {
+       /* Server has completed search - remove from idle list */
+       printf( "addressbook_ldap_idle_end... completed" );
+       if( _idleID_ != 0 ) {
+               gtk_idle_remove( _idleID_ );
+       }
+       _idleID_ = 0;
+}
+
+/*
+ * Perform lookup for LDAP search.
+ * Enter: ads     Adapter for data source.
+ *        sLookup Lookup string.
+ */
+static void addressbook_ldap_lookup( AdapterDSource *ads, gchar *sLookup ) {
+       AddressDataSource *ds = NULL;
+       AddressInterface *iface = NULL;
+       LdapServer *server;
+
+       printf( "addressbook_ldap_lookup/Searching for '%s'\n", sLookup );
+       ds = ads->dataSource;
+       if( ds == NULL ) return;
+       iface = ds->interface;
+       if( ! iface->haveLibrary ) return;
+       server = ds->rawDataSource;
+       if( server ) {
+               printf( "addressbook_ldap_lookup/Starting.../1\n" );
+               if( *sLookup == '\0' || strlen( sLookup ) < 1 ) return;
+
+               /* Setup a query */
+               printf( "addressbook_ldap_lookup/Starting.../2\n" );
+
+               /* Sit back and wait for something to happen */
+               _idleID_ = gtk_idle_add(
+                       ( GtkFunction ) addressbook_ldap_idle, NULL );
+               addrindex_search_ldap_noid( server, sLookup, addressbook_ldap_idle_end  );
+               addressbook_status_show( _tempMessage_ );
+       }
+}
 #endif
 
 /*
@@ -3334,45 +3441,28 @@ static void ldapsearch_callback( SyldapServer *sls ) {
 static void addressbook_lup_clicked( GtkButton *button, gpointer data ) {
        GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
        AddressObject *obj;
-#ifdef USE_LDAP
        AdapterDSource *ads = NULL;
-       AddressDataSource *ds = NULL;
-       AddressInterface *iface = NULL;
-#endif
        gchar *sLookup;
 
-       sLookup = gtk_editable_get_chars( GTK_EDITABLE(addrbook.entry), 0, -1 );
-       g_strchomp( sLookup );
-
        if( ! addrbook.treeSelected ) return;
        if( GTK_CTREE_ROW( addrbook.treeSelected )->level == 1 ) return;
 
        obj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
        if( obj == NULL ) return;
 
-#ifdef USE_LDAP
+       sLookup = gtk_editable_get_chars( GTK_EDITABLE(addrbook.entry), 0, -1 );
+       g_strchomp( sLookup );
+
        if( obj->type == ADDR_DATASOURCE ) {
                ads = ADAPTER_DSOURCE(obj);
+#ifdef USE_LDAP
                if( ads->subType == ADDR_LDAP ) {
-                       SyldapServer *server;
-
-                       ds = ads->dataSource;
-                       if( ds == NULL ) return;
-                       iface = ds->interface;
-                       if( ! iface->haveLibrary ) return;
-                       server = ds->rawDataSource;
-                       if( server ) {
-                               syldap_cancel_read( server );
-                               if( *sLookup == '\0' || strlen( sLookup ) < 1 ) return;
-                               syldap_set_search_value( server, sLookup );
-                               syldap_set_callback( server, ldapsearch_callback );
-                               syldap_read_data_th( server );
-                               addressbook_ldap_show_message( server );
-                       }
+                       addressbook_ldap_lookup( ads, sLookup );
                }
+#endif /* USE_LDAP */
        }
-#endif
 
+       g_free( sLookup );
 }
 
 /* **********************************************************************
@@ -3388,13 +3478,13 @@ static void addressbook_lup_clicked( GtkButton *button, gpointer data ) {
 ItemObjectType addressbook_type2item( AddressObjectType abType ) {
        ItemObjectType ioType;
 
-       ioType = ITEMTYPE_NONE;
        switch( abType ) {
                case ADDR_ITEM_PERSON: ioType = ITEMTYPE_PERSON;     break;
                case ADDR_ITEM_EMAIL:  ioType = ITEMTYPE_EMAIL;      break;
                case ADDR_ITEM_FOLDER: ioType = ITEMTYPE_FOLDER;     break;
                case ADDR_ITEM_GROUP:  ioType = ITEMTYPE_GROUP;      break;
                case ADDR_DATASOURCE:  ioType = ITEMTYPE_DATASOURCE; break;
+               default:               ioType = ITEMTYPE_NONE;       break;
        }
        return ioType;
 }
@@ -4003,6 +4093,30 @@ void addressbook_harvest(
        }
 }
 
+/*
+* Export HTML file.
+*/
+static void addressbook_export_html_cb( void ) {
+       GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
+       AddressObject *obj;
+       AddressDataSource *ds = NULL;
+       AddrBookBase *adbase;
+       AddressCache *cache;
+       GtkCTreeNode *node = NULL;
+
+       if( ! addrbook.treeSelected ) return;
+       node = addrbook.treeSelected;
+       if( GTK_CTREE_ROW(node)->level == 1 ) return;
+       obj = gtk_ctree_node_get_row_data( ctree, node );
+       if( obj == NULL ) return;
+
+       ds = addressbook_find_datasource( node );
+       if( ds == NULL ) return;
+       adbase = ( AddrBookBase * ) ds->rawDataSource;
+       cache = adbase->addressCache;
+       addressbook_exp_html( cache );
+}
+
 /*
 * End of Source.
 */