new matcher parser / bounce fix / rrr fixed
[claws.git] / src / addressbook.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2001 Hiroyuki Yamamoto
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2 of the License, or
8  * (at your option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software
17  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18  */
19
20 #ifdef HAVE_CONFIG_H
21 #  include "config.h"
22 #endif
23
24 #include "defs.h"
25
26 #include <glib.h>
27 #include <gdk/gdkkeysyms.h>
28 #include <gtk/gtkwindow.h>
29 #include <gtk/gtksignal.h>
30 #include <gtk/gtkvbox.h>
31 #include <gtk/gtkscrolledwindow.h>
32 #include <gtk/gtkhpaned.h>
33 #include <gtk/gtkhbox.h>
34 #include <gtk/gtklabel.h>
35 #include <gtk/gtkentry.h>
36 #include <gtk/gtkctree.h>
37 #include <gtk/gtkclist.h>
38 #include <gtk/gtktable.h>
39 #include <gtk/gtkhbbox.h>
40 #include <gtk/gtkbutton.h>
41 #include <gtk/gtkmenu.h>
42 #include <gtk/gtkmenuitem.h>
43 #include <gtk/gtkitemfactory.h>
44 #include <string.h>
45 #include <setjmp.h>
46
47 #include "intl.h"
48 #include "main.h"
49 #include "addressbook.h"
50 #include "manage_window.h"
51 #include "prefs_common.h"
52 #include "alertpanel.h"
53 #include "inputdialog.h"
54 #include "menu.h"
55 #include "xml.h"
56 #include "prefs.h"
57 #include "procmime.h"
58 #include "utils.h"
59 #include "gtkutils.h"
60 #include "codeconv.h"
61 #include "about.h"
62 #include "addr_compl.h"
63
64 #include "mgutils.h"
65 #include "addressitem.h"
66 #include "addritem.h"
67 #include "addrcache.h"
68 #include "addrbook.h"
69 #include "addrindex.h"
70 #include "addressadd.h"
71 #include "vcard.h"
72 #include "editvcard.h"
73 #include "editgroup.h"
74 #include "editaddress.h"
75 #include "editbook.h"
76 #include "ldif.h"
77 #include "importldif.h"
78 #include "mutt.h"
79 #include "importmutt.h"
80
81 #ifdef USE_JPILOT
82 #include "jpilot.h"
83 #include "editjpilot.h"
84 #endif
85
86 #ifdef USE_LDAP
87 #include <pthread.h>
88 #include "syldap.h"
89 #include "editldap.h"
90
91 #define ADDRESSBOOK_LDAP_BUSYMSG "Busy"
92 #endif
93
94 #include "pixmaps/dir-close.xpm"
95 #include "pixmaps/dir-open.xpm"
96 #include "pixmaps/group.xpm"
97 #include "pixmaps/interface.xpm"
98 #include "pixmaps/book.xpm"
99 #include "pixmaps/address.xpm"
100 #include "pixmaps/vcard.xpm"
101 #include "pixmaps/jpilot.xpm"
102 #include "pixmaps/category.xpm"
103 #include "pixmaps/ldap.xpm"
104
105 typedef enum
106 {
107         COL_NAME        = 0,
108         COL_ADDRESS     = 1,
109         COL_REMARKS     = 2
110 } AddressBookColumnPos;
111
112 #define N_COLS  3
113 #define COL_NAME_WIDTH          164
114 #define COL_ADDRESS_WIDTH       156
115
116 #define COL_FOLDER_WIDTH        170
117 #define ADDRESSBOOK_WIDTH       640
118 #define ADDRESSBOOK_HEIGHT      360
119
120 #define ADDRESSBOOK_MSGBUF_SIZE 2048
121
122 static GdkPixmap *folderxpm;
123 static GdkBitmap *folderxpmmask;
124 static GdkPixmap *folderopenxpm;
125 static GdkBitmap *folderopenxpmmask;
126 static GdkPixmap *groupxpm;
127 static GdkBitmap *groupxpmmask;
128 static GdkPixmap *interfacexpm;
129 static GdkBitmap *interfacexpmmask;
130 static GdkPixmap *bookxpm;
131 static GdkBitmap *bookxpmmask;
132 static GdkPixmap *addressxpm;
133 static GdkBitmap *addressxpmmask;
134 static GdkPixmap *vcardxpm;
135 static GdkBitmap *vcardxpmmask;
136 static GdkPixmap *jpilotxpm;
137 static GdkBitmap *jpilotxpmmask;
138 static GdkPixmap *categoryxpm;
139 static GdkBitmap *categoryxpmmask;
140 static GdkPixmap *ldapxpm;
141 static GdkBitmap *ldapxpmmask;
142
143 /* Message buffer */
144 static gchar addressbook_msgbuf[ ADDRESSBOOK_MSGBUF_SIZE ];
145
146 /* Address list selection */
147 static GList *_addressListSelection_ = NULL;
148
149 /* Address index file and interfaces */
150 static AddressIndex *_addressIndex_ = NULL;
151 static GList *_addressInterfaceList_ = NULL;
152 static GList *_addressIFaceSelection_ = NULL;
153 #define ADDRESSBOOK_IFACE_SELECTION "1/y,3/y,4/y,2/n"
154
155 static AddressBook_win addrbook;
156
157 static GHashTable *_addressBookTypeHash_ = NULL;
158 static GList *_addressBookTypeList_ = NULL;
159
160 static void addressbook_create                  (void);
161 static gint addressbook_close                   (void);
162 static void addressbook_button_set_sensitive    (void);
163
164 /* callback functions */
165 static void addressbook_del_clicked             (GtkButton      *button,
166                                                  gpointer        data);
167 static void addressbook_reg_clicked             (GtkButton      *button,
168                                                  gpointer        data);
169 static void addressbook_to_clicked              (GtkButton      *button,
170                                                  gpointer        data);
171 static void addressbook_lup_clicked             (GtkButton      *button,
172                                                  gpointer       data);
173
174 static void addressbook_tree_selected           (GtkCTree       *ctree,
175                                                  GtkCTreeNode   *node,
176                                                  gint            column,
177                                                  gpointer        data);
178 static void addressbook_list_selected           (GtkCList       *clist,
179                                                  gint            row,
180                                                  gint            column,
181                                                  GdkEvent       *event,
182                                                  gpointer        data);
183 static void addressbook_list_row_selected       (GtkCTree       *clist,
184                                                  GtkCTreeNode   *node,
185                                                  gint            column,
186                                                  gpointer        data);
187 static void addressbook_list_row_unselected     (GtkCTree       *clist,
188                                                  GtkCTreeNode   *node,
189                                                  gint            column,
190                                                  gpointer        data);
191 static void addressbook_person_expand_node      (GtkCTree       *ctree,
192                                                  GList          *node,
193                                                  gpointer       *data );
194 static void addressbook_person_collapse_node    (GtkCTree       *ctree,
195                                                  GList          *node,
196                                                  gpointer       *data );
197 static void addressbook_entry_gotfocus          (GtkWidget      *widget);
198
199 #if 0
200 static void addressbook_entry_changed           (GtkWidget      *widget);
201 #endif
202
203 static void addressbook_list_button_pressed     (GtkWidget      *widget,
204                                                  GdkEventButton *event,
205                                                  gpointer        data);
206 static void addressbook_list_button_released    (GtkWidget      *widget,
207                                                  GdkEventButton *event,
208                                                  gpointer        data);
209 static void addressbook_tree_button_pressed     (GtkWidget      *ctree,
210                                                  GdkEventButton *event,
211                                                  gpointer        data);
212 static void addressbook_tree_button_released    (GtkWidget      *ctree,
213                                                  GdkEventButton *event,
214                                                  gpointer        data);
215 static void addressbook_popup_close             (GtkMenuShell   *menu_shell,
216                                                  gpointer        data);
217
218 static void addressbook_new_folder_cb           (gpointer        data,
219                                                  guint           action,
220                                                  GtkWidget      *widget);
221 static void addressbook_new_group_cb            (gpointer        data,
222                                                  guint           action,
223                                                  GtkWidget      *widget);
224 static void addressbook_treenode_edit_cb        (gpointer        data,
225                                                  guint           action,
226                                                  GtkWidget      *widget);
227 static void addressbook_treenode_delete_cb      (gpointer        data,
228                                                  guint           action,
229                                                  GtkWidget      *widget);
230
231 static void addressbook_change_node_name        (GtkCTreeNode   *node,
232                                                  const gchar    *name);
233
234 static void addressbook_new_address_cb          (gpointer        data,
235                                                  guint           action,
236                                                  GtkWidget      *widget);
237 static void addressbook_edit_address_cb         (gpointer        data,
238                                                  guint           action,
239                                                  GtkWidget      *widget);
240 static void addressbook_delete_address_cb       (gpointer        data,
241                                                  guint           action,
242                                                  GtkWidget      *widget);
243
244 static void close_cb                            (gpointer        data,
245                                                  guint           action,
246                                                  GtkWidget      *widget);
247 static void addressbook_file_save_cb            (gpointer        data,
248                                                  guint           action,
249                                                  GtkWidget      *widget);
250
251 /* Data source edit stuff */
252 static void addressbook_new_book_cb             (gpointer        data,
253                                                  guint           action,
254                                                  GtkWidget      *widget);
255 static void addressbook_new_vcard_cb            (gpointer        data,
256                                                  guint           action,
257                                                  GtkWidget      *widget);
258
259 #ifdef USE_JPILOT
260 static void addressbook_new_jpilot_cb           (gpointer        data,
261                                                  guint           action,
262                                                  GtkWidget      *widget);
263 #endif
264
265 #ifdef USE_LDAP
266 static void addressbook_new_ldap_cb             (gpointer        data,
267                                                  guint           action,
268                                                  GtkWidget      *widget);
269 #endif
270
271 static void addressbook_set_clist               (AddressObject  *obj);
272
273 static void addressbook_load_tree               (void);
274 void addressbook_read_file                      (void);
275
276 static GtkCTreeNode *addressbook_add_object     (GtkCTreeNode   *node,
277                                                  AddressObject  *obj);
278 static AddressDataSource *addressbook_find_datasource
279                                                 (GtkCTreeNode   *node );
280
281 static AddressBookFile *addressbook_get_book_file();
282
283 static GtkCTreeNode *addressbook_node_add_folder
284                                                 (GtkCTreeNode   *node,
285                                                 AddressDataSource *ds,
286                                                 ItemFolder      *itemFolder,
287                                                 AddressObjectType otype);
288 static GtkCTreeNode *addressbook_node_add_group (GtkCTreeNode   *node,
289                                                 AddressDataSource *ds,
290                                                 ItemGroup       *itemGroup);
291 /* static GtkCTreeNode *addressbook_node_add_category */
292 /*                                              (GtkCTreeNode   *node, */
293 /*                                               AddressDataSource *ds, */
294 /*                                               ItemFolder     *itemFolder); */
295 static void addressbook_tree_remove_children    (GtkCTree       *ctree,
296                                                 GtkCTreeNode    *parent);
297 static void addressbook_move_nodes_up           (GtkCTree       *ctree,
298                                                 GtkCTreeNode    *node);
299 static GtkCTreeNode *addressbook_find_group_node (GtkCTreeNode  *parent,
300                                                 ItemGroup       *group);
301
302 static void addressbook_delete_object           (AddressObject  *obj);
303
304 static void key_pressed                         (GtkWidget      *widget,
305                                                  GdkEventKey    *event,
306                                                  gpointer        data);
307 static gint addressbook_list_compare_func       (GtkCList       *clist,
308                                                  gconstpointer   ptr1,
309                                                  gconstpointer   ptr2);
310 /* static gint addressbook_obj_name_compare     (gconstpointer   a, */
311 /*                                               gconstpointer   b); */
312
313 static void addressbook_book_show_message       (AddressBookFile *book);
314 static void addressbook_vcard_show_message      (VCardFile *vcf);
315 #ifdef USE_JPILOT
316 static void addressbook_jpilot_show_message     (JPilotFile *jpf);
317 #endif
318 #ifdef USE_LDAP
319 static void addressbook_ldap_show_message       (SyldapServer *server);
320 #endif
321
322 /* LUT's and IF stuff */
323 static void addressbook_free_adapter            (GtkCTreeNode   *node);
324 static void addressbook_free_child_adapters     (GtkCTreeNode   *node);
325 AddressTypeControlItem *addrbookctl_lookup      (gint            ot);
326 AddressTypeControlItem *addrbookctl_lookup_iface(AddressIfType   ifType);
327
328 void addrbookctl_build_map                      (GtkWidget      *window);
329 void addrbookctl_build_iflist                   (void);
330 AdapterInterface *addrbookctl_find_interface    (AddressIfType   ifType);
331 void addrbookctl_build_ifselect                 (void);
332
333 static void addrbookctl_free_interface          (AdapterInterface *adapter);
334 static void addrbookctl_free_datasource         (AdapterDSource   *adapter);
335 static void addrbookctl_free_folder             (AdapterFolder    *adapter);
336 static void addrbookctl_free_group              (AdapterGroup     *adapter);
337
338 static void addressbook_list_select_clear       (void);
339 static void addressbook_list_select_add         (AddressObject *obj);
340 static void addressbook_list_select_remove      (AddressObject *obj);
341
342 static void addressbook_import_ldif_cb          ();
343 static void addressbook_import_mutt_cb          ();
344
345 static GtkItemFactoryEntry addressbook_entries[] =
346 {
347         {N_("/_File"),                  NULL,           NULL, 0, "<Branch>"},
348         {N_("/_File/New _Book"),        "<alt>B",       addressbook_new_book_cb,        0, NULL},
349         {N_("/_File/New _V-Card"),      "<alt>D",       addressbook_new_vcard_cb,       0, NULL},
350 #ifdef USE_JPILOT
351         {N_("/_File/New _J-Pilot"),     "<alt>J",       addressbook_new_jpilot_cb,      0, NULL},
352 #endif
353 #ifdef USE_LDAP
354         {N_("/_File/New _Server"),      "<alt>S",       addressbook_new_ldap_cb,        0, NULL},
355 #endif
356         {N_("/_File/---"),              NULL,           NULL, 0, "<Separator>"},
357         {N_("/_File/_Edit"),            NULL,           addressbook_treenode_edit_cb,   0, NULL},
358         {N_("/_File/_Delete"),          NULL,           addressbook_treenode_delete_cb, 0, NULL},
359         {N_("/_File/---"),              NULL,           NULL, 0, "<Separator>"},
360         {N_("/_File/_Save"),            "<alt>S",       addressbook_file_save_cb,       0, NULL},
361         {N_("/_File/_Close"),           "<alt>W",       close_cb, 0, NULL},
362         {N_("/_Address"),               NULL,           NULL, 0, "<Branch>"},
363         {N_("/_Address/New _Address"),  "<alt>N",       addressbook_new_address_cb,     0, NULL},
364         {N_("/_Address/New _Group"),    "<alt>G",       addressbook_new_group_cb,       0, NULL},
365         {N_("/_Address/New _Folder"),   "<alt>R",       addressbook_new_folder_cb,      0, NULL},
366         {N_("/_Address/---"),           NULL,           NULL, 0, "<Separator>"},
367         {N_("/_Address/_Edit"),         "<alt>Return",  addressbook_edit_address_cb,    0, NULL},
368         {N_("/_Address/_Delete"),       NULL,           addressbook_delete_address_cb,  0, NULL},
369         {N_("/_Tools/---"),             NULL,           NULL, 0, "<Separator>"},
370         {N_("/_Tools/Import _LDIF"),    NULL,           addressbook_import_ldif_cb,     0, NULL},
371         {N_("/_Tools/Import M_utt"),    NULL,           addressbook_import_mutt_cb,     0, NULL},
372         {N_("/_Help"),                  NULL,           NULL, 0, "<LastBranch>"},
373         {N_("/_Help/_About"),           NULL,           about_show, 0, NULL}
374 };
375
376 /* New options to be added. */
377 /*
378         {N_("/_Edit"),                  NULL,           NULL, 0, "<Branch>"},
379         {N_("/_Edit/C_ut"),             "<ctl>X",       NULL,                           0, NULL},
380         {N_("/_Edit/_Copy"),            "<ctl>C",       NULL,                           0, NULL},
381         {N_("/_Edit/_Paste"),           "<ctl>V",       NULL,                           0, NULL},
382         {N_("/_Tools"),                 NULL,           NULL, 0, "<Branch>"},
383         {N_("/_Tools/Import _Mozilla"), NULL,           NULL,                           0, NULL},
384         {N_("/_Tools/Import _V-Card"),  NULL,           NULL,                           0, NULL},
385         {N_("/_Tools/---"),             NULL,           NULL, 0, "<Separator>"},
386         {N_("/_Tools/Export _LDIF"),    NULL,           NULL,                           0, NULL},
387         {N_("/_Tools/Export V-_Card"),  NULL,           NULL,                           0, NULL},
388 */
389
390 static GtkItemFactoryEntry addressbook_tree_popup_entries[] =
391 {
392         {N_("/New _Address"),   NULL, addressbook_new_address_cb, 0, NULL},
393         {N_("/New _Group"),     NULL, addressbook_new_group_cb,   0, NULL},
394         {N_("/New _Folder"),    NULL, addressbook_new_folder_cb,  0, NULL},
395         {N_("/---"),            NULL, NULL, 0, "<Separator>"},
396         {N_("/_Edit"),          NULL, addressbook_treenode_edit_cb,   0, NULL},
397         {N_("/_Delete"),        NULL, addressbook_treenode_delete_cb, 0, NULL}
398 };
399
400 static GtkItemFactoryEntry addressbook_list_popup_entries[] =
401 {
402         {N_("/New _Address"),   NULL, addressbook_new_address_cb,  0, NULL},
403         {N_("/New _Group"),     NULL, addressbook_new_group_cb,    0, NULL},
404         {N_("/New _Folder"),    NULL, addressbook_new_folder_cb,   0, NULL},
405         {N_("/---"),            NULL, NULL, 0, "<Separator>"},
406         {N_("/_Edit"),          NULL, addressbook_edit_address_cb,   0, NULL},
407         {N_("/_Delete"),        NULL, addressbook_delete_address_cb, 0, NULL}
408 };
409
410 void addressbook_open(Compose *target)
411 {
412         if (!addrbook.window) {
413                 addressbook_read_file();
414                 addressbook_create();
415                 addressbook_load_tree();
416                 gtk_ctree_select(GTK_CTREE(addrbook.ctree),
417                                  GTK_CTREE_NODE(GTK_CLIST(addrbook.ctree)->row_list));
418         } else
419                 gtk_widget_hide(addrbook.window);
420
421         gtk_widget_show_all(addrbook.window);
422
423         addressbook_set_target_compose(target);
424 }
425
426 void addressbook_set_target_compose(Compose *target)
427 {
428         addrbook.target_compose = target;
429
430         addressbook_button_set_sensitive();
431 }
432
433 Compose *addressbook_get_target_compose(void)
434 {
435         return addrbook.target_compose;
436 }
437
438 void addressbook_refresh(void)
439 {
440         if (addrbook.window) {
441                 if (addrbook.treeSelected) {
442                         gtk_ctree_select(GTK_CTREE(addrbook.ctree),
443                                          addrbook.treeSelected);
444                 }
445         }
446         addressbook_export_to_file();
447 }
448
449 /*
450 * Create the address book widgets. The address book contains two CTree widgets: the
451 * address index tree on the left and the address list on the right.
452 *
453 * The address index tree displays a hierarchy of interfaces and groups. Each node in
454 * this tree is linked to an address Adapter. Adapters have been created for interfaces,
455 * data sources and folder objects.
456 *
457 * The address list displays group, person and email objects. These items are linked
458 * directly to ItemGroup, ItemPerson and ItemEMail objects inside the address book data
459 * sources.
460 *
461 * In the tradition of MVC architecture, the data stores have been separated from the
462 * GUI components. The addrindex.c file provides the interface to all data stores.
463 */
464 static void addressbook_create(void)
465 {
466         GtkWidget *window;
467         GtkWidget *vbox;
468         GtkWidget *menubar;
469         GtkWidget *vbox2;
470         GtkWidget *ctree_swin;
471         GtkWidget *ctree;
472         GtkWidget *clist_vbox;
473         GtkWidget *clist_swin;
474         GtkWidget *clist;
475         GtkWidget *paned;
476         GtkWidget *hbox;
477         GtkWidget *label;
478         GtkWidget *entry;
479         GtkWidget *statusbar;
480         GtkWidget *hbbox;
481         GtkWidget *hsbox;
482         GtkWidget *del_btn;
483         GtkWidget *reg_btn;
484         GtkWidget *lup_btn;
485         GtkWidget *to_btn;
486         GtkWidget *cc_btn;
487         GtkWidget *bcc_btn;
488         GtkWidget *tree_popup;
489         GtkWidget *list_popup;
490         GtkItemFactory *tree_factory;
491         GtkItemFactory *list_factory;
492         GtkItemFactory *menu_factory;
493         gint n_entries;
494         GList *nodeIf;
495
496         gchar *titles[N_COLS] = {_("Name"), _("E-Mail address"), _("Remarks")};
497         gchar *text;
498         gint i;
499
500         debug_print("Creating addressbook window...\n");
501
502         window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
503         gtk_window_set_title(GTK_WINDOW(window), _("Address book"));
504         gtk_widget_set_usize(window, ADDRESSBOOK_WIDTH, ADDRESSBOOK_HEIGHT);
505         gtk_window_set_policy(GTK_WINDOW(window), TRUE, TRUE, TRUE);
506         gtk_widget_realize(window);
507
508         gtk_signal_connect(GTK_OBJECT(window), "delete_event",
509                            GTK_SIGNAL_FUNC(addressbook_close), NULL);
510         gtk_signal_connect(GTK_OBJECT(window), "key_press_event",
511                            GTK_SIGNAL_FUNC(key_pressed), NULL);
512         gtk_signal_connect(GTK_OBJECT(window), "focus_in_event",
513                            GTK_SIGNAL_FUNC(manage_window_focus_in), NULL);
514         gtk_signal_connect(GTK_OBJECT(window), "focus_out_event",
515                            GTK_SIGNAL_FUNC(manage_window_focus_out), NULL);
516
517         vbox = gtk_vbox_new(FALSE, 0);
518         gtk_container_add(GTK_CONTAINER(window), vbox);
519
520         n_entries = sizeof(addressbook_entries) /
521                 sizeof(addressbook_entries[0]);
522         menubar = menubar_create(window, addressbook_entries, n_entries,
523                                  "<AddressBook>", NULL);
524         gtk_box_pack_start(GTK_BOX(vbox), menubar, FALSE, TRUE, 0);
525         menu_factory = gtk_item_factory_from_widget(menubar);
526
527         vbox2 = gtk_vbox_new(FALSE, 4);
528         gtk_container_set_border_width(GTK_CONTAINER(vbox2), BORDER_WIDTH);
529         gtk_box_pack_start(GTK_BOX(vbox), vbox2, TRUE, TRUE, 0);
530
531         ctree_swin = gtk_scrolled_window_new(NULL, NULL);
532         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(ctree_swin),
533                                        GTK_POLICY_AUTOMATIC,
534                                        GTK_POLICY_ALWAYS);
535         gtk_widget_set_usize(ctree_swin, COL_FOLDER_WIDTH + 40, -1);
536
537         /* Address index */
538         ctree = gtk_ctree_new(1, 0);
539         gtk_container_add(GTK_CONTAINER(ctree_swin), ctree);
540         gtk_clist_set_selection_mode(GTK_CLIST(ctree), GTK_SELECTION_BROWSE);
541         gtk_clist_set_column_width(GTK_CLIST(ctree), 0, COL_FOLDER_WIDTH);
542         gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_DOTTED);
543         gtk_ctree_set_expander_style(GTK_CTREE(ctree),
544                                      GTK_CTREE_EXPANDER_SQUARE);
545         gtk_ctree_set_indent(GTK_CTREE(ctree), CTREE_INDENT);
546         gtk_clist_set_compare_func(GTK_CLIST(ctree),
547                                    addressbook_list_compare_func);
548
549         gtk_signal_connect(GTK_OBJECT(ctree), "tree_select_row",
550                            GTK_SIGNAL_FUNC(addressbook_tree_selected), NULL);
551         gtk_signal_connect(GTK_OBJECT(ctree), "button_press_event",
552                            GTK_SIGNAL_FUNC(addressbook_tree_button_pressed),
553                            NULL);
554         gtk_signal_connect(GTK_OBJECT(ctree), "button_release_event",
555                            GTK_SIGNAL_FUNC(addressbook_tree_button_released),
556                            NULL);
557
558         clist_vbox = gtk_vbox_new(FALSE, 4);
559
560         clist_swin = gtk_scrolled_window_new(NULL, NULL);
561         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(clist_swin),
562                                        GTK_POLICY_AUTOMATIC,
563                                        GTK_POLICY_ALWAYS);
564         gtk_box_pack_start(GTK_BOX(clist_vbox), clist_swin, TRUE, TRUE, 0);
565
566         /* Address list */
567         clist = gtk_ctree_new_with_titles(N_COLS, 0, titles);
568         gtk_container_add(GTK_CONTAINER(clist_swin), clist);
569         gtk_clist_set_selection_mode(GTK_CLIST(clist), GTK_SELECTION_EXTENDED);
570         gtk_ctree_set_line_style(GTK_CTREE(clist), GTK_CTREE_LINES_NONE);
571         gtk_ctree_set_expander_style(GTK_CTREE(clist), GTK_CTREE_EXPANDER_SQUARE);
572         gtk_ctree_set_indent(GTK_CTREE(clist), CTREE_INDENT);
573         gtk_clist_set_column_width(GTK_CLIST(clist), COL_NAME,
574                                    COL_NAME_WIDTH);
575         gtk_clist_set_column_width(GTK_CLIST(clist), COL_ADDRESS,
576                                    COL_ADDRESS_WIDTH);
577         gtk_clist_set_compare_func(GTK_CLIST(clist),
578                                    addressbook_list_compare_func);
579
580         for (i = 0; i < N_COLS; i++)
581                 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist)->column[i].button,
582                                        GTK_CAN_FOCUS);
583
584         gtk_signal_connect(GTK_OBJECT(clist), "tree_select_row",
585                            GTK_SIGNAL_FUNC(addressbook_list_row_selected), NULL);
586         gtk_signal_connect(GTK_OBJECT(clist), "tree_unselect_row",
587                            GTK_SIGNAL_FUNC(addressbook_list_row_unselected), NULL);
588         gtk_signal_connect(GTK_OBJECT(clist), "button_press_event",
589                            GTK_SIGNAL_FUNC(addressbook_list_button_pressed),
590                            NULL);
591         gtk_signal_connect(GTK_OBJECT(clist), "button_release_event",
592                            GTK_SIGNAL_FUNC(addressbook_list_button_released),
593                            NULL);
594         gtk_signal_connect(GTK_OBJECT(clist), "select_row",
595                            GTK_SIGNAL_FUNC(addressbook_list_selected), NULL);
596         gtk_signal_connect(GTK_OBJECT(clist), "tree_expand",
597                            GTK_SIGNAL_FUNC(addressbook_person_expand_node), NULL );
598         gtk_signal_connect(GTK_OBJECT(clist), "tree_collapse",
599                            GTK_SIGNAL_FUNC(addressbook_person_collapse_node), NULL );
600
601         hbox = gtk_hbox_new(FALSE, 4);
602         gtk_box_pack_start(GTK_BOX(clist_vbox), hbox, FALSE, FALSE, 0);
603
604         label = gtk_label_new(_("Name:"));
605         gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
606
607         entry = gtk_entry_new();
608         gtk_box_pack_start(GTK_BOX(hbox), entry, TRUE, TRUE, 0);
609
610         address_completion_register_entry(GTK_ENTRY(entry));
611         gtk_signal_connect(GTK_OBJECT(entry), "focus_in_event",
612                            GTK_SIGNAL_FUNC(addressbook_entry_gotfocus), NULL);
613
614 #if 0
615         gtk_signal_connect(GTK_OBJECT(entry), "changed",
616                            GTK_SIGNAL_FUNC(addressbook_entry_changed), NULL);
617 #endif
618
619         paned = gtk_hpaned_new();
620         gtk_box_pack_start(GTK_BOX(vbox2), paned, TRUE, TRUE, 0);
621         gtk_paned_add1(GTK_PANED(paned), ctree_swin);
622         gtk_paned_add2(GTK_PANED(paned), clist_vbox);
623
624         /* Status bar */
625         hsbox = gtk_hbox_new(FALSE, 0);
626         gtk_box_pack_end(GTK_BOX(vbox), hsbox, FALSE, FALSE, BORDER_WIDTH);
627         statusbar = gtk_statusbar_new();
628         gtk_box_pack_start(GTK_BOX(hsbox), statusbar, TRUE, TRUE, BORDER_WIDTH);
629
630         /* Button panel */
631         hbbox = gtk_hbutton_box_new();
632         gtk_button_box_set_layout(GTK_BUTTON_BOX(hbbox), GTK_BUTTONBOX_END);
633         gtk_button_box_set_spacing(GTK_BUTTON_BOX(hbbox), 2);
634         gtk_box_pack_end(GTK_BOX(vbox), hbbox, FALSE, FALSE, 0);
635
636         del_btn = gtk_button_new_with_label(_("Delete"));
637         GTK_WIDGET_SET_FLAGS(del_btn, GTK_CAN_DEFAULT);
638         gtk_box_pack_start(GTK_BOX(hbbox), del_btn, TRUE, TRUE, 0);
639         reg_btn = gtk_button_new_with_label(_("Add"));
640         GTK_WIDGET_SET_FLAGS(reg_btn, GTK_CAN_DEFAULT);
641         gtk_box_pack_start(GTK_BOX(hbbox), reg_btn, TRUE, TRUE, 0);
642         lup_btn = gtk_button_new_with_label(_("Lookup"));
643         GTK_WIDGET_SET_FLAGS(lup_btn, GTK_CAN_DEFAULT);
644         gtk_box_pack_start(GTK_BOX(hbbox), lup_btn, TRUE, TRUE, 0);
645
646         gtk_signal_connect(GTK_OBJECT(del_btn), "clicked",
647                            GTK_SIGNAL_FUNC(addressbook_del_clicked), NULL);
648         gtk_signal_connect(GTK_OBJECT(reg_btn), "clicked",
649                            GTK_SIGNAL_FUNC(addressbook_reg_clicked), NULL);
650         gtk_signal_connect(GTK_OBJECT(lup_btn), "clicked",
651                            GTK_SIGNAL_FUNC(addressbook_lup_clicked), NULL);
652
653         to_btn = gtk_button_new_with_label
654                 (prefs_common.trans_hdr ? _("To:") : "To:");
655         GTK_WIDGET_SET_FLAGS(to_btn, GTK_CAN_DEFAULT);
656         gtk_box_pack_start(GTK_BOX(hbbox), to_btn, TRUE, TRUE, 0);
657         cc_btn = gtk_button_new_with_label
658                 (prefs_common.trans_hdr ? _("Cc:") : "Cc:");
659         GTK_WIDGET_SET_FLAGS(cc_btn, GTK_CAN_DEFAULT);
660         gtk_box_pack_start(GTK_BOX(hbbox), cc_btn, TRUE, TRUE, 0);
661         bcc_btn = gtk_button_new_with_label
662                 (prefs_common.trans_hdr ? _("Bcc:") : "Bcc:");
663         GTK_WIDGET_SET_FLAGS(bcc_btn, GTK_CAN_DEFAULT);
664         gtk_box_pack_start(GTK_BOX(hbbox), bcc_btn, TRUE, TRUE, 0);
665
666         gtk_signal_connect(GTK_OBJECT(to_btn), "clicked",
667                            GTK_SIGNAL_FUNC(addressbook_to_clicked),
668                            GINT_TO_POINTER(COMPOSE_TO));
669         gtk_signal_connect(GTK_OBJECT(cc_btn), "clicked",
670                            GTK_SIGNAL_FUNC(addressbook_to_clicked),
671                            GINT_TO_POINTER(COMPOSE_CC));
672         gtk_signal_connect(GTK_OBJECT(bcc_btn), "clicked",
673                            GTK_SIGNAL_FUNC(addressbook_to_clicked),
674                            GINT_TO_POINTER(COMPOSE_BCC));
675
676         /* Build icons for interface */
677         PIXMAP_CREATE( window, interfacexpm, interfacexpmmask, interface_xpm );
678
679         /* Build control tables */
680         addrbookctl_build_map(window);
681         addrbookctl_build_iflist();
682         addrbookctl_build_ifselect();
683
684         /* Add each interface into the tree as a root level folder */
685         nodeIf = _addressInterfaceList_;
686         while( nodeIf ) {
687                 AdapterInterface *adapter = nodeIf->data;
688                 AddressInterface *iface = adapter->interface;
689                 nodeIf = g_list_next(nodeIf);
690
691                 if(iface->useInterface) {
692                         AddressTypeControlItem *atci = adapter->atci;
693                         text = atci->displayName;
694                         adapter->treeNode =
695                                 gtk_ctree_insert_node( GTK_CTREE(ctree),
696                                         NULL, NULL, &text, FOLDER_SPACING,
697                                         interfacexpm, interfacexpmmask,
698                                         interfacexpm, interfacexpmmask,
699                                         FALSE, FALSE );
700                         menu_set_sensitive( menu_factory, atci->menuCommand, adapter->haveLibrary );
701                         gtk_ctree_node_set_row_data( GTK_CTREE(ctree), adapter->treeNode, adapter );
702                 }
703         }
704
705         /* Popup menu */
706         n_entries = sizeof(addressbook_tree_popup_entries) /
707                 sizeof(addressbook_tree_popup_entries[0]);
708         tree_popup = menu_create_items(addressbook_tree_popup_entries,
709                                        n_entries,
710                                        "<AddressBookTree>", &tree_factory,
711                                        NULL);
712         gtk_signal_connect(GTK_OBJECT(tree_popup), "selection_done",
713                            GTK_SIGNAL_FUNC(addressbook_popup_close), NULL);
714         n_entries = sizeof(addressbook_list_popup_entries) /
715                 sizeof(addressbook_list_popup_entries[0]);
716         list_popup = menu_create_items(addressbook_list_popup_entries,
717                                        n_entries,
718                                        "<AddressBookList>", &list_factory,
719                                        NULL);
720
721         addrbook.window  = window;
722         addrbook.menubar = menubar;
723         addrbook.ctree   = ctree;
724         addrbook.clist   = clist;
725         addrbook.entry   = entry;
726         addrbook.statusbar = statusbar;
727         addrbook.status_cid = gtk_statusbar_get_context_id( GTK_STATUSBAR(statusbar), "Addressbook Window" );
728
729         addrbook.del_btn = del_btn;
730         addrbook.reg_btn = reg_btn;
731         addrbook.lup_btn = lup_btn;
732         addrbook.to_btn  = to_btn;
733         addrbook.cc_btn  = cc_btn;
734         addrbook.bcc_btn = bcc_btn;
735
736         addrbook.tree_popup   = tree_popup;
737         addrbook.list_popup   = list_popup;
738         addrbook.tree_factory = tree_factory;
739         addrbook.list_factory = list_factory;
740         addrbook.menu_factory = menu_factory;
741
742         addrbook.listSelected = NULL;
743         address_completion_start(window);
744         gtk_widget_show_all(window);
745
746 }
747
748 static gint addressbook_close(void)
749 {
750         gtk_widget_hide(addrbook.window);
751         addressbook_export_to_file();
752         return TRUE;
753 }
754
755 static void addressbook_status_show( gchar *msg ) {
756         if( addrbook.statusbar != NULL ) {
757                 gtk_statusbar_pop( GTK_STATUSBAR(addrbook.statusbar), addrbook.status_cid );
758                 if( msg ) {
759                         gtk_statusbar_push( GTK_STATUSBAR(addrbook.statusbar), addrbook.status_cid, msg );
760                 }
761         }
762 }
763
764 static void addressbook_ds_show_message( AddressDataSource *ds ) {
765         gint retVal;
766         gchar *name;
767         *addressbook_msgbuf = '\0';
768         if( ds ) {
769                 name = addrindex_ds_get_name( ds );
770                 retVal = addrindex_ds_get_status_code( ds );
771                 if( retVal == MGU_SUCCESS ) {
772                         if( ds ) {
773                                 sprintf( addressbook_msgbuf, "%s", name );
774                         }
775                 }
776                 else {
777                         if( ds == NULL ) {
778                                 sprintf( addressbook_msgbuf, "%s", mgu_error2string( retVal ) );
779                         }
780                         else {
781                                 sprintf( addressbook_msgbuf, "%s: %s", name, mgu_error2string( retVal ) );
782                         }
783                 }
784         }
785         addressbook_status_show( addressbook_msgbuf );
786 }
787
788 static void addressbook_button_set_sensitive(void)
789 {
790         gboolean to_sens  = FALSE;
791         gboolean cc_sens  = FALSE;
792         gboolean bcc_sens = FALSE;
793
794         if (!addrbook.window) return;
795
796         if (addrbook.target_compose) {
797                 to_sens = TRUE;
798                 cc_sens = TRUE;
799                 if (addrbook.target_compose->use_bcc)
800                         bcc_sens = TRUE;
801         }
802
803         gtk_widget_set_sensitive(addrbook.to_btn, to_sens);
804         gtk_widget_set_sensitive(addrbook.cc_btn, cc_sens);
805         gtk_widget_set_sensitive(addrbook.bcc_btn, bcc_sens);
806 }
807
808 /*
809 * Delete one or more objects from address list.
810 */
811 static void addressbook_del_clicked(GtkButton *button, gpointer data)
812 {
813         GtkCTree *clist = GTK_CTREE(addrbook.clist);
814         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
815         AddressObject *pobj, *obj;
816         AdapterDSource *ads = NULL;
817         GtkCTreeNode *nodeList;
818         gboolean procFlag;
819         AlertValue aval;
820         AddressBookFile *abf = NULL;
821         AddressDataSource *ds = NULL;
822
823         pobj = gtk_ctree_node_get_row_data(ctree, addrbook.opened );
824         g_return_if_fail(pobj != NULL);
825
826         nodeList = addrbook.listSelected;
827         obj = gtk_ctree_node_get_row_data( clist, nodeList );
828         if( obj == NULL) return;
829         ds = addressbook_find_datasource( addrbook.treeSelected );
830         if( ds == NULL ) return;
831
832         procFlag = FALSE;
833         if( pobj->type == ADDR_DATASOURCE ) {
834                 ads = ADAPTER_DSOURCE(pobj);
835                 if( ads->subType == ADDR_BOOK ) procFlag = TRUE;
836         }
837         else if( pobj->type == ADDR_ITEM_FOLDER ) {
838                 procFlag = TRUE;
839         }
840         else if( pobj->type == ADDR_ITEM_GROUP ) {
841                 procFlag = TRUE;
842         }
843         if( ! procFlag ) return;
844         abf = ds->rawDataSource;
845         if( abf == NULL ) return;
846
847         /* Confirm deletion */
848         aval = alertpanel( _("Delete address(es)"),
849                         _("Really delete the address(es)?"),
850                         _("Yes"), _("No"), NULL );
851         if( aval != G_ALERTDEFAULT ) return;
852
853         /* Process deletions */
854         if( pobj->type == ADDR_DATASOURCE || pobj->type == ADDR_ITEM_FOLDER ) {
855                 /* Items inside folders */
856                 GList *node;
857                 node = _addressListSelection_;
858                 while( node ) {
859                         AddrItemObject *aio = node->data;
860                         node = g_list_next( node );
861                         if( aio->type == ADDR_ITEM_GROUP ) {
862                                 ItemGroup *item = ( ItemGroup * ) aio;
863                                 GtkCTreeNode *nd = NULL;
864
865                                 nd = addressbook_find_group_node( addrbook.opened, item );
866                                 item = addrbook_remove_group( abf, item );
867                                 if( item ) {
868                                         addritem_free_item_group( item );
869                                         item = NULL;
870                                 }
871                                 /* Remove group from parent node */
872                                 gtk_ctree_remove_node( ctree, nd );
873                         }
874                         else if( aio->type == ADDR_ITEM_PERSON ) {
875                                 ItemPerson *item = ( ItemPerson * ) aio;
876                                 item = addrbook_remove_person( abf, item );
877                                 if( item ) {
878                                         addritem_free_item_person( item );
879                                         item = NULL;
880                                 }
881                         }
882                         else if( aio->type == ADDR_ITEM_EMAIL ) {
883                                 ItemEMail *item = ( ItemEMail * ) aio;
884                                 ItemPerson *person = ( ItemPerson * ) ADDRITEM_PARENT(item);
885                                 item = addrbook_person_remove_email( abf, person, item );
886                                 if( item ) {
887                                         addritem_free_item_email( item );
888                                         item = NULL;
889                                 }
890                         }
891                 }
892                 addressbook_list_select_clear();
893                 gtk_ctree_select( ctree, addrbook.opened);
894                 return;
895         }
896         else if( pobj->type == ADDR_ITEM_GROUP ) {
897                 /* Items inside groups */
898                 GList *node;
899                 node = _addressListSelection_;
900                 while( node ) {
901                         AddrItemObject *aio = node->data;
902                         node = g_list_next( node );
903                         if( aio->type == ADDR_ITEM_EMAIL ) {
904                                 ItemEMail *item = ( ItemEMail * ) aio;
905                                 ItemPerson *person = ( ItemPerson * ) ADDRITEM_PARENT(item);
906                                 item = addrbook_person_remove_email( abf, person, item );
907                                 if( item ) {
908                                         addritem_print_item_email( item, stdout );
909                                         addritem_free_item_email( item );
910                                         item = NULL;
911                                 }
912                         }
913                 }
914                 addressbook_list_select_clear();
915                 gtk_ctree_select( ctree, addrbook.opened);
916                 return;
917         }
918
919         gtk_ctree_node_set_row_data( clist, nodeList, NULL );
920         gtk_ctree_remove_node( clist, nodeList );
921         addressbook_list_select_remove( obj );
922
923 }
924
925 static void addressbook_reg_clicked(GtkButton *button, gpointer data)
926 {
927         addressbook_new_address_cb( NULL, 0, NULL );
928 }
929
930 gchar *addressbook_format_address( AddressObject * obj ) {
931         gchar *buf = NULL;
932         gchar *name = NULL;
933         gchar *address = NULL;
934
935         if( obj->type == ADDR_ITEM_EMAIL ) {
936                 ItemPerson *person = NULL;
937                 ItemEMail *email = ( ItemEMail * ) obj;
938
939                 person = ( ItemPerson * ) ADDRITEM_PARENT(email);
940                 if( email->address ) {
941                         if( ADDRITEM_NAME(email) ) {
942                                 name = ADDRITEM_NAME(email);
943                                 if( *name == '\0' ) {
944                                         name = ADDRITEM_NAME(person);
945                                 }
946                         }
947                         else if( ADDRITEM_NAME(person) ) {
948                                 name = ADDRITEM_NAME(person);
949                         }
950                         else {
951                                 buf = g_strdup( email->address );
952                         }
953                         address = email->address;
954                 }
955         }
956         else if( obj->type == ADDR_ITEM_PERSON ) {
957                 ItemPerson *person = ( ItemPerson * ) obj;
958                 GList *node = person->listEMail;
959
960                 name = ADDRITEM_NAME(person);
961                 if( node ) {
962                         ItemEMail *email = ( ItemEMail * ) node->data;
963                         address = email->address;
964                 }
965         }
966         if( address ) {
967                 if( name ) {
968                         buf = g_strdup_printf( "%s <%s>", name, address );
969                 }
970                 else {
971                         buf = g_strdup( address );
972                 }
973         }
974
975         return buf;
976 }
977
978 static void addressbook_to_clicked(GtkButton *button, gpointer data)
979 {
980         GList *node = _addressListSelection_;
981         if (!addrbook.target_compose) return;
982         while( node ) {
983                 AddressObject *obj = node->data;
984                 Compose *compose = addrbook.target_compose;
985                 node = g_list_next( node );
986                 if( obj->type == ADDR_ITEM_PERSON || obj->type == ADDR_ITEM_EMAIL ) {
987                         gchar *addr = addressbook_format_address( obj );
988                         compose_entry_append( compose, addr, (ComposeEntryType) data );
989                         g_free( addr );
990                         addr = NULL;
991                 }
992                 else if( obj->type == ADDR_ITEM_GROUP ) {
993                         ItemGroup *group = ( ItemGroup * ) obj;
994                         GList *nodeMail = group->listEMail;
995                         while( nodeMail ) {
996                                 ItemEMail *email = nodeMail->data;
997                                 gchar *addr = addressbook_format_address( ( AddressObject * ) email );
998                                 compose_entry_append( compose, addr, (ComposeEntryType) data );
999                                 g_free( addr );
1000                                 nodeMail = g_list_next( nodeMail );
1001                         }
1002                 }
1003         }
1004 }
1005
1006 static void addressbook_menubar_set_sensitive( gboolean sensitive ) {
1007         menu_set_sensitive( addrbook.menu_factory, "/File/New Book",    sensitive );
1008         menu_set_sensitive( addrbook.menu_factory, "/File/New V-Card",  sensitive );
1009 #ifdef USE_JPILOT
1010         menu_set_sensitive( addrbook.menu_factory, "/File/New J-Pilot", sensitive );
1011 #endif
1012 #ifdef USE_LDAP
1013         menu_set_sensitive( addrbook.menu_factory, "/File/New Server",  sensitive );
1014 #endif
1015         menu_set_sensitive( addrbook.menu_factory, "/File/Edit",        sensitive );
1016         menu_set_sensitive( addrbook.menu_factory, "/File/Delete",      sensitive );
1017
1018         menu_set_sensitive( addrbook.menu_factory, "/Address/New Address", sensitive );
1019         menu_set_sensitive( addrbook.menu_factory, "/Address/New Group",   sensitive );
1020         menu_set_sensitive( addrbook.menu_factory, "/Address/New Folder",  sensitive );
1021         gtk_widget_set_sensitive( addrbook.reg_btn, sensitive );
1022         gtk_widget_set_sensitive( addrbook.del_btn, sensitive );
1023 }
1024
1025 static void addressbook_menuitem_set_sensitive( AddressObject *obj, GtkCTreeNode *node ) {
1026         gboolean canEdit = FALSE;
1027         gboolean canAdd = FALSE;
1028         gboolean canEditTr = TRUE;
1029         gboolean editAddress = FALSE;
1030         AddressTypeControlItem *atci = NULL;
1031         AddressDataSource *ds = NULL;
1032         AddressInterface *iface = NULL;
1033
1034         if( obj == NULL ) return;
1035         if( obj->type == ADDR_INTERFACE ) {
1036                 AdapterInterface *adapter = ADAPTER_INTERFACE(obj);
1037                 iface = adapter->interface;
1038                 if( iface ) {
1039                         if( iface->haveLibrary ) {
1040                                 /* Enable appropriate File / New command */
1041                                 atci = adapter->atci;
1042                                 menu_set_sensitive( addrbook.menu_factory, atci->menuCommand, TRUE );
1043                         }
1044                 }
1045                 canEditTr = FALSE;
1046         }
1047         else if( obj->type == ADDR_DATASOURCE ) {
1048                 AdapterDSource *ads = ADAPTER_DSOURCE(obj);
1049                 ds = ads->dataSource;
1050                 iface = ds->interface;
1051                 if( ! iface->readOnly ) {
1052                         canAdd = canEdit = editAddress = TRUE;
1053                 }
1054                 if( ! iface->haveLibrary ) {
1055                         canAdd = canEdit = editAddress = FALSE;
1056                 }
1057         }
1058         else if( obj->type == ADDR_ITEM_FOLDER ) {
1059                 ds = addressbook_find_datasource( addrbook.treeSelected );
1060                 if( ds ) {
1061                         iface = ds->interface;
1062                         if( ! iface->readOnly ) {
1063                                 canAdd = editAddress = TRUE;
1064                         }
1065                 }
1066         }
1067         else if( obj->type == ADDR_ITEM_GROUP ) {
1068                 ds = addressbook_find_datasource( addrbook.treeSelected );
1069                 if( ds ) {
1070                         iface = ds->interface;
1071                         if( ! iface->readOnly ) {
1072                                 editAddress = TRUE;
1073                         }
1074                 }
1075         }
1076
1077         if( addrbook.listSelected == NULL ) canEdit = FALSE;
1078
1079         /* Enable add */
1080         menu_set_sensitive( addrbook.menu_factory, "/Address/New Address", editAddress );
1081         menu_set_sensitive( addrbook.menu_factory, "/Address/New Group",   canAdd );
1082         menu_set_sensitive( addrbook.menu_factory, "/Address/New Folder",  canAdd );
1083         gtk_widget_set_sensitive( addrbook.reg_btn, editAddress );
1084
1085         /* Enable edit */
1086         menu_set_sensitive( addrbook.menu_factory, "/Address/Edit",   canEdit );
1087         menu_set_sensitive( addrbook.menu_factory, "/Address/Delete", canEdit );
1088         gtk_widget_set_sensitive( addrbook.del_btn, canEdit );
1089
1090         menu_set_sensitive( addrbook.menu_factory, "/File/Edit",      canEditTr );
1091         menu_set_sensitive( addrbook.menu_factory, "/File/Delete",    canEditTr );
1092 }
1093
1094 static void addressbook_tree_selected(GtkCTree *ctree, GtkCTreeNode *node,
1095                                       gint column, gpointer data)
1096 {
1097         AddressObject *obj = NULL;
1098         AdapterDSource *ads = NULL;
1099         AddressDataSource *ds = NULL;
1100         ItemFolder *rootFolder = NULL;
1101
1102         addrbook.treeSelected = node;
1103         addrbook.listSelected = NULL;
1104         addressbook_status_show( "" );
1105         if( addrbook.entry != NULL ) gtk_entry_set_text(GTK_ENTRY(addrbook.entry), "");
1106
1107         if( addrbook.clist ) gtk_clist_clear( GTK_CLIST(addrbook.clist) );
1108         if( node ) obj = gtk_ctree_node_get_row_data( ctree, node );
1109         if( obj == NULL ) return;
1110
1111         addrbook.opened = node;
1112
1113         if( obj->type == ADDR_DATASOURCE ) {
1114                 /* Read from file */
1115                 static gboolean tVal = TRUE;
1116
1117                 ads = ADAPTER_DSOURCE(obj);
1118                 if( ads == NULL ) return;
1119                 ds = ads->dataSource;
1120                 if( ds == NULL ) return;                
1121
1122                 if( addrindex_ds_get_modify_flag( ds ) ) {
1123                         addrindex_ds_read_data( ds );
1124                 }
1125
1126                 if( ! addrindex_ds_get_read_flag( ds ) ) {
1127                         addrindex_ds_read_data( ds );
1128                 }
1129                 addressbook_ds_show_message( ds );
1130
1131                 if( ! addrindex_ds_get_access_flag( ds ) ) {
1132                         /* Remove existing folders and groups */
1133                         gtk_clist_freeze( GTK_CLIST(ctree) );
1134                         addressbook_tree_remove_children( ctree, node );
1135                         gtk_clist_thaw( GTK_CLIST(ctree) );
1136
1137                         /* Load folders into the tree */
1138                         rootFolder = addrindex_ds_get_root_folder( ds );
1139                         if( ds->type == ADDR_IF_JPILOT ) {
1140                                 addressbook_node_add_folder( node, ds, rootFolder, ADDR_CATEGORY );
1141                         }
1142                         else {
1143                                 addressbook_node_add_folder( node, ds, rootFolder, ADDR_ITEM_FOLDER );
1144                         }
1145                         addrindex_ds_set_access_flag( ds, &tVal );
1146                         gtk_ctree_expand( ctree, node );
1147                 }
1148         }
1149
1150         /* Update address list */
1151         addressbook_set_clist( obj );
1152
1153         /* Setup main menu selections */
1154         addressbook_menubar_set_sensitive( FALSE );
1155         addressbook_menuitem_set_sensitive( obj, node );
1156
1157         addressbook_list_select_clear();
1158
1159 }
1160
1161 static void addressbook_list_selected(GtkCList *clist, gint row, gint column,
1162                                       GdkEvent *event, gpointer data)
1163 {
1164         if (event && event->type == GDK_2BUTTON_PRESS) {
1165                 /* Handle double click */
1166                 if (prefs_common.add_address_by_click &&
1167                     addrbook.target_compose)
1168                         addressbook_to_clicked(NULL, NULL);
1169                 else
1170                         addressbook_edit_address_cb(NULL, 0, NULL);
1171         }
1172 }
1173
1174 #if 0
1175 static void addressbook_list_select_show() {
1176         GList *node = _addressListSelection_;
1177         gchar *addr = NULL;
1178         printf( "show selection...>>>\n" );
1179         while( node != NULL ) {
1180                 AddressObject *obj = ( AddressObject * ) node->data;
1181                 if( obj ) {
1182                         printf( "- %d : '%s'\n", obj->type, obj->name );
1183                         if( obj->type == ADDR_ITEM_GROUP ) {
1184                                 ItemGroup *group = ( ItemGroup * ) obj;
1185                                 GList *node = group->listEMail;
1186                                 while( node ) {
1187                                         ItemEMail *email = node->data;
1188                                         addr = addressbook_format_address( ( AddressObject * ) email );
1189                                         if( addr ) {
1190                                                 printf( "\tgrp >%s<\n", addr );
1191                                                 g_free( addr );
1192                                         }
1193                                         node = g_list_next( node );
1194                                 }
1195                         }
1196                         else {
1197                                 addr = addressbook_format_address( obj );
1198                                 if( addr ) {
1199                                         printf( "\t>%s<\n", addr );
1200                                         g_free( addr );
1201                                 }
1202                         }
1203                 }
1204                 else {
1205                         printf( "- NULL" );
1206                 }
1207                 node = g_list_next( node );
1208         }
1209         printf( "show selection...<<<\n" );
1210 }
1211 #endif
1212
1213 static void addressbook_list_select_clear() {
1214         if( _addressListSelection_ ) {
1215                 g_list_free( _addressListSelection_ );
1216         }
1217         _addressListSelection_ = NULL;
1218 }
1219
1220 static void addressbook_list_select_add( AddressObject *obj ) {
1221         if( obj ) {
1222                 if(     obj->type == ADDR_ITEM_PERSON ||
1223                         obj->type == ADDR_ITEM_EMAIL ||
1224                         obj->type == ADDR_ITEM_GROUP ) {
1225                         if( ! g_list_find( _addressListSelection_, obj ) ) {
1226                                 _addressListSelection_ = g_list_append( _addressListSelection_, obj );
1227                         }
1228                 }
1229         }
1230         /* addressbook_list_select_show(); */
1231 }
1232
1233 static void addressbook_list_select_remove( AddressObject *obj ) {
1234         if( obj == NULL ) return;
1235         if( _addressListSelection_ ) {
1236                 _addressListSelection_ = g_list_remove( _addressListSelection_, obj );
1237         }
1238         /* addressbook_list_select_show(); */
1239 }
1240
1241 static void addressbook_list_row_selected( GtkCTree *clist, GtkCTreeNode *node, gint column, gpointer data ) {
1242         GtkEntry *entry = GTK_ENTRY(addrbook.entry);
1243         AddressObject *obj = NULL;
1244         AddressObject *pobj = NULL;
1245         AdapterDSource *ads = NULL;
1246         AddressInterface *iface = NULL;
1247         AddressDataSource *ds = NULL;
1248         gboolean canEdit = FALSE;
1249         gboolean canDelete = FALSE;
1250
1251         gtk_entry_set_text( entry, "" );
1252         addrbook.listSelected = node;
1253         obj = gtk_ctree_node_get_row_data( clist, node );
1254         if( obj != NULL ) {
1255                 /* printf( "list select: %d : '%s'\n", obj->type, obj->name ); */
1256                 addressbook_list_select_add( obj );
1257         }
1258
1259         pobj = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), addrbook.treeSelected );
1260         if( pobj == NULL ) return;
1261
1262         menu_set_insensitive_all( GTK_MENU_SHELL(addrbook.list_popup) );
1263
1264         if( pobj->type == ADDR_DATASOURCE ) {
1265                 ads = ADAPTER_DSOURCE(pobj);
1266                 ds = ads->dataSource;
1267                 iface = ds->interface;
1268                 if( ! iface->readOnly ) {
1269                         canEdit = TRUE;
1270                         menu_set_sensitive( addrbook.list_factory, "/New Address", TRUE );
1271                         menu_set_sensitive( addrbook.list_factory, "/New Folder", TRUE );
1272                         menu_set_sensitive( addrbook.list_factory, "/New Group", TRUE );
1273                         gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
1274                 }
1275         }
1276         else if( pobj->type != ADDR_INTERFACE ) {
1277                 ds = addressbook_find_datasource( addrbook.treeSelected );
1278                 iface = ds->interface;
1279                 if( ! iface->readOnly ) {
1280                         if( pobj->type == ADDR_ITEM_FOLDER || pobj->type == ADDR_ITEM_GROUP ) {
1281                                 canEdit = TRUE;
1282                                 menu_set_sensitive( addrbook.list_factory, "/New Address", TRUE );
1283                                 gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
1284                         }
1285                         if( pobj->type == ADDR_ITEM_FOLDER ) {
1286                                 canEdit = TRUE;
1287                                 menu_set_sensitive( addrbook.list_factory, "/New Folder", TRUE );
1288                                 menu_set_sensitive( addrbook.list_factory, "/New Group", TRUE );
1289                         }
1290                 }
1291         }
1292         if( obj == NULL ) canEdit = FALSE;
1293         canDelete = canEdit;
1294         if( GTK_CLIST(clist)->selection && GTK_CLIST(clist)->selection->next ) canEdit = FALSE;
1295
1296         menu_set_sensitive( addrbook.list_factory, "/Edit",   canEdit );
1297         menu_set_sensitive( addrbook.list_factory, "/Delete", canDelete );
1298
1299         menu_set_sensitive( addrbook.menu_factory, "/Address/Edit",   canEdit );
1300         menu_set_sensitive( addrbook.menu_factory, "/Address/Delete", canDelete );
1301
1302         gtk_widget_set_sensitive( addrbook.del_btn, canDelete );
1303
1304 }
1305
1306 static void addressbook_list_row_unselected( GtkCTree *ctree, GtkCTreeNode *node, gint column, gpointer data ) {
1307         AddressObject *obj;
1308
1309         obj = gtk_ctree_node_get_row_data( ctree, node );
1310         if( obj != NULL ) {
1311                 /* g_print( "list unselect: %d : '%s'\n", obj->type, obj->name ); */
1312                 addressbook_list_select_remove( obj );
1313         }
1314 }
1315
1316 static void addressbook_entry_gotfocus( GtkWidget *widget ) {
1317         gtk_editable_select_region( GTK_EDITABLE(addrbook.entry), 0, -1 );
1318 }
1319
1320 static void addressbook_list_button_pressed(GtkWidget *widget,
1321                                             GdkEventButton *event,
1322                                             gpointer data)
1323 {
1324         if( ! event ) return;
1325         if( event->button == 3 ) {
1326                 gtk_menu_popup( GTK_MENU(addrbook.list_popup), NULL, NULL, NULL, NULL,
1327                        event->button, event->time );
1328         }
1329 }
1330
1331 static void addressbook_list_button_released(GtkWidget *widget,
1332                                              GdkEventButton *event,
1333                                              gpointer data)
1334 {
1335 }
1336
1337 static void addressbook_tree_button_pressed(GtkWidget *ctree,
1338                                             GdkEventButton *event,
1339                                             gpointer data)
1340 {
1341         GtkCList *clist = GTK_CLIST(ctree);
1342         gint row, column;
1343         AddressObject *obj = NULL;
1344         /* GtkCTreeNode *node; */
1345         AdapterDSource *ads = NULL;
1346         AddressInterface *iface = NULL;
1347         AddressDataSource *ds = NULL;
1348         /* AddressTypeControlItem *atci = NULL; */
1349         gboolean canEdit = FALSE;
1350
1351         if( ! event ) return;
1352         addressbook_menubar_set_sensitive( FALSE );
1353 /* */
1354         if( gtk_clist_get_selection_info( clist, event->x, event->y, &row, &column ) ) {
1355                 gtk_clist_select_row( clist, row, column );
1356                 gtkut_clist_set_focus_row(clist, row);
1357                 obj = gtk_clist_get_row_data( clist, row );
1358         }
1359 /* */
1360         menu_set_insensitive_all(GTK_MENU_SHELL(addrbook.tree_popup));
1361
1362         if( obj == NULL ) return;
1363         if (obj->type == ADDR_DATASOURCE) {
1364                 ads = ADAPTER_DSOURCE(obj);
1365                 ds = ads->dataSource;
1366                 iface = ds->interface;
1367                 canEdit = TRUE;
1368                 if( ! iface->readOnly ) {
1369                         menu_set_sensitive( addrbook.tree_factory, "/New Address", TRUE );
1370                         menu_set_sensitive( addrbook.tree_factory, "/New Folder", TRUE );
1371                         menu_set_sensitive( addrbook.tree_factory, "/New Group", TRUE );
1372                         gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
1373                 }
1374         }
1375         else if (obj->type == ADDR_ITEM_FOLDER) {
1376                 ds = addressbook_find_datasource( addrbook.treeSelected );
1377                 iface = ds->interface;
1378                 if( ! iface->readOnly ) {
1379                         canEdit = TRUE;
1380                         menu_set_sensitive( addrbook.tree_factory, "/New Address", TRUE );
1381                         menu_set_sensitive( addrbook.tree_factory, "/New Folder", TRUE );
1382                         menu_set_sensitive( addrbook.tree_factory, "/New Group", TRUE );
1383                         gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
1384                 }
1385         }
1386         else if (obj->type == ADDR_ITEM_GROUP) {
1387                 ds = addressbook_find_datasource( addrbook.treeSelected );
1388                 iface = ds->interface;
1389                 if( ! iface->readOnly ) {
1390                         canEdit = TRUE;
1391                         menu_set_sensitive( addrbook.tree_factory, "/New Address", TRUE );
1392                         gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
1393                 }
1394         }
1395
1396         /* Enable edit */
1397         menu_set_sensitive( addrbook.tree_factory, "/Edit",   canEdit );
1398         menu_set_sensitive( addrbook.tree_factory, "/Delete", canEdit );
1399         menu_set_sensitive( addrbook.menu_factory, "/File/Edit",   canEdit );
1400         menu_set_sensitive( addrbook.menu_factory, "/File/Delete", canEdit );
1401
1402         if( event->button == 3 ) {
1403                 gtk_menu_popup(GTK_MENU(addrbook.tree_popup), NULL, NULL, NULL, NULL,
1404                                event->button, event->time);
1405         }
1406
1407 }
1408
1409 static void addressbook_tree_button_released(GtkWidget *ctree,
1410                                              GdkEventButton *event,
1411                                              gpointer data)
1412 {
1413         gtk_ctree_select(GTK_CTREE(addrbook.ctree), addrbook.opened);
1414         gtkut_ctree_set_focus_row(GTK_CTREE(addrbook.ctree), addrbook.opened);
1415 }
1416
1417 static void addressbook_popup_close(GtkMenuShell *menu_shell, gpointer data)
1418 {
1419         if (!addrbook.opened) return;
1420
1421         gtk_ctree_select(GTK_CTREE(addrbook.ctree), addrbook.opened);
1422         gtkut_ctree_set_focus_row(GTK_CTREE(addrbook.ctree),
1423                                   addrbook.opened);
1424 }
1425
1426 static void addressbook_new_folder_cb(gpointer data, guint action,
1427                                       GtkWidget *widget)
1428 {
1429         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
1430         AddressObject *obj = NULL;
1431         AddressDataSource *ds = NULL;
1432         AddressBookFile *abf = NULL;
1433         ItemFolder *parentFolder = NULL;
1434         ItemFolder *folder = NULL;
1435
1436         if( ! addrbook.treeSelected ) return;
1437         obj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
1438         if( obj == NULL ) return;
1439         ds = addressbook_find_datasource( addrbook.treeSelected );
1440         if( ds == NULL ) return;
1441
1442         if( obj->type == ADDR_DATASOURCE ) {
1443                 if( ADAPTER_DSOURCE(obj)->subType != ADDR_BOOK ) return;
1444         }
1445         else if( obj->type == ADDR_ITEM_FOLDER ) {
1446                 parentFolder = ADAPTER_FOLDER(obj)->itemFolder;
1447         }
1448         else {
1449                 return;
1450         }
1451
1452         abf = ds->rawDataSource;
1453         if( abf == NULL ) return;
1454         folder = addressbook_edit_folder( abf, parentFolder, NULL );
1455         if( folder ) {
1456                 GtkCTreeNode *nn;
1457                 nn = addressbook_node_add_folder( addrbook.treeSelected, ds, folder, ADDR_ITEM_FOLDER );
1458                 gtk_ctree_expand( ctree, addrbook.treeSelected );
1459                 if( addrbook.treeSelected == addrbook.opened ) addressbook_set_clist(obj);
1460         }
1461
1462 }
1463
1464 static void addressbook_new_group_cb(gpointer data, guint action,
1465                                      GtkWidget *widget)
1466 {
1467         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
1468         AddressObject *obj = NULL;
1469         AddressDataSource *ds = NULL;
1470         AddressBookFile *abf = NULL;
1471         ItemFolder *parentFolder = NULL;
1472         ItemGroup *group = NULL;
1473
1474         if( ! addrbook.treeSelected ) return;
1475         obj = gtk_ctree_node_get_row_data(ctree, addrbook.treeSelected);
1476         if( obj == NULL ) return;
1477         ds = addressbook_find_datasource( addrbook.treeSelected );
1478         if( ds == NULL ) return;
1479
1480         if( obj->type == ADDR_DATASOURCE ) {
1481                 if( ADAPTER_DSOURCE(obj)->subType != ADDR_BOOK ) return;
1482         }
1483         else if( obj->type == ADDR_ITEM_FOLDER ) {
1484                 parentFolder = ADAPTER_FOLDER(obj)->itemFolder;
1485         }
1486         else {
1487                 return;
1488         }
1489
1490         abf = ds->rawDataSource;
1491         if( abf == NULL ) return;
1492         group = addressbook_edit_group( abf, parentFolder, NULL );
1493         if( group ) {
1494                 GtkCTreeNode *nn;
1495                 nn = addressbook_node_add_group( addrbook.treeSelected, ds, group );
1496                 gtk_ctree_expand( ctree, addrbook.treeSelected );
1497                 if( addrbook.treeSelected == addrbook.opened ) addressbook_set_clist(obj);
1498         }
1499
1500 }
1501
1502 static void addressbook_change_node_name(GtkCTreeNode *node, const gchar *name)
1503 {
1504         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
1505         gchar *text[1];
1506         guint8 spacing;
1507         GdkPixmap *pix_cl, *pix_op;
1508         GdkBitmap *mask_cl, *mask_op;
1509         gboolean is_leaf, expanded;
1510
1511         gtk_ctree_get_node_info(ctree, node, text, &spacing,
1512                                 &pix_cl, &mask_cl, &pix_op, &mask_op,
1513                                 &is_leaf, &expanded);
1514         gtk_ctree_set_node_info(ctree, node, name, spacing,
1515                                 pix_cl, mask_cl, pix_op, mask_op,
1516                                 is_leaf, expanded);
1517 }
1518
1519 /*
1520 * Edit data source.
1521 * Enter: obj   Address object to edit.
1522 *        node  Node in tree.
1523 * Return: New name of data source.
1524 */
1525 static gchar *addressbook_edit_datasource( AddressObject *obj, GtkCTreeNode *node ) {
1526         gchar *newName = NULL;
1527         AddressDataSource *ds = NULL;
1528         AddressInterface *iface = NULL;
1529         AdapterDSource *ads = NULL;
1530
1531         ds = addressbook_find_datasource( node );
1532         if( ds == NULL ) return NULL;
1533         iface = ds->interface;
1534         if( ! iface->haveLibrary ) return NULL;
1535
1536         /* Read data from data source */
1537         if( ! addrindex_ds_get_read_flag( ds ) ) {
1538                 addrindex_ds_read_data( ds );
1539         }
1540
1541         /* Handle edit */
1542         ads = ADAPTER_DSOURCE(obj);
1543         if( ads->subType == ADDR_BOOK ) {
1544                 if( addressbook_edit_book( _addressIndex_, ads ) == NULL ) return NULL;
1545         }
1546         else if( ads->subType == ADDR_VCARD ) {
1547                 if( addressbook_edit_vcard( _addressIndex_, ads ) == NULL ) return NULL;
1548         }
1549 #ifdef USE_JPILOT
1550         else if( ads->subType == ADDR_JPILOT ) {
1551                 if( addressbook_edit_jpilot( _addressIndex_, ads ) == NULL ) return NULL;
1552         }
1553 #endif
1554 #ifdef USE_LDAP
1555         else if( ads->subType == ADDR_LDAP ) {
1556                 if( addressbook_edit_ldap( _addressIndex_, ads ) == NULL ) return NULL;
1557         }
1558 #endif
1559         else {
1560                 return NULL;
1561         }
1562         newName = obj->name;
1563         return newName;
1564 }
1565
1566 /*
1567 * Edit an object that is in the address tree area.
1568 */
1569 static void addressbook_treenode_edit_cb(gpointer data, guint action,
1570                                        GtkWidget *widget)
1571 {
1572         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
1573         AddressObject *obj;
1574         AddressDataSource *ds = NULL;
1575         AddressBookFile *abf = NULL;
1576         GtkCTreeNode *node = NULL, *parentNode = NULL;
1577         gchar *name = NULL;
1578
1579         if( ! addrbook.treeSelected ) return;
1580         node = addrbook.treeSelected;
1581         if( GTK_CTREE_ROW(node)->level == 1 ) return;
1582         obj = gtk_ctree_node_get_row_data( ctree, node );
1583         if( obj == NULL ) return;
1584         parentNode = GTK_CTREE_ROW(node)->parent;
1585
1586         ds = addressbook_find_datasource( node );
1587         if( ds == NULL ) return;
1588
1589         if( obj->type == ADDR_DATASOURCE ) {
1590                 name = addressbook_edit_datasource( obj, node );
1591                 if( name == NULL ) return;
1592         }
1593         else {
1594                 abf = ds->rawDataSource;
1595                 if( abf == NULL ) return;
1596                 if( obj->type == ADDR_ITEM_FOLDER ) {
1597                         AdapterFolder *adapter = ADAPTER_FOLDER(obj);
1598                         ItemFolder *item = adapter->itemFolder;
1599                         ItemFolder *parentFolder = NULL;
1600                         parentFolder = ( ItemFolder * ) ADDRITEM_PARENT(item);
1601                         if( addressbook_edit_folder( abf, parentFolder, item ) == NULL ) return;
1602                         name = ADDRITEM_NAME(item);
1603                 }
1604                 else if( obj->type == ADDR_ITEM_GROUP ) {
1605                         AdapterGroup *adapter = ADAPTER_GROUP(obj);
1606                         ItemGroup *item = adapter->itemGroup;
1607                         ItemFolder *parentFolder = NULL;
1608                         parentFolder = ( ItemFolder * ) ADDRITEM_PARENT(item);
1609                         if( addressbook_edit_group( abf, parentFolder, item ) == NULL ) return;
1610                         name = ADDRITEM_NAME(item);
1611                 }
1612         }
1613         if( name && parentNode ) {
1614                 /* Update node in tree view */
1615                 addressbook_change_node_name( node, name );
1616                 gtk_ctree_sort_node(ctree, parentNode);
1617                 gtk_ctree_expand( ctree, node );
1618                 gtk_ctree_select( ctree, node );
1619         }
1620 }
1621
1622 /*
1623 * Delete an item from the tree widget.
1624 */
1625 static void addressbook_treenode_delete_cb(gpointer data, guint action,
1626                                          GtkWidget *widget)
1627 {
1628         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
1629         GtkCTreeNode *node = NULL;
1630         AddressObject *obj;
1631         gchar *message;
1632         AlertValue aval;
1633         AddressBookFile *abf = NULL;
1634         AdapterDSource *ads = NULL;
1635         AddressInterface *iface = NULL;
1636         AddressDataSource *ds = NULL;
1637         gboolean remFlag = FALSE;
1638
1639         if( ! addrbook.treeSelected ) return;
1640         node = addrbook.treeSelected;
1641         if( GTK_CTREE_ROW(node)->level == 1 ) return;
1642
1643         obj = gtk_ctree_node_get_row_data( ctree, node );
1644         g_return_if_fail(obj != NULL);
1645
1646         if( obj->type == ADDR_DATASOURCE ) {
1647                 ads = ADAPTER_DSOURCE(obj);
1648                 if( ads == NULL ) return;
1649                 ds = ads->dataSource;
1650                 if( ds == NULL ) return;
1651         }
1652         else {
1653                 /* Must be folder or something else */
1654                 ds = addressbook_find_datasource( node );
1655                 if( ds == NULL ) return;
1656
1657                 /* Only allow deletion from non-readOnly data sources */
1658                 iface = ds->interface;
1659                 if( iface->readOnly ) return;
1660         }
1661
1662         /* Confirm deletion */
1663         if( obj->type == ADDR_ITEM_FOLDER ) {
1664                 message = g_strdup_printf( _(
1665                                 "Do you want to delete the folder AND all addresses in `%s' ? \n" \
1666                                 "If deleting the folder only, addresses will be moved into parent folder." ),
1667                                 obj->name );
1668                 aval = alertpanel( _("Delete"), message, _("Folder only"), _("Folder and Addresses"), _("Cancel") );
1669                 g_free(message);
1670                 if( aval == G_ALERTOTHER ) return;
1671         }
1672         else {
1673                 message = g_strdup_printf(_("Really delete `%s' ?"), obj->name);
1674                 aval = alertpanel(_("Delete"), message, _("Yes"), _("No"), NULL);
1675                 g_free(message);
1676                 if (aval != G_ALERTDEFAULT) return;
1677         }
1678
1679         /* Proceed with deletion */
1680         if( obj->type == ADDR_DATASOURCE ) {
1681                 /* Remove data source. */
1682                 if( addrindex_index_remove_datasource( _addressIndex_, ds ) ) {
1683                         addressbook_free_child_adapters( node );
1684                         remFlag = TRUE;
1685                 }
1686         }
1687         else {
1688                 abf = addressbook_get_book_file();
1689                 if( abf == NULL ) return;
1690         }
1691
1692         if( obj->type == ADDR_ITEM_FOLDER ) {
1693                 AdapterFolder *adapter = ADAPTER_FOLDER(obj);
1694                 ItemFolder *item = adapter->itemFolder;
1695                 if( aval == G_ALERTDEFAULT ) {
1696                         /* Remove folder only */
1697                         item = addrbook_remove_folder( abf, item );
1698                         if( item ) {
1699                                 addritem_free_item_folder( item );
1700                                 addressbook_move_nodes_up( ctree, node );
1701                                 remFlag = TRUE;
1702                         }
1703                 }
1704                 else if( aval == G_ALERTALTERNATE ) {
1705                         /* Remove folder and addresses */
1706                         item = addrbook_remove_folder_delete( abf, item );
1707                         if( item ) {
1708                                 addritem_free_item_folder( item );
1709                                 addressbook_free_child_adapters( node );
1710                                 remFlag = TRUE;
1711                         }
1712                 }
1713         }
1714         else if( obj->type == ADDR_ITEM_GROUP ) {
1715                 AdapterGroup *adapter = ADAPTER_GROUP(obj);
1716                 ItemGroup *item = adapter->itemGroup;
1717
1718                 item = addrbook_remove_group( abf, item );
1719                 if( item ) {
1720                         addritem_free_item_group( item );
1721                         remFlag = TRUE;
1722                 }
1723         }
1724
1725         if( remFlag ) {
1726                 /* Free up adapter and remove node. */
1727                 addressbook_free_adapter( node );
1728                 gtk_ctree_remove_node(ctree, node );
1729         }
1730 }
1731
1732 static void addressbook_new_address_cb( gpointer data, guint action, GtkWidget *widget ) {
1733         AddressObject *pobj = NULL;
1734         AddressDataSource *ds = NULL;
1735         AddressBookFile *abf = NULL;
1736
1737         pobj = gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree), addrbook.treeSelected);
1738         if( pobj == NULL ) return;
1739         ds = addressbook_find_datasource( GTK_CTREE_NODE(addrbook.treeSelected) );
1740         if( ds == NULL ) return;
1741
1742         abf = ds->rawDataSource;
1743         if( abf == NULL ) return;
1744
1745         if( pobj->type == ADDR_DATASOURCE ) {
1746                 if( ADAPTER_DSOURCE(pobj)->subType == ADDR_BOOK ) {
1747                         /* New address */
1748                         ItemPerson *person = addressbook_edit_person( abf, NULL, NULL, FALSE );
1749                         if( person ) {
1750                                 if( addrbook.treeSelected == addrbook.opened ) {
1751                                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
1752                                 }
1753                         }
1754                 }
1755         }
1756         else if( pobj->type == ADDR_ITEM_FOLDER ) {
1757                 /* New address */
1758                 ItemFolder *folder = ADAPTER_FOLDER(pobj)->itemFolder;
1759                 ItemPerson *person = addressbook_edit_person( abf, folder, NULL, FALSE );
1760                 if( person ) {
1761                         if (addrbook.treeSelected == addrbook.opened) {
1762                                 gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
1763                         }
1764                 }
1765         }
1766         else if( pobj->type == ADDR_ITEM_GROUP ) {
1767                 /* New address in group */
1768                 ItemGroup *group = ADAPTER_GROUP(pobj)->itemGroup;
1769                 if( addressbook_edit_group( abf, NULL, group ) == NULL ) return;
1770                 if (addrbook.treeSelected == addrbook.opened) {
1771                         /* Change node name in tree. */
1772                         addressbook_change_node_name( addrbook.treeSelected, ADDRITEM_NAME(group) );
1773                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
1774                 }
1775         }
1776 }
1777
1778 /*
1779 * Search for specified group in address index tree.
1780 */
1781 static GtkCTreeNode *addressbook_find_group_node( GtkCTreeNode *parent, ItemGroup *group ) {
1782         GtkCTreeNode *node = NULL;
1783         GtkCTreeRow *currRow;
1784
1785         currRow = GTK_CTREE_ROW( parent );
1786         if( currRow ) {
1787                 node = currRow->children;
1788                 while( node ) {
1789                         AddressObject *obj = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), node );
1790                         if( obj->type == ADDR_ITEM_GROUP ) {
1791                                 ItemGroup *g = ADAPTER_GROUP(obj)->itemGroup;
1792                                 if( g == group ) return node;
1793                         }
1794                         currRow = GTK_CTREE_ROW(node);
1795                         node = currRow->sibling;
1796                 }
1797         }
1798         return NULL;
1799 }
1800
1801 static AddressBookFile *addressbook_get_book_file() {
1802         AddressBookFile *abf = NULL;
1803         AddressDataSource *ds = NULL;
1804
1805         ds = addressbook_find_datasource( addrbook.treeSelected );
1806         if( ds == NULL ) return NULL;
1807         if( ds->type == ADDR_IF_BOOK ) abf = ds->rawDataSource;
1808         return abf;
1809 }
1810
1811 static void addressbook_tree_remove_children( GtkCTree *ctree, GtkCTreeNode *parent ) {
1812         GtkCTreeNode *node;
1813         GtkCTreeRow *row;
1814
1815         /* Remove existing folders and groups */
1816         row = GTK_CTREE_ROW( parent );
1817         if( row ) {
1818                 while( (node = row->children) ) {
1819                         gtk_ctree_remove_node( ctree, node );
1820                 }
1821         }
1822 }
1823
1824 static void addressbook_move_nodes_up( GtkCTree *ctree, GtkCTreeNode *node ) {
1825         GtkCTreeNode *parent, *child;
1826         GtkCTreeRow *currRow;
1827         currRow = GTK_CTREE_ROW( node );
1828         if( currRow ) {
1829                 parent = currRow->parent;
1830                 while( (child = currRow->children) ) {
1831                         gtk_ctree_move( ctree, child, parent, node );
1832                 }
1833                 gtk_ctree_sort_node( ctree, parent );
1834         }
1835 }
1836
1837 static void addressbook_edit_address_cb( gpointer data, guint action, GtkWidget *widget ) {
1838         GtkCTree *clist = GTK_CTREE(addrbook.clist);
1839         GtkCTree *ctree;
1840         AddressObject *obj = NULL, *pobj = NULL;
1841         AddressDataSource *ds = NULL;
1842         GtkCTreeNode *node = NULL, *parentNode = NULL;
1843         gchar *name = NULL;
1844         AddressBookFile *abf = NULL;
1845
1846         if( addrbook.listSelected == NULL ) return;
1847         obj = gtk_ctree_node_get_row_data( clist, addrbook.listSelected );
1848         g_return_if_fail(obj != NULL);
1849
1850         ctree = GTK_CTREE( addrbook.ctree );
1851         pobj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
1852         node = gtk_ctree_find_by_row_data( ctree, addrbook.treeSelected, obj );
1853
1854         ds = addressbook_find_datasource( GTK_CTREE_NODE(addrbook.treeSelected) );
1855         if( ds == NULL ) return;
1856
1857         abf = addressbook_get_book_file();
1858         if( abf == NULL ) return;
1859         if( obj->type == ADDR_ITEM_EMAIL ) {
1860                 ItemEMail *email = ( ItemEMail * ) obj;
1861                 ItemPerson *person;
1862                 if( email == NULL ) return;
1863                 if( pobj && pobj->type == ADDR_ITEM_GROUP ) {
1864                         /* Edit parent group */
1865                         AdapterGroup *adapter = ADAPTER_GROUP(pobj);
1866                         ItemGroup *itemGrp = adapter->itemGroup;
1867                         if( addressbook_edit_group( abf, NULL, itemGrp ) == NULL ) return;
1868                         name = ADDRITEM_NAME(itemGrp);
1869                         node = addrbook.treeSelected;
1870                         parentNode = GTK_CTREE_ROW(node)->parent;
1871                 }
1872                 else {
1873                         /* Edit person - email page */
1874                         person = ( ItemPerson * ) ADDRITEM_PARENT(email);
1875                         if( addressbook_edit_person( abf, NULL, person, TRUE ) == NULL ) return;
1876                         gtk_ctree_select( ctree, addrbook.opened );
1877                         invalidate_address_completion();
1878                         return;
1879                 }
1880         }
1881         else if( obj->type == ADDR_ITEM_PERSON ) {
1882                 /* Edit person - basic page */
1883                 ItemPerson *person = ( ItemPerson * ) obj;
1884                 if( addressbook_edit_person( abf, NULL, person, FALSE ) == NULL ) return;
1885                 gtk_ctree_select( ctree, addrbook.opened);
1886                 invalidate_address_completion();
1887                 return;
1888         }
1889         else if( obj->type == ADDR_ITEM_GROUP ) {
1890                 ItemGroup *itemGrp = ( ItemGroup * ) obj;
1891                 if( addressbook_edit_group( abf, NULL, itemGrp ) == NULL ) return;
1892                 parentNode = addrbook.treeSelected;
1893                 node = addressbook_find_group_node( parentNode, itemGrp );
1894                 name = ADDRITEM_NAME(itemGrp);
1895         }
1896         else {
1897                 return;
1898         }
1899
1900         /* Update tree node with node name */
1901         if( node == NULL ) return;
1902         addressbook_change_node_name( node, name );
1903         gtk_ctree_sort_node( ctree, parentNode );
1904         gtk_ctree_select( ctree, addrbook.opened );
1905 }
1906
1907 static void addressbook_delete_address_cb(gpointer data, guint action,
1908                                           GtkWidget *widget)
1909 {
1910         addressbook_del_clicked(NULL, NULL);
1911 }
1912
1913 static void close_cb(gpointer data, guint action, GtkWidget *widget)
1914 {
1915         addressbook_close();
1916 }
1917
1918 static void addressbook_file_save_cb( gpointer data, guint action, GtkWidget *widget ) {
1919         addressbook_export_to_file();
1920 }
1921
1922 static void addressbook_person_expand_node( GtkCTree *ctree, GList *node, gpointer *data ) {
1923         if( node ) {
1924                 ItemPerson *person = gtk_ctree_node_get_row_data( ctree, GTK_CTREE_NODE(node) );
1925                 if( person ) addritem_person_set_opened( person, TRUE );
1926         }
1927 }
1928
1929 static void addressbook_person_collapse_node( GtkCTree *ctree, GList *node, gpointer *data ) {
1930         if( node ) {
1931                 ItemPerson *person = gtk_ctree_node_get_row_data( ctree, GTK_CTREE_NODE(node) );
1932                 if( person ) addritem_person_set_opened( person, FALSE );
1933         }
1934 }
1935
1936 static gchar *addressbook_format_item_clist( ItemPerson *person, ItemEMail *email ) {
1937         gchar *str = NULL;
1938         gchar *eMailAlias = ADDRITEM_NAME(email);
1939         if( eMailAlias && *eMailAlias != '\0' ) {
1940                 if( person ) {
1941                         str = g_strdup_printf( "%s - %s", ADDRITEM_NAME(person), eMailAlias );
1942                 }
1943                 else {
1944                         str = g_strdup( eMailAlias );
1945                 }
1946         }
1947         return str;
1948 }
1949
1950 static void addressbook_load_group( GtkCTree *clist, ItemGroup *itemGroup ) {
1951         GList *items = itemGroup->listEMail;
1952         AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_EMAIL );
1953         for( ; items != NULL; items = g_list_next( items ) ) {
1954                 GtkCTreeNode *nodeEMail = NULL;
1955                 gchar *text[N_COLS];
1956                 ItemEMail *email = items->data;
1957                 ItemPerson *person;
1958                 gchar *str = NULL;
1959
1960                 if( ! email ) continue;
1961
1962                 person = ( ItemPerson * ) ADDRITEM_PARENT(email);
1963                 str = addressbook_format_item_clist( person, email );
1964                 if( str ) {
1965                         text[COL_NAME] = str;
1966                 }
1967                 else {
1968                         text[COL_NAME] = ADDRITEM_NAME(person);
1969                 }
1970                 text[COL_ADDRESS] = email->address;
1971                 text[COL_REMARKS] = email->remarks;
1972                 nodeEMail = gtk_ctree_insert_node(
1973                                 clist, NULL, NULL,
1974                                 text, FOLDER_SPACING,
1975                                 atci->iconXpm, atci->maskXpm,
1976                                 atci->iconXpmOpen, atci->maskXpmOpen,
1977                                 FALSE, FALSE );
1978                 gtk_ctree_node_set_row_data( clist, nodeEMail, email );
1979                 g_free( str );
1980                 str = NULL;
1981         }
1982 }
1983
1984 static void addressbook_folder_load_person( GtkCTree *clist, ItemFolder *itemFolder ) {
1985         GList *items;
1986         AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_PERSON );
1987         AddressTypeControlItem *atciMail = addrbookctl_lookup( ADDR_ITEM_EMAIL );
1988
1989         if( atci == NULL ) return;
1990         if( atciMail == NULL ) return;
1991
1992         /* Load email addresses */
1993         items = addritem_folder_get_person_list( itemFolder );
1994         for( ; items != NULL; items = g_list_next( items ) ) {
1995                 GtkCTreeNode *nodePerson = NULL;
1996                 GtkCTreeNode *nodeEMail = NULL;
1997                 gchar *text[N_COLS];
1998                 gboolean flgFirst = TRUE, haveAddr = FALSE;
1999                 /* gint row; */
2000                 ItemPerson *person;
2001                 GList *node;
2002
2003                 person = ( ItemPerson * ) items->data;
2004                 if( person == NULL ) continue;
2005
2006                 text[COL_NAME] = NULL;
2007                 node = person->listEMail;
2008                 while( node ) {
2009                         ItemEMail *email = node->data;
2010                         gchar *eMailAddr = NULL;
2011                         node = g_list_next( node );
2012
2013                         text[COL_ADDRESS] = email->address;
2014                         text[COL_REMARKS] = email->remarks;
2015                         eMailAddr = ADDRITEM_NAME(email);
2016                         if( eMailAddr && *eMailAddr == '\0' ) eMailAddr = NULL;
2017                         if( flgFirst ) {
2018                                 /* First email belongs with person */
2019                                 gchar *str = addressbook_format_item_clist( person, email );
2020                                 if( str ) {
2021                                         text[COL_NAME] = str;
2022                                 }
2023                                 else {
2024                                         text[COL_NAME] = ADDRITEM_NAME(person);
2025                                 }
2026                                 nodePerson = gtk_ctree_insert_node(
2027                                                 clist, NULL, NULL,
2028                                                 text, FOLDER_SPACING,
2029                                                 atci->iconXpm, atci->maskXpm,
2030                                                 atci->iconXpmOpen, atci->maskXpmOpen,
2031                                                 FALSE, person->isOpened );
2032                                 g_free( str );
2033                                 str = NULL;
2034                                 gtk_ctree_node_set_row_data(clist, nodePerson, person );
2035                         }
2036                         else {
2037                                 /* Subsequent email is a child node of person */
2038                                 text[COL_NAME] = ADDRITEM_NAME(email);
2039                                 nodeEMail = gtk_ctree_insert_node(
2040                                                 clist, nodePerson, NULL,
2041                                                 text, FOLDER_SPACING,
2042                                                 atciMail->iconXpm, atciMail->maskXpm,
2043                                                 atciMail->iconXpmOpen, atciMail->maskXpmOpen,
2044                                                 FALSE, TRUE );
2045                                 gtk_ctree_node_set_row_data(clist, nodeEMail, email );
2046                         }
2047                         flgFirst = FALSE;
2048                         haveAddr = TRUE;
2049                 }
2050                 if( ! haveAddr ) {
2051                         /* Have name without EMail */
2052                         text[COL_NAME] = ADDRITEM_NAME(person);
2053                         text[COL_ADDRESS] = NULL;
2054                         text[COL_REMARKS] = NULL;
2055                         nodePerson = gtk_ctree_insert_node(
2056                                         clist, NULL, NULL,
2057                                         text, FOLDER_SPACING,
2058                                         atci->iconXpm, atci->maskXpm,
2059                                         atci->iconXpmOpen, atci->maskXpmOpen,
2060                                         FALSE, person->isOpened );
2061                         gtk_ctree_node_set_row_data(clist, nodePerson, person );
2062                 }
2063                 gtk_ctree_sort_node(GTK_CTREE(clist), NULL);
2064         }
2065         /* Free up the list */
2066         mgu_clear_list( items );
2067         g_list_free( items );
2068 }
2069
2070 static void addressbook_folder_load_group( GtkCTree *clist, ItemFolder *itemFolder ) {
2071         GList *items;
2072         AddressTypeControlItem *atci =  addrbookctl_lookup( ADDR_ITEM_GROUP );
2073
2074         /* Load any groups */
2075         if( ! atci ) return;
2076         items = addritem_folder_get_group_list( itemFolder );
2077         for( ; items != NULL; items = g_list_next( items ) ) {
2078                 GtkCTreeNode *nodeGroup = NULL;
2079                 gchar *text[N_COLS];
2080                 ItemGroup *group = items->data;
2081                 if( group == NULL ) continue;
2082                 text[COL_NAME] = ADDRITEM_NAME(group);
2083                 text[COL_ADDRESS] = NULL;
2084                 text[COL_REMARKS] = NULL;
2085                 nodeGroup = gtk_ctree_insert_node(clist, NULL, NULL,
2086                                       text, FOLDER_SPACING,
2087                                       atci->iconXpm, atci->maskXpm,
2088                                       atci->iconXpmOpen, atci->maskXpmOpen,
2089                                       FALSE, FALSE);
2090                 gtk_ctree_node_set_row_data(clist, nodeGroup, group );
2091                 gtk_ctree_sort_node(clist, NULL);
2092         }
2093         /* Free up the list */
2094         mgu_clear_list( items );
2095         g_list_free( items );
2096 }
2097
2098 /*
2099  * Load data sources into list.
2100  */
2101 static void addressbook_node_load_datasource( GtkCTree *clist, AddressObject *obj ) {
2102         AdapterInterface *adapter;
2103         AddressInterface *iface;
2104         AddressTypeControlItem *atci = NULL;
2105         /* AddressDataSource *ds; */
2106         GtkCTreeNode *newNode, *node;
2107         GtkCTreeRow *row;
2108         GtkCell *cell = NULL;
2109         gchar *text[N_COLS];
2110
2111         adapter = ADAPTER_INTERFACE(obj);
2112         if( adapter == NULL ) return;
2113         iface = adapter->interface;
2114         atci = adapter->atci;
2115         if( atci == NULL ) return;
2116
2117         /* Create nodes in list copying values for data sources in tree */
2118         row = GTK_CTREE_ROW( adapter->treeNode );
2119         if( row ) {
2120                 node = row->children;
2121                 while( node ) {
2122                         gpointer data = gtk_ctree_node_get_row_data( clist, node );
2123                         row = GTK_CTREE_ROW( node );
2124                         cell = ( ( GtkCListRow * )row )->cell;
2125                         text[COL_NAME] = cell->u.text;
2126                         text[COL_ADDRESS] = NULL;
2127                         text[COL_REMARKS] = NULL;
2128                         newNode = gtk_ctree_insert_node( clist, NULL, NULL,
2129                                       text, FOLDER_SPACING,
2130                                       atci->iconXpm, atci->maskXpm,
2131                                       atci->iconXpmOpen, atci->maskXpmOpen,
2132                                       FALSE, FALSE);
2133                         gtk_ctree_node_set_row_data( clist, newNode, data );
2134                         node = row->sibling;
2135
2136                 }
2137         }
2138         gtk_ctree_sort_node( clist, NULL );
2139 }
2140
2141 static AddressDataSource *addressbook_find_datasource( GtkCTreeNode *node ) {
2142         AddressDataSource *ds = NULL;
2143         AddressObject *ao;
2144
2145         g_return_val_if_fail(addrbook.ctree != NULL, NULL);
2146
2147         while( node ) {
2148                 if( GTK_CTREE_ROW(node)->level < 2 ) return NULL;
2149                 ao = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), node );
2150                 if( ao ) {
2151 /*                      printf( "ao->type = %d\n", ao->type ); */
2152                         if( ao->type == ADDR_DATASOURCE ) {
2153                                 AdapterDSource *ads = ADAPTER_DSOURCE(ao);
2154 /*                              printf( "found it\n" ); */
2155                                 ds = ads->dataSource;
2156                                 break;
2157                         }
2158                 }
2159                 node = GTK_CTREE_ROW(node)->parent;
2160         }
2161         return ds;
2162 }
2163
2164 /*
2165 * Load address list widget with children of specified object.
2166 * Enter: obj    Parent object to be loaded.
2167 */
2168 static void addressbook_set_clist( AddressObject *obj ) {
2169         GtkCTree *ctreelist = GTK_CTREE(addrbook.clist);
2170         GtkCList *clist = GTK_CLIST(addrbook.clist);
2171         AddressDataSource *ds = NULL;
2172         AdapterDSource *ads = NULL;
2173
2174         if( obj == NULL ) {
2175                 gtk_clist_clear(clist);
2176                 return;
2177         }
2178
2179         if( obj->type == ADDR_INTERFACE ) {
2180                 /* printf( "set_clist: loading datasource...\n" ); */
2181                 /* addressbook_node_load_datasource( clist, obj ); */
2182                 return;
2183         }
2184
2185         gtk_clist_freeze(clist);
2186         gtk_clist_clear(clist);
2187
2188         if( obj->type == ADDR_DATASOURCE ) {
2189                 ads = ADAPTER_DSOURCE(obj);
2190                 ds = ADAPTER_DSOURCE(obj)->dataSource;
2191                 if( ds ) {
2192                         /* Load root folder */
2193                         ItemFolder *rootFolder = NULL;
2194                         rootFolder = addrindex_ds_get_root_folder( ds );
2195                         addressbook_folder_load_person( ctreelist, addrindex_ds_get_root_folder( ds ) );
2196                         addressbook_folder_load_group( ctreelist, addrindex_ds_get_root_folder( ds ) );
2197                 }
2198         }
2199         else {
2200                 if( obj->type == ADDR_ITEM_GROUP ) {
2201                         /* Load groups */
2202                         ItemGroup *itemGroup = ADAPTER_GROUP(obj)->itemGroup;
2203                         addressbook_load_group( ctreelist, itemGroup );
2204                 }
2205                 else if( obj->type == ADDR_ITEM_FOLDER ) {
2206                         /* Load folders */
2207                         ItemFolder *itemFolder = ADAPTER_FOLDER(obj)->itemFolder;
2208                         addressbook_folder_load_person( ctreelist, itemFolder );
2209                         addressbook_folder_load_group( ctreelist, itemFolder );
2210                 }
2211         }
2212
2213         gtk_clist_sort(clist);
2214         gtk_clist_thaw(clist);
2215 }
2216
2217 /*
2218 * Free adaptor for specified node.
2219 */
2220 static void addressbook_free_adapter( GtkCTreeNode *node ) {
2221         AddressObject *ao;
2222
2223         g_return_if_fail(addrbook.ctree != NULL);
2224
2225         if( node ) {
2226                 if( GTK_CTREE_ROW(node)->level < 2 ) return;
2227                 ao = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), node );
2228                 if( ao == NULL ) return;
2229                 if( ao->type == ADDR_INTERFACE ) {
2230                         AdapterInterface *ai = ADAPTER_INTERFACE(ao);
2231                         addrbookctl_free_interface( ai );
2232                 }
2233                 else if( ao->type == ADDR_DATASOURCE ) {
2234                         AdapterDSource *ads = ADAPTER_DSOURCE(ao);
2235                         addrbookctl_free_datasource( ads );
2236                 }
2237                 else if( ao->type == ADDR_ITEM_FOLDER ) {
2238                         AdapterFolder *af = ADAPTER_FOLDER(ao);
2239                         addrbookctl_free_folder( af );
2240                 }
2241                 else if( ao->type == ADDR_ITEM_GROUP ) {
2242                         AdapterGroup *ag = ADAPTER_GROUP(ao);
2243                         addrbookctl_free_group( ag );
2244                 }
2245                 gtk_ctree_node_set_row_data( GTK_CTREE(addrbook.ctree), node, NULL );
2246         }
2247 }
2248
2249 /*
2250 * Free all children adapters.
2251 */
2252 static void addressbook_free_child_adapters( GtkCTreeNode *node ) {
2253         GtkCTreeNode *parent, *child;
2254         GtkCTreeRow *currRow;
2255
2256         if( node == NULL ) return;
2257         currRow = GTK_CTREE_ROW( node );
2258         if( currRow ) {
2259                 parent = currRow->parent;
2260                 child = currRow->children;
2261                 while( child ) {
2262                         addressbook_free_child_adapters( child );
2263                         addressbook_free_adapter( child );
2264                         currRow = GTK_CTREE_ROW( child );
2265                         child = currRow->sibling;
2266                 }
2267         }
2268 }
2269
2270 AdapterDSource *addressbook_create_ds_adapter( AddressDataSource *ds,
2271                                 AddressObjectType otype, gchar *name )
2272 {
2273         AdapterDSource *adapter = g_new0( AdapterDSource, 1 );
2274         ADDRESS_OBJECT(adapter)->type = ADDR_DATASOURCE;
2275         ADDRESS_OBJECT_NAME(adapter) = g_strdup( name );
2276         adapter->dataSource = ds;
2277         adapter->subType = otype;
2278         return adapter;
2279 }
2280
2281 void addressbook_ads_set_name( AdapterDSource *adapter, gchar *value ) {
2282         ADDRESS_OBJECT_NAME(adapter) = mgu_replace_string( ADDRESS_OBJECT_NAME(adapter), value );
2283 }
2284
2285 /*
2286  * Load tree from address index with the initial data.
2287  */
2288 static void addressbook_load_tree( void ) {
2289         GtkCTree *ctree = GTK_CTREE( addrbook.ctree );
2290         GList *nodeIf, *nodeDS;
2291         AdapterInterface *adapter;
2292         AddressInterface *iface;
2293         AddressTypeControlItem *atci;
2294         AddressDataSource *ds;
2295         AdapterDSource *ads;
2296         GtkCTreeNode *node, *newNode;
2297         gchar *name;
2298
2299         nodeIf = _addressInterfaceList_;
2300         while( nodeIf ) {
2301                 adapter = nodeIf->data;
2302                 node = adapter->treeNode;
2303                 iface = adapter->interface;
2304                 atci = adapter->atci;
2305                 if( iface ) {
2306                         if( iface->useInterface ) {
2307                                 /* Load data sources below interface node */
2308                                 nodeDS = iface->listSource;
2309                                 while( nodeDS ) {
2310                                         ds = nodeDS->data;
2311                                         newNode = NULL;
2312                                         name = addrindex_ds_get_name( ds );
2313                                         ads = addressbook_create_ds_adapter( ds, atci->objectType, name );
2314                                         newNode = addressbook_add_object( node, ADDRESS_OBJECT(ads) );
2315                                         nodeDS = g_list_next( nodeDS );
2316                                 }
2317                                 gtk_ctree_expand( ctree, node );
2318                         }
2319                 }
2320                 nodeIf = g_list_next( nodeIf );
2321         }
2322 }
2323
2324 /*
2325  * Convert the old address book to new format.
2326  */
2327 static gboolean addressbook_convert( AddressIndex *addrIndex ) {
2328         gboolean retVal = FALSE;
2329         gboolean errFlag = TRUE;
2330         gchar *msg = NULL;
2331
2332         /* Read old address book, performing conversion */
2333         debug_print( "Reading and converting old address book...\n" );
2334         addrindex_set_file_name( addrIndex, ADDRESSBOOK_OLD_FILE );
2335         addrindex_read_data( addrIndex );
2336         if( addrIndex->retVal == MGU_NO_FILE ) {
2337                 /* We do not have a file - new user */
2338                 debug_print( "New user... create new books...\n" );
2339                 addrindex_create_new_books( addrIndex );
2340                 if( addrIndex->retVal == MGU_SUCCESS ) {
2341                         /* Save index file */
2342                         addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2343                         addrindex_save_data( addrIndex );
2344                         if( addrIndex->retVal == MGU_SUCCESS ) {
2345                                 retVal = TRUE;
2346                                 errFlag = FALSE;
2347                         }
2348                         else {
2349                                 msg = _( "New user, could not save index file." );
2350                         }
2351                 }
2352                 else {
2353                         msg = _( "New user, could not save address book files." );
2354                 }
2355         }
2356         else {
2357                 /* We have an old file */
2358                 if( addrIndex->wasConverted ) {
2359                         /* Converted successfully - save address index */
2360                         addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2361                         addrindex_save_data( addrIndex );
2362                         if( addrIndex->retVal == MGU_SUCCESS ) {
2363                                 msg = _( "Old address book converted successfully." );
2364                                 retVal = TRUE;
2365                                 errFlag = FALSE;
2366                         }
2367                         else {
2368                                 msg = _("Old address book converted,\n"
2369                                         "could not save new address index file" );
2370                         }
2371                 }
2372                 else {
2373                         /* File conversion failed - just create new books */
2374                         debug_print( "File conversion failed... just create new books...\n" );
2375                         addrindex_create_new_books( addrIndex );
2376                         if( addrIndex->retVal == MGU_SUCCESS ) {
2377                                 /* Save index */
2378                                 addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2379                                 addrindex_save_data( addrIndex );
2380                                 if( addrIndex->retVal == MGU_SUCCESS ) {
2381                                         msg = _("Could not convert address book,\n"
2382                                                 "but created empty new address book files." );
2383                                         retVal = TRUE;
2384                                         errFlag = FALSE;
2385                                 }
2386                                 else {
2387                                         msg = _("Could not convert address book,\n"
2388                                                 "could not create new address book files." );
2389                                 }
2390                         }
2391                         else {
2392                                 msg = _("Could not convert address book\n"
2393                                         "and could not create new address book files." );
2394                         }
2395                 }
2396         }
2397         if( errFlag ) {
2398                 debug_print( "Error\n%s\n", msg );
2399                 alertpanel( _( "Addressbook conversion error" ), msg, _( "Close" ), NULL, NULL );
2400         }
2401         else if( msg ) {
2402                 debug_print( "Warning\n%s\n", msg );
2403                 alertpanel( _( "Addressbook conversion" ), msg, _( "Close" ), NULL, NULL );
2404         }
2405
2406         return retVal;
2407 }
2408
2409 void addressbook_read_file( void ) {
2410         AddressIndex *addrIndex = NULL;
2411
2412         debug_print( "Reading address index...\n" );
2413         if( _addressIndex_ ) {
2414                 debug_print( "address book already read!!!\n" );
2415                 return;
2416         }
2417
2418         addrIndex = addrindex_create_index();
2419
2420         /* Use new address book index. */
2421         addrindex_set_file_path( addrIndex, get_rc_dir() );
2422         addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2423         addrindex_read_data( addrIndex );
2424         if( addrIndex->retVal == MGU_NO_FILE ) {
2425                 /* Conversion required */
2426                 debug_print( "Converting...\n" );
2427                 if( addressbook_convert( addrIndex ) ) {
2428                         _addressIndex_ = addrIndex;
2429                 }
2430         }
2431         else if( addrIndex->retVal == MGU_SUCCESS ) {
2432                 _addressIndex_ = addrIndex;
2433         }
2434         else {
2435                 /* Error reading address book */
2436                 debug_print( "Could not read address index.\n" );
2437                 addrindex_print_index( addrIndex, stdout );
2438                 alertpanel( _( "Addressbook Error" ),
2439                             _( "Could not read address index" ),
2440                             _( "Close" ), NULL, NULL );
2441         }
2442         debug_print( "done.\n" );
2443 }
2444
2445 #if 0
2446 void addressbook_read_file_old( void ) {
2447         AddressIndex *addrIndex = NULL;
2448         gboolean errFlag = TRUE;
2449         gchar *msg = NULL;
2450
2451         if( _addressIndex_ ) {
2452                 debug_print( "address book already read!!!\n" );
2453                 return;
2454         }
2455
2456         addrIndex = addrindex_create_index();
2457
2458         /* Use use new address book. */
2459         /* addrindex_set_file_path( addrIndex, "/home/match/tmp/empty-dir" ); */
2460         addrindex_set_file_path( addrIndex, get_rc_dir() );
2461         addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2462
2463         debug_print( "Reading address index...\n" );
2464         addrindex_read_data( addrIndex );
2465         if( addrIndex->retVal == MGU_NO_FILE ) {
2466                 /* Read old address book, performing conversion */
2467                 debug_print( "Reading and converting old address book...\n" );
2468                 addrindex_set_file_name( addrIndex, ADDRESSBOOK_OLD_FILE );
2469                 addrindex_read_data( addrIndex );
2470                 if( addrIndex->retVal == MGU_NO_FILE ) {
2471                         /* We do not have a file - new user */
2472                         debug_print( "New user... create new books...\n" );
2473                         addrindex_create_new_books( addrIndex );
2474                         if( addrIndex->retVal == MGU_SUCCESS ) {
2475                                 /* Save index file */
2476                                 addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2477                                 addrindex_save_data( addrIndex );
2478                                 if( addrIndex->retVal == MGU_SUCCESS ) {
2479                                         errFlag = FALSE;
2480                                 }
2481                                 else {
2482                                         msg = g_strdup( _( "New user, could not save index file." ) );
2483                                 }
2484                         }
2485                         else {
2486                                 msg = g_strdup( _( "New user, could not save address book files." ) );
2487                         }
2488                 }
2489                 else {
2490                         /* We have an old file */
2491                         if( addrIndex->wasConverted ) {
2492                                 /* Converted successfully - save address index */
2493                                 addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2494                                 addrindex_save_data( addrIndex );
2495                                 if( addrIndex->retVal == MGU_SUCCESS ) {
2496                                         msg = g_strdup( _( "Old address book converted successfully." ) );
2497                                         errFlag = FALSE;
2498                                 }
2499                                 else {
2500                                         msg = g_strdup( _(
2501                                                 "Old address book converted, " \
2502                                                 "could not save new address index file" ) );
2503                                 }
2504                         }
2505                         else {
2506                                 /* File conversion failed - just create new books */
2507                                 debug_print( "File conversion failed... just create new books...\n" );
2508                                 addrindex_create_new_books( addrIndex );
2509                                 if( addrIndex->retVal == MGU_SUCCESS ) {
2510                                         /* Save index */
2511                                         addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
2512                                         addrindex_save_data( addrIndex );
2513                                         if( addrIndex->retVal == MGU_SUCCESS ) {
2514                                                 msg = g_strdup( _(
2515                                                         "Could not convert address book, " \
2516                                                         "but created empty new address book files." ) );
2517                                                 errFlag = FALSE;
2518                                         }
2519                                         else {
2520                                                 msg = g_strdup( _(
2521                                                         "Could not convert address book, " \
2522                                                         "could not create new address book files." ) );
2523                                         }
2524                                 }
2525                                 else {
2526                                         msg = g_strdup( _(
2527                                                 "Could not convert address book " \
2528                                                 "and could not create new address book files." ) );
2529                                 }
2530                         }
2531                 }
2532         }
2533         else if( addrIndex->retVal == MGU_SUCCESS ) {
2534                 errFlag = FALSE;
2535         }
2536         else {
2537                 debug_print( "Could not read address index.\n" );
2538                 addrindex_print_index( addrIndex, stdout );
2539                 msg = g_strdup( _( "Could not read address index" ) );
2540         }
2541         _addressIndex_ = addrIndex;
2542
2543         if( errFlag ) {
2544                 debug_print( "Error\n%s\n", msg );
2545                 alertpanel( _( "Addressbook Conversion Error" ), msg,
2546                             _( "Close" ), NULL, NULL );
2547         }
2548         else {
2549                 if( msg ) {
2550                         debug_print( "Warning\n%s\n", msg );
2551                         alertpanel( _( "Addressbook Conversion" ), msg,
2552                                     _( "Close" ), NULL, NULL );
2553                 }
2554         }
2555         if( msg ) g_free( msg );
2556         debug_print( "done.\n" );
2557 }
2558 #endif
2559
2560 /*
2561 * Add object into the address index tree widget.
2562 * Enter: node   Parent node.
2563 *        obj    Object to add.
2564 * Return: Node that was added, or NULL if object not added.
2565 */
2566 static GtkCTreeNode *addressbook_add_object(GtkCTreeNode *node,
2567                                             AddressObject *obj)
2568 {
2569         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2570         GtkCTreeNode *added;
2571         AddressObject *pobj;
2572         AddressObjectType otype;
2573         AddressTypeControlItem *atci = NULL;
2574
2575         g_return_val_if_fail(node != NULL, NULL);
2576         g_return_val_if_fail(obj  != NULL, NULL);
2577
2578         pobj = gtk_ctree_node_get_row_data(ctree, node);
2579         g_return_val_if_fail(pobj != NULL, NULL);
2580
2581         /* Determine object type to be displayed */
2582         if( obj->type == ADDR_DATASOURCE ) {
2583                 otype = ADAPTER_DSOURCE(obj)->subType;
2584         }
2585         else {
2586                 otype = obj->type;
2587         }
2588
2589         /* Handle any special conditions. */
2590         added = node;
2591         atci = addrbookctl_lookup( otype );
2592         if( atci ) {
2593                 if( atci->showInTree ) {
2594                         /* Add object to tree */
2595                         gchar **name;
2596                         name = &obj->name;
2597                         added = gtk_ctree_insert_node( ctree, node, NULL, name, FOLDER_SPACING,
2598                                 atci->iconXpm, atci->maskXpm, atci->iconXpmOpen, atci->maskXpmOpen,
2599                                 atci->treeLeaf, atci->treeExpand );
2600                         gtk_ctree_node_set_row_data(ctree, added, obj);
2601                 }
2602         }
2603
2604         gtk_ctree_sort_node(ctree, node);
2605
2606         return added;
2607 }
2608
2609 /*
2610 * Add group into the address index tree.
2611 * Enter: node      Parent node.
2612 *        ds        Data source.
2613 *        itemGroup Group to add.
2614 * Return: Inserted node.
2615 */
2616 static GtkCTreeNode *addressbook_node_add_group( GtkCTreeNode *node, AddressDataSource *ds, ItemGroup *itemGroup ) {
2617         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2618         GtkCTreeNode *newNode;
2619         AdapterGroup *adapter;
2620         AddressTypeControlItem *atci = NULL;
2621         gchar **name;
2622
2623         if( ds == NULL ) return NULL;
2624         if( node == NULL || itemGroup == NULL ) return NULL;
2625
2626         name = &itemGroup->obj.name;
2627
2628         atci = addrbookctl_lookup( ADDR_ITEM_GROUP );
2629
2630         adapter = g_new0( AdapterGroup, 1 );
2631         ADDRESS_OBJECT_TYPE(adapter) = ADDR_ITEM_GROUP;
2632         ADDRESS_OBJECT_NAME(adapter) = g_strdup( ADDRITEM_NAME(itemGroup) );
2633         adapter->itemGroup = itemGroup;
2634
2635         newNode = gtk_ctree_insert_node( ctree, node, NULL, name, FOLDER_SPACING,
2636                         atci->iconXpm, atci->maskXpm, atci->iconXpm, atci->maskXpm,
2637                         atci->treeLeaf, atci->treeExpand );
2638         gtk_ctree_node_set_row_data( ctree, newNode, adapter );
2639         gtk_ctree_sort_node( ctree, node );
2640         return newNode;
2641 }
2642
2643 /*
2644 * Add folder into the address index tree.
2645 * Enter: node       Parent node.
2646 *        ds         Data source.
2647 *        itemFolder Folder to add.
2648 *        otype      Object type to display.
2649 * Return: Inserted node.
2650 */
2651 static GtkCTreeNode *addressbook_node_add_folder(
2652                 GtkCTreeNode *node, AddressDataSource *ds, ItemFolder *itemFolder, AddressObjectType otype )
2653 {
2654         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2655         GtkCTreeNode *newNode = NULL;
2656         AdapterFolder *adapter;
2657         AddressTypeControlItem *atci = NULL;
2658         GList *listItems = NULL;
2659         gchar **name;
2660         ItemFolder *rootFolder;
2661
2662         if( ds == NULL ) return NULL;
2663         if( node == NULL || itemFolder == NULL ) return NULL;
2664
2665         /* Determine object type */
2666         atci = addrbookctl_lookup( otype );
2667         if( atci == NULL ) return NULL;
2668
2669         rootFolder = addrindex_ds_get_root_folder( ds );
2670         if( itemFolder == rootFolder ) {
2671                 newNode = node;
2672         }
2673         else {
2674                 name = &itemFolder->obj.name;
2675
2676                 adapter = g_new0( AdapterFolder, 1 );
2677                 ADDRESS_OBJECT_TYPE(adapter) = ADDR_ITEM_FOLDER;
2678                 ADDRESS_OBJECT_NAME(adapter) = g_strdup( ADDRITEM_NAME(itemFolder) );
2679                 adapter->itemFolder = itemFolder;
2680
2681                 newNode = gtk_ctree_insert_node( ctree, node, NULL, name, FOLDER_SPACING,
2682                                 atci->iconXpm, atci->maskXpm, atci->iconXpm, atci->maskXpm,
2683                                 atci->treeLeaf, atci->treeExpand );
2684                 gtk_ctree_node_set_row_data( ctree, newNode, adapter );
2685         }
2686
2687         listItems = itemFolder->listFolder;
2688         while( listItems ) {
2689                 ItemFolder *item = listItems->data;
2690                 addressbook_node_add_folder( newNode, ds, item, otype );
2691                 listItems = g_list_next( listItems );
2692         }
2693         listItems = itemFolder->listGroup;
2694         while( listItems ) {
2695                 ItemGroup *item = listItems->data;
2696                 addressbook_node_add_group( newNode, ds, item );
2697                 listItems = g_list_next( listItems );
2698         }
2699         gtk_ctree_sort_node( ctree, node );
2700         return newNode;
2701 }
2702
2703 static void addressbook_delete_object(AddressObject *obj) {
2704         AdapterDSource *ads = NULL;
2705         AddressDataSource *ds = NULL;
2706         if (!obj) return;
2707
2708         /* Remove data source. */
2709         /* printf( "Delete obj type : %d\n", obj->type ); */
2710
2711         ads = ADAPTER_DSOURCE(obj);
2712         if( ads == NULL ) return;
2713         ds = ads->dataSource;
2714         if( ds == NULL ) return;
2715
2716         /* Remove data source */
2717         if( addrindex_index_remove_datasource( _addressIndex_, ds ) ) {
2718                 addrindex_free_datasource( _addressIndex_, ds );
2719         }
2720         /* Free up Adapter object */
2721         g_free( ADAPTER_DSOURCE(obj) );
2722 }
2723
2724 void addressbook_export_to_file( void ) {
2725         if( _addressIndex_ ) {
2726                 /* Save all new address book data */
2727                 debug_print( "Saving address books...\n" );
2728                 addrindex_save_all_books( _addressIndex_ );
2729
2730                 debug_print( "Exporting addressbook to file...\n" );
2731                 addrindex_save_data( _addressIndex_ );
2732                 if( _addressIndex_->retVal != MGU_SUCCESS ) {
2733                         addrindex_print_index( _addressIndex_, stdout );
2734                 }
2735
2736                 /* Notify address completion of new data */
2737                 invalidate_address_completion();
2738         }
2739 }
2740
2741 static void key_pressed(GtkWidget *widget, GdkEventKey *event, gpointer data)
2742 {
2743         if (event && event->keyval == GDK_Escape)
2744                 addressbook_close();
2745 }
2746
2747 /*
2748 * Comparsion using names of AddressItem objects.
2749 */
2750 /*
2751 static gint addressbook_list_compare_func(GtkCList *clist,
2752                                           gconstpointer ptr1,
2753                                           gconstpointer ptr2)
2754 {
2755         AddressObject *obj1 = ((GtkCListRow *)ptr1)->data;
2756         AddressObject *obj2 = ((GtkCListRow *)ptr2)->data;
2757         gchar *name1 = NULL, *name2 = NULL;
2758         if( obj1 ) name1 = obj1->name;
2759         if( obj2 ) name2 = obj2->name;
2760         if( ! name1 ) return ( name2 != NULL );
2761         if( ! name2 ) return -1;
2762         return strcasecmp(name1, name2);
2763 }
2764 */
2765
2766 /*
2767 * Comparison using cell contents (text in first column).
2768 */
2769 static gint addressbook_list_compare_func( GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2 ) {
2770         GtkCell *cell1 = ((GtkCListRow *)ptr1)->cell;
2771         GtkCell *cell2 = ((GtkCListRow *)ptr2)->cell;
2772         gchar *name1 = NULL, *name2 = NULL;
2773         if( cell1 ) name1 = cell1->u.text;
2774         if( cell2 ) name2 = cell2->u.text;
2775         if( ! name1 ) return ( name2 != NULL );
2776         if( ! name2 ) return -1;
2777         return strcasecmp( name1, name2 );
2778 }
2779
2780 /* static */ 
2781 gint addressbook_obj_name_compare(gconstpointer a, gconstpointer b)
2782 {
2783         const AddressObject *obj = a;
2784         const gchar *name = b;
2785         AddressTypeControlItem *atci = NULL;
2786
2787         if (!obj || !name) return -1;
2788
2789         atci = addrbookctl_lookup( obj->type );
2790         if( ! atci ) return -1;
2791         if( ! obj->name ) return -1;
2792         return strcasecmp(obj->name, name);
2793 }
2794
2795 static void addressbook_book_show_message( AddressBookFile *abf ) {
2796         *addressbook_msgbuf = '\0';
2797         if( abf ) {
2798                 if( abf->retVal == MGU_SUCCESS ) {
2799                         sprintf( addressbook_msgbuf, "%s", abf->name );
2800                 }
2801                 else {
2802                         sprintf( addressbook_msgbuf, "%s: %s", abf->name, mgu_error2string( abf->retVal ) );
2803                 }
2804         }
2805         addressbook_status_show( addressbook_msgbuf );
2806 }
2807
2808 static void addressbook_new_book_cb( gpointer data, guint action, GtkWidget *widget ) {
2809         AdapterDSource *ads;
2810         AdapterInterface *adapter;
2811
2812         adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
2813         if( adapter == NULL ) return;
2814         if( addrbook.treeSelected == NULL ) return;
2815         if( addrbook.treeSelected != adapter->treeNode ) return;
2816         ads = addressbook_edit_book( _addressIndex_, NULL );
2817         if( ads ) {
2818                 addressbook_add_object( addrbook.treeSelected, ADDRESS_OBJECT(ads) );
2819                 if( addrbook.treeSelected == addrbook.opened ) {
2820                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
2821                 }
2822         }
2823 }
2824
2825 static void addressbook_new_vcard_cb( gpointer data, guint action, GtkWidget *widget ) {
2826         AdapterDSource *ads;
2827         AdapterInterface *adapter;
2828
2829         adapter = addrbookctl_find_interface( ADDR_IF_VCARD );
2830         if( adapter == NULL ) return;
2831         if( addrbook.treeSelected != adapter->treeNode ) return;
2832         ads = addressbook_edit_vcard( _addressIndex_, NULL );
2833         if( ads ) {
2834                 addressbook_add_object( addrbook.treeSelected, ADDRESS_OBJECT(ads) );
2835                 if( addrbook.treeSelected == addrbook.opened ) {
2836                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
2837                 }
2838         }
2839 }
2840
2841 static void addressbook_vcard_show_message( VCardFile *vcf ) {
2842         *addressbook_msgbuf = '\0';
2843         if( vcf ) {
2844                 if( vcf->retVal == MGU_SUCCESS ) {
2845                         sprintf( addressbook_msgbuf, "%s", vcf->name );
2846                 }
2847                 else {
2848                         sprintf( addressbook_msgbuf, "%s: %s", vcf->name, mgu_error2string( vcf->retVal ) );
2849                 }
2850         }
2851         addressbook_status_show( addressbook_msgbuf );
2852 }
2853
2854 #ifdef USE_JPILOT
2855 static void addressbook_new_jpilot_cb( gpointer data, guint action, GtkWidget *widget ) {
2856         AdapterDSource *ads;
2857         AdapterInterface *adapter;
2858         AddressInterface *iface;
2859
2860         adapter = addrbookctl_find_interface( ADDR_IF_JPILOT );
2861         if( adapter == NULL ) return;
2862         if( addrbook.treeSelected != adapter->treeNode ) return;
2863         iface = adapter->interface;
2864         if( ! iface->haveLibrary ) return;
2865         ads = addressbook_edit_jpilot( _addressIndex_, NULL );
2866         if( ads ) {
2867                 addressbook_add_object( addrbook.treeSelected, ADDRESS_OBJECT(ads) );
2868                 if( addrbook.treeSelected == addrbook.opened ) {
2869                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
2870                 }
2871         }
2872 }
2873
2874 static void addressbook_jpilot_show_message( JPilotFile *jpf ) {
2875         *addressbook_msgbuf = '\0';
2876         if( jpf ) {
2877                 if( jpf->retVal == MGU_SUCCESS ) {
2878                         sprintf( addressbook_msgbuf, "%s", jpf->name );
2879                 }
2880                 else {
2881                         sprintf( addressbook_msgbuf, "%s: %s", jpf->name, mgu_error2string( jpf->retVal ) );
2882                 }
2883         }
2884         addressbook_status_show( addressbook_msgbuf );
2885 }
2886
2887 #endif
2888
2889 #ifdef USE_LDAP
2890 static void addressbook_new_ldap_cb( gpointer data, guint action, GtkWidget *widget ) {
2891         AdapterDSource *ads;
2892         AdapterInterface *adapter;
2893         AddressInterface *iface;
2894
2895         adapter = addrbookctl_find_interface( ADDR_IF_LDAP );
2896         if( adapter == NULL ) return;
2897         if( addrbook.treeSelected != adapter->treeNode ) return;
2898         iface = adapter->interface;
2899         if( ! iface->haveLibrary ) return;
2900         ads = addressbook_edit_ldap( _addressIndex_, NULL );
2901         if( ads ) {
2902                 addressbook_add_object( addrbook.treeSelected, ADDRESS_OBJECT(ads) );
2903                 if( addrbook.treeSelected == addrbook.opened ) {
2904                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), addrbook.opened );
2905                 }
2906         }
2907 }
2908
2909 static void addressbook_ldap_show_message( SyldapServer *svr ) {
2910         *addressbook_msgbuf = '\0';
2911         if( svr ) {
2912                 if( svr->busyFlag ) {
2913                         sprintf( addressbook_msgbuf, "%s: %s", svr->name, ADDRESSBOOK_LDAP_BUSYMSG );
2914                 }
2915                 else {
2916                         if( svr->retVal == MGU_SUCCESS ) {
2917                                 sprintf( addressbook_msgbuf, "%s", svr->name );
2918                         }
2919                         else {
2920                                 sprintf( addressbook_msgbuf, "%s: %s", svr->name, mgu_error2string( svr->retVal ) );
2921                         }
2922                 }
2923         }
2924         addressbook_status_show( addressbook_msgbuf );
2925 }
2926
2927 static void ldapsearch_callback( SyldapServer *sls ) {
2928         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2929         AddressObject *obj;
2930         AdapterDSource *ads = NULL;
2931         AddressDataSource *ds = NULL;
2932         AddressInterface *iface = NULL;
2933
2934         if( sls == NULL ) return;
2935         if( ! addrbook.treeSelected ) return;
2936         if( GTK_CTREE_ROW( addrbook.treeSelected )->level == 1 ) return;
2937
2938         obj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
2939         if( obj == NULL ) return;
2940         if( obj->type == ADDR_DATASOURCE ) {
2941                 ads = ADAPTER_DSOURCE(obj);
2942                 if( ads->subType == ADDR_LDAP ) {
2943                         SyldapServer *server;
2944
2945                         ds = ads->dataSource;
2946                         if( ds == NULL ) return;
2947                         iface = ds->interface;
2948                         if( ! iface->haveLibrary ) return;
2949                         server = ds->rawDataSource;
2950                         if( server == sls ) {
2951                                 /* Read from cache */
2952                                 gtk_widget_show_all(addrbook.window);
2953                                 addressbook_set_clist( obj );
2954                                 addressbook_ldap_show_message( sls );
2955                                 gtk_widget_show_all(addrbook.window);
2956                                 gtk_entry_set_text( GTK_ENTRY(addrbook.entry), "" );
2957                         }
2958                 }
2959         }
2960 }
2961 #endif
2962
2963 /*
2964  * Lookup button handler.
2965  */
2966 static void addressbook_lup_clicked( GtkButton *button, gpointer data ) {
2967         GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2968         AddressObject *obj;
2969 #ifdef USE_LDAP
2970         AdapterDSource *ads = NULL;
2971         AddressDataSource *ds = NULL;
2972         AddressInterface *iface = NULL;
2973 #endif
2974         gchar *sLookup;
2975
2976         sLookup = gtk_editable_get_chars( GTK_EDITABLE(addrbook.entry), 0, -1 );
2977         g_strchomp( sLookup );
2978
2979         if( ! addrbook.treeSelected ) return;
2980         if( GTK_CTREE_ROW( addrbook.treeSelected )->level == 1 ) return;
2981
2982         obj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
2983         if( obj == NULL ) return;
2984
2985 #ifdef USE_LDAP
2986         if( obj->type == ADDR_DATASOURCE ) {
2987                 ads = ADAPTER_DSOURCE(obj);
2988                 if( ads->subType == ADDR_LDAP ) {
2989                         SyldapServer *server;
2990
2991                         ds = ads->dataSource;
2992                         if( ds == NULL ) return;
2993                         iface = ds->interface;
2994                         if( ! iface->haveLibrary ) return;
2995                         server = ds->rawDataSource;
2996                         if( server ) {
2997                                 syldap_cancel_read( server );
2998                                 if( *sLookup == '\0' || strlen( sLookup ) < 1 ) return;
2999                                 syldap_set_search_value( server, sLookup );
3000                                 syldap_set_callback( server, ldapsearch_callback );
3001                                 syldap_read_data_th( server );
3002                                 addressbook_ldap_show_message( server );
3003                         }
3004                 }
3005         }
3006 #endif
3007
3008 }
3009
3010 /* **********************************************************************
3011 * Build lookup tables.
3012 * ***********************************************************************
3013 */
3014
3015 /*
3016 * Build table that controls the rendering of object types.
3017 */
3018 void addrbookctl_build_map( GtkWidget *window ) {
3019         AddressTypeControlItem *atci;
3020
3021         /* Build icons */
3022         PIXMAP_CREATE(window, folderxpm, folderxpmmask, dir_close_xpm);
3023         PIXMAP_CREATE(window, folderopenxpm, folderopenxpmmask, dir_open_xpm);
3024         PIXMAP_CREATE(window, groupxpm, groupxpmmask, group_xpm);
3025         PIXMAP_CREATE(window, vcardxpm, vcardxpmmask, vcard_xpm);
3026         PIXMAP_CREATE(window, bookxpm, bookxpmmask, book_xpm);
3027         PIXMAP_CREATE(window, addressxpm, addressxpmmask, address_xpm);
3028         PIXMAP_CREATE(window, jpilotxpm, jpilotxpmmask, jpilot_xpm);
3029         PIXMAP_CREATE(window, categoryxpm, categoryxpmmask, category_xpm);
3030         PIXMAP_CREATE(window, ldapxpm, ldapxpmmask, ldap_xpm);
3031
3032         _addressBookTypeHash_ = g_hash_table_new( g_int_hash, g_int_equal );
3033         _addressBookTypeList_ = NULL;
3034
3035         /* Interface */
3036         atci = g_new0( AddressTypeControlItem, 1 );
3037         atci->objectType = ADDR_INTERFACE;
3038         atci->interfaceType = ADDR_IF_NONE;
3039         atci->showInTree = TRUE;
3040         atci->treeExpand = TRUE;
3041         atci->treeLeaf = FALSE;
3042         atci->displayName = _( "Interface" );
3043         atci->iconXpm = folderxpm;
3044         atci->maskXpm = folderxpmmask;
3045         atci->iconXpmOpen = folderopenxpm;
3046         atci->maskXpmOpen = folderopenxpmmask;
3047         atci->menuCommand = NULL;
3048         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3049         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3050
3051         /* Address book */
3052         atci = g_new0( AddressTypeControlItem, 1 );
3053         atci->objectType = ADDR_BOOK;
3054         atci->interfaceType = ADDR_IF_BOOK;
3055         atci->showInTree = TRUE;
3056         atci->treeExpand = TRUE;
3057         atci->treeLeaf = FALSE;
3058         atci->displayName = _( "Address Book" );
3059         atci->iconXpm = bookxpm;
3060         atci->maskXpm = bookxpmmask;
3061         atci->iconXpmOpen = bookxpm;
3062         atci->maskXpmOpen = bookxpmmask;
3063         atci->menuCommand = "/File/New Book";
3064         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3065         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3066
3067         /* Item person */
3068         atci = g_new0( AddressTypeControlItem, 1 );
3069         atci->objectType = ADDR_ITEM_PERSON;
3070         atci->interfaceType = ADDR_IF_NONE;
3071         atci->showInTree = FALSE;
3072         atci->treeExpand = FALSE;
3073         atci->treeLeaf = FALSE;
3074         atci->displayName = _( "Person" );
3075         atci->iconXpm = NULL;
3076         atci->maskXpm = NULL;
3077         atci->iconXpmOpen = NULL;
3078         atci->maskXpmOpen = NULL;
3079         atci->menuCommand = NULL;
3080         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3081         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3082
3083         /* Item email */
3084         atci = g_new0( AddressTypeControlItem, 1 );
3085         atci->objectType = ADDR_ITEM_EMAIL;
3086         atci->interfaceType = ADDR_IF_NONE;
3087         atci->showInTree = FALSE;
3088         atci->treeExpand = FALSE;
3089         atci->treeLeaf = TRUE;
3090         atci->displayName = _( "EMail Address" );
3091         atci->iconXpm = addressxpm;
3092         atci->maskXpm = addressxpmmask;
3093         atci->iconXpmOpen = addressxpm;
3094         atci->maskXpmOpen = addressxpmmask;
3095         atci->menuCommand = NULL;
3096         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3097         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3098
3099         /* Item group */
3100         atci = g_new0( AddressTypeControlItem, 1 );
3101         atci->objectType = ADDR_ITEM_GROUP;
3102         atci->interfaceType = ADDR_IF_BOOK;
3103         atci->showInTree = TRUE;
3104         atci->treeExpand = FALSE;
3105         atci->treeLeaf = FALSE;
3106         atci->displayName = _( "Group" );
3107         atci->iconXpm = groupxpm;
3108         atci->maskXpm = groupxpmmask;
3109         atci->iconXpmOpen = groupxpm;
3110         atci->maskXpmOpen = groupxpmmask;
3111         atci->menuCommand = NULL;
3112         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3113         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3114
3115         /* Item folder */
3116         atci = g_new0( AddressTypeControlItem, 1 );
3117         atci->objectType = ADDR_ITEM_FOLDER;
3118         atci->interfaceType = ADDR_IF_BOOK;
3119         atci->showInTree = TRUE;
3120         atci->treeExpand = FALSE;
3121         atci->treeLeaf = FALSE;
3122         atci->displayName = _( "Folder" );
3123         atci->iconXpm = folderxpm;
3124         atci->maskXpm = folderxpmmask;
3125         atci->iconXpmOpen = folderopenxpm;
3126         atci->maskXpmOpen = folderopenxpmmask;
3127         atci->menuCommand = NULL;
3128         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3129         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3130
3131         /* V-Card */
3132         atci = g_new0( AddressTypeControlItem, 1 );
3133         atci->objectType = ADDR_VCARD;
3134         atci->interfaceType = ADDR_IF_VCARD;
3135         atci->showInTree = TRUE;
3136         atci->treeExpand = TRUE;
3137         atci->treeLeaf = TRUE;
3138         atci->displayName = _( "V-Card" );
3139         atci->iconXpm = vcardxpm;
3140         atci->maskXpm = vcardxpmmask;
3141         atci->iconXpmOpen = vcardxpm;
3142         atci->maskXpmOpen = vcardxpmmask;
3143         atci->menuCommand = "/File/New V-Card";
3144         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3145         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3146
3147         /* J-Pilot */
3148         atci = g_new0( AddressTypeControlItem, 1 );
3149         atci->objectType = ADDR_JPILOT;
3150         atci->interfaceType = ADDR_IF_JPILOT;
3151         atci->showInTree = TRUE;
3152         atci->treeExpand = TRUE;
3153         atci->treeLeaf = FALSE;
3154         atci->displayName = _( "J-Pilot" );
3155         atci->iconXpm = jpilotxpm;
3156         atci->maskXpm = jpilotxpmmask;
3157         atci->iconXpmOpen = jpilotxpm;
3158         atci->maskXpmOpen = jpilotxpmmask;
3159         atci->menuCommand = "/File/New J-Pilot";
3160         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3161         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3162
3163         /* Category */
3164         atci = g_new0( AddressTypeControlItem, 1 );
3165         atci->objectType = ADDR_CATEGORY;
3166         atci->interfaceType = ADDR_IF_JPILOT;
3167         atci->showInTree = TRUE;
3168         atci->treeExpand = TRUE;
3169         atci->treeLeaf = TRUE;
3170         atci->displayName = _( "J-Pilot" );
3171         atci->iconXpm = categoryxpm;
3172         atci->maskXpm = categoryxpmmask;
3173         atci->iconXpmOpen = categoryxpm;
3174         atci->maskXpmOpen = categoryxpmmask;
3175         atci->menuCommand = NULL;
3176         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3177         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3178
3179         /* LDAP Server */
3180         atci = g_new0( AddressTypeControlItem, 1 );
3181         atci->objectType = ADDR_LDAP;
3182         atci->interfaceType = ADDR_IF_LDAP;
3183         atci->showInTree = TRUE;
3184         atci->treeExpand = TRUE;
3185         atci->treeLeaf = TRUE;
3186         atci->displayName = _( "LDAP Server" );
3187         atci->iconXpm = ldapxpm;
3188         atci->maskXpm = ldapxpmmask;
3189         atci->iconXpmOpen = ldapxpm;
3190         atci->maskXpmOpen = ldapxpmmask;
3191         atci->menuCommand = "/File/New Server";
3192         g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
3193         _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
3194
3195 }
3196
3197 /*
3198 * Search for specified object type.
3199 */
3200 AddressTypeControlItem *addrbookctl_lookup( gint ot ) {
3201         gint objType = ot;
3202         return ( AddressTypeControlItem * ) g_hash_table_lookup( _addressBookTypeHash_, &objType );
3203 }
3204
3205 /*
3206 * Search for specified interface type.
3207 */
3208 AddressTypeControlItem *addrbookctl_lookup_iface( AddressIfType ifType ) {
3209         GList *node = _addressBookTypeList_;
3210         while( node ) {
3211                 AddressTypeControlItem *atci = node->data;
3212                 if( atci->interfaceType == ifType ) return atci;
3213                 node = g_list_next( node );
3214         }
3215         return NULL;
3216 }
3217
3218 static void addrbookctl_free_address( AddressObject *obj ) {
3219         g_free( obj->name );
3220         obj->type = ADDR_NONE;
3221         obj->name = NULL;
3222 }
3223
3224 static void addrbookctl_free_interface( AdapterInterface *adapter ) {
3225         addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
3226         adapter->interface = NULL;
3227         adapter->interfaceType = ADDR_IF_NONE;
3228         adapter->atci = NULL;
3229         adapter->enabled = FALSE;
3230         adapter->haveLibrary = FALSE;
3231         adapter->treeNode = NULL;
3232         g_free( adapter );
3233 }
3234
3235 static void addrbookctl_free_datasource( AdapterDSource *adapter ) {
3236         addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
3237         adapter->dataSource = NULL;
3238         adapter->subType = ADDR_NONE;
3239         g_free( adapter );
3240 }
3241
3242 static void addrbookctl_free_folder( AdapterFolder *adapter ) {
3243         addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
3244         adapter->itemFolder = NULL;
3245         g_free( adapter );
3246 }
3247
3248 static void addrbookctl_free_group( AdapterGroup *adapter ) {
3249         addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
3250         adapter->itemGroup = NULL;
3251         g_free( adapter );
3252 }
3253
3254 /*
3255  * Build GUI interface list.
3256  */
3257 void addrbookctl_build_iflist() {
3258         AddressTypeControlItem *atci;
3259         AdapterInterface *adapter;
3260         GList *list = NULL;
3261
3262         if( _addressIndex_ == NULL ) {
3263                 _addressIndex_ = addrindex_create_index();
3264         }
3265         _addressInterfaceList_ = NULL;
3266         list = addrindex_get_interface_list( _addressIndex_ );
3267         while( list ) {
3268                 AddressInterface *interface = list->data;
3269                 atci = addrbookctl_lookup_iface( interface->type );
3270                 if( atci ) {
3271                         adapter = g_new0( AdapterInterface, 1 );
3272                         adapter->interfaceType = interface->type;
3273                         adapter->atci = atci;
3274                         adapter->interface = interface;
3275                         adapter->treeNode = NULL;
3276                         adapter->enabled = TRUE;
3277                         adapter->haveLibrary = interface->haveLibrary;
3278                         ADDRESS_OBJECT(adapter)->type = ADDR_INTERFACE;
3279                         ADDRESS_OBJECT_NAME(adapter) = g_strdup( atci->displayName );
3280                         _addressInterfaceList_ = g_list_append( _addressInterfaceList_, adapter );
3281                 }
3282                 list = g_list_next( list );
3283         }
3284 }
3285
3286 void addrbookctl_free_selection( GList *list ) {
3287         GList *node = list;
3288         while( node ) {
3289                 AdapterInterface *adapter = node->data;
3290                 adapter = NULL;
3291                 node = g_list_next( node );
3292         }
3293         g_list_free( list );
3294 }
3295
3296 /*
3297 * Find GUI interface type specified interface type.
3298 * Return: Interface item, or NULL if not found.
3299 */
3300 AdapterInterface *addrbookctl_find_interface( AddressIfType ifType ) {
3301         GList *node = _addressInterfaceList_;
3302         while( node ) {
3303                 AdapterInterface *adapter = node->data;
3304                 if( adapter->interfaceType == ifType ) return adapter;
3305                 node = g_list_next( node );
3306         }
3307         return NULL;
3308 }
3309
3310 /*
3311 * Build interface list selection.
3312 */
3313 void addrbookctl_build_ifselect() {
3314         GList *newList = NULL;
3315         gchar *selectStr;
3316         gchar **splitStr;
3317         gint ifType;
3318         gint i;
3319         gchar *endptr = NULL;
3320         gboolean enabled;
3321         AdapterInterface *adapter;
3322         /* GList *node; */
3323
3324         selectStr = g_strdup( ADDRESSBOOK_IFACE_SELECTION );
3325
3326         /* Parse string */
3327         splitStr = g_strsplit( selectStr, ",", -1 );
3328         for( i = 0; i < ADDRESSBOOK_MAX_IFACE; i++ ) {
3329                 if( splitStr[i] ) {
3330                         /* printf( "%d : %s\n", i, splitStr[i] ); */
3331                         ifType = strtol( splitStr[i], &endptr, 10 );
3332                         enabled = TRUE;
3333                         if( *endptr ) {
3334                                 if( strcmp( endptr, "/n" ) == 0 ) {
3335                                         enabled = FALSE;
3336                                 }
3337                         }
3338                         /* printf( "\t%d : %s\n", ifType, enabled ? "yes" : "no" ); */
3339                         adapter = addrbookctl_find_interface( ifType );
3340                         if( adapter ) {
3341                                 newList = g_list_append( newList, adapter );
3342                         }
3343                 }
3344                 else {
3345                         break;
3346                 }
3347         }
3348         /* printf( "i=%d\n", i ); */
3349         g_strfreev( splitStr );
3350         g_free( selectStr );
3351
3352         /* Replace existing list */
3353         mgu_clear_list( _addressIFaceSelection_ );
3354         g_list_free( _addressIFaceSelection_ );
3355         _addressIFaceSelection_ = newList;
3356         newList = NULL;
3357
3358 }
3359
3360 /* **********************************************************************
3361 * Add sender to address book.
3362 * ***********************************************************************
3363 */
3364
3365 /*
3366  * This function is used by the Add sender to address book function.
3367  */
3368 gboolean addressbook_add_contact( const gchar *name, const gchar *address, const gchar *remarks ) {
3369         debug_print( "addressbook_add_contact: name/address: %s - %s\n", name, address );
3370         if( addressadd_selection( _addressIndex_, name, address, remarks ) ) {
3371                 debug_print( "addressbook_add_contact - added\n" );
3372                 addressbook_refresh();
3373         }
3374         return TRUE;
3375 }
3376
3377 /* **********************************************************************
3378 * Address completion support.
3379 * ***********************************************************************
3380 */
3381
3382 /*
3383 * This function is used by the address completion function to load
3384 * addresses.
3385 * Enter: callBackFunc Function to be called when an address is
3386 *                     to be loaded.
3387 * Return: TRUE if data loaded, FALSE if address index not loaded.
3388 */
3389 gboolean addressbook_load_completion( gint (*callBackFunc) ( const gchar *, const gchar * ) ) {
3390         /* AddressInterface *interface; */
3391         AddressDataSource *ds;
3392         GList *nodeIf, *nodeDS;
3393         GList *listP, *nodeP;
3394         GList *nodeM;
3395         gchar *sName, *sAddress, *sAlias, *sFriendly;
3396
3397         debug_print( "addressbook_load_completion\n" );
3398
3399         if( _addressIndex_ == NULL ) return FALSE;
3400
3401         nodeIf = addrindex_get_interface_list( _addressIndex_ );
3402         while( nodeIf ) {
3403                 AddressInterface *interface = nodeIf->data;
3404                 nodeDS = interface->listSource;
3405                 while( nodeDS ) {
3406                         ds = nodeDS->data;
3407
3408                         /* Read address book */
3409                         if( ! addrindex_ds_get_read_flag( ds ) ) {
3410                                 addrindex_ds_read_data( ds );
3411                         }
3412
3413                         /* Get all persons */
3414                         listP = addrindex_ds_get_all_persons( ds );
3415                         nodeP = listP;
3416                         while( nodeP ) {
3417                                 ItemPerson *person = nodeP->data;
3418                                 nodeM = person->listEMail;
3419
3420                                 /* Figure out name to use */
3421                                 sName = person->nickName;
3422                                 if( sName == NULL || *sName == '\0' ) {
3423                                         sName = ADDRITEM_NAME(person);
3424                                 }
3425
3426                                 /* Process each E-Mail address */
3427                                 while( nodeM ) {
3428                                         ItemEMail *email = nodeM->data;
3429                                         /* Have mail */
3430                                         sFriendly = sName;
3431                                         sAddress = email->address;
3432                                         if( sAddress || *sAddress != '\0' ) {
3433                                                 sAlias = ADDRITEM_NAME(email);
3434                                                 if( sAlias && *sAlias != '\0' ) {
3435                                                         sFriendly = sAlias;
3436                                                 }
3437                                                 ( callBackFunc ) ( sFriendly, sAddress );
3438                                         }
3439
3440                                         nodeM = g_list_next( nodeM );
3441                                 }
3442                                 nodeP = g_list_next( nodeP );
3443                         }
3444                         /* Free up the list */
3445                         g_list_free( listP );
3446
3447                         nodeDS = g_list_next( nodeDS );
3448                 }
3449                 nodeIf = g_list_next( nodeIf );
3450         }
3451         debug_print( "addressbook_load_completion... done\n" );
3452
3453         return TRUE;
3454 }
3455
3456 /* **********************************************************************
3457 * Address Import.
3458 * ***********************************************************************
3459 */
3460
3461 /*
3462 * Import LDIF file.
3463 */
3464 static void addressbook_import_ldif_cb() {
3465         AddressDataSource *ds = NULL;
3466         AdapterDSource *ads = NULL;
3467         AddressBookFile *abf = NULL;
3468         AdapterInterface *adapter;
3469         GtkCTreeNode *newNode;
3470
3471         adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
3472         if( adapter ) {
3473                 if( adapter->treeNode ) {
3474                         abf = addressbook_imp_ldif( _addressIndex_ );
3475                         if( abf ) {
3476                                 ds = addrindex_index_add_datasource( _addressIndex_, ADDR_IF_BOOK, abf );
3477                                 ads = addressbook_create_ds_adapter( ds, ADDR_BOOK, NULL );
3478                                 addressbook_ads_set_name( ads, abf->name );
3479                                 newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
3480                                 if( newNode ) {
3481                                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), newNode );
3482                                         addrbook.treeSelected = newNode;
3483                                 }
3484
3485                                 /* Notify address completion */
3486                                 invalidate_address_completion();
3487                         }
3488                 }
3489         }
3490
3491 }
3492
3493 /*
3494 * Import MUTT file.
3495 */
3496 static void addressbook_import_mutt_cb() {
3497         AddressDataSource *ds = NULL;
3498         AdapterDSource *ads = NULL;
3499         AddressBookFile *abf = NULL;
3500         AdapterInterface *adapter;
3501         GtkCTreeNode *newNode;
3502
3503         adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
3504         if( adapter ) {
3505                 if( adapter->treeNode ) {
3506                         abf = addressbook_imp_mutt( _addressIndex_ );
3507                         if( abf ) {
3508                                 ds = addrindex_index_add_datasource( _addressIndex_, ADDR_IF_BOOK, abf );
3509                                 ads = addressbook_create_ds_adapter( ds, ADDR_BOOK, NULL );
3510                                 addressbook_ads_set_name( ads, abf->name );
3511                                 newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
3512                                 if( newNode ) {
3513                                         gtk_ctree_select( GTK_CTREE(addrbook.ctree), newNode );
3514                                         addrbook.treeSelected = newNode;
3515                                 }
3516
3517                                 /* Notify address completion */
3518                                 invalidate_address_completion();
3519                         }
3520                 }
3521         }
3522
3523 }
3524
3525 /*
3526 * End of Source.
3527 */
3528