add new ctree sorting code
[claws.git] / src / addressbook.c
index 6edc109247918d20b876f64477a53cd1ff709661..bb1073088be597c87e3ff804809dab755f64a4b6 100644 (file)
 #include "editgroup.h"
 #include "editaddress.h"
 #include "editbook.h"
-#include "ldif.h"
 #include "importldif.h"
-#include "mutt.h"
 #include "importmutt.h"
+#include "importpine.h"
 
 #ifdef USE_JPILOT
 #include "jpilot.h"
 
 #include "addrselect.h"
 #include "addrclip.h"
-
-/*
-#include "pixmaps/dir-close.xpm"
-#include "pixmaps/dir-open.xpm"
-#include "pixmaps/group.xpm"
-#include "pixmaps/interface.xpm"
-#include "pixmaps/book.xpm"
-#include "pixmaps/address.xpm"
-#include "pixmaps/vcard.xpm"
-#include "pixmaps/jpilot.xpm"
-#include "pixmaps/category.xpm"
-#include "pixmaps/ldap.xpm"
-*/
+#include "addrgather.h"
+#include "adbookbase.h"
+#include "exphtmldlg.h"
 
 typedef enum
 {
@@ -303,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);
@@ -344,6 +328,8 @@ 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 );
@@ -383,22 +369,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 _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},
@@ -447,18 +426,34 @@ 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);
 }
 
+void addressbook_destroy() {
+       /* Free up address stuff */
+       if( _addressSelect_ != NULL ) {
+               addrselect_list_free( _addressSelect_ );
+       }
+       if( _clipBoard_ != NULL ) {
+               addrclip_free( _clipBoard_ );
+       }
+       if( _addressIndex_ != NULL ) {
+              addrindex_free_index( _addressIndex_ );
+       }
+       _addressSelect_ = NULL;
+       _clipBoard_ = NULL;
+       _addressIndex_ = NULL;
+}
+
 void addressbook_set_target_compose(Compose *target)
 {
        addrbook.target_compose = target;
-
        addressbook_button_set_sensitive();
 }
 
@@ -525,12 +520,16 @@ static void addressbook_create(void)
        gint n_entries;
        GList *nodeIf;
 
-       gchar *titles[N_COLS] = {_("Name"), _("E-Mail address"), _("Remarks")};
+       gchar *titles[N_COLS];
        gchar *text;
        gint i;
 
        debug_print("Creating addressbook window...\n");
 
+       titles[COL_NAME]    = _("Name");
+       titles[COL_ADDRESS] = _("E-Mail address");
+       titles[COL_REMARKS] = _("Remarks");
+
        window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
        gtk_window_set_title(GTK_WINDOW(window), _("Address book"));
        gtk_widget_set_usize(window, ADDRESSBOOK_WIDTH, ADDRESSBOOK_HEIGHT);
@@ -542,10 +541,7 @@ static void addressbook_create(void)
                           GTK_SIGNAL_FUNC(addressbook_close), NULL);
        gtk_signal_connect(GTK_OBJECT(window), "key_press_event",
                           GTK_SIGNAL_FUNC(key_pressed), NULL);
-       gtk_signal_connect(GTK_OBJECT(window), "focus_in_event",
-                          GTK_SIGNAL_FUNC(manage_window_focus_in), NULL);
-       gtk_signal_connect(GTK_OBJECT(window), "focus_out_event",
-                          GTK_SIGNAL_FUNC(manage_window_focus_out), NULL);
+       MANAGE_WINDOW_SIGNALS_CONNECT(window);
 
        vbox = gtk_vbox_new(FALSE, 0);
        gtk_container_add(GTK_CONTAINER(window), vbox);
@@ -782,8 +778,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;
@@ -888,7 +883,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;
@@ -1053,6 +1048,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 ) {
@@ -1105,6 +1108,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;
@@ -1120,7 +1124,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);
@@ -1130,7 +1134,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 ) {
@@ -1170,6 +1174,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,
@@ -1236,7 +1243,6 @@ static void addressbook_tree_selected(GtkCTree *ctree, GtkCTreeNode *node,
        addressbook_menuitem_set_sensitive( obj, node );
 
        addressbook_list_select_clear();
-
 }
 
 /*
@@ -1933,6 +1939,10 @@ static gchar *addressbook_edit_datasource( AddressObject *obj, GtkCTreeNode *nod
        if( ! iface->haveLibrary ) return NULL;
 
        /* Read data from data source */
+       if( addrindex_ds_get_modify_flag( ds ) ) {
+               addrindex_ds_read_data( ds );
+       }
+
        if( ! addrindex_ds_get_read_flag( ds ) ) {
                addrindex_ds_read_data( ds );
        }
@@ -2012,7 +2022,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 );
        }
@@ -2200,33 +2210,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;
@@ -2259,7 +2242,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 );
        }
 }
 
@@ -2329,7 +2312,7 @@ 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_sctree_sort_node( ctree, parentNode );
        gtk_ctree_select( ctree, addrbook.opened );
 }
 
@@ -2488,7 +2471,7 @@ static void addressbook_folder_load_person( GtkCTree *clist, ItemFolder *itemFol
                                        FALSE, person->isOpened );
                        gtk_ctree_node_set_row_data(clist, nodePerson, person );
                }
-               gtk_ctree_sort_node(GTK_CTREE(clist), NULL);
+               gtk_sctree_sort_node(GTK_CTREE(clist), NULL);
        }
        /* Free up the list */
        mgu_clear_list( items );
@@ -2516,59 +2499,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.
@@ -2989,7 +2926,7 @@ static GtkCTreeNode *addressbook_add_object(GtkCTreeNode *node,
                }
        }
 
-       gtk_ctree_sort_node(ctree, node);
+       gtk_sctree_sort_node(ctree, node);
 
        return added;
 }
@@ -3025,7 +2962,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;
 }
 
@@ -3086,31 +3023,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 */
@@ -3765,8 +3681,7 @@ gboolean addressbook_add_contact( const gchar *name, const gchar *address, const
 *                     to be loaded.
 * Return: TRUE if data loaded, FALSE if address index not loaded.
 */
-gboolean addressbook_load_completion( gint (*callBackFunc) ( const gchar *, const gchar * ) ) {
-       /* AddressInterface *interface; */
+gboolean addressbook_load_completion( gint (*callBackFunc) ( const gchar *, const gchar *, const gchar * ) ) {
        AddressDataSource *ds;
        GList *nodeIf, *nodeDS;
        GList *listP, *nodeP;
@@ -3785,6 +3700,10 @@ gboolean addressbook_load_completion( gint (*callBackFunc) ( const gchar *, cons
                        ds = nodeDS->data;
 
                        /* Read address book */
+                       if( addrindex_ds_get_modify_flag( ds ) ) {
+                               addrindex_ds_read_data( ds );
+                       }
+
                        if( ! addrindex_ds_get_read_flag( ds ) ) {
                                addrindex_ds_read_data( ds );
                        }
@@ -3813,7 +3732,7 @@ gboolean addressbook_load_completion( gint (*callBackFunc) ( const gchar *, cons
                                                if( sAlias && *sAlias != '\0' ) {
                                                        sFriendly = sAlias;
                                                }
-                                               ( callBackFunc ) ( sFriendly, sAddress );
+                                               ( callBackFunc ) ( sFriendly, sAddress, sName );
                                        }
 
                                        nodeM = g_list_next( nodeM );
@@ -3852,12 +3771,19 @@ static void addressbook_import_ldif_cb() {
                if( adapter->treeNode ) {
                        abf = addressbook_imp_ldif( _addressIndex_ );
                        if( abf ) {
-                               ds = addrindex_index_add_datasource( _addressIndex_, ADDR_IF_BOOK, abf );
-                               ads = addressbook_create_ds_adapter( ds, ADDR_BOOK, NULL );
-                               addressbook_ads_set_name( ads, addrbook_get_name( abf ) );
-                               newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
+                               ds = addrindex_index_add_datasource(
+                                       _addressIndex_, ADDR_IF_BOOK, abf );
+                               ads = addressbook_create_ds_adapter(
+                                       ds, ADDR_BOOK, NULL );
+                               addressbook_ads_set_name(
+                                       ads, addrbook_get_name( abf ) );
+                               newNode = addressbook_add_object(
+                                       adapter->treeNode,
+                                       ADDRESS_OBJECT(ads) );
                                if( newNode ) {
-                                       gtk_ctree_select( GTK_CTREE(addrbook.ctree), newNode );
+                                       gtk_ctree_select(
+                                               GTK_CTREE(addrbook.ctree),
+                                               newNode );
                                        addrbook.treeSelected = newNode;
                                }
 
@@ -3866,7 +3792,6 @@ static void addressbook_import_ldif_cb() {
                        }
                }
        }
-
 }
 
 /*
@@ -3884,12 +3809,57 @@ static void addressbook_import_mutt_cb() {
                if( adapter->treeNode ) {
                        abf = addressbook_imp_mutt( _addressIndex_ );
                        if( abf ) {
-                               ds = addrindex_index_add_datasource( _addressIndex_, ADDR_IF_BOOK, abf );
-                               ads = addressbook_create_ds_adapter( ds, ADDR_BOOK, NULL );
-                               addressbook_ads_set_name( ads, addrbook_get_name( abf ) );
-                               newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
+                               ds = addrindex_index_add_datasource(
+                                       _addressIndex_, ADDR_IF_BOOK, abf );
+                               ads = addressbook_create_ds_adapter(
+                                       ds, ADDR_BOOK, NULL );
+                               addressbook_ads_set_name(
+                                       ads, addrbook_get_name( abf ) );
+                               newNode = addressbook_add_object(
+                                       adapter->treeNode,
+                                       ADDRESS_OBJECT(ads) );
+                               if( newNode ) {
+                                       gtk_ctree_select(
+                                               GTK_CTREE(addrbook.ctree),
+                                               newNode );
+                                       addrbook.treeSelected = newNode;
+                               }
+
+                               /* Notify address completion */
+                               invalidate_address_completion();
+                       }
+               }
+       }
+}
+
+/*
+* Import Pine file.
+*/
+static void addressbook_import_pine_cb() {
+       AddressDataSource *ds = NULL;
+       AdapterDSource *ads = NULL;
+       AddressBookFile *abf = NULL;
+       AdapterInterface *adapter;
+       GtkCTreeNode *newNode;
+
+       adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
+       if( adapter ) {
+               if( adapter->treeNode ) {
+                       abf = addressbook_imp_pine( _addressIndex_ );
+                       if( abf ) {
+                               ds = addrindex_index_add_datasource(
+                                       _addressIndex_, ADDR_IF_BOOK, abf );
+                               ads = addressbook_create_ds_adapter(
+                                       ds, ADDR_BOOK, NULL );
+                               addressbook_ads_set_name(
+                                       ads, addrbook_get_name( abf ) );
+                               newNode = addressbook_add_object(
+                                       adapter->treeNode,
+                                       ADDRESS_OBJECT(ads) );
                                if( newNode ) {
-                                       gtk_ctree_select( GTK_CTREE(addrbook.ctree), newNode );
+                                       gtk_ctree_select(
+                                               GTK_CTREE(addrbook.ctree),
+                                               newNode );
                                        addrbook.treeSelected = newNode;
                                }
 
@@ -3898,7 +3868,67 @@ static void addressbook_import_mutt_cb() {
                        }
                }
        }
+}
+
+/*
+ * Harvest addresses.
+ * Enter: folderItem Folder to import.
+ *        sourceInd  Source indicator: FALSE - Folder, TRUE - Messages.
+ *        msgList    List of message numbers, or NULL to process folder.
+ */
+void addressbook_harvest(
+       FolderItem *folderItem, gboolean sourceInd, GList *msgList )
+{
+       AddressDataSource *ds = NULL;
+       AdapterDSource *ads = NULL;
+       AddressBookFile *abf = NULL;
+       AdapterInterface *adapter;
+       GtkCTreeNode *newNode;
+
+       abf = addrgather_dlg_execute(
+               folderItem, _addressIndex_, sourceInd, msgList );
+       if( abf ) {
+               ds = addrindex_index_add_datasource(
+                       _addressIndex_, ADDR_IF_BOOK, abf );
+
+               adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
+               if( adapter ) {
+                       if( adapter->treeNode ) {
+                               ads = addressbook_create_ds_adapter(
+                                       ds, ADDR_BOOK, addrbook_get_name( abf ) );
+                               newNode = addressbook_add_object(
+                                               adapter->treeNode,
+                                               ADDRESS_OBJECT(ads) );
+                       }
+               }
+
+               /* Notify address completion */
+               invalidate_address_completion();
+       }
+}
+
+/*
+* 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 );
 }
 
 /*