2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2006 Hiroyuki Yamamoto and the Sylpheed-Claws team
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.
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.
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
27 #include <glib/gi18n.h>
28 #include <gdk/gdkkeysyms.h>
29 #include <gtk/gtkwindow.h>
30 #include <gtk/gtksignal.h>
31 #include <gtk/gtkvbox.h>
32 #include <gtk/gtkscrolledwindow.h>
33 #include <gtk/gtkhpaned.h>
34 #include <gtk/gtkhbox.h>
35 #include <gtk/gtklabel.h>
36 #include <gtk/gtkentry.h>
37 #include <gtk/gtkctree.h>
38 #include <gtk/gtkclist.h>
39 #include <gtk/gtktable.h>
40 #include <gtk/gtkhbbox.h>
41 #include <gtk/gtkbutton.h>
42 #include <gtk/gtkmenu.h>
43 #include <gtk/gtkmenuitem.h>
44 #include <gtk/gtkitemfactory.h>
49 #include "addressbook.h"
50 #include "manage_window.h"
51 #include "prefs_common.h"
52 #include "alertpanel.h"
53 #include "inputdialog.h"
55 #include "stock_pixmap.h"
57 #include "prefs_gtk.h"
63 #include "addr_compl.h"
66 #include "addressitem.h"
68 #include "addrcache.h"
70 #include "addrindex.h"
71 #include "addressadd.h"
72 #include "addressbook_foldersel.h"
74 #include "editvcard.h"
75 #include "editgroup.h"
76 #include "editaddress.h"
78 #include "importldif.h"
79 #include "importmutt.h"
80 #include "importpine.h"
85 #include "editjpilot.h"
90 #include "ldapserver.h"
93 #define ADDRESSBOOK_LDAP_BUSYMSG "Busy"
96 #include "addrquery.h"
97 #include "addrselect.h"
99 #include "addrgather.h"
100 #include "adbookbase.h"
101 #include "exphtmldlg.h"
102 #include "expldifdlg.h"
103 #include "browseldap.h"
109 } AddressIndexColumns;
117 } AddressListColumns;
120 AddressBookFile *book;
128 AddressDataSource *book;
132 static gchar *list_titles[] = { N_("Name"),
136 #define COL_NAME_WIDTH 164
137 #define COL_ADDRESS_WIDTH 156
139 #define COL_FOLDER_WIDTH 170
140 #define ADDRESSBOOK_WIDTH 640
141 #define ADDRESSBOOK_HEIGHT 360
143 #define ADDRESSBOOK_MSGBUF_SIZE 2048
145 static GdkPixmap *folderxpm;
146 static GdkBitmap *folderxpmmask;
147 static GdkPixmap *folderopenxpm;
148 static GdkBitmap *folderopenxpmmask;
149 static GdkPixmap *groupxpm;
150 static GdkBitmap *groupxpmmask;
151 static GdkPixmap *interfacexpm;
152 static GdkBitmap *interfacexpmmask;
153 static GdkPixmap *bookxpm;
154 static GdkBitmap *bookxpmmask;
155 static GdkPixmap *addressxpm;
156 static GdkBitmap *addressxpmmask;
157 static GdkPixmap *vcardxpm;
158 static GdkBitmap *vcardxpmmask;
159 static GdkPixmap *jpilotxpm;
160 static GdkBitmap *jpilotxpmmask;
161 static GdkPixmap *categoryxpm;
162 static GdkBitmap *categoryxpmmask;
163 static GdkPixmap *ldapxpm;
164 static GdkBitmap *ldapxpmmask;
165 static GdkPixmap *addrsearchxpm;
166 static GdkPixmap *addrsearchxpmmask;
169 static gchar addressbook_msgbuf[ ADDRESSBOOK_MSGBUF_SIZE ];
171 /* Address list selection */
172 static AddrSelectList *_addressSelect_ = NULL;
173 static AddressClipboard *_clipBoard_ = NULL;
175 /* Address index file and interfaces */
176 static AddressIndex *_addressIndex_ = NULL;
177 static GList *_addressInterfaceList_ = NULL;
178 static GList *_addressIFaceSelection_ = NULL;
179 #define ADDRESSBOOK_IFACE_SELECTION "1/y,3/y,4/y,2/n"
181 static AddressBook_win addrbook;
183 static GHashTable *_addressBookTypeHash_ = NULL;
184 static GList *_addressBookTypeList_ = NULL;
186 static void addressbook_create (void);
187 static gint addressbook_close (void);
188 static void addressbook_button_set_sensitive (void);
190 /* callback functions */
191 static void addressbook_del_clicked (GtkButton *button,
193 static void addressbook_reg_clicked (GtkButton *button,
195 static void addressbook_to_clicked (GtkButton *button,
197 static void addressbook_lup_clicked (GtkButton *button,
199 static void addressbook_close_clicked (GtkButton *button,
202 static void addressbook_tree_selected (GtkCTree *ctree,
206 static void addressbook_select_row_tree (GtkCTree *ctree,
210 static void addressbook_list_row_selected (GtkCTree *clist,
214 static void addressbook_list_row_unselected (GtkCTree *clist,
218 static void addressbook_person_expand_node (GtkCTree *ctree,
221 static void addressbook_person_collapse_node (GtkCTree *ctree,
225 static gboolean addressbook_list_button_pressed (GtkWidget *widget,
226 GdkEventButton *event,
228 static gboolean addressbook_list_button_released(GtkWidget *widget,
229 GdkEventButton *event,
231 static gboolean addressbook_tree_button_pressed (GtkWidget *ctree,
232 GdkEventButton *event,
234 static gboolean addressbook_tree_button_released(GtkWidget *ctree,
235 GdkEventButton *event,
238 static void addressbook_new_folder_cb (gpointer data,
241 static void addressbook_new_group_cb (gpointer data,
244 static void addressbook_treenode_edit_cb (gpointer data,
247 static void addressbook_treenode_delete_cb (gpointer data,
251 static void addressbook_change_node_name (GtkCTreeNode *node,
254 static void addressbook_new_address_cb (gpointer data,
257 static void addressbook_edit_address_cb (gpointer data,
260 static void addressbook_delete_address_cb (gpointer data,
264 static void close_cb (gpointer data,
267 static void addressbook_file_save_cb (gpointer data,
271 /* Data source edit stuff */
272 static void addressbook_new_book_cb (gpointer data,
275 static void addressbook_new_vcard_cb (gpointer data,
280 static void addressbook_new_jpilot_cb (gpointer data,
286 static void addressbook_new_ldap_cb (gpointer data,
291 static void addressbook_set_clist (AddressObject *obj,
294 static void addressbook_load_tree (void);
295 void addressbook_read_file (void);
297 static GtkCTreeNode *addressbook_add_object (GtkCTreeNode *node,
299 static void addressbook_treenode_remove_item ( void );
301 static AddressDataSource *addressbook_find_datasource
302 (GtkCTreeNode *node );
304 static AddressBookFile *addressbook_get_book_file(void);
306 static GtkCTreeNode *addressbook_node_add_folder
308 AddressDataSource *ds,
309 ItemFolder *itemFolder,
310 AddressObjectType otype);
311 static GtkCTreeNode *addressbook_node_add_group (GtkCTreeNode *node,
312 AddressDataSource *ds,
313 ItemGroup *itemGroup);
314 static void addressbook_tree_remove_children (GtkCTree *ctree,
315 GtkCTreeNode *parent);
316 static void addressbook_move_nodes_up (GtkCTree *ctree,
318 static GtkCTreeNode *addressbook_find_group_node (GtkCTreeNode *parent,
320 static gboolean addressbook_entry_key_pressed (GtkWidget *widget,
323 static gint addressbook_treenode_compare_func (GtkCList *clist,
326 static void addressbook_folder_load_one_person (GtkCTree *clist,
328 AddressTypeControlItem *atci,
329 AddressTypeControlItem *atciMail);
330 static void addressbook_folder_refresh_one_person(GtkCTree *clist,
332 static void addressbook_folder_remove_one_person(GtkCTree *clist,
334 static void addressbook_folder_remove_node (GtkCTree *clist,
337 /* LUT's and IF stuff */
338 static void addressbook_free_treenode ( gpointer data );
339 AddressTypeControlItem *addrbookctl_lookup (gint ot);
340 AddressTypeControlItem *addrbookctl_lookup_iface(AddressIfType ifType);
342 void addrbookctl_build_map (GtkWidget *window);
343 void addrbookctl_build_iflist (void);
344 AdapterInterface *addrbookctl_find_interface (AddressIfType ifType);
345 void addrbookctl_build_ifselect (void);
347 static void addrbookctl_free_interface (AdapterInterface *adapter);
348 static void addrbookctl_free_datasource (AdapterDSource *adapter);
349 static void addrbookctl_free_folder (AdapterFolder *adapter);
350 static void addrbookctl_free_group (AdapterGroup *adapter);
352 static void addressbook_list_select_clear ( void );
353 static void addressbook_list_select_add ( AddrItemObject *aio,
354 AddressDataSource *ds );
355 static void addressbook_list_select_remove ( AddrItemObject *aio );
357 static void addressbook_import_ldif_cb ( void );
358 static void addressbook_import_mutt_cb ( void );
359 static void addressbook_import_pine_cb ( void );
360 static void addressbook_export_html_cb ( void );
361 static void addressbook_export_ldif_cb ( void );
362 static void addressbook_select_all_cb ( void );
363 static void addressbook_clip_cut_cb ( void );
364 static void addressbook_clip_copy_cb ( void );
365 static void addressbook_clip_paste_cb ( void );
366 static void addressbook_treenode_cut_cb ( void );
367 static void addressbook_treenode_copy_cb ( void );
368 static void addressbook_treenode_paste_cb ( void );
370 static void addressbook_mail_to_cb ( void );
373 static void addressbook_browse_entry_cb ( void );
375 static void addressbook_edit_clicked(GtkButton *button, gpointer data);
377 static void addressbook_start_drag(GtkWidget *widget, gint button,
380 static void addressbook_drag_data_get(GtkWidget *widget,
381 GdkDragContext *drag_context,
382 GtkSelectionData *selection_data,
386 static gboolean addressbook_drag_motion_cb(GtkWidget *widget,
387 GdkDragContext *context,
392 static void addressbook_drag_leave_cb(GtkWidget *widget,
393 GdkDragContext *context,
396 static void addressbook_drag_received_cb(GtkWidget *widget,
397 GdkDragContext *drag_context,
400 GtkSelectionData *data,
404 static void addressbook_list_menu_setup( void );
406 static GtkTargetEntry addressbook_drag_types[] =
408 {"sylpheed-claws/internal", GTK_TARGET_SAME_APP, TARGET_DUMMY}
411 static GtkTargetList *addressbook_target_list = NULL;
414 static GtkItemFactoryEntry addressbook_entries[] =
416 {N_("/_Book"), NULL, NULL, 0, "<Branch>"},
417 {N_("/_Book/New _Book"), "<control>B", addressbook_new_book_cb, 0, NULL},
418 {N_("/_Book/New _Folder"), "<control>R", addressbook_new_folder_cb, 0, NULL},
419 {N_("/_Book/New _vCard"), "<control><shift>D", addressbook_new_vcard_cb, 0, NULL},
421 {N_("/_Book/New _JPilot"), "<control>J", addressbook_new_jpilot_cb, 0, NULL},
424 {N_("/_Book/New LDAP _Server"), "<control><shift>S", addressbook_new_ldap_cb, 0, NULL},
426 {N_("/_Book/---"), NULL, NULL, 0, "<Separator>"},
427 {N_("/_Book/_Edit book"), NULL, addressbook_treenode_edit_cb, 0, NULL},
428 {N_("/_Book/_Delete book"), NULL, addressbook_treenode_delete_cb, 0, NULL},
429 {N_("/_Book/---"), NULL, NULL, 0, "<Separator>"},
430 {N_("/_Book/_Save"), "<control>S", addressbook_file_save_cb, 0, NULL},
431 {N_("/_Book/_Close"), "<control>W", close_cb, 0, NULL},
432 {N_("/_Address"), NULL, NULL, 0, "<Branch>"},
433 {N_("/_Address/_Select all"), "<control>A", addressbook_select_all_cb, 0, NULL},
434 {N_("/_Address/---"), NULL, NULL, 0, "<Separator>"},
435 {N_("/_Address/C_ut"), "<control>X", addressbook_clip_cut_cb, 0, NULL},
436 {N_("/_Address/_Copy"), "<control>C", addressbook_clip_copy_cb, 0, NULL},
437 {N_("/_Address/_Paste"), "<control>V", addressbook_clip_paste_cb, 0, NULL},
438 {N_("/_Address/---"), NULL, NULL, 0, "<Separator>"},
439 {N_("/_Address/_Edit"), "<control>Return",addressbook_edit_address_cb, 0, NULL},
440 {N_("/_Address/_Delete"), "<control>D", addressbook_delete_address_cb, 0, NULL},
441 {N_("/_Address/---"), NULL, NULL, 0, "<Separator>"},
442 {N_("/_Address/New _Address"), "<control>N", addressbook_new_address_cb, 0, NULL},
443 {N_("/_Address/New _Group"), "<control>G", addressbook_new_group_cb, 0, NULL},
444 {N_("/_Address/---"), NULL, NULL, 0, "<Separator>"},
445 {N_("/_Address/_Mail To"), NULL, addressbook_mail_to_cb, 0, NULL},
446 {N_("/_Tools"), NULL, NULL, 0, "<Branch>"},
447 {N_("/_Tools/Import _LDIF file..."), NULL, addressbook_import_ldif_cb, 0, NULL},
448 {N_("/_Tools/Import M_utt file..."), NULL, addressbook_import_mutt_cb, 0, NULL},
449 {N_("/_Tools/Import _Pine file..."), NULL, addressbook_import_pine_cb, 0, NULL},
450 {N_("/_Tools/---"), NULL, NULL, 0, "<Separator>"},
451 {N_("/_Tools/Export _HTML..."), NULL, addressbook_export_html_cb, 0, NULL},
452 {N_("/_Tools/Export LDI_F..."), NULL, addressbook_export_ldif_cb, 0, NULL},
453 {N_("/_Help"), NULL, NULL, 0, "<Branch>"},
454 {N_("/_Help/_About"), NULL, about_show, 0, NULL}
457 static GtkItemFactoryEntry addressbook_tree_popup_entries[] =
459 {N_("/_Edit"), NULL, addressbook_treenode_edit_cb, 0, NULL},
460 {N_("/_Delete"), NULL, addressbook_treenode_delete_cb, 0, NULL},
461 {N_("/---"), NULL, NULL, 0, "<Separator>"},
462 {N_("/New _Folder"), NULL, addressbook_new_folder_cb, 0, NULL},
463 {N_("/---"), NULL, NULL, 0, "<Separator>"},
464 {N_("/C_ut"), NULL, addressbook_treenode_cut_cb, 0, NULL},
465 {N_("/_Copy"), NULL, addressbook_treenode_copy_cb, 0, NULL},
466 {N_("/_Paste"), NULL, addressbook_treenode_paste_cb, 0, NULL}
469 static GtkItemFactoryEntry addressbook_list_popup_entries[] =
471 {N_("/_Select all"), NULL, addressbook_select_all_cb, 0, NULL},
472 {N_("/---"), NULL, NULL, 0, "<Separator>"},
473 {N_("/_Edit"), NULL, addressbook_edit_address_cb, 0, NULL},
474 {N_("/_Delete"), NULL, addressbook_delete_address_cb, 0, NULL},
475 {N_("/---"), NULL, NULL, 0, "<Separator>"},
476 {N_("/New _Address"), NULL, addressbook_new_address_cb, 0, NULL},
477 {N_("/New _Group"), NULL, addressbook_new_group_cb, 0, NULL},
478 {N_("/---"), NULL, NULL, 0, "<Separator>"},
479 {N_("/C_ut"), NULL, addressbook_clip_cut_cb, 0, NULL},
480 {N_("/_Copy"), NULL, addressbook_clip_copy_cb, 0, NULL},
481 {N_("/_Paste"), NULL, addressbook_clip_paste_cb, 0, NULL},
482 {N_("/---"), NULL, NULL, 0, "<Separator>"},
483 /* {N_("/Pa_ste Address"), NULL, addressbook_clip_paste_address_cb, 0, NULL},*/
484 {N_("/_Mail To"), NULL, addressbook_mail_to_cb, 0, NULL},
486 {N_("/_Browse Entry"), NULL, addressbook_browse_entry_cb, 0, NULL},
491 * Structure of error message table.
493 typedef struct _ErrMsgTableEntry ErrMsgTableEntry;
494 struct _ErrMsgTableEntry {
499 static gchar *_errMsgUnknown_ = N_( "Unknown" );
502 * Lookup table of error messages for general errors. Note that a NULL
503 * description signifies the end of the table.
505 static ErrMsgTableEntry _lutErrorsGeneral_[] = {
506 { MGU_SUCCESS, N_("Success") },
507 { MGU_BAD_ARGS, N_("Bad arguments") },
508 { MGU_NO_FILE, N_("File not specified") },
509 { MGU_OPEN_FILE, N_("Error opening file") },
510 { MGU_ERROR_READ, N_("Error reading file") },
511 { MGU_EOF, N_("End of file encountered") },
512 { MGU_OO_MEMORY, N_("Error allocating memory") },
513 { MGU_BAD_FORMAT, N_("Bad file format") },
514 { MGU_ERROR_WRITE, N_("Error writing to file") },
515 { MGU_OPEN_DIRECTORY, N_("Error opening directory") },
516 { MGU_NO_PATH, N_("No path specified") },
522 * Lookup table of error messages for LDAP errors.
524 static ErrMsgTableEntry _lutErrorsLDAP_[] = {
525 { LDAPRC_SUCCESS, N_("Success") },
526 { LDAPRC_CONNECT, N_("Error connecting to LDAP server") },
527 { LDAPRC_INIT, N_("Error initializing LDAP") },
528 { LDAPRC_BIND, N_("Error binding to LDAP server") },
529 { LDAPRC_SEARCH, N_("Error searching LDAP database") },
530 { LDAPRC_TIMEOUT, N_("Timeout performing LDAP operation") },
531 { LDAPRC_CRITERIA, N_("Error in LDAP search criteria") },
532 { LDAPRC_NOENTRIES, N_("No LDAP entries found for search criteria") },
533 { LDAPRC_STOP_FLAG, N_("LDAP search terminated on request") },
534 { LDAPRC_TLS, N_("Error starting TLS connection") },
540 * Lookup message for specified error code.
541 * \param lut Lookup table.
542 * \param code Code to lookup.
543 * \return Description associated to code.
545 static gchar *addressbook_err2string( ErrMsgTableEntry lut[], gint code ) {
547 ErrMsgTableEntry entry;
550 for( i = 0; ; i++ ) {
552 if( entry.description == NULL ) break;
553 if( entry.code == code ) {
554 desc = entry.description;
559 desc = _errMsgUnknown_;
564 static gboolean lastCanLookup = FALSE;
566 void addressbook_show_buttons(gboolean add_and_delete, gboolean lookup, gboolean mail_ops)
568 if (add_and_delete) {
569 gtk_widget_show(addrbook.edit_btn);
570 gtk_widget_show(addrbook.del_btn);
571 gtk_widget_show(addrbook.reg_btn);
573 gtk_widget_hide(addrbook.edit_btn);
574 gtk_widget_hide(addrbook.del_btn);
575 gtk_widget_hide(addrbook.reg_btn);
579 gtk_widget_show(addrbook.lup_btn);
580 gtk_widget_show(addrbook.entry);
581 gtk_widget_show(addrbook.label);
583 gtk_widget_hide(addrbook.lup_btn);
584 gtk_widget_hide(addrbook.entry);
585 gtk_widget_hide(addrbook.label);
588 lastCanLookup = lookup;
591 gtk_widget_show(addrbook.to_btn);
592 gtk_widget_show(addrbook.cc_btn);
593 gtk_widget_show(addrbook.bcc_btn);
595 gtk_widget_hide(addrbook.to_btn);
596 gtk_widget_hide(addrbook.cc_btn);
597 gtk_widget_hide(addrbook.bcc_btn);
601 void addressbook_open(Compose *target)
603 /* Initialize all static members */
604 if( _clipBoard_ == NULL ) {
605 _clipBoard_ = addrclip_create();
607 if( _addressIndex_ != NULL ) {
608 addrclip_set_index( _clipBoard_, _addressIndex_ );
610 if( _addressSelect_ == NULL ) {
611 _addressSelect_ = addrselect_list_create();
613 if (!addrbook.window) {
614 addressbook_read_file();
615 addressbook_create();
616 addressbook_load_tree();
617 gtk_sctree_select( GTK_SCTREE(addrbook.ctree),
618 GTK_CTREE_NODE(GTK_CLIST(addrbook.ctree)->row_list));
621 gtk_widget_hide(addrbook.window);
624 gtk_widget_show_all(addrbook.window);
625 addressbook_show_buttons(target == NULL, lastCanLookup, target != NULL);
626 addressbook_set_target_compose(target);
630 * Destroy addressbook.
632 void addressbook_destroy( void ) {
633 /* Free up address stuff */
634 if( _addressSelect_ != NULL ) {
635 addrselect_list_free( _addressSelect_ );
637 if( _clipBoard_ != NULL ) {
638 addrclip_free( _clipBoard_ );
640 if( _addressIndex_ != NULL ) {
641 addrindex_free_index( _addressIndex_ );
642 addrindex_teardown();
644 _addressSelect_ = NULL;
646 _addressIndex_ = NULL;
649 void addressbook_set_target_compose(Compose *target)
651 addrbook.target_compose = target;
652 addressbook_button_set_sensitive();
655 Compose *addressbook_get_target_compose(void)
657 return addrbook.target_compose;
661 * Refresh addressbook and save to file(s).
663 void addressbook_refresh( void )
665 if (addrbook.window) {
666 if (addrbook.treeSelected) {
667 gtk_sctree_select( GTK_SCTREE(addrbook.ctree),
668 addrbook.treeSelected);
669 addressbook_set_clist(
670 gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree),
671 addrbook.treeSelected),
676 addressbook_export_to_file();
679 static gboolean key_pressed(GtkWidget *widget, GdkEventKey *event, gpointer data)
681 if (event && event->keyval == GDK_Escape)
683 else if (event && event->keyval == GDK_Delete)
684 addressbook_del_clicked(NULL, NULL);
689 *\brief Save Gtk object size to prefs dataset
691 static void addressbook_size_allocate_cb(GtkWidget *widget,
692 GtkAllocation *allocation)
694 g_return_if_fail(allocation != NULL);
696 prefs_common.addressbookwin_width = allocation->width;
697 prefs_common.addressbookwin_height = allocation->height;
700 static gint sort_column_number = 0;
701 static GtkSortType sort_column_type = GTK_SORT_ASCENDING;
703 static gint list_case_sort(
704 GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2 )
706 GtkCListRow *row1 = (GtkCListRow *) ptr1;
707 GtkCListRow *row2 = (GtkCListRow *) ptr2;
708 gchar *name1 = NULL, *name2 = NULL;
709 AddrItemObject *aio1 = ((GtkCListRow *)ptr1)->data;
710 AddrItemObject *aio2 = ((GtkCListRow *)ptr2)->data;
712 if( aio1->type == aio2->type ) {
714 name1 = GTK_CELL_TEXT (row1->cell[sort_column_number])->text;
716 name2 = GTK_CELL_TEXT (row2->cell[sort_column_number])->text;
717 if( ! name1 ) return ( name2 != NULL );
718 if( ! name2 ) return -1;
719 return strcasecmp( name1, name2 );
721 /* Order groups before person */
722 if( aio1->type == ITEMTYPE_GROUP ) {
723 return (sort_column_type==GTK_SORT_ASCENDING) ? -1:+1;
724 } else if( aio2->type == ITEMTYPE_GROUP ) {
725 return (sort_column_type==GTK_SORT_ASCENDING) ? +1:-1;
731 static void addressbook_sort_list(GtkCList *clist, const gint col,
732 const GtkSortType sort_type)
735 GtkWidget *hbox, *label, *arrow;
737 sort_column_number = col;
738 sort_column_type = sort_type;
739 gtk_clist_set_compare_func(clist, list_case_sort);
740 gtk_clist_set_sort_type(clist, sort_type);
741 gtk_clist_set_sort_column(clist, col);
743 gtk_clist_freeze(clist);
744 gtk_clist_sort(clist);
746 for(pos = 0 ; pos < N_LIST_COLS ; pos++) {
747 hbox = gtk_hbox_new(FALSE, 4);
748 label = gtk_label_new(gettext(list_titles[pos]));
749 gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
752 arrow = gtk_arrow_new(sort_type == GTK_SORT_ASCENDING ?
753 GTK_ARROW_DOWN : GTK_ARROW_UP, GTK_SHADOW_IN);
754 gtk_box_pack_end(GTK_BOX(hbox), arrow, FALSE, FALSE, 0);
757 gtk_widget_show_all(hbox);
758 gtk_clist_set_column_widget(clist, pos, hbox);
761 gtk_clist_thaw(clist);
764 static void addressbook_name_clicked(GtkWidget *button, GtkCList *clist)
766 static GtkSortType sort_type = GTK_SORT_ASCENDING;
768 sort_type = (sort_type == GTK_SORT_ASCENDING) ? GTK_SORT_DESCENDING :
770 addressbook_sort_list(clist, COL_NAME, sort_type);
773 static void addressbook_address_clicked(GtkWidget *button, GtkCList *clist)
775 static GtkSortType sort_type = GTK_SORT_ASCENDING;
777 sort_type = (sort_type == GTK_SORT_ASCENDING) ? GTK_SORT_DESCENDING :
779 addressbook_sort_list(clist, COL_ADDRESS, sort_type);
782 static void addressbook_remarks_clicked(GtkWidget *button, GtkCList *clist)
784 static GtkSortType sort_type = GTK_SORT_ASCENDING;
786 sort_type = (sort_type == GTK_SORT_ASCENDING) ? GTK_SORT_DESCENDING :
788 addressbook_sort_list(clist, COL_REMARKS, sort_type);
792 * Create the address book widgets. The address book contains two CTree widgets: the
793 * address index tree on the left and the address list on the right.
795 * The address index tree displays a hierarchy of interfaces and groups. Each node in
796 * this tree is linked to an address Adapter. Adapters have been created for interfaces,
797 * data sources and folder objects.
799 * The address list displays group, person and email objects. These items are linked
800 * directly to ItemGroup, ItemPerson and ItemEMail objects inside the address book data
803 * In the tradition of MVC architecture, the data stores have been separated from the
804 * GUI components. The addrindex.c file provides the interface to all data stores.
806 static void addressbook_create(void)
812 GtkWidget *ctree_swin;
814 GtkWidget *clist_vbox;
815 GtkWidget *clist_swin;
821 GtkWidget *statusbar;
832 GtkWidget *close_btn;
833 GtkWidget *tree_popup;
834 GtkWidget *list_popup;
835 GtkItemFactory *tree_factory;
836 GtkItemFactory *list_factory;
837 GtkItemFactory *menu_factory;
841 gchar *index_titles[N_INDEX_COLS];
845 static GdkGeometry geometry;
847 debug_print("Creating addressbook window...\n");
849 index_titles[COL_SOURCES] = _("Sources");
851 /* Address book window */
852 window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
853 gtk_window_set_title(GTK_WINDOW(window), _("Address book"));
854 gtk_window_set_resizable(GTK_WINDOW(window), TRUE);
855 gtk_widget_realize(window);
857 g_signal_connect(G_OBJECT(window), "delete_event",
858 G_CALLBACK(addressbook_close), NULL);
859 g_signal_connect(G_OBJECT(window), "size_allocate",
860 G_CALLBACK(addressbook_size_allocate_cb), NULL);
861 g_signal_connect(G_OBJECT(window), "key_press_event",
862 G_CALLBACK(key_pressed), NULL);
863 MANAGE_WINDOW_SIGNALS_CONNECT(window);
865 vbox = gtk_vbox_new(FALSE, 0);
866 gtk_container_add(GTK_CONTAINER(window), vbox);
869 n_entries = sizeof(addressbook_entries) /
870 sizeof(addressbook_entries[0]);
871 menubar = menubar_create(window, addressbook_entries, n_entries,
872 "<AddressBook>", NULL);
873 gtk_box_pack_start(GTK_BOX(vbox), menubar, FALSE, TRUE, 0);
874 menu_factory = gtk_item_factory_from_widget(menubar);
876 vbox2 = gtk_vbox_new(FALSE, BORDER_WIDTH);
877 gtk_container_set_border_width(GTK_CONTAINER(vbox2), BORDER_WIDTH);
878 gtk_box_pack_start(GTK_BOX(vbox), vbox2, TRUE, TRUE, 0);
880 ctree_swin = gtk_scrolled_window_new(NULL, NULL);
881 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(ctree_swin),
882 GTK_POLICY_AUTOMATIC,
883 GTK_POLICY_AUTOMATIC);
884 gtk_widget_set_size_request(ctree_swin, COL_FOLDER_WIDTH + 40, -1);
887 ctree = gtk_sctree_new_with_titles(N_INDEX_COLS, 0, index_titles);
888 gtk_container_add(GTK_CONTAINER(ctree_swin), ctree);
889 gtk_clist_set_selection_mode(GTK_CLIST(ctree), GTK_SELECTION_BROWSE);
890 gtk_clist_set_column_width(GTK_CLIST(ctree), 0, COL_FOLDER_WIDTH);
891 gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_DOTTED);
892 gtk_ctree_set_expander_style(GTK_CTREE(ctree),
893 GTK_CTREE_EXPANDER_SQUARE);
894 gtk_ctree_set_indent(GTK_CTREE(ctree), CTREE_INDENT);
895 gtk_clist_set_compare_func(GTK_CLIST(ctree),
896 addressbook_treenode_compare_func);
898 g_signal_connect(G_OBJECT(ctree), "tree_select_row",
899 G_CALLBACK(addressbook_tree_selected), NULL);
900 g_signal_connect(G_OBJECT(ctree), "button_press_event",
901 G_CALLBACK(addressbook_tree_button_pressed),
903 g_signal_connect(G_OBJECT(ctree), "button_release_event",
904 G_CALLBACK(addressbook_tree_button_released),
907 g_signal_connect(G_OBJECT(ctree), "select_row",
908 G_CALLBACK(addressbook_select_row_tree), NULL);
910 gtk_drag_dest_set(ctree, GTK_DEST_DEFAULT_ALL & ~GTK_DEST_DEFAULT_HIGHLIGHT,
911 addressbook_drag_types, 1,
912 GDK_ACTION_MOVE | GDK_ACTION_COPY | GDK_ACTION_DEFAULT);
913 g_signal_connect(G_OBJECT(ctree), "drag_motion",
914 G_CALLBACK(addressbook_drag_motion_cb),
916 g_signal_connect(G_OBJECT(ctree), "drag_leave",
917 G_CALLBACK(addressbook_drag_leave_cb),
919 g_signal_connect(G_OBJECT(ctree), "drag_data_received",
920 G_CALLBACK(addressbook_drag_received_cb),
923 clist_vbox = gtk_vbox_new(FALSE, 4);
925 clist_swin = gtk_scrolled_window_new(NULL, NULL);
926 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(clist_swin),
927 GTK_POLICY_AUTOMATIC,
928 GTK_POLICY_AUTOMATIC);
929 gtk_box_pack_start(GTK_BOX(clist_vbox), clist_swin, TRUE, TRUE, 0);
932 clist = gtk_sctree_new_with_titles(N_LIST_COLS, 0, list_titles);
933 gtk_container_add(GTK_CONTAINER(clist_swin), clist);
934 gtk_clist_set_selection_mode(GTK_CLIST(clist), GTK_SELECTION_EXTENDED);
935 gtk_ctree_set_line_style(GTK_CTREE(clist), GTK_CTREE_LINES_NONE);
936 gtk_ctree_set_expander_style(GTK_CTREE(clist), GTK_CTREE_EXPANDER_SQUARE);
937 gtk_ctree_set_indent(GTK_CTREE(clist), CTREE_INDENT);
938 gtk_clist_set_column_width(GTK_CLIST(clist), COL_NAME,
940 gtk_clist_set_column_width(GTK_CLIST(clist), COL_ADDRESS,
943 addressbook_sort_list(GTK_CLIST(clist), COL_NAME, GTK_SORT_ASCENDING);
944 g_signal_connect(G_OBJECT(GTK_CLIST(clist)->column[COL_NAME].button),
945 "clicked", G_CALLBACK(addressbook_name_clicked), clist);
946 g_signal_connect(G_OBJECT(GTK_CLIST(clist)->column[COL_ADDRESS].button),
947 "clicked", G_CALLBACK(addressbook_address_clicked), clist);
948 g_signal_connect(G_OBJECT(GTK_CLIST(clist)->column[COL_REMARKS].button),
949 "clicked", G_CALLBACK(addressbook_remarks_clicked), clist);
951 for (i = 0; i < N_LIST_COLS; i++)
952 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist)->column[i].button,
955 g_signal_connect(G_OBJECT(clist), "tree_select_row",
956 G_CALLBACK(addressbook_list_row_selected), NULL);
957 g_signal_connect(G_OBJECT(clist), "tree_unselect_row",
958 G_CALLBACK(addressbook_list_row_unselected), NULL);
959 g_signal_connect(G_OBJECT(clist), "button_press_event",
960 G_CALLBACK(addressbook_list_button_pressed),
962 g_signal_connect(G_OBJECT(clist), "button_release_event",
963 G_CALLBACK(addressbook_list_button_released),
965 g_signal_connect(G_OBJECT(clist), "tree_expand",
966 G_CALLBACK(addressbook_person_expand_node), NULL );
967 g_signal_connect(G_OBJECT(clist), "tree_collapse",
968 G_CALLBACK(addressbook_person_collapse_node), NULL );
969 g_signal_connect(G_OBJECT(clist), "start_drag",
970 G_CALLBACK(addressbook_start_drag), NULL);
971 g_signal_connect(G_OBJECT(clist), "drag_data_get",
972 G_CALLBACK(addressbook_drag_data_get), NULL);
973 hbox = gtk_hbox_new(FALSE, 4);
974 gtk_box_pack_start(GTK_BOX(clist_vbox), hbox, FALSE, FALSE, 0);
976 label = gtk_label_new(_("Lookup name:"));
977 gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
979 entry = gtk_entry_new();
980 gtk_box_pack_start(GTK_BOX(hbox), entry, TRUE, TRUE, 0);
982 address_completion_register_entry(GTK_ENTRY(entry));
984 g_signal_connect(G_OBJECT(entry), "key_press_event",
985 G_CALLBACK(addressbook_entry_key_pressed),
987 paned = gtk_hpaned_new();
988 gtk_box_pack_start(GTK_BOX(vbox2), paned, TRUE, TRUE, 0);
989 gtk_paned_add1(GTK_PANED(paned), ctree_swin);
990 gtk_paned_add2(GTK_PANED(paned), clist_vbox);
993 hsbox = gtk_hbox_new(FALSE, 0);
994 gtk_box_pack_end(GTK_BOX(vbox), hsbox, FALSE, FALSE, BORDER_WIDTH);
995 statusbar = gtk_statusbar_new();
996 gtk_box_pack_start(GTK_BOX(hsbox), statusbar, TRUE, TRUE, BORDER_WIDTH);
999 hbbox = gtk_hbutton_box_new();
1000 gtk_button_box_set_layout(GTK_BUTTON_BOX(hbbox), GTK_BUTTONBOX_END);
1001 gtk_box_set_spacing(GTK_BOX(hbbox), 2);
1002 gtk_container_set_border_width(GTK_CONTAINER(hbbox), 4);
1003 gtk_box_pack_end(GTK_BOX(vbox), hbbox, FALSE, FALSE, 0);
1005 gtkut_stock_button_add_help(hbbox, &help_btn);
1007 edit_btn = gtk_button_new_from_stock(GTK_STOCK_EDIT);
1008 GTK_WIDGET_SET_FLAGS(edit_btn, GTK_CAN_DEFAULT);
1009 gtk_box_pack_start(GTK_BOX(hbbox), edit_btn, TRUE, TRUE, 0);
1010 del_btn = gtk_button_new_from_stock(GTK_STOCK_DELETE);
1011 GTK_WIDGET_SET_FLAGS(del_btn, GTK_CAN_DEFAULT);
1012 gtk_box_pack_start(GTK_BOX(hbbox), del_btn, TRUE, TRUE, 0);
1013 reg_btn = gtk_button_new_from_stock(GTK_STOCK_NEW);
1014 GTK_WIDGET_SET_FLAGS(reg_btn, GTK_CAN_DEFAULT);
1015 gtk_box_pack_start(GTK_BOX(hbbox), reg_btn, TRUE, TRUE, 0);
1018 lup_btn = gtk_button_new_from_stock(GTK_STOCK_FIND);
1019 GTK_WIDGET_SET_FLAGS(lup_btn, GTK_CAN_DEFAULT);
1020 gtk_box_pack_start(GTK_BOX(hbox), lup_btn, TRUE, TRUE, 0);
1022 g_signal_connect(G_OBJECT(help_btn), "clicked",
1023 G_CALLBACK(manual_open_with_anchor_cb),
1024 MANUAL_ANCHOR_ADDRBOOK);
1026 g_signal_connect(G_OBJECT(edit_btn), "clicked",
1027 G_CALLBACK(addressbook_edit_clicked), NULL);
1028 g_signal_connect(G_OBJECT(del_btn), "clicked",
1029 G_CALLBACK(addressbook_del_clicked), NULL);
1030 g_signal_connect(G_OBJECT(reg_btn), "clicked",
1031 G_CALLBACK(addressbook_reg_clicked), NULL);
1032 g_signal_connect(G_OBJECT(lup_btn), "clicked",
1033 G_CALLBACK(addressbook_lup_clicked), NULL);
1035 to_btn = gtk_button_new_with_label
1036 (prefs_common.trans_hdr ? _("To:") : "To:");
1037 GTK_WIDGET_SET_FLAGS(to_btn, GTK_CAN_DEFAULT);
1038 gtk_box_pack_start(GTK_BOX(hbbox), to_btn, TRUE, TRUE, 0);
1039 cc_btn = gtk_button_new_with_label
1040 (prefs_common.trans_hdr ? _("Cc:") : "Cc:");
1041 GTK_WIDGET_SET_FLAGS(cc_btn, GTK_CAN_DEFAULT);
1042 gtk_box_pack_start(GTK_BOX(hbbox), cc_btn, TRUE, TRUE, 0);
1043 bcc_btn = gtk_button_new_with_label
1044 (prefs_common.trans_hdr ? _("Bcc:") : "Bcc:");
1045 GTK_WIDGET_SET_FLAGS(bcc_btn, GTK_CAN_DEFAULT);
1046 gtk_box_pack_start(GTK_BOX(hbbox), bcc_btn, TRUE, TRUE, 0);
1048 close_btn = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
1049 GTK_WIDGET_SET_FLAGS(close_btn, GTK_CAN_DEFAULT);
1050 gtk_box_pack_start(GTK_BOX(hbbox), close_btn, TRUE, TRUE, 0);
1052 g_signal_connect(G_OBJECT(to_btn), "clicked",
1053 G_CALLBACK(addressbook_to_clicked),
1054 GINT_TO_POINTER(COMPOSE_TO));
1055 g_signal_connect(G_OBJECT(cc_btn), "clicked",
1056 G_CALLBACK(addressbook_to_clicked),
1057 GINT_TO_POINTER(COMPOSE_CC));
1058 g_signal_connect(G_OBJECT(bcc_btn), "clicked",
1059 G_CALLBACK(addressbook_to_clicked),
1060 GINT_TO_POINTER(COMPOSE_BCC));
1061 g_signal_connect(G_OBJECT(close_btn), "clicked",
1062 G_CALLBACK(addressbook_close_clicked), NULL);
1064 /* Build icons for interface */
1065 stock_pixmap_gdk( window, STOCK_PIXMAP_INTERFACE,
1066 &interfacexpm, &interfacexpmmask );
1068 /* Build control tables */
1069 addrbookctl_build_map(window);
1070 addrbookctl_build_iflist();
1071 addrbookctl_build_ifselect();
1073 addrbook.clist = NULL;
1075 /* Add each interface into the tree as a root level folder */
1076 nodeIf = _addressInterfaceList_;
1078 AdapterInterface *adapter = nodeIf->data;
1079 AddressInterface *iface = adapter->interface;
1080 nodeIf = g_list_next(nodeIf);
1082 if(iface->useInterface) {
1083 AddressTypeControlItem *atci = adapter->atci;
1084 text = atci->displayName;
1086 gtk_sctree_insert_node( GTK_CTREE(ctree),
1087 NULL, NULL, &text, FOLDER_SPACING,
1088 interfacexpm, interfacexpmmask,
1089 interfacexpm, interfacexpmmask,
1091 menu_set_sensitive( menu_factory, atci->menuCommand, adapter->haveLibrary );
1092 gtk_ctree_node_set_row_data_full(
1093 GTK_CTREE(ctree), adapter->treeNode, adapter,
1094 addressbook_free_treenode );
1099 n_entries = sizeof(addressbook_tree_popup_entries) /
1100 sizeof(addressbook_tree_popup_entries[0]);
1101 tree_popup = menu_create_items(addressbook_tree_popup_entries,
1103 "<AddressBookTree>", &tree_factory,
1105 n_entries = sizeof(addressbook_list_popup_entries) /
1106 sizeof(addressbook_list_popup_entries[0]);
1107 list_popup = menu_create_items(addressbook_list_popup_entries,
1109 "<AddressBookList>", &list_factory,
1112 addrbook.window = window;
1113 addrbook.menubar = menubar;
1114 addrbook.ctree = ctree;
1117 addrbook.clist = clist;
1118 addrbook.label = label;
1119 addrbook.entry = entry;
1120 addrbook.statusbar = statusbar;
1121 addrbook.status_cid = gtk_statusbar_get_context_id(
1122 GTK_STATUSBAR(statusbar), "Addressbook Window" );
1124 addrbook.help_btn = help_btn;
1125 addrbook.edit_btn = edit_btn;
1126 addrbook.del_btn = del_btn;
1127 addrbook.reg_btn = reg_btn;
1128 addrbook.lup_btn = lup_btn;
1129 addrbook.to_btn = to_btn;
1130 addrbook.cc_btn = cc_btn;
1131 addrbook.bcc_btn = bcc_btn;
1133 addrbook.tree_popup = tree_popup;
1134 addrbook.list_popup = list_popup;
1135 addrbook.tree_factory = tree_factory;
1136 addrbook.list_factory = list_factory;
1137 addrbook.menu_factory = menu_factory;
1139 addrbook.listSelected = NULL;
1140 address_completion_start(window);
1142 if (!geometry.min_height) {
1143 geometry.min_width = ADDRESSBOOK_WIDTH;
1144 geometry.min_height = ADDRESSBOOK_HEIGHT;
1147 gtk_window_set_geometry_hints(GTK_WINDOW(window), NULL, &geometry,
1149 gtk_widget_set_size_request(window, prefs_common.addressbookwin_width,
1150 prefs_common.addressbookwin_height);
1152 gtk_widget_show_all(window);
1156 * Close address book window and save to file(s).
1158 static gint addressbook_close( void ) {
1159 gtk_widget_hide(addrbook.window);
1160 addressbook_export_to_file();
1165 * Display message in status line.
1166 * \param msg Message to display.
1168 static void addressbook_status_show( gchar *msg ) {
1169 if( addrbook.statusbar != NULL ) {
1171 GTK_STATUSBAR(addrbook.statusbar),
1172 addrbook.status_cid );
1175 GTK_STATUSBAR(addrbook.statusbar),
1176 addrbook.status_cid, msg );
1181 static void addressbook_ds_status_message( AddressDataSource *ds, gchar *msg ) {
1182 *addressbook_msgbuf = '\0';
1186 name = addrindex_ds_get_name( ds );
1187 g_snprintf( addressbook_msgbuf, sizeof(addressbook_msgbuf),
1188 "%s: %s", name, msg );
1191 g_snprintf( addressbook_msgbuf, sizeof(addressbook_msgbuf),
1194 addressbook_status_show( addressbook_msgbuf );
1197 static void addressbook_ds_show_message( AddressDataSource *ds ) {
1201 *addressbook_msgbuf = '\0';
1203 name = addrindex_ds_get_name( ds );
1204 retVal = addrindex_ds_get_status_code( ds );
1205 if( retVal == MGU_SUCCESS ) {
1206 g_snprintf( addressbook_msgbuf,
1207 sizeof(addressbook_msgbuf), "%s", name );
1210 desc = addressbook_err2string( _lutErrorsGeneral_, retVal );
1211 g_snprintf( addressbook_msgbuf,
1212 sizeof(addressbook_msgbuf), "%s: %s", name, desc );
1215 addressbook_status_show( addressbook_msgbuf );
1218 static void addressbook_button_set_sensitive(void)
1220 gboolean to_sens = FALSE;
1221 gboolean cc_sens = FALSE;
1222 gboolean bcc_sens = FALSE;
1224 if (!addrbook.window) return;
1226 if (addrbook.target_compose) {
1230 if (addrbook.target_compose->use_bcc)
1235 gtk_widget_set_sensitive(addrbook.to_btn, to_sens);
1236 gtk_widget_set_sensitive(addrbook.cc_btn, cc_sens);
1237 gtk_widget_set_sensitive(addrbook.bcc_btn, bcc_sens);
1240 static void addressbook_edit_clicked(GtkButton *button, gpointer data)
1242 addressbook_edit_address_cb(NULL, 0, NULL);
1246 * Delete one or more objects from address list.
1248 static void addressbook_del_clicked(GtkButton *button, gpointer data)
1250 GtkCTree *clist = GTK_CTREE(addrbook.clist);
1251 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
1252 AddressObject *pobj;
1253 AdapterDSource *ads = NULL;
1254 GtkCTreeNode *nodeList;
1257 AddressBookFile *abf = NULL;
1258 AddressDataSource *ds = NULL;
1259 AddressInterface *iface;
1260 AddrItemObject *aio;
1261 AddrSelectItem *item;
1263 gboolean refreshList = FALSE;
1265 pobj = gtk_ctree_node_get_row_data(ctree, addrbook.opened );
1266 g_return_if_fail(pobj != NULL);
1268 /* Test whether anything selected for deletion */
1269 nodeList = addrbook.listSelected;
1271 aio = gtk_ctree_node_get_row_data( clist, nodeList );
1272 if( aio == NULL) return;
1273 ds = addressbook_find_datasource( addrbook.treeSelected );
1274 if( ds == NULL ) return;
1276 /* Test for read only */
1277 iface = ds->interface;
1278 if( iface->readOnly ) {
1279 alertpanel( _("Delete address(es)"),
1280 _("This address data is readonly and cannot be deleted."),
1281 GTK_STOCK_CLOSE, NULL, NULL );
1285 /* Test whether Ok to proceed */
1287 if( pobj->type == ADDR_DATASOURCE ) {
1288 ads = ADAPTER_DSOURCE(pobj);
1289 if( ads->subType == ADDR_BOOK ) procFlag = TRUE;
1291 else if( pobj->type == ADDR_ITEM_FOLDER ) {
1294 else if( pobj->type == ADDR_ITEM_GROUP ) {
1297 if( ! procFlag ) return;
1298 abf = ds->rawDataSource;
1299 if( abf == NULL ) return;
1301 /* Confirm deletion */
1302 aval = alertpanel( _("Delete address(es)"),
1303 _("Really delete the address(es)?"),
1304 GTK_STOCK_CANCEL, "+"GTK_STOCK_DELETE, NULL );
1305 if( aval != G_ALERTALTERNATE ) return;
1307 /* Process deletions */
1308 if( pobj->type == ADDR_DATASOURCE || pobj->type == ADDR_ITEM_FOLDER ) {
1309 /* Items inside folders */
1310 list = addrselect_get_list( _addressSelect_ );
1315 node = g_list_next( node );
1316 aio = ( AddrItemObject * ) item->addressItem;
1317 if( aio->type == ADDR_ITEM_GROUP ) {
1318 ItemGroup *item = ( ItemGroup * ) aio;
1319 GtkCTreeNode *nd = NULL;
1321 nd = addressbook_find_group_node( addrbook.opened, item );
1322 item = addrbook_remove_group( abf, item );
1324 addritem_free_item_group( item );
1326 /* Remove group from parent node */
1327 gtk_ctree_remove_node( ctree, nd );
1330 else if( aio->type == ADDR_ITEM_PERSON ) {
1331 ItemPerson *item = ( ItemPerson * ) aio;
1332 addressbook_folder_remove_one_person( clist, item );
1333 if (pobj->type == ADDR_ITEM_FOLDER)
1334 addritem_folder_remove_person(ADAPTER_FOLDER(pobj)->itemFolder, item);
1335 item = addrbook_remove_person( abf, item );
1337 addritem_free_item_person( item );
1340 else if( aio->type == ADDR_ITEM_EMAIL ) {
1341 ItemEMail *item = ( ItemEMail * ) aio;
1342 ItemPerson *person = ( ItemPerson * ) ADDRITEM_PARENT(item);
1343 item = addrbook_person_remove_email( abf, person, item );
1345 addrcache_remove_email(abf->addressCache, item);
1346 addritem_free_item_email( item );
1348 addressbook_folder_refresh_one_person( clist, person );
1351 g_list_free( list );
1352 addressbook_list_select_clear();
1354 gtk_sctree_select( GTK_SCTREE(ctree), addrbook.opened);
1355 addressbook_set_clist(
1356 gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree),
1360 addrbook_set_dirty(abf, TRUE);
1361 addressbook_export_to_file();
1362 addressbook_list_menu_setup();
1365 else if( pobj->type == ADDR_ITEM_GROUP ) {
1366 /* Items inside groups */
1367 list = addrselect_get_list( _addressSelect_ );
1371 node = g_list_next( node );
1372 aio = ( AddrItemObject * ) item->addressItem;
1373 if( aio->type == ADDR_ITEM_EMAIL ) {
1374 ItemEMail *item = ( ItemEMail * ) aio;
1375 ItemPerson *person = ( ItemPerson * ) ADDRITEM_PARENT(item);
1376 item = addrbook_person_remove_email( abf, person, item );
1378 addritem_free_item_email( item );
1382 g_list_free( list );
1383 addressbook_list_select_clear();
1384 gtk_sctree_select( GTK_SCTREE(ctree), addrbook.opened);
1385 addressbook_set_clist(
1386 gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree),
1390 addrbook_set_dirty(abf, TRUE);
1391 addressbook_export_to_file();
1392 addressbook_list_menu_setup();
1396 gtk_ctree_node_set_row_data( clist, nodeList, NULL );
1397 gtk_ctree_remove_node( clist, nodeList );
1401 static void addressbook_reg_clicked(GtkButton *button, gpointer data)
1403 addressbook_new_address_cb( NULL, 0, NULL );
1406 gchar *addressbook_format_address( AddrItemObject * aio ) {
1409 gchar *address = NULL;
1411 if( aio->type == ADDR_ITEM_EMAIL ) {
1412 ItemPerson *person = NULL;
1413 ItemEMail *email = ( ItemEMail * ) aio;
1415 person = ( ItemPerson * ) ADDRITEM_PARENT(email);
1416 if( email->address ) {
1417 if( ADDRITEM_NAME(email) ) {
1418 name = ADDRITEM_NAME(email);
1419 if( *name == '\0' ) {
1420 name = ADDRITEM_NAME(person);
1423 else if( ADDRITEM_NAME(person) ) {
1424 name = ADDRITEM_NAME(person);
1427 buf = g_strdup( email->address );
1429 address = email->address;
1432 else if( aio->type == ADDR_ITEM_PERSON ) {
1433 ItemPerson *person = ( ItemPerson * ) aio;
1434 GList *node = person->listEMail;
1436 name = ADDRITEM_NAME(person);
1438 ItemEMail *email = ( ItemEMail * ) node->data;
1439 address = email->address;
1443 if( name && name[0] != '\0' ) {
1444 if( strchr_with_skip_quote( name, '"', ',' ) )
1445 buf = g_strdup_printf( "\"%s\" <%s>", name, address );
1447 buf = g_strdup_printf( "%s <%s>", name, address );
1450 buf = g_strdup( address );
1457 static void addressbook_to_clicked(GtkButton *button, gpointer data)
1461 AddrSelectItem *item;
1462 AddrItemObject *aio;
1465 compose = addrbook.target_compose;
1466 if( ! compose ) return;
1468 /* Nothing selected, but maybe there is something in text entry */
1469 addr = (char *)gtk_entry_get_text( GTK_ENTRY( addrbook.entry) );
1471 compose_entry_append(
1472 compose, addr, (ComposeEntryType)data );
1475 /* Select from address list */
1476 list = addrselect_get_list( _addressSelect_ );
1480 node = g_list_next( node );
1481 aio = item->addressItem;
1482 if( aio->type == ADDR_ITEM_PERSON ||
1483 aio->type == ADDR_ITEM_EMAIL ) {
1484 addr = addressbook_format_address( aio );
1485 compose_entry_append(
1486 compose, addr, (ComposeEntryType) data );
1489 else if( aio->type == ADDR_ITEM_GROUP ) {
1490 ItemGroup *group = ( ItemGroup * ) aio;
1491 AddressDataSource *ds = NULL;
1492 AddressBookFile *abf = NULL;
1493 GList *nodeMail = group->listEMail;
1494 if (nodeMail == NULL) {
1495 if(addrbook.treeSelected ) {
1496 ds = addressbook_find_datasource( addrbook.treeSelected );
1497 abf = ds->rawDataSource;
1499 nodeMail = addrbook_get_available_email_list( abf, group );
1503 ItemEMail *email = nodeMail->data;
1505 addr = addressbook_format_address(
1506 ( AddrItemObject * ) email );
1507 compose_entry_append(
1508 compose, addr, (ComposeEntryType) data );
1510 nodeMail = g_list_next( nodeMail );
1514 g_list_free( list );
1517 static void addressbook_menubar_set_sensitive( gboolean sensitive ) {
1518 menu_set_sensitive( addrbook.menu_factory, "/Book/Edit book", sensitive );
1519 menu_set_sensitive( addrbook.menu_factory, "/Book/Delete book", sensitive );
1520 menu_set_sensitive( addrbook.menu_factory, "/Book/New Folder", sensitive );
1522 menu_set_sensitive( addrbook.menu_factory, "/Address/Select all", TRUE );
1523 menu_set_sensitive( addrbook.menu_factory, "/Address/Cut", sensitive );
1524 menu_set_sensitive( addrbook.menu_factory, "/Address/Copy", sensitive );
1525 menu_set_sensitive( addrbook.menu_factory, "/Address/Paste", sensitive );
1527 menu_set_sensitive( addrbook.menu_factory, "/Address/New Address", sensitive );
1528 menu_set_sensitive( addrbook.menu_factory, "/Address/New Group", sensitive );
1529 menu_set_sensitive( addrbook.menu_factory, "/Address/Mail To", sensitive );
1530 gtk_widget_set_sensitive( addrbook.reg_btn, sensitive );
1531 gtk_widget_set_sensitive( addrbook.edit_btn, sensitive );
1532 gtk_widget_set_sensitive( addrbook.del_btn, sensitive );
1535 static void addressbook_menuitem_set_sensitive( AddressObject *obj, GtkCTreeNode *node ) {
1536 gboolean canEdit = FALSE;
1537 gboolean canDelete = TRUE;
1538 gboolean canAdd = FALSE;
1539 gboolean canEditTr = TRUE;
1540 gboolean editAddress = FALSE;
1541 gboolean canExport = TRUE;
1542 AddressTypeControlItem *atci = NULL;
1543 AddressDataSource *ds = NULL;
1544 AddressInterface *iface = NULL;
1546 if( obj == NULL ) return;
1547 if( obj->type == ADDR_INTERFACE ) {
1548 AdapterInterface *adapter = ADAPTER_INTERFACE(obj);
1549 iface = adapter->interface;
1551 if( iface->haveLibrary ) {
1552 /* Enable appropriate File / New command */
1553 atci = adapter->atci;
1554 menu_set_sensitive( addrbook.menu_factory, atci->menuCommand, TRUE );
1557 canEditTr = canExport = FALSE;
1559 else if( obj->type == ADDR_DATASOURCE ) {
1560 AdapterDSource *ads = ADAPTER_DSOURCE(obj);
1561 ds = ads->dataSource;
1562 iface = ds->interface;
1563 if( ! iface->readOnly ) {
1564 canAdd = canEdit = editAddress = canDelete = TRUE;
1566 if( ! iface->haveLibrary ) {
1567 canAdd = canEdit = editAddress = canExport = canDelete = FALSE;
1570 else if( obj->type == ADDR_ITEM_FOLDER ) {
1571 ds = addressbook_find_datasource( addrbook.treeSelected );
1573 iface = ds->interface;
1574 if( iface->readOnly ) {
1579 canAdd = editAddress = TRUE;
1583 else if( obj->type == ADDR_ITEM_GROUP ) {
1584 ds = addressbook_find_datasource( addrbook.treeSelected );
1586 iface = ds->interface;
1587 if( ! iface->readOnly ) {
1593 if( addrbook.listSelected == NULL ) canEdit = FALSE;
1596 menu_set_sensitive( addrbook.menu_factory, "/Address/New Address", editAddress );
1597 menu_set_sensitive( addrbook.menu_factory, "/Address/New Group", canAdd );
1598 menu_set_sensitive( addrbook.menu_factory, "/Book/New Folder", canAdd );
1599 gtk_widget_set_sensitive( addrbook.reg_btn, editAddress );
1602 menu_set_sensitive( addrbook.menu_factory, "/Address/Edit", canEdit );
1603 menu_set_sensitive( addrbook.menu_factory, "/Address/Delete", canDelete );
1604 gtk_widget_set_sensitive( addrbook.edit_btn, canEdit );
1605 gtk_widget_set_sensitive( addrbook.del_btn, canDelete );
1607 menu_set_sensitive( addrbook.menu_factory, "/Book/Edit book", canEditTr );
1608 menu_set_sensitive( addrbook.menu_factory, "/Book/Delete book", canEditTr );
1611 menu_set_sensitive( addrbook.menu_factory, "/Tools/Export HTML...", canExport );
1612 menu_set_sensitive( addrbook.menu_factory, "/Tools/Export LDIF...", canExport );
1615 static void addressbook_list_menu_setup( void );
1618 * Address book tree callback function that responds to selection of tree
1621 * \param ctree Tree widget.
1622 * \param node Node that was selected.
1623 * \param column Column number where selected occurred.
1624 * \param data Pointer to user data.
1626 static void addressbook_tree_selected(GtkCTree *ctree, GtkCTreeNode *node,
1627 gint column, gpointer data)
1629 AddressObject *obj = NULL;
1630 AdapterDSource *ads = NULL;
1631 AddressDataSource *ds = NULL;
1632 ItemFolder *rootFolder = NULL;
1633 AddressObjectType aot;
1635 addrbook.treeSelected = node;
1636 addrbook.listSelected = NULL;
1637 addressbook_status_show( "" );
1638 if( addrbook.entry != NULL ) gtk_entry_set_text(GTK_ENTRY(addrbook.entry), "");
1640 if( node ) obj = gtk_ctree_node_get_row_data( ctree, node );
1642 addressbook_set_clist(NULL, TRUE);
1645 addrbook.opened = node;
1647 if( obj->type == ADDR_DATASOURCE ) {
1648 /* Read from file */
1649 static gboolean tVal = TRUE;
1651 ads = ADAPTER_DSOURCE(obj);
1652 if( ads == NULL ) return;
1653 ds = ads->dataSource;
1654 if( ds == NULL ) return;
1656 if( addrindex_ds_get_modify_flag( ds ) ) {
1657 addrindex_ds_read_data( ds );
1660 if( ! addrindex_ds_get_read_flag( ds ) ) {
1661 addrindex_ds_read_data( ds );
1663 addressbook_ds_show_message( ds );
1665 if( ! addrindex_ds_get_access_flag( ds ) ) {
1666 /* Remove existing folders and groups */
1667 gtk_clist_freeze( GTK_CLIST(ctree) );
1668 addressbook_tree_remove_children( ctree, node );
1669 gtk_clist_thaw( GTK_CLIST(ctree) );
1671 /* Load folders into the tree */
1672 rootFolder = addrindex_ds_get_root_folder( ds );
1673 if( ds->type == ADDR_IF_JPILOT ) {
1674 aot = ADDR_CATEGORY;
1676 else if( ds->type == ADDR_IF_LDAP ) {
1677 aot = ADDR_LDAP_QUERY;
1680 aot = ADDR_ITEM_FOLDER;
1682 addressbook_node_add_folder( node, ds, rootFolder, aot );
1683 addrindex_ds_set_access_flag( ds, &tVal );
1684 gtk_ctree_expand( ctree, node );
1687 addressbook_set_clist(NULL, TRUE);
1690 /* Update address list */
1691 g_signal_handlers_block_by_func
1693 G_CALLBACK(addressbook_tree_selected), NULL);
1694 addressbook_set_clist( obj, FALSE );
1695 g_signal_handlers_unblock_by_func
1697 G_CALLBACK(addressbook_tree_selected), NULL);
1699 /* Setup main menu selections */
1700 addressbook_menubar_set_sensitive( FALSE );
1701 addressbook_list_menu_setup();
1702 addressbook_menuitem_set_sensitive( obj, node );
1704 addressbook_list_select_clear();
1705 addressbook_list_menu_setup();
1710 * Setup address list popup menu items. Items are enabled or disabled as
1713 static void addressbook_list_menu_setup( void ) {
1714 GtkCTree *clist = NULL;
1715 AddressObject *pobj = NULL;
1716 AddressObject *obj = NULL;
1717 AdapterDSource *ads = NULL;
1718 AddressInterface *iface = NULL;
1719 AddressDataSource *ds = NULL;
1720 gboolean canEdit = FALSE;
1721 gboolean canDelete = FALSE;
1722 gboolean canCut = FALSE;
1723 gboolean canCopy = FALSE;
1724 gboolean canPaste = FALSE;
1725 gboolean canBrowse = FALSE;
1727 pobj = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), addrbook.treeSelected );
1728 if( pobj == NULL ) return;
1730 clist = GTK_CTREE(addrbook.clist);
1731 obj = gtk_ctree_node_get_row_data( clist, addrbook.listSelected );
1732 if( obj == NULL ) canEdit = FALSE;
1734 menu_set_insensitive_all( GTK_MENU_SHELL(addrbook.list_popup) );
1735 menu_set_sensitive( addrbook.list_factory, "/Select all", TRUE );
1737 if( pobj->type == ADDR_DATASOURCE ) {
1738 /* Parent object is a data source */
1739 ads = ADAPTER_DSOURCE(pobj);
1740 ds = ads->dataSource;
1741 iface = ds->interface;
1742 if( ! iface->readOnly ) {
1743 menu_set_sensitive( addrbook.list_factory, "/New Address", TRUE );
1744 menu_set_sensitive( addrbook.list_factory, "/New Group", TRUE );
1745 gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
1746 if( ! addrclip_is_empty( _clipBoard_ ) ) canPaste = TRUE;
1747 if( ! addrselect_test_empty( _addressSelect_ ) ) canCut = TRUE;
1748 if( obj ) canEdit = TRUE;
1750 canDelete = canEdit;
1752 else if( pobj->type != ADDR_INTERFACE ) {
1753 /* Parent object is not an interface */
1754 ds = addressbook_find_datasource( addrbook.treeSelected );
1755 iface = ds->interface;
1756 if( ! iface->readOnly ) {
1757 /* Folder or group */
1758 if( pobj->type == ADDR_ITEM_FOLDER || pobj->type == ADDR_ITEM_GROUP ) {
1759 menu_set_sensitive( addrbook.list_factory, "/New Address", TRUE );
1760 gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
1761 if( obj ) canEdit = TRUE;
1764 if( pobj->type == ADDR_ITEM_FOLDER ) {
1765 menu_set_sensitive( addrbook.list_factory, "/New Group", TRUE );
1766 if( obj ) canEdit = TRUE;
1768 if( ! addrclip_is_empty( _clipBoard_ ) ) canPaste = TRUE;
1769 if( ! addrselect_test_empty( _addressSelect_ ) ) canCut = TRUE;
1770 canDelete = canEdit;
1772 if( iface->type == ADDR_IF_LDAP ) {
1773 if( obj ) canBrowse = TRUE;
1778 if( ! addrselect_test_empty( _addressSelect_ ) ) canCopy = TRUE;
1780 /* Disable edit or browse if more than one row selected */
1781 if( GTK_CLIST(clist)->selection && GTK_CLIST(clist)->selection->next ) {
1786 /* Now go finalize menu items */
1787 menu_set_sensitive( addrbook.list_factory, "/Edit", canEdit );
1788 menu_set_sensitive( addrbook.list_factory, "/Delete", canDelete );
1790 menu_set_sensitive( addrbook.list_factory, "/Cut", canCut );
1791 menu_set_sensitive( addrbook.list_factory, "/Copy", canCopy );
1792 menu_set_sensitive( addrbook.list_factory, "/Paste", canPaste );
1793 /* menu_set_sensitive( addrbook.list_factory, "/Paste Address", canPaste );*/
1795 menu_set_sensitive( addrbook.list_factory, "/Mail To", canCopy );
1797 menu_set_sensitive( addrbook.menu_factory, "/Address/Cut", canCut );
1798 menu_set_sensitive( addrbook.menu_factory, "/Address/Copy", canCopy );
1799 menu_set_sensitive( addrbook.menu_factory, "/Address/Paste", canPaste );
1800 /* menu_set_sensitive( addrbook.menu_factory, "/Edit/Paste Address", canPaste );*/
1802 menu_set_sensitive( addrbook.tree_factory, "/Cut", canCut );
1803 menu_set_sensitive( addrbook.tree_factory, "/Copy", canCopy );
1804 menu_set_sensitive( addrbook.tree_factory, "/Paste", canPaste );
1806 menu_set_sensitive( addrbook.menu_factory, "/Address/Edit", canEdit );
1807 menu_set_sensitive( addrbook.menu_factory, "/Address/Delete", canDelete );
1808 menu_set_sensitive( addrbook.menu_factory, "/Address/Mail To", canCopy );
1810 gtk_widget_set_sensitive( addrbook.edit_btn, canEdit );
1811 gtk_widget_set_sensitive( addrbook.del_btn, canDelete );
1814 menu_set_sensitive( addrbook.list_factory, "/Browse Entry", canBrowse );
1818 static void addressbook_select_row_tree (GtkCTree *ctree,
1826 * Add list of items into tree node below specified tree node.
1827 * \param treeNode Tree node.
1828 * \param ds Data source.
1829 * \param listItems List of items.
1831 static void addressbook_treenode_add_list(
1832 GtkCTreeNode *treeNode, AddressDataSource *ds, GList *listItems )
1838 AddrItemObject *aio;
1842 if( ADDRESS_OBJECT_TYPE(aio) == ITEMTYPE_GROUP ) {
1845 group = ( ItemGroup * ) aio;
1846 nn = addressbook_node_add_group( treeNode, ds, group );
1848 else if( ADDRESS_OBJECT_TYPE(aio) == ITEMTYPE_FOLDER ) {
1851 folder = ( ItemFolder * ) aio;
1852 nn = addressbook_node_add_folder(
1853 treeNode, ds, folder, ADDR_ITEM_FOLDER );
1855 node = g_list_next( node );
1859 static void addressbook_select_all_cb( void ) {
1860 gtk_clist_select_all(GTK_CLIST(addrbook.clist));
1864 * Cut from address list widget.
1866 static void addressbook_clip_cut_cb( void ) {
1867 _clipBoard_->cutFlag = TRUE;
1868 addrclip_clear( _clipBoard_ );
1869 addrclip_add( _clipBoard_, _addressSelect_ );
1870 /* addrclip_list_show( _clipBoard_, stdout ); */
1874 * Copy from address list widget.
1876 static void addressbook_clip_copy_cb( void ) {
1877 _clipBoard_->cutFlag = FALSE;
1878 addrclip_clear( _clipBoard_ );
1879 addrclip_add( _clipBoard_, _addressSelect_ );
1880 /* addrclip_list_show( _clipBoard_, stdout ); */
1884 * Paste clipboard into address list widget.
1886 static void addressbook_clip_paste_cb( void ) {
1887 GtkCTree *ctree = GTK_CTREE( addrbook.ctree );
1888 AddressObject *pobj = NULL;
1889 AddressDataSource *ds = NULL;
1890 AddressBookFile *abf = NULL;
1891 ItemFolder *folder = NULL;
1892 GList *folderGroup = NULL;
1894 ds = addressbook_find_datasource( GTK_CTREE_NODE(addrbook.treeSelected) );
1895 if( ds == NULL ) return;
1896 if( addrindex_ds_get_readonly( ds ) ) {
1897 addressbook_ds_status_message(
1898 ds, _( "Cannot paste. Target address book is readonly." ) );
1902 pobj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
1904 if( pobj->type == ADDR_ITEM_FOLDER ) {
1905 folder = ADAPTER_FOLDER(pobj)->itemFolder;
1907 else if( pobj->type == ADDR_ITEM_GROUP ) {
1908 addressbook_ds_status_message(
1909 ds, _( "Cannot paste into an address group." ) );
1914 /* Get an address book */
1915 abf = addressbook_get_book_file();
1916 if( abf == NULL ) return;
1918 if( _clipBoard_->cutFlag ) {
1920 folderGroup = addrclip_paste_cut( _clipBoard_, abf, folder );
1922 /* Remove all groups and folders in clipboard from tree node */
1923 addressbook_treenode_remove_item();
1925 /* Remove all "cut" items */
1926 addrclip_delete_item( _clipBoard_ );
1928 /* Clear clipboard - cut items??? */
1929 addrclip_clear( _clipBoard_ );
1933 folderGroup = addrclip_paste_copy( _clipBoard_, abf, folder );
1936 /* addrclip_list_show( _clipBoard_, stdout ); */
1938 /* Update tree by inserting node for each folder or group */
1939 addressbook_treenode_add_list(
1940 addrbook.treeSelected, ds, folderGroup );
1941 gtk_ctree_expand( ctree, addrbook.treeSelected );
1942 g_list_free( folderGroup );
1946 /* Display items pasted */
1947 gtk_sctree_select( GTK_SCTREE(ctree), addrbook.opened );
1948 addressbook_set_clist(
1949 gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree),
1957 * Add current treenode object to clipboard. Note that widget only allows
1958 * one entry from the tree list to be selected.
1960 static void addressbook_treenode_to_clipboard( void ) {
1961 AddressObject *obj = NULL;
1962 AddressDataSource *ds = NULL;
1963 AddrSelectItem *item;
1964 GtkCTree *ctree = GTK_CTREE( addrbook.ctree );
1967 node = addrbook.treeSelected;
1968 if( node == NULL ) return;
1969 obj = gtk_ctree_node_get_row_data( ctree, node );
1970 if( obj == NULL ) return;
1972 ds = addressbook_find_datasource( node );
1973 if( ds == NULL ) return;
1976 if( obj->type == ADDR_ITEM_FOLDER ) {
1977 AdapterFolder *adapter = ADAPTER_FOLDER(obj);
1978 ItemFolder *folder = adapter->itemFolder;
1980 item = addrselect_create_node( obj );
1981 item->uid = g_strdup( ADDRITEM_ID(folder) );
1983 else if( obj->type == ADDR_ITEM_GROUP ) {
1984 AdapterGroup *adapter = ADAPTER_GROUP(obj);
1985 ItemGroup *group = adapter->itemGroup;
1987 item = addrselect_create_node( obj );
1988 item->uid = g_strdup( ADDRITEM_ID(group) );
1990 else if( obj->type == ADDR_DATASOURCE ) {
1992 item = addrselect_create_node( obj );
1997 /* Clear existing list and add item into list */
2000 addressbook_list_select_clear();
2001 cacheID = addrindex_get_cache_id( _addressIndex_, ds );
2002 addrselect_list_add( _addressSelect_, item, cacheID );
2008 * Cut from tree widget.
2010 static void addressbook_treenode_cut_cb( void ) {
2011 _clipBoard_->cutFlag = TRUE;
2012 addressbook_treenode_to_clipboard();
2013 addrclip_clear( _clipBoard_ );
2014 addrclip_add( _clipBoard_, _addressSelect_ );
2015 /* addrclip_list_show( _clipBoard_, stdout ); */
2019 * Copy from tree widget.
2021 static void addressbook_treenode_copy_cb( void ) {
2022 _clipBoard_->cutFlag = FALSE;
2023 addressbook_treenode_to_clipboard();
2024 addrclip_clear( _clipBoard_ );
2025 addrclip_add( _clipBoard_, _addressSelect_ );
2026 /* addrclip_list_show( _clipBoard_, stdout ); */
2030 * Paste clipboard into address tree widget.
2032 static void addressbook_treenode_paste_cb( void ) {
2033 addressbook_clip_paste_cb();
2037 * Clear selected entries in clipboard.
2039 static void addressbook_list_select_clear( void ) {
2040 addrselect_list_clear( _addressSelect_ );
2044 * Add specified address item to selected address list.
2045 * \param aio Address item object.
2046 * \param ds Datasource.
2048 static void addressbook_list_select_add( AddrItemObject *aio, AddressDataSource *ds ) {
2051 if( ds == NULL ) return;
2052 cacheID = addrindex_get_cache_id( _addressIndex_, ds );
2053 addrselect_list_add_obj( _addressSelect_, aio, cacheID );
2058 * Remove specified address item from selected address list.
2059 * \param aio Address item object.
2061 static void addressbook_list_select_remove( AddrItemObject *aio ) {
2062 addrselect_list_remove( _addressSelect_, aio );
2066 * Invoke EMail compose window with addresses in selected address list.
2068 static void addressbook_mail_to_cb( void ) {
2071 if( ! addrselect_test_empty( _addressSelect_ ) ) {
2072 listAddress = addrselect_build_list( _addressSelect_ );
2073 compose_new_with_list( NULL, listAddress );
2074 mgu_free_dlist( listAddress );
2079 static void addressbook_list_row_selected( GtkCTree *clist,
2084 GtkEntry *entry = GTK_ENTRY(addrbook.entry);
2085 AddrItemObject *aio = NULL;
2086 AddressObject *pobj = NULL;
2087 AdapterDSource *ads = NULL;
2088 AddressDataSource *ds = NULL;
2090 gtk_entry_set_text( entry, "" );
2091 addrbook.listSelected = node;
2093 pobj = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), addrbook.treeSelected );
2094 if( pobj == NULL ) return;
2096 if( pobj->type == ADDR_DATASOURCE ) {
2097 ads = ADAPTER_DSOURCE(pobj);
2098 ds = ads->dataSource;
2100 else if( pobj->type != ADDR_INTERFACE ) {
2101 ds = addressbook_find_datasource( addrbook.treeSelected );
2104 aio = gtk_ctree_node_get_row_data( clist, node );
2106 /* printf( "list select: %d : '%s'\n", aio->type, aio->name ); */
2107 addressbook_list_select_add( aio, ds );
2110 addressbook_list_menu_setup();
2113 static void addressbook_list_row_unselected( GtkCTree *ctree,
2118 AddrItemObject *aio;
2120 aio = gtk_ctree_node_get_row_data( ctree, node );
2122 /* printf( "list unselect: %d : '%s'\n", aio->type, aio->name ); */
2123 addressbook_list_select_remove( aio );
2127 /* from gdkevents.c */
2128 #define DOUBLE_CLICK_TIME 250
2130 static gboolean addressbook_list_button_pressed(GtkWidget *widget,
2131 GdkEventButton *event,
2134 static guint32 lasttime = 0;
2135 if( ! event ) return FALSE;
2137 addressbook_list_menu_setup();
2139 if( event->button == 3 ) {
2140 gtk_menu_popup( GTK_MENU(addrbook.list_popup), NULL, NULL, NULL, NULL,
2141 event->button, event->time );
2142 } else if (event->button == 1) {
2143 if (event->time - lasttime < DOUBLE_CLICK_TIME) {
2144 if (prefs_common.add_address_by_click &&
2145 addrbook.target_compose)
2146 addressbook_to_clicked(NULL, GINT_TO_POINTER(COMPOSE_TO));
2148 addressbook_edit_address_cb(NULL, 0, NULL);
2152 lasttime = event->time;
2158 static gboolean addressbook_list_button_released(GtkWidget *widget,
2159 GdkEventButton *event,
2165 static gboolean addressbook_tree_button_pressed(GtkWidget *ctree,
2166 GdkEventButton *event,
2169 GtkCList *clist = GTK_CLIST(ctree);
2171 AddressObject *obj = NULL;
2172 AdapterDSource *ads = NULL;
2173 AddressInterface *iface = NULL;
2174 AddressDataSource *ds = NULL;
2175 gboolean canEdit = FALSE;
2176 gboolean canDelete = FALSE;
2177 gboolean canCut = FALSE;
2178 gboolean canCopy = FALSE;
2179 gboolean canPaste = FALSE;
2180 gboolean canTreeCut = FALSE;
2181 gboolean canTreeCopy = FALSE;
2182 gboolean canTreePaste = FALSE;
2183 gboolean canLookup = FALSE;
2185 if( ! event ) return FALSE;
2186 addressbook_menubar_set_sensitive( FALSE );
2188 if( gtk_clist_get_selection_info( clist, event->x, event->y, &row, &column ) ) {
2189 gtkut_clist_set_focus_row(clist, row);
2190 obj = gtk_clist_get_row_data( clist, row );
2193 menu_set_insensitive_all(GTK_MENU_SHELL(addrbook.tree_popup));
2195 if( obj == NULL ) return FALSE;
2197 if( ! addrclip_is_empty( _clipBoard_ ) ) {
2198 canTreePaste = TRUE;
2201 if (obj->type == ADDR_DATASOURCE) {
2202 ads = ADAPTER_DSOURCE(obj);
2203 ds = ads->dataSource;
2206 iface = ds->interface;
2211 if( iface->readOnly ) {
2212 canTreePaste = FALSE;
2215 menu_set_sensitive( addrbook.tree_factory, "/New Folder", TRUE );
2216 gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
2219 if( iface->externalQuery ) canLookup = TRUE;
2221 else if (obj->type == ADDR_ITEM_FOLDER) {
2222 ds = addressbook_find_datasource( addrbook.treeSelected );
2225 iface = ds->interface;
2228 if( iface->readOnly ) {
2229 canTreePaste = FALSE;
2235 menu_set_sensitive( addrbook.tree_factory, "/New Folder", TRUE );
2236 gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
2240 if( iface->externalQuery ) {
2241 /* Enable deletion of LDAP folder */
2246 else if (obj->type == ADDR_ITEM_GROUP) {
2247 ds = addressbook_find_datasource( addrbook.treeSelected );
2250 iface = ds->interface;
2253 if( ! iface->readOnly ) {
2256 menu_set_sensitive( addrbook.list_factory, "/New Address", TRUE );
2257 gtk_widget_set_sensitive( addrbook.reg_btn, TRUE );
2260 else if (obj->type == ADDR_INTERFACE) {
2261 canTreePaste = FALSE;
2265 if( ! addrselect_test_empty( _addressSelect_ ) ) canCut = TRUE;
2267 if( ! addrselect_test_empty( _addressSelect_ ) ) canCopy = TRUE;
2268 if( ! addrclip_is_empty( _clipBoard_ ) ) canPaste = TRUE;
2272 menu_set_sensitive( addrbook.tree_factory, "/Edit", canEdit );
2273 menu_set_sensitive( addrbook.tree_factory, "/Delete", canDelete );
2274 menu_set_sensitive( addrbook.tree_factory, "/Cut", canTreeCut );
2275 menu_set_sensitive( addrbook.tree_factory, "/Copy", canTreeCopy );
2276 menu_set_sensitive( addrbook.tree_factory, "/Paste", canTreePaste );
2278 menu_set_sensitive( addrbook.menu_factory, "/Book/Edit book", canEdit );
2279 menu_set_sensitive( addrbook.menu_factory, "/Book/Delete book", canEdit );
2280 menu_set_sensitive( addrbook.menu_factory, "/Address/Cut", canCut );
2281 menu_set_sensitive( addrbook.menu_factory, "/Address/Copy", canCopy );
2282 menu_set_sensitive( addrbook.menu_factory, "/Address/Paste", canPaste );
2283 /* menu_set_sensitive( addrbook.menu_factory, "/Edit/Paste Address", canPaste );*/
2285 addressbook_show_buttons(addrbook.target_compose == NULL, canLookup, addrbook.target_compose != NULL);
2286 if( event->button == 3 ) {
2287 gtk_menu_popup(GTK_MENU(addrbook.tree_popup), NULL, NULL, NULL, NULL,
2288 event->button, event->time);
2294 static gboolean addressbook_tree_button_released(GtkWidget *ctree,
2295 GdkEventButton *event,
2298 gtkut_ctree_set_focus_row(GTK_CTREE(addrbook.ctree), addrbook.opened);
2302 static void addressbook_new_folder_cb(gpointer data, guint action,
2305 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2306 AddressObject *obj = NULL;
2307 AddressDataSource *ds = NULL;
2308 AddressBookFile *abf = NULL;
2309 ItemFolder *parentFolder = NULL;
2310 ItemFolder *folder = NULL;
2312 if( ! addrbook.treeSelected ) return;
2313 obj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
2314 if( obj == NULL ) return;
2315 ds = addressbook_find_datasource( addrbook.treeSelected );
2316 if( ds == NULL ) return;
2318 if( obj->type == ADDR_DATASOURCE ) {
2319 if( ADAPTER_DSOURCE(obj)->subType != ADDR_BOOK ) return;
2321 else if( obj->type == ADDR_ITEM_FOLDER ) {
2322 parentFolder = ADAPTER_FOLDER(obj)->itemFolder;
2328 abf = ds->rawDataSource;
2329 if( abf == NULL ) return;
2330 folder = addressbook_edit_folder( abf, parentFolder, NULL );
2333 nn = addressbook_node_add_folder(
2334 addrbook.treeSelected, ds, folder, ADDR_ITEM_FOLDER );
2335 gtk_ctree_expand( ctree, addrbook.treeSelected );
2336 if( addrbook.treeSelected == addrbook.opened )
2337 addressbook_set_clist(obj, TRUE);
2342 static void addressbook_new_group_cb(gpointer data, guint action,
2345 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2346 AddressObject *obj = NULL;
2347 AddressDataSource *ds = NULL;
2348 AddressBookFile *abf = NULL;
2349 ItemFolder *parentFolder = NULL;
2350 ItemGroup *group = NULL;
2352 if( ! addrbook.treeSelected ) return;
2353 obj = gtk_ctree_node_get_row_data(ctree, addrbook.treeSelected);
2354 if( obj == NULL ) return;
2355 ds = addressbook_find_datasource( addrbook.treeSelected );
2356 if( ds == NULL ) return;
2358 if( obj->type == ADDR_DATASOURCE ) {
2359 if( ADAPTER_DSOURCE(obj)->subType != ADDR_BOOK ) return;
2361 else if( obj->type == ADDR_ITEM_FOLDER ) {
2362 parentFolder = ADAPTER_FOLDER(obj)->itemFolder;
2368 abf = ds->rawDataSource;
2369 if( abf == NULL ) return;
2370 group = addressbook_edit_group( abf, parentFolder, NULL );
2373 nn = addressbook_node_add_group( addrbook.treeSelected, ds, group );
2374 gtk_ctree_expand( ctree, addrbook.treeSelected );
2375 if( addrbook.treeSelected == addrbook.opened )
2376 addressbook_set_clist(obj, TRUE);
2381 static void addressbook_change_node_name(GtkCTreeNode *node, const gchar *name)
2383 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2386 GdkPixmap *pix_cl, *pix_op;
2387 GdkBitmap *mask_cl, *mask_op;
2388 gboolean is_leaf, expanded;
2390 gtk_ctree_get_node_info(ctree, node, text, &spacing,
2391 &pix_cl, &mask_cl, &pix_op, &mask_op,
2392 &is_leaf, &expanded);
2393 gtk_sctree_set_node_info(ctree, node, name, spacing,
2394 pix_cl, mask_cl, pix_op, mask_op,
2400 * \param obj Address object to edit.
2401 * \param node Node in tree.
2402 * \return New name of data source.
2404 static gchar *addressbook_edit_datasource( AddressObject *obj, GtkCTreeNode *node ) {
2405 gchar *newName = NULL;
2406 AddressDataSource *ds = NULL;
2407 AddressInterface *iface = NULL;
2408 AdapterDSource *ads = NULL;
2410 ds = addressbook_find_datasource( node );
2411 if( ds == NULL ) return NULL;
2412 iface = ds->interface;
2413 if( ! iface->haveLibrary ) return NULL;
2415 /* Read data from data source */
2416 if( addrindex_ds_get_modify_flag( ds ) ) {
2417 addrindex_ds_read_data( ds );
2420 if( ! addrindex_ds_get_read_flag( ds ) ) {
2421 addrindex_ds_read_data( ds );
2425 ads = ADAPTER_DSOURCE(obj);
2426 if( ads->subType == ADDR_BOOK ) {
2427 if( addressbook_edit_book( _addressIndex_, ads ) == NULL ) return NULL;
2429 else if( ads->subType == ADDR_VCARD ) {
2430 if( addressbook_edit_vcard( _addressIndex_, ads ) == NULL ) return NULL;
2433 else if( ads->subType == ADDR_JPILOT ) {
2434 if( addressbook_edit_jpilot( _addressIndex_, ads ) == NULL ) return NULL;
2438 else if( ads->subType == ADDR_LDAP ) {
2439 if( addressbook_edit_ldap( _addressIndex_, ads ) == NULL ) return NULL;
2445 newName = obj->name;
2450 * Edit an object that is in the address tree area.
2452 static void addressbook_treenode_edit_cb(gpointer data, guint action,
2455 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2457 AddressDataSource *ds = NULL;
2458 AddressBookFile *abf = NULL;
2459 GtkCTreeNode *node = NULL, *parentNode = NULL;
2462 if( ! addrbook.treeSelected ) return;
2463 node = addrbook.treeSelected;
2464 if( GTK_CTREE_ROW(node)->level == 1 ) return;
2465 obj = gtk_ctree_node_get_row_data( ctree, node );
2466 if( obj == NULL ) return;
2467 parentNode = GTK_CTREE_ROW(node)->parent;
2469 ds = addressbook_find_datasource( node );
2470 if( ds == NULL ) return;
2472 if( obj->type == ADDR_DATASOURCE ) {
2473 name = addressbook_edit_datasource( obj, node );
2474 if( name == NULL ) return;
2477 abf = ds->rawDataSource;
2478 if( abf == NULL ) return;
2479 if( obj->type == ADDR_ITEM_FOLDER ) {
2480 AdapterFolder *adapter = ADAPTER_FOLDER(obj);
2481 ItemFolder *item = adapter->itemFolder;
2482 ItemFolder *parentFolder = NULL;
2483 parentFolder = ( ItemFolder * ) ADDRITEM_PARENT(item);
2484 if( addressbook_edit_folder( abf, parentFolder, item ) == NULL ) return;
2485 name = ADDRITEM_NAME(item);
2487 else if( obj->type == ADDR_ITEM_GROUP ) {
2488 AdapterGroup *adapter = ADAPTER_GROUP(obj);
2489 ItemGroup *item = adapter->itemGroup;
2490 ItemFolder *parentFolder = NULL;
2491 parentFolder = ( ItemFolder * ) ADDRITEM_PARENT(item);
2492 if( addressbook_edit_group( abf, parentFolder, item ) == NULL ) return;
2493 name = ADDRITEM_NAME(item);
2496 if( name && parentNode ) {
2497 /* Update node in tree view */
2498 addressbook_change_node_name( node, name );
2499 gtk_sctree_sort_node(ctree, parentNode);
2500 gtk_ctree_expand( ctree, node );
2501 gtk_sctree_select( GTK_SCTREE( ctree), node );
2508 ADDRTREE_DEL_FOLDER_ONLY,
2509 ADDRTREE_DEL_FOLDER_ADDR
2513 * Delete an item from the tree widget.
2514 * \param data Data passed in.
2515 * \param action Action.
2516 * \param widget Widget issuing callback.
2518 static void addressbook_treenode_delete_cb(
2519 gpointer data, guint action, GtkWidget *widget )
2521 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
2522 GtkCTreeNode *node = NULL;
2526 AddrBookBase *adbase;
2527 AddressCache *cache;
2528 AdapterDSource *ads = NULL;
2529 AddressInterface *iface = NULL;
2530 AddressDataSource *ds = NULL;
2531 gboolean remFlag = FALSE;
2532 TreeItemDelType delType;
2534 if( ! addrbook.treeSelected ) return;
2535 node = addrbook.treeSelected;
2536 if( GTK_CTREE_ROW(node)->level == 1 ) return;
2538 obj = gtk_ctree_node_get_row_data( ctree, node );
2539 g_return_if_fail(obj != NULL);
2541 if( obj->type == ADDR_DATASOURCE ) {
2542 ads = ADAPTER_DSOURCE(obj);
2543 if( ads == NULL ) return;
2544 ds = ads->dataSource;
2545 if( ds == NULL ) return;
2548 /* Must be folder or something else */
2549 ds = addressbook_find_datasource( node );
2550 if( ds == NULL ) return;
2552 /* Only allow deletion from non-readOnly */
2553 iface = ds->interface;
2554 if( iface->readOnly ) {
2555 /* Allow deletion of query results */
2556 if( ! iface->externalQuery ) return;
2560 /* Confirm deletion */
2561 delType = ADDRTREE_DEL_NONE;
2562 if( obj->type == ADDR_ITEM_FOLDER ) {
2563 if( iface->externalQuery ) {
2564 message = g_strdup_printf( _(
2565 "Do you want to delete the query " \
2566 "results and addresses in '%s' ?" ),
2568 aval = alertpanel( _("Delete"), message,
2569 GTK_STOCK_CANCEL, "+"GTK_STOCK_DELETE, NULL );
2571 if( aval == G_ALERTALTERNATE ) {
2572 delType = ADDRTREE_DEL_FOLDER_ADDR;
2576 message = g_strdup_printf
2577 ( _( "Do you want to delete '%s' ?"
2578 "If you delete the folder only, the addresses it contains will be moved into the parent folder." ),
2580 aval = alertpanel( _("Delete folder"), message,
2581 GTK_STOCK_CANCEL, _("+Delete _folder only"), _("Delete folder and _addresses"));
2583 if( aval == G_ALERTALTERNATE ) {
2584 delType = ADDRTREE_DEL_FOLDER_ONLY;
2586 else if( aval == G_ALERTOTHER ) {
2587 delType = ADDRTREE_DEL_FOLDER_ADDR;
2592 message = g_strdup_printf(_("Do you want to delete '%s'?\n"
2593 "The addresses it contains will be lost."), obj->name);
2594 aval = alertpanel(_("Delete"), message, GTK_STOCK_CANCEL,
2595 "+" GTK_STOCK_DELETE, NULL);
2597 if( aval == G_ALERTALTERNATE ) delType = ADDRTREE_DEL_DATA;
2599 if( delType == ADDRTREE_DEL_NONE ) return;
2601 /* Proceed with deletion */
2602 if( obj->type == ADDR_DATASOURCE ) {
2603 /* Remove node from tree */
2604 gtk_ctree_remove_node( ctree, node );
2606 /* Remove data source. */
2607 if( addrindex_index_remove_datasource( _addressIndex_, ds ) ) {
2608 addrindex_free_datasource( ds );
2613 /* Get reference to cache */
2614 adbase = ( AddrBookBase * ) ds->rawDataSource;
2615 if( adbase == NULL ) return;
2616 cache = adbase->addressCache;
2618 /* Remove query results folder */
2619 if( iface->externalQuery ) {
2620 AdapterFolder *adapter = ADAPTER_FOLDER(obj);
2621 ItemFolder *folder = adapter->itemFolder;
2623 adapter->itemFolder = NULL;
2625 printf( "remove folder for ::%s::\n", obj->name );
2626 printf( " folder name ::%s::\n", ADDRITEM_NAME(folder) );
2627 printf( "-------------- remove results\n" );
2629 addrindex_remove_results( ds, folder );
2630 /* printf( "-------------- remove node\n" ); */
2631 gtk_ctree_remove_node( ctree, node );
2635 /* Code below is valid for regular address book deletion */
2636 if( obj->type == ADDR_ITEM_FOLDER ) {
2637 AdapterFolder *adapter = ADAPTER_FOLDER(obj);
2638 ItemFolder *item = adapter->itemFolder;
2640 if( delType == ADDRTREE_DEL_FOLDER_ONLY ) {
2641 /* Remove folder only */
2642 item = addrcache_remove_folder( cache, item );
2644 addritem_free_item_folder( item );
2645 addressbook_move_nodes_up( ctree, node );
2649 else if( delType == ADDRTREE_DEL_FOLDER_ADDR ) {
2650 /* Remove folder and addresses */
2651 item = addrcache_remove_folder_delete( cache, item );
2653 addritem_free_item_folder( item );
2658 else if( obj->type == ADDR_ITEM_GROUP ) {
2659 AdapterGroup *adapter = ADAPTER_GROUP(obj);
2660 ItemGroup *item = adapter->itemGroup;
2662 item = addrcache_remove_group( cache, item );
2664 addritem_free_item_group( item );
2671 gtk_ctree_remove_node(ctree, node );
2675 static void addressbook_new_address_cb( gpointer data, guint action, GtkWidget *widget ) {
2676 AddressObject *pobj = NULL;
2677 AddressDataSource *ds = NULL;
2678 AddressBookFile *abf = NULL;
2680 pobj = gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree), addrbook.treeSelected);
2681 if( pobj == NULL ) return;
2682 ds = addressbook_find_datasource( GTK_CTREE_NODE(addrbook.treeSelected) );
2683 if( ds == NULL ) return;
2685 abf = ds->rawDataSource;
2686 if( abf == NULL ) return;
2688 if( pobj->type == ADDR_DATASOURCE ) {
2689 if( ADAPTER_DSOURCE(pobj)->subType == ADDR_BOOK ) {
2691 ItemPerson *person = addressbook_edit_person( abf, NULL, NULL, FALSE );
2692 if( person && addrbook.treeSelected == addrbook.opened ) {
2693 gtk_clist_unselect_all( GTK_CLIST(addrbook.clist) );
2694 addressbook_folder_refresh_one_person(
2695 GTK_CTREE(addrbook.clist), person );
2699 else if( pobj->type == ADDR_ITEM_FOLDER ) {
2701 ItemFolder *folder = ADAPTER_FOLDER(pobj)->itemFolder;
2702 ItemPerson *person = addressbook_edit_person( abf, folder, NULL, FALSE );
2704 if (addrbook.treeSelected == addrbook.opened) {
2705 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), addrbook.opened );
2706 addressbook_set_clist(
2707 gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree),
2713 else if( pobj->type == ADDR_ITEM_GROUP ) {
2714 /* New address in group */
2715 ItemGroup *group = ADAPTER_GROUP(pobj)->itemGroup;
2716 if( addressbook_edit_group( abf, NULL, group ) == NULL ) return;
2717 if (addrbook.treeSelected == addrbook.opened) {
2718 /* Change node name in tree. */
2719 addressbook_change_node_name( addrbook.treeSelected, ADDRITEM_NAME(group) );
2720 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), addrbook.opened );
2721 addressbook_set_clist(
2722 gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree),
2730 * Search for specified child group node in address index tree.
2731 * \param parent Parent node.
2732 * \param group Group to find.
2734 static GtkCTreeNode *addressbook_find_group_node( GtkCTreeNode *parent, ItemGroup *group ) {
2735 GtkCTreeNode *node = NULL;
2736 GtkCTreeRow *currRow;
2738 currRow = GTK_CTREE_ROW( parent );
2740 node = currRow->children;
2744 obj = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), node );
2745 if( obj->type == ADDR_ITEM_GROUP ) {
2746 ItemGroup *g = ADAPTER_GROUP(obj)->itemGroup;
2747 if( g == group ) return node;
2749 currRow = GTK_CTREE_ROW(node);
2750 node = currRow->sibling;
2756 static AddressBookFile *addressbook_get_book_file() {
2757 AddressBookFile *abf = NULL;
2758 AddressDataSource *ds = NULL;
2760 ds = addressbook_find_datasource( addrbook.treeSelected );
2761 if( ds == NULL ) return NULL;
2762 if( ds->type == ADDR_IF_BOOK ) abf = ds->rawDataSource;
2766 static void addressbook_tree_remove_children( GtkCTree *ctree, GtkCTreeNode *parent ) {
2770 /* Remove existing folders and groups */
2771 row = GTK_CTREE_ROW( parent );
2773 while( (node = row->children) ) {
2774 gtk_ctree_remove_node( ctree, node );
2779 static void addressbook_move_nodes_up( GtkCTree *ctree, GtkCTreeNode *node ) {
2780 GtkCTreeNode *parent, *child;
2781 GtkCTreeRow *currRow;
2782 currRow = GTK_CTREE_ROW( node );
2784 parent = currRow->parent;
2785 while( (child = currRow->children) ) {
2786 gtk_ctree_move( ctree, child, parent, node );
2788 gtk_sctree_sort_node( ctree, parent );
2792 static void addressbook_edit_address_cb( gpointer data, guint action, GtkWidget *widget ) {
2793 GtkCTree *clist = GTK_CTREE(addrbook.clist);
2795 AddressObject *obj = NULL, *pobj = NULL;
2796 AddressDataSource *ds = NULL;
2797 GtkCTreeNode *node = NULL, *parentNode = NULL;
2799 AddressBookFile *abf = NULL;
2801 if( addrbook.listSelected == NULL ) return;
2802 obj = gtk_ctree_node_get_row_data( clist, addrbook.listSelected );
2803 g_return_if_fail(obj != NULL);
2805 ctree = GTK_CTREE( addrbook.ctree );
2806 pobj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
2807 node = gtk_ctree_find_by_row_data( ctree, addrbook.treeSelected, obj );
2809 ds = addressbook_find_datasource( GTK_CTREE_NODE(addrbook.treeSelected) );
2810 if( ds == NULL ) return;
2812 abf = addressbook_get_book_file();
2814 if( obj->type == ADDR_ITEM_EMAIL ) {
2815 ItemEMail *email = ( ItemEMail * ) obj;
2816 if( email == NULL ) return;
2817 if( pobj && pobj->type == ADDR_ITEM_GROUP ) {
2818 /* Edit parent group */
2819 AdapterGroup *adapter = ADAPTER_GROUP(pobj);
2820 ItemGroup *itemGrp = adapter->itemGroup;
2821 if( abf == NULL ) return;
2822 if( addressbook_edit_group( abf, NULL, itemGrp ) == NULL ) return;
2823 name = ADDRITEM_NAME(itemGrp);
2824 node = addrbook.treeSelected;
2825 parentNode = GTK_CTREE_ROW(node)->parent;
2828 /* Edit person - email page */
2830 person = ( ItemPerson * ) ADDRITEM_PARENT(email);
2831 if( addressbook_edit_person( abf, NULL, person, TRUE ) == NULL ) return;
2832 addressbook_folder_refresh_one_person( clist, person );
2833 invalidate_address_completion();
2837 else if( obj->type == ADDR_ITEM_PERSON ) {
2838 /* Edit person - basic page */
2839 ItemPerson *person = ( ItemPerson * ) obj;
2840 if( addressbook_edit_person( abf, NULL, person, FALSE ) == NULL ) return;
2841 invalidate_address_completion();
2842 addressbook_folder_refresh_one_person( clist, person );
2845 else if( obj->type == ADDR_ITEM_GROUP ) {
2846 ItemGroup *itemGrp = ( ItemGroup * ) obj;
2847 if( addressbook_edit_group( abf, NULL, itemGrp ) == NULL ) return;
2848 parentNode = addrbook.treeSelected;
2849 node = addressbook_find_group_node( parentNode, itemGrp );
2850 name = ADDRITEM_NAME(itemGrp);
2856 /* Update tree node with node name */
2857 if( node == NULL ) return;
2858 addressbook_change_node_name( node, name );
2859 gtk_sctree_sort_node( ctree, parentNode );
2860 gtk_sctree_select( GTK_SCTREE(ctree), addrbook.opened );
2861 addressbook_set_clist(
2862 gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree),
2867 static void addressbook_delete_address_cb(gpointer data, guint action,
2870 gtk_clist_freeze(GTK_CLIST(addrbook.clist));
2871 addressbook_del_clicked(NULL, NULL);
2872 gtk_clist_thaw(GTK_CLIST(addrbook.clist));
2875 static void close_cb(gpointer data, guint action, GtkWidget *widget)
2877 addressbook_close();
2880 static void addressbook_file_save_cb( gpointer data, guint action, GtkWidget *widget ) {
2881 addressbook_export_to_file();
2884 static void addressbook_person_expand_node( GtkCTree *ctree, GList *node, gpointer *data ) {
2886 ItemPerson *person = gtk_ctree_node_get_row_data( ctree, GTK_CTREE_NODE(node) );
2887 if( person ) addritem_person_set_opened( person, TRUE );
2891 static void addressbook_person_collapse_node( GtkCTree *ctree, GList *node, gpointer *data ) {
2893 ItemPerson *person = gtk_ctree_node_get_row_data( ctree, GTK_CTREE_NODE(node) );
2894 if( person ) addritem_person_set_opened( person, FALSE );
2898 static gchar *addressbook_format_item_clist( ItemPerson *person, ItemEMail *email ) {
2900 gchar *eMailAlias = ADDRITEM_NAME(email);
2901 if( eMailAlias && *eMailAlias != '\0' ) {
2903 str = g_strdup_printf( "%s - %s", ADDRITEM_NAME(person), eMailAlias );
2906 str = g_strdup( eMailAlias );
2912 static void addressbook_load_group( GtkCTree *clist, ItemGroup *itemGroup ) {
2913 GList *items = itemGroup->listEMail;
2914 AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_EMAIL );
2915 for( ; items != NULL; items = g_list_next( items ) ) {
2916 GtkCTreeNode *nodeEMail = NULL;
2917 gchar *text[N_LIST_COLS];
2918 ItemEMail *email = items->data;
2922 if( ! email ) continue;
2924 person = ( ItemPerson * ) ADDRITEM_PARENT(email);
2925 str = addressbook_format_item_clist( person, email );
2927 text[COL_NAME] = str;
2930 text[COL_NAME] = ADDRITEM_NAME(person);
2932 text[COL_ADDRESS] = email->address;
2933 text[COL_REMARKS] = email->remarks;
2934 nodeEMail = gtk_sctree_insert_node(
2936 text, FOLDER_SPACING,
2937 atci->iconXpm, atci->maskXpm,
2938 atci->iconXpmOpen, atci->maskXpmOpen,
2940 gtk_ctree_node_set_row_data( clist, nodeEMail, email );
2946 static void addressbook_folder_load_one_person(
2947 GtkCTree *clist, ItemPerson *person,
2948 AddressTypeControlItem *atci,
2949 AddressTypeControlItem *atciMail )
2951 GtkCTreeNode *nodePerson = NULL;
2952 GtkCTreeNode *nodeEMail = NULL;
2953 gchar *text[N_LIST_COLS];
2954 gboolean flgFirst = TRUE, haveAddr = FALSE;
2957 if( person == NULL ) return;
2959 text[COL_NAME] = "";
2960 node = person->listEMail;
2962 ItemEMail *email = node->data;
2963 gchar *eMailAddr = NULL;
2964 node = g_list_next( node );
2966 text[COL_ADDRESS] = email->address;
2967 text[COL_REMARKS] = email->remarks;
2968 eMailAddr = ADDRITEM_NAME(email);
2969 if( eMailAddr && *eMailAddr == '\0' ) eMailAddr = NULL;
2971 /* First email belongs with person */
2972 gchar *str = addressbook_format_item_clist( person, email );
2974 text[COL_NAME] = str;
2977 text[COL_NAME] = ADDRITEM_NAME(person);
2979 nodePerson = gtk_sctree_insert_node(
2981 text, FOLDER_SPACING,
2982 atci->iconXpm, atci->maskXpm,
2983 atci->iconXpmOpen, atci->maskXpmOpen,
2984 FALSE, person->isOpened );
2987 gtk_ctree_node_set_row_data(clist, nodePerson, person );
2990 /* Subsequent email is a child node of person */
2991 text[COL_NAME] = ADDRITEM_NAME(email);
2992 nodeEMail = gtk_sctree_insert_node(
2993 clist, nodePerson, NULL,
2994 text, FOLDER_SPACING,
2995 atciMail->iconXpm, atciMail->maskXpm,
2996 atciMail->iconXpmOpen, atciMail->maskXpmOpen,
2998 gtk_ctree_node_set_row_data(clist, nodeEMail, email );
3004 /* Have name without EMail */
3005 text[COL_NAME] = ADDRITEM_NAME(person);
3006 text[COL_ADDRESS] = "";
3007 text[COL_REMARKS] = "";
3008 nodePerson = gtk_sctree_insert_node(
3010 text, FOLDER_SPACING,
3011 atci->iconXpm, atci->maskXpm,
3012 atci->iconXpmOpen, atci->maskXpmOpen,
3013 FALSE, person->isOpened );
3014 gtk_ctree_node_set_row_data(clist, nodePerson, person );
3019 static void addressbook_folder_load_person( GtkCTree *clist, ItemFolder *itemFolder ) {
3021 AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_PERSON );
3022 AddressTypeControlItem *atciMail = addrbookctl_lookup( ADDR_ITEM_EMAIL );
3024 if( atci == NULL ) return;
3025 if( atciMail == NULL ) return;
3027 /* Load email addresses */
3028 items = addritem_folder_get_person_list( itemFolder );
3029 for( ; items != NULL; items = g_list_next( items ) ) {
3030 addressbook_folder_load_one_person( clist, items->data, atci, atciMail );
3032 /* Free up the list */
3033 mgu_clear_list( items );
3034 g_list_free( items );
3037 static void addressbook_folder_remove_node( GtkCTree *clist, GtkCTreeNode *node ) {
3038 addrbook.listSelected = NULL;
3039 gtk_ctree_remove_node( clist, node );
3040 addressbook_menubar_set_sensitive( FALSE );
3041 addressbook_menuitem_set_sensitive(
3042 gtk_ctree_node_get_row_data(
3043 GTK_CTREE(clist), addrbook.treeSelected ),
3044 addrbook.treeSelected );
3047 static void addressbook_folder_refresh_one_person( GtkCTree *clist, ItemPerson *person ) {
3048 AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_PERSON );
3049 AddressTypeControlItem *atciMail = addrbookctl_lookup( ADDR_ITEM_EMAIL );
3051 if( atci == NULL ) return;
3052 if( atciMail == NULL ) return;
3053 if( person == NULL ) return;
3054 /* unload the person */
3056 node = gtk_ctree_find_by_row_data( clist, NULL, person );
3058 addressbook_folder_remove_node( clist, node );
3059 addressbook_folder_load_one_person( clist, person, atci, atciMail );
3060 gtk_sctree_sort_node( clist, NULL );
3061 node = gtk_ctree_find_by_row_data( clist, NULL, person );
3063 gtk_sctree_select( GTK_SCTREE(clist), node );
3064 if (!gtk_ctree_node_is_visible( clist, node ) )
3065 gtk_ctree_node_moveto( clist, node, 0, 0, 0 );
3069 static void addressbook_folder_remove_one_person( GtkCTree *clist, ItemPerson *person ) {
3073 if( person == NULL ) return;
3074 node = gtk_ctree_find_by_row_data( clist, NULL, person );
3075 row = gtk_clist_find_row_from_data( GTK_CLIST(clist), person );
3077 addressbook_folder_remove_node( clist, node );
3081 static void addressbook_folder_load_group( GtkCTree *clist, ItemFolder *itemFolder ) {
3083 AddressTypeControlItem *atci = addrbookctl_lookup( ADDR_ITEM_GROUP );
3085 /* Load any groups */
3086 if( ! atci ) return;
3087 items = addritem_folder_get_group_list( itemFolder );
3088 for( ; items != NULL; items = g_list_next( items ) ) {
3089 GtkCTreeNode *nodeGroup = NULL;
3090 gchar *text[N_LIST_COLS];
3091 ItemGroup *group = items->data;
3092 if( group == NULL ) continue;
3093 text[COL_NAME] = ADDRITEM_NAME(group);
3094 text[COL_ADDRESS] = "";
3095 text[COL_REMARKS] = "";
3096 nodeGroup = gtk_sctree_insert_node(clist, NULL, NULL,
3097 text, FOLDER_SPACING,
3098 atci->iconXpm, atci->maskXpm,
3099 atci->iconXpmOpen, atci->maskXpmOpen,
3101 gtk_ctree_node_set_row_data(clist, nodeGroup, group );
3102 gtk_sctree_sort_node(clist, NULL);
3104 /* Free up the list */
3105 mgu_clear_list( items );
3106 g_list_free( items );
3110 * Search ctree widget callback function.
3111 * \param pA Pointer to node.
3112 * \param pB Pointer to data item being sought.
3113 * \return Zero (0) if group found.
3115 static int addressbook_treenode_find_group_cb( gconstpointer pA, gconstpointer pB ) {
3118 aoA = ( AddressObject * ) pA;
3119 if( aoA->type == ADDR_ITEM_GROUP ) {
3120 ItemGroup *group, *grp;
3122 grp = ADAPTER_GROUP(aoA)->itemGroup;
3123 group = ( ItemGroup * ) pB;
3124 if( grp == group ) return 0; /* Found group */
3130 * Search ctree widget callback function.
3131 * \param pA Pointer to node.
3132 * \param pB Pointer to data item being sought.
3133 * \return Zero (0) if folder found.
3135 static int addressbook_treenode_find_folder_cb( gconstpointer pA, gconstpointer pB ) {
3138 aoA = ( AddressObject * ) pA;
3139 if( aoA->type == ADDR_ITEM_FOLDER ) {
3140 ItemFolder *folder, *fld;
3142 fld = ADAPTER_FOLDER(aoA)->itemFolder;
3143 folder = ( ItemFolder * ) pB;
3144 if( fld == folder ) return 0; /* Found folder */
3150 * Remove folder and group nodes from tree widget for items contained ("cut")
3153 static void addressbook_treenode_remove_item( void ) {
3155 AddrSelectItem *cutItem;
3156 AddressCache *cache;
3157 AddrItemObject *aio;
3158 GtkCTree *ctree = GTK_CTREE( addrbook.ctree );
3161 node = _clipBoard_->objectList;
3163 cutItem = node->data;
3164 node = g_list_next( node );
3165 cache = addrindex_get_cache(
3166 _clipBoard_->addressIndex, cutItem->cacheID );
3167 if( cache == NULL ) continue;
3168 aio = addrcache_get_object( cache, cutItem->uid );
3171 if( ADDRITEM_TYPE(aio) == ITEMTYPE_FOLDER ) {
3174 folder = ( ItemFolder * ) aio;
3175 tn = gtk_ctree_find_by_row_data_custom(
3176 ctree, NULL, folder,
3177 addressbook_treenode_find_folder_cb );
3179 else if( ADDRITEM_TYPE(aio) == ITEMTYPE_GROUP ) {
3182 group = ( ItemGroup * ) aio;
3183 tn = gtk_ctree_find_by_row_data_custom(
3185 addressbook_treenode_find_group_cb );
3189 /* Free up adapter and remove node. */
3190 gtk_ctree_remove_node( ctree, tn );
3197 * Find parent datasource for specified tree node.
3198 * \param node Node to test.
3199 * \return Data source, or NULL if not found.
3201 static AddressDataSource *addressbook_find_datasource( GtkCTreeNode *node ) {
3202 AddressDataSource *ds = NULL;
3205 g_return_val_if_fail(addrbook.ctree != NULL, NULL);
3208 if( GTK_CTREE_ROW(node)->level < 2 ) return NULL;
3209 ao = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), node );
3211 /* printf( "ao->type = %d\n", ao->type ); */
3212 if( ao->type == ADDR_DATASOURCE ) {
3213 AdapterDSource *ads = ADAPTER_DSOURCE(ao);
3214 /* printf( "found it\n" ); */
3215 ds = ads->dataSource;
3219 node = GTK_CTREE_ROW(node)->parent;
3225 * Load address list widget with children of specified object.
3226 * \param obj Parent object to be loaded.
3228 static void addressbook_set_clist( AddressObject *obj, gboolean refresh ) {
3229 GtkCTree *ctreelist = GTK_CTREE(addrbook.clist);
3230 GtkCList *clist = GTK_CLIST(addrbook.clist);
3231 AddressDataSource *ds = NULL;
3232 AdapterDSource *ads = NULL;
3233 static AddressObject *last_obj = NULL;
3235 if (addrbook.clist == NULL) {
3238 if (obj == last_obj && !refresh)
3243 gtk_clist_clear(clist);
3247 if( obj->type == ADDR_INTERFACE ) {
3248 /* printf( "set_clist: loading datasource...\n" ); */
3249 /* addressbook_node_load_datasource( GTK_CTREE(clist), obj ); */
3253 gtk_clist_freeze(clist);
3254 gtk_clist_clear(clist);
3256 if( obj->type == ADDR_DATASOURCE ) {
3257 ads = ADAPTER_DSOURCE(obj);
3258 ds = ADAPTER_DSOURCE(obj)->dataSource;
3260 /* Load root folder */
3261 ItemFolder *rootFolder = NULL;
3262 rootFolder = addrindex_ds_get_root_folder( ds );
3263 addressbook_folder_load_person(
3264 ctreelist, addrindex_ds_get_root_folder( ds ) );
3265 addressbook_folder_load_group(
3266 ctreelist, addrindex_ds_get_root_folder( ds ) );
3270 if( obj->type == ADDR_ITEM_GROUP ) {
3272 ItemGroup *itemGroup = ADAPTER_GROUP(obj)->itemGroup;
3273 addressbook_load_group( ctreelist, itemGroup );
3275 else if( obj->type == ADDR_ITEM_FOLDER ) {
3277 ItemFolder *itemFolder = ADAPTER_FOLDER(obj)->itemFolder;
3278 addressbook_folder_load_person( ctreelist, itemFolder );
3279 addressbook_folder_load_group( ctreelist, itemFolder );
3282 gtk_sctree_sort_recursive(GTK_CTREE(clist), NULL);
3283 clist->focus_row = -1;
3284 gtk_clist_thaw(clist);
3288 * Call back function to free adaptor. Call back is setup by function
3289 * gtk_ctree_node_set_row_data_full() when node is populated. This function is
3290 * called when the address book tree widget node is removed by calling
3291 * function gtk_ctree_remove_node().
3293 * \param data Tree node's row data.
3295 static void addressbook_free_treenode( gpointer data ) {
3298 ao = ( AddressObject * ) data;
3299 if( ao == NULL ) return;
3300 if( ao->type == ADDR_INTERFACE ) {
3301 AdapterInterface *ai = ADAPTER_INTERFACE(ao);
3302 addrbookctl_free_interface( ai );
3304 else if( ao->type == ADDR_DATASOURCE ) {
3305 AdapterDSource *ads = ADAPTER_DSOURCE(ao);
3306 addrbookctl_free_datasource( ads );
3308 else if( ao->type == ADDR_ITEM_FOLDER ) {
3309 AdapterFolder *af = ADAPTER_FOLDER(ao);
3310 addrbookctl_free_folder( af );
3312 else if( ao->type == ADDR_ITEM_GROUP ) {
3313 AdapterGroup *ag = ADAPTER_GROUP(ao);
3314 addrbookctl_free_group( ag );
3319 * Create new adaptor for specified data source.
3321 AdapterDSource *addressbook_create_ds_adapter( AddressDataSource *ds,
3322 AddressObjectType otype, gchar *name )
3324 AdapterDSource *adapter = g_new0( AdapterDSource, 1 );
3325 ADDRESS_OBJECT(adapter)->type = ADDR_DATASOURCE;
3326 ADDRESS_OBJECT_NAME(adapter) = g_strdup( name );
3327 adapter->dataSource = ds;
3328 adapter->subType = otype;
3332 void addressbook_ads_set_name( AdapterDSource *adapter, gchar *value ) {
3333 ADDRESS_OBJECT_NAME(adapter) =
3334 mgu_replace_string( ADDRESS_OBJECT_NAME(adapter), value );
3338 * Load tree from address index with the initial data.
3340 static void addressbook_load_tree( void ) {
3341 GtkCTree *ctree = GTK_CTREE( addrbook.ctree );
3342 GList *nodeIf, *nodeDS;
3343 AdapterInterface *adapter;
3344 AddressInterface *iface;
3345 AddressTypeControlItem *atci;
3346 AddressDataSource *ds;
3347 AdapterDSource *ads;
3348 GtkCTreeNode *node, *newNode;
3351 nodeIf = _addressInterfaceList_;
3353 adapter = nodeIf->data;
3354 node = adapter->treeNode;
3355 iface = adapter->interface;
3356 atci = adapter->atci;
3358 if( iface->useInterface ) {
3359 /* Load data sources below interface node */
3360 nodeDS = iface->listSource;
3364 name = addrindex_ds_get_name( ds );
3365 ads = addressbook_create_ds_adapter(
3366 ds, atci->objectType, name );
3367 newNode = addressbook_add_object(
3368 node, ADDRESS_OBJECT(ads) );
3369 nodeDS = g_list_next( nodeDS );
3371 gtk_ctree_expand( ctree, node );
3374 nodeIf = g_list_next( nodeIf );
3379 * Convert the old address book to new format.
3381 static gboolean addressbook_convert( AddressIndex *addrIndex ) {
3382 gboolean retVal = FALSE;
3383 gboolean errFlag = TRUE;
3386 /* Read old address book, performing conversion */
3387 debug_print( "Reading and converting old address book...\n" );
3388 addrindex_set_file_name( addrIndex, ADDRESSBOOK_OLD_FILE );
3389 addrindex_read_data( addrIndex );
3390 if( addrIndex->retVal == MGU_NO_FILE ) {
3391 /* We do not have a file - new user */
3392 debug_print( "New user... create new books...\n" );
3393 addrindex_create_new_books( addrIndex );
3394 if( addrIndex->retVal == MGU_SUCCESS ) {
3395 /* Save index file */
3396 addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
3397 addrindex_save_data( addrIndex );
3398 if( addrIndex->retVal == MGU_SUCCESS ) {
3403 msg = _( "New user, could not save index file." );
3407 msg = _( "New user, could not save address book files." );
3411 /* We have an old file */
3412 if( addrIndex->wasConverted ) {
3413 /* Converted successfully - save address index */
3414 addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
3415 addrindex_save_data( addrIndex );
3416 if( addrIndex->retVal == MGU_SUCCESS ) {
3417 msg = _( "Old address book converted successfully." );
3422 msg = _("Old address book converted,\n"
3423 "could not save new address index file." );
3427 /* File conversion failed - just create new books */
3428 debug_print( "File conversion failed... just create new books...\n" );
3429 addrindex_create_new_books( addrIndex );
3430 if( addrIndex->retVal == MGU_SUCCESS ) {
3432 addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
3433 addrindex_save_data( addrIndex );
3434 if( addrIndex->retVal == MGU_SUCCESS ) {
3435 msg = _("Could not convert address book,\n"
3436 "but created empty new address book files." );
3441 msg = _("Could not convert address book,\n"
3442 "could not save new address index file." );
3446 msg = _("Could not convert address book\n"
3447 "and could not create new address book files." );
3452 debug_print( "Error\n%s\n", msg );
3453 alertpanel_full(_("Addressbook conversion error"), msg,
3454 GTK_STOCK_CLOSE, NULL, NULL, FALSE,
3455 NULL, ALERT_ERROR, G_ALERTDEFAULT);
3458 debug_print( "Warning\n%s\n", msg );
3459 alertpanel_full(_("Addressbook conversion error"), msg,
3460 GTK_STOCK_CLOSE, NULL, NULL, FALSE,
3461 NULL, ALERT_WARNING, G_ALERTDEFAULT);
3467 void addressbook_read_file( void ) {
3468 AddressIndex *addrIndex = NULL;
3470 debug_print( "Reading address index...\n" );
3471 if( _addressIndex_ ) {
3472 debug_print( "address book already read!!!\n" );
3476 addrIndex = addrindex_create_index();
3477 addrindex_initialize();
3479 /* Use new address book index. */
3480 addrindex_set_file_path( addrIndex, get_rc_dir() );
3481 addrindex_set_file_name( addrIndex, ADDRESSBOOK_INDEX_FILE );
3482 addrindex_read_data( addrIndex );
3483 if( addrIndex->retVal == MGU_NO_FILE ) {
3484 /* Conversion required */
3485 debug_print( "Converting...\n" );
3486 if( addressbook_convert( addrIndex ) ) {
3487 _addressIndex_ = addrIndex;
3490 else if( addrIndex->retVal == MGU_SUCCESS ) {
3491 _addressIndex_ = addrIndex;
3494 /* Error reading address book */
3495 debug_print( "Could not read address index.\n" );
3496 addrindex_print_index( addrIndex, stdout );
3497 alertpanel_full(_("Addressbook Error"),
3498 _("Could not read address index"),
3499 GTK_STOCK_CLOSE, NULL, NULL, FALSE,
3500 NULL, ALERT_ERROR, G_ALERTDEFAULT);
3502 debug_print( "done.\n" );
3506 * Add object into the address index tree widget.
3507 * Enter: node Parent node.
3508 * obj Object to add.
3509 * Return: Node that was added, or NULL if object not added.
3511 static GtkCTreeNode *addressbook_add_object(GtkCTreeNode *node,
3514 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
3515 GtkCTreeNode *added;
3516 AddressObject *pobj;
3517 AddressObjectType otype;
3518 AddressTypeControlItem *atci = NULL;
3520 g_return_val_if_fail(node != NULL, NULL);
3521 g_return_val_if_fail(obj != NULL, NULL);
3523 pobj = gtk_ctree_node_get_row_data(ctree, node);
3524 g_return_val_if_fail(pobj != NULL, NULL);
3526 /* Determine object type to be displayed */
3527 if( obj->type == ADDR_DATASOURCE ) {
3528 otype = ADAPTER_DSOURCE(obj)->subType;
3534 /* Handle any special conditions. */
3536 atci = addrbookctl_lookup( otype );
3538 if( atci->showInTree ) {
3539 /* Add object to tree */
3542 added = gtk_sctree_insert_node( ctree, node, NULL, name, FOLDER_SPACING,
3543 atci->iconXpm, atci->maskXpm, atci->iconXpmOpen, atci->maskXpmOpen,
3544 atci->treeLeaf, atci->treeExpand );
3545 gtk_ctree_node_set_row_data_full( ctree, added, obj,
3546 addressbook_free_treenode );
3550 gtk_sctree_sort_node(ctree, node);
3556 * Add group into the address index tree.
3557 * \param node Parent node.
3558 * \param ds Data source.
3559 * \param itemGroup Group to add.
3560 * \return Inserted node.
3562 static GtkCTreeNode *addressbook_node_add_group(
3563 GtkCTreeNode *node, AddressDataSource *ds,
3564 ItemGroup *itemGroup )
3566 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
3567 GtkCTreeNode *newNode;
3568 AdapterGroup *adapter;
3569 AddressTypeControlItem *atci = NULL;
3572 if( ds == NULL ) return NULL;
3573 if( node == NULL || itemGroup == NULL ) return NULL;
3575 name = &itemGroup->obj.name;
3577 atci = addrbookctl_lookup( ADDR_ITEM_GROUP );
3579 adapter = g_new0( AdapterGroup, 1 );
3580 ADDRESS_OBJECT_TYPE(adapter) = ADDR_ITEM_GROUP;
3581 ADDRESS_OBJECT_NAME(adapter) = g_strdup( ADDRITEM_NAME(itemGroup) );
3582 adapter->itemGroup = itemGroup;
3584 newNode = gtk_sctree_insert_node( ctree, node, NULL, name, FOLDER_SPACING,
3585 atci->iconXpm, atci->maskXpm, atci->iconXpm, atci->maskXpm,
3586 atci->treeLeaf, atci->treeExpand );
3587 gtk_ctree_node_set_row_data_full( ctree, newNode, adapter,
3588 addressbook_free_treenode );
3589 gtk_sctree_sort_node( ctree, node );
3594 * Add folder into the address index tree. Only visible folders are loaded into
3595 * the address index tree. Note that the root folder is not inserted into the
3598 * \param node Parent node.
3599 * \param ds Data source.
3600 * \param itemFolder Folder to add.
3601 * \param otype Object type to display.
3602 * \return Inserted node for the folder.
3604 static GtkCTreeNode *addressbook_node_add_folder(
3605 GtkCTreeNode *node, AddressDataSource *ds,
3606 ItemFolder *itemFolder, AddressObjectType otype )
3608 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
3609 GtkCTreeNode *newNode = NULL;
3610 AdapterFolder *adapter;
3611 AddressTypeControlItem *atci = NULL;
3612 GList *listItems = NULL;
3614 ItemFolder *rootFolder;
3616 /* Only visible folders */
3617 if( itemFolder->isHidden ) return NULL;
3619 if( ds == NULL ) return NULL;
3620 if( node == NULL || itemFolder == NULL ) return NULL;
3622 /* Determine object type */
3623 atci = addrbookctl_lookup( otype );
3624 if( atci == NULL ) return NULL;
3626 rootFolder = addrindex_ds_get_root_folder( ds );
3627 if( itemFolder == rootFolder ) {
3631 adapter = g_new0( AdapterFolder, 1 );
3632 ADDRESS_OBJECT_TYPE(adapter) = ADDR_ITEM_FOLDER;
3633 ADDRESS_OBJECT_NAME(adapter) = g_strdup( ADDRITEM_NAME(itemFolder) );
3634 adapter->itemFolder = itemFolder;
3636 name = ADDRITEM_NAME(itemFolder);
3637 newNode = gtk_sctree_insert_node( ctree, node, NULL, &name, FOLDER_SPACING,
3638 atci->iconXpm, atci->maskXpm, atci->iconXpm, atci->maskXpm,
3639 atci->treeLeaf, atci->treeExpand );
3641 gtk_ctree_node_set_row_data_full( ctree, newNode, adapter,
3642 addressbook_free_treenode );
3646 listItems = itemFolder->listFolder;
3647 while( listItems ) {
3648 ItemFolder *item = listItems->data;
3649 addressbook_node_add_folder( newNode, ds, item, otype );
3650 listItems = g_list_next( listItems );
3652 listItems = itemFolder->listGroup;
3653 while( listItems ) {
3654 ItemGroup *item = listItems->data;
3655 addressbook_node_add_group( newNode, ds, item );
3656 listItems = g_list_next( listItems );
3658 gtk_sctree_sort_node( ctree, node );
3662 void addressbook_export_to_file( void ) {
3663 if( _addressIndex_ ) {
3664 /* Save all new address book data */
3665 debug_print( "Saving address books...\n" );
3666 addrindex_save_all_books( _addressIndex_ );
3668 debug_print( "Exporting addressbook to file...\n" );
3669 addrindex_save_data( _addressIndex_ );
3670 if( _addressIndex_->retVal != MGU_SUCCESS ) {
3671 addrindex_print_index( _addressIndex_, stdout );
3674 /* Notify address completion of new data */
3675 invalidate_address_completion();
3679 static gboolean addressbook_entry_key_pressed(GtkWidget *widget, GdkEventKey *event, gpointer data)
3681 if (event && event->keyval == GDK_Return)
3682 addressbook_lup_clicked(NULL, NULL);
3687 * Comparison using cell contents (text in first column). Used for sort
3688 * address index widget.
3690 static gint addressbook_treenode_compare_func(
3691 GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2 )
3693 GtkCell *cell1 = ((GtkCListRow *)ptr1)->cell;
3694 GtkCell *cell2 = ((GtkCListRow *)ptr2)->cell;
3695 gchar *name1 = NULL, *name2 = NULL;
3696 if( cell1 ) name1 = cell1->u.text;
3697 if( cell2 ) name2 = cell2->u.text;
3698 if( ! name1 ) return ( name2 != NULL );
3699 if( ! name2 ) return -1;
3700 return g_utf8_collate( name1, name2 );
3703 static void addressbook_new_book_cb( gpointer data, guint action, GtkWidget *widget ) {
3704 AdapterDSource *ads;
3705 AdapterInterface *adapter;
3706 GtkCTreeNode *newNode;
3708 adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
3709 if( adapter == NULL ) return;
3710 ads = addressbook_edit_book( _addressIndex_, NULL );
3712 newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
3714 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), newNode );
3715 addrbook.treeSelected = newNode;
3720 static void addressbook_new_vcard_cb( gpointer data, guint action, GtkWidget *widget ) {
3721 AdapterDSource *ads;
3722 AdapterInterface *adapter;
3723 GtkCTreeNode *newNode;
3725 adapter = addrbookctl_find_interface( ADDR_IF_VCARD );
3726 if( adapter == NULL ) return;
3727 ads = addressbook_edit_vcard( _addressIndex_, NULL );
3729 newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
3731 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), newNode );
3732 addrbook.treeSelected = newNode;
3738 static void addressbook_new_jpilot_cb( gpointer data, guint action, GtkWidget *widget ) {
3739 AdapterDSource *ads;
3740 AdapterInterface *adapter;
3741 AddressInterface *iface;
3742 GtkCTreeNode *newNode;
3744 adapter = addrbookctl_find_interface( ADDR_IF_JPILOT );
3745 if( adapter == NULL ) return;
3746 iface = adapter->interface;
3747 if( ! iface->haveLibrary ) return;
3748 ads = addressbook_edit_jpilot( _addressIndex_, NULL );
3750 newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
3752 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), newNode );
3753 addrbook.treeSelected = newNode;
3760 static void addressbook_new_ldap_cb( gpointer data, guint action, GtkWidget *widget ) {
3761 AdapterDSource *ads;
3762 AdapterInterface *adapter;
3763 AddressInterface *iface;
3764 GtkCTreeNode *newNode;
3766 adapter = addrbookctl_find_interface( ADDR_IF_LDAP );
3767 if( adapter == NULL ) return;
3768 iface = adapter->interface;
3769 if( ! iface->haveLibrary ) return;
3770 ads = addressbook_edit_ldap( _addressIndex_, NULL );
3772 newNode = addressbook_add_object( adapter->treeNode, ADDRESS_OBJECT(ads) );
3774 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), newNode );
3775 addrbook.treeSelected = newNode;
3782 * Display address search status message.
3783 * \param queryType Query type.
3784 * \param status Status/Error code.
3786 static void addressbook_search_message( gint queryType, gint sts ) {
3788 *addressbook_msgbuf = '\0';
3790 if( sts != MGU_SUCCESS ) {
3791 if( queryType == ADDRQUERY_LDAP ) {
3793 desc = addressbook_err2string( _lutErrorsLDAP_, sts );
3798 g_snprintf( addressbook_msgbuf,
3799 sizeof(addressbook_msgbuf), "%s", desc );
3800 addressbook_status_show( addressbook_msgbuf );
3803 addressbook_status_show( "" );
3808 * Refresh addressbook by forcing refresh of current selected object in
3811 static void addressbook_refresh_current( void ) {
3815 ctree = GTK_CTREE(addrbook.ctree);
3816 obj = gtk_ctree_node_get_row_data( ctree, addrbook.treeSelected );
3817 if( obj == NULL ) return;
3818 addressbook_set_clist( obj, TRUE );
3822 * Message that is displayed whilst a query is executing in a background
3825 static gchar *_tempMessage_ = N_( "Busy searching..." );
3828 * Address search idle function. This function is called during UI idle time
3829 * while a search is in progress.
3831 * \param data Idler data.
3833 static void addressbook_search_idle( gpointer data ) {
3837 queryID = GPOINTER_TO_INT( data );
3838 printf( "addressbook_ldap_idle... queryID=%d\n", queryID );
3843 * Search completion callback function. This removes the query from the idle
3846 * \param queryID Query ID of search request.
3848 void addressbook_clear_idler( gint queryID ) {
3851 /* Remove idler function */
3852 /* printf( "addressbook_clear_idler::%d::\n", queryID ); */
3853 ptrQID = GINT_TO_POINTER( queryID );
3855 gtk_idle_remove_by_data( ptrQID );
3860 * Search completion callback function. This removes the query from the idle
3863 * \param sender Sender of query.
3864 * \param queryID Query ID of search request.
3865 * \param status Search status.
3866 * \param data Query data.
3868 static void addressbook_search_callback_end(
3869 gpointer sender, gint queryID, gint status, gpointer data )
3873 AddrQueryObject *aqo;
3875 /* Remove idler function */
3876 ptrQID = GINT_TO_POINTER( queryID );
3878 gtk_idle_remove_by_data( ptrQID );
3881 /* Refresh addressbook contents */
3882 addressbook_refresh_current();
3883 req = qrymgr_find_request( queryID );
3885 aqo = ( AddrQueryObject * ) req->queryList->data;
3886 addressbook_search_message( aqo->queryType, status );
3889 /* Stop the search */
3890 addrindex_stop_search( queryID );
3894 * Label (a format string) that is used to name each folder.
3896 static gchar *_queryFolderLabel_ = N_( "Search '%s'" );
3901 * \param ds Data source to search.
3902 * \param searchTerm String to lookup.
3903 * \param pNode Parent data source node.
3905 static void addressbook_perform_search(
3906 AddressDataSource *ds, gchar *searchTerm,
3907 GtkCTreeNode *pNode )
3909 AddrBookBase *adbase;
3910 AddressCache *cache;
3913 GtkCTreeNode *nNode;
3917 AddressObjectType aoType = ADDR_NONE;
3920 if( *searchTerm == '\0' || strlen( searchTerm ) < 1 ) return;
3922 if( ds->type == ADDR_IF_LDAP ) {
3924 aoType = ADDR_LDAP_QUERY;
3931 /* Get reference to address cache */
3932 adbase = ( AddrBookBase * ) ds->rawDataSource;
3933 cache = adbase->addressCache;
3935 /* Create a folder for the search results */
3936 folder = addrcache_add_new_folder( cache, NULL );
3937 name = g_strdup_printf( _queryFolderLabel_, searchTerm );
3938 addritem_folder_set_name( folder, name );
3939 addritem_folder_set_remarks( folder, "" );
3942 /* Now let's see the folder */
3943 ctree = GTK_CTREE(addrbook.ctree);
3944 nNode = addressbook_node_add_folder( pNode, ds, folder, aoType );
3945 gtk_ctree_expand( ctree, pNode );
3947 gtk_sctree_select( GTK_SCTREE(ctree), nNode );
3948 addrbook.treeSelected = nNode;
3951 /* Setup the search */
3952 queryID = addrindex_setup_explicit_search(
3953 ds, searchTerm, folder, addressbook_search_callback_end, NULL );
3954 if( queryID == 0 ) return;
3956 /* Set up idler function */
3957 idleID = gtk_idle_add(
3958 ( GtkFunction ) addressbook_search_idle,
3959 GINT_TO_POINTER( queryID ) );
3961 /* Start search, sit back and wait for something to happen */
3962 addrindex_start_search( queryID );
3964 addressbook_status_show( _tempMessage_ );
3968 * Lookup button handler. Address search is only performed against
3969 * address interfaces for external queries.
3971 * \param button Lookup button widget.
3972 * \param data Data object.
3974 static void addressbook_lup_clicked( GtkButton *button, gpointer data ) {
3977 AddressDataSource *ds;
3978 AddressInterface *iface;
3980 GtkCTreeNode *node, *parentNode;
3982 node = addrbook.treeSelected;
3983 if( ! node ) return;
3984 if( GTK_CTREE_ROW(node)->level == 1 ) return;
3986 ctree = GTK_CTREE(addrbook.ctree);
3987 obj = gtk_ctree_node_get_row_data( ctree, node );
3988 if( obj == NULL ) return;
3990 ds = addressbook_find_datasource( node );
3991 if( ds == NULL ) return;
3993 /* We must have a datasource that is an external interface */
3994 iface = ds->interface;
3995 if( ! iface->haveLibrary ) return;
3996 if( ! iface->externalQuery ) return;
3999 gtk_editable_get_chars( GTK_EDITABLE(addrbook.entry), 0, -1 );
4000 g_strchomp( searchTerm );
4002 if( obj->type == ADDR_ITEM_FOLDER ) {
4003 parentNode = GTK_CTREE_ROW(node)->parent;
4008 addressbook_perform_search( ds, searchTerm, parentNode );
4010 gtk_widget_grab_focus( addrbook.entry );
4012 g_free( searchTerm );
4015 static void addressbook_close_clicked( GtkButton *button, gpointer data ) {
4016 addressbook_close();
4021 * Browse address entry for highlighted entry.
4023 static void addressbook_browse_entry_cb(void)
4025 GtkCTree *clist = GTK_CTREE(addrbook.clist);
4027 AddressDataSource *ds;
4028 AddressInterface *iface;
4032 if(addrbook.listSelected == NULL)
4035 obj = gtk_ctree_node_get_row_data(clist, addrbook.listSelected);
4039 ds = addressbook_find_datasource(GTK_CTREE_NODE(addrbook.treeSelected));
4043 iface = ds->interface;
4044 if(! iface->haveLibrary )
4048 if (obj->type == ADDR_ITEM_EMAIL) {
4049 email = ( ItemEMail * ) obj;
4053 person = (ItemPerson *) ADDRITEM_PARENT(email);
4055 else if (obj->type == ADDR_ITEM_PERSON) {
4056 person = (ItemPerson *) obj;
4063 if( iface->type == ADDR_IF_LDAP ) {
4064 browseldap_entry(ds, person->externalID);
4069 /* **********************************************************************
4070 * Build lookup tables.
4071 * ***********************************************************************
4075 * Remap object types.
4076 * Enter: abType AddressObjectType (used in tree node).
4077 * Return: ItemObjectType (used in address cache data).
4079 ItemObjectType addressbook_type2item( AddressObjectType abType ) {
4080 ItemObjectType ioType;
4083 case ADDR_ITEM_PERSON: ioType = ITEMTYPE_PERSON; break;
4084 case ADDR_ITEM_EMAIL: ioType = ITEMTYPE_EMAIL; break;
4085 case ADDR_ITEM_FOLDER: ioType = ITEMTYPE_FOLDER; break;
4086 case ADDR_ITEM_GROUP: ioType = ITEMTYPE_GROUP; break;
4087 case ADDR_DATASOURCE: ioType = ITEMTYPE_DATASOURCE; break;
4088 default: ioType = ITEMTYPE_NONE; break;
4094 * Build table that controls the rendering of object types.
4096 void addrbookctl_build_map( GtkWidget *window ) {
4097 AddressTypeControlItem *atci;
4100 stock_pixmap_gdk(window, STOCK_PIXMAP_DIR_CLOSE, &folderxpm, &folderxpmmask);
4101 stock_pixmap_gdk(window, STOCK_PIXMAP_DIR_OPEN, &folderopenxpm, &folderopenxpmmask);
4102 stock_pixmap_gdk(window, STOCK_PIXMAP_GROUP, &groupxpm, &groupxpmmask);
4103 stock_pixmap_gdk(window, STOCK_PIXMAP_VCARD, &vcardxpm, &vcardxpmmask);
4104 stock_pixmap_gdk(window, STOCK_PIXMAP_BOOK, &bookxpm, &bookxpmmask);
4105 stock_pixmap_gdk(window, STOCK_PIXMAP_ADDRESS, &addressxpm, &addressxpmmask);
4106 stock_pixmap_gdk(window, STOCK_PIXMAP_JPILOT, &jpilotxpm, &jpilotxpmmask);
4107 stock_pixmap_gdk(window, STOCK_PIXMAP_CATEGORY, &categoryxpm, &categoryxpmmask);
4108 stock_pixmap_gdk(window, STOCK_PIXMAP_LDAP, &ldapxpm, &ldapxpmmask);
4109 stock_pixmap_gdk(window, STOCK_PIXMAP_ADDRESS_SEARCH, &addrsearchxpm, &addrsearchxpmmask);
4111 _addressBookTypeHash_ = g_hash_table_new( g_int_hash, g_int_equal );
4112 _addressBookTypeList_ = NULL;
4115 atci = g_new0( AddressTypeControlItem, 1 );
4116 atci->objectType = ADDR_INTERFACE;
4117 atci->interfaceType = ADDR_IF_NONE;
4118 atci->showInTree = TRUE;
4119 atci->treeExpand = TRUE;
4120 atci->treeLeaf = FALSE;
4121 atci->displayName = _( "Interface" );
4122 atci->iconXpm = folderxpm;
4123 atci->maskXpm = folderxpmmask;
4124 atci->iconXpmOpen = folderopenxpm;
4125 atci->maskXpmOpen = folderopenxpmmask;
4126 atci->menuCommand = NULL;
4127 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4128 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4131 atci = g_new0( AddressTypeControlItem, 1 );
4132 atci->objectType = ADDR_BOOK;
4133 atci->interfaceType = ADDR_IF_BOOK;
4134 atci->showInTree = TRUE;
4135 atci->treeExpand = TRUE;
4136 atci->treeLeaf = FALSE;
4137 atci->displayName = _( "Address Book" );
4138 atci->iconXpm = bookxpm;
4139 atci->maskXpm = bookxpmmask;
4140 atci->iconXpmOpen = bookxpm;
4141 atci->maskXpmOpen = bookxpmmask;
4142 atci->menuCommand = "/Book/New Book";
4143 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4144 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4147 atci = g_new0( AddressTypeControlItem, 1 );
4148 atci->objectType = ADDR_ITEM_PERSON;
4149 atci->interfaceType = ADDR_IF_NONE;
4150 atci->showInTree = FALSE;
4151 atci->treeExpand = FALSE;
4152 atci->treeLeaf = FALSE;
4153 atci->displayName = _( "Person" );
4154 atci->iconXpm = NULL;
4155 atci->maskXpm = NULL;
4156 atci->iconXpmOpen = NULL;
4157 atci->maskXpmOpen = NULL;
4158 atci->menuCommand = NULL;
4159 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4160 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4163 atci = g_new0( AddressTypeControlItem, 1 );
4164 atci->objectType = ADDR_ITEM_EMAIL;
4165 atci->interfaceType = ADDR_IF_NONE;
4166 atci->showInTree = FALSE;
4167 atci->treeExpand = FALSE;
4168 atci->treeLeaf = TRUE;
4169 atci->displayName = _( "EMail Address" );
4170 atci->iconXpm = addressxpm;
4171 atci->maskXpm = addressxpmmask;
4172 atci->iconXpmOpen = addressxpm;
4173 atci->maskXpmOpen = addressxpmmask;
4174 atci->menuCommand = NULL;
4175 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4176 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4179 atci = g_new0( AddressTypeControlItem, 1 );
4180 atci->objectType = ADDR_ITEM_GROUP;
4181 atci->interfaceType = ADDR_IF_BOOK;
4182 atci->showInTree = TRUE;
4183 atci->treeExpand = FALSE;
4184 atci->treeLeaf = FALSE;
4185 atci->displayName = _( "Group" );
4186 atci->iconXpm = groupxpm;
4187 atci->maskXpm = groupxpmmask;
4188 atci->iconXpmOpen = groupxpm;
4189 atci->maskXpmOpen = groupxpmmask;
4190 atci->menuCommand = NULL;
4191 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4192 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4195 atci = g_new0( AddressTypeControlItem, 1 );
4196 atci->objectType = ADDR_ITEM_FOLDER;
4197 atci->interfaceType = ADDR_IF_BOOK;
4198 atci->showInTree = TRUE;
4199 atci->treeExpand = FALSE;
4200 atci->treeLeaf = FALSE;
4201 atci->displayName = _( "Folder" );
4202 atci->iconXpm = folderxpm;
4203 atci->maskXpm = folderxpmmask;
4204 atci->iconXpmOpen = folderopenxpm;
4205 atci->maskXpmOpen = folderopenxpmmask;
4206 atci->menuCommand = NULL;
4207 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4208 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4211 atci = g_new0( AddressTypeControlItem, 1 );
4212 atci->objectType = ADDR_VCARD;
4213 atci->interfaceType = ADDR_IF_VCARD;
4214 atci->showInTree = TRUE;
4215 atci->treeExpand = TRUE;
4216 atci->treeLeaf = TRUE;
4217 atci->displayName = _( "vCard" );
4218 atci->iconXpm = vcardxpm;
4219 atci->maskXpm = vcardxpmmask;
4220 atci->iconXpmOpen = vcardxpm;
4221 atci->maskXpmOpen = vcardxpmmask;
4222 atci->menuCommand = "/Book/New vCard";
4223 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4224 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4227 atci = g_new0( AddressTypeControlItem, 1 );
4228 atci->objectType = ADDR_JPILOT;
4229 atci->interfaceType = ADDR_IF_JPILOT;
4230 atci->showInTree = TRUE;
4231 atci->treeExpand = TRUE;
4232 atci->treeLeaf = FALSE;
4233 atci->displayName = _( "JPilot" );
4234 atci->iconXpm = jpilotxpm;
4235 atci->maskXpm = jpilotxpmmask;
4236 atci->iconXpmOpen = jpilotxpm;
4237 atci->maskXpmOpen = jpilotxpmmask;
4238 atci->menuCommand = "/Book/New JPilot";
4239 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4240 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4243 atci = g_new0( AddressTypeControlItem, 1 );
4244 atci->objectType = ADDR_CATEGORY;
4245 atci->interfaceType = ADDR_IF_JPILOT;
4246 atci->showInTree = TRUE;
4247 atci->treeExpand = TRUE;
4248 atci->treeLeaf = TRUE;
4249 atci->displayName = _( "JPilot" );
4250 atci->iconXpm = categoryxpm;
4251 atci->maskXpm = categoryxpmmask;
4252 atci->iconXpmOpen = categoryxpm;
4253 atci->maskXpmOpen = categoryxpmmask;
4254 atci->menuCommand = NULL;
4255 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4256 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4259 atci = g_new0( AddressTypeControlItem, 1 );
4260 atci->objectType = ADDR_LDAP;
4261 atci->interfaceType = ADDR_IF_LDAP;
4262 atci->showInTree = TRUE;
4263 atci->treeExpand = TRUE;
4264 atci->treeLeaf = FALSE;
4265 atci->displayName = _( "LDAP servers" );
4266 atci->iconXpm = ldapxpm;
4267 atci->maskXpm = ldapxpmmask;
4268 atci->iconXpmOpen = ldapxpm;
4269 atci->maskXpmOpen = ldapxpmmask;
4270 atci->menuCommand = "/Book/New LDAP Server";
4271 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4272 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4275 atci = g_new0( AddressTypeControlItem, 1 );
4276 atci->objectType = ADDR_LDAP_QUERY;
4277 atci->interfaceType = ADDR_IF_LDAP;
4278 atci->showInTree = TRUE;
4279 atci->treeExpand = FALSE;
4280 atci->treeLeaf = TRUE;
4281 atci->displayName = _( "LDAP Query" );
4282 atci->iconXpm = addrsearchxpm;
4283 atci->maskXpm = addrsearchxpmmask;
4284 atci->iconXpmOpen = addrsearchxpm;
4285 atci->maskXpmOpen = addrsearchxpmmask;
4286 atci->menuCommand = NULL;
4287 g_hash_table_insert( _addressBookTypeHash_, &atci->objectType, atci );
4288 _addressBookTypeList_ = g_list_append( _addressBookTypeList_, atci );
4293 * Search for specified object type.
4295 AddressTypeControlItem *addrbookctl_lookup( gint ot ) {
4297 return ( AddressTypeControlItem * ) g_hash_table_lookup( _addressBookTypeHash_, &objType );
4301 * Search for specified interface type.
4303 AddressTypeControlItem *addrbookctl_lookup_iface( AddressIfType ifType ) {
4304 GList *node = _addressBookTypeList_;
4306 AddressTypeControlItem *atci = node->data;
4307 if( atci->interfaceType == ifType ) return atci;
4308 node = g_list_next( node );
4313 static void addrbookctl_free_address( AddressObject *obj ) {
4314 g_free( obj->name );
4315 obj->type = ADDR_NONE;
4319 static void addrbookctl_free_interface( AdapterInterface *adapter ) {
4320 addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
4321 adapter->interface = NULL;
4322 adapter->interfaceType = ADDR_IF_NONE;
4323 adapter->atci = NULL;
4324 adapter->enabled = FALSE;
4325 adapter->haveLibrary = FALSE;
4326 adapter->treeNode = NULL;
4330 static void addrbookctl_free_datasource( AdapterDSource *adapter ) {
4331 addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
4332 adapter->dataSource = NULL;
4333 adapter->subType = ADDR_NONE;
4337 static void addrbookctl_free_folder( AdapterFolder *adapter ) {
4338 addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
4339 adapter->itemFolder = NULL;
4343 static void addrbookctl_free_group( AdapterGroup *adapter ) {
4344 addrbookctl_free_address( ADDRESS_OBJECT(adapter) );
4345 adapter->itemGroup = NULL;
4350 * Build GUI interface list.
4352 void addrbookctl_build_iflist( void ) {
4353 AddressTypeControlItem *atci;
4354 AdapterInterface *adapter;
4357 if( _addressIndex_ == NULL ) {
4358 _addressIndex_ = addrindex_create_index();
4359 if( _clipBoard_ == NULL ) {
4360 _clipBoard_ = addrclip_create();
4362 addrclip_set_index( _clipBoard_, _addressIndex_ );
4364 _addressInterfaceList_ = NULL;
4365 list = addrindex_get_interface_list( _addressIndex_ );
4367 AddressInterface *interface = list->data;
4368 atci = addrbookctl_lookup_iface( interface->type );
4370 adapter = g_new0( AdapterInterface, 1 );
4371 adapter->interfaceType = interface->type;
4372 adapter->atci = atci;
4373 adapter->interface = interface;
4374 adapter->treeNode = NULL;
4375 adapter->enabled = TRUE;
4376 adapter->haveLibrary = interface->haveLibrary;
4377 ADDRESS_OBJECT(adapter)->type = ADDR_INTERFACE;
4378 ADDRESS_OBJECT_NAME(adapter) = g_strdup( atci->displayName );
4379 _addressInterfaceList_ =
4380 g_list_append( _addressInterfaceList_, adapter );
4382 list = g_list_next( list );
4387 * Find GUI interface type specified interface type.
4388 * \param ifType Interface type.
4389 * \return Interface item, or NULL if not found.
4391 AdapterInterface *addrbookctl_find_interface( AddressIfType ifType ) {
4392 GList *node = _addressInterfaceList_;
4394 AdapterInterface *adapter = node->data;
4395 if( adapter->interfaceType == ifType ) return adapter;
4396 node = g_list_next( node );
4402 * Build interface list selection.
4404 void addrbookctl_build_ifselect( void ) {
4405 GList *newList = NULL;
4410 gchar *endptr = NULL;
4412 AdapterInterface *adapter;
4414 selectStr = g_strdup( ADDRESSBOOK_IFACE_SELECTION );
4417 splitStr = g_strsplit( selectStr, ",", -1 );
4418 for( i = 0; i < ADDRESSBOOK_MAX_IFACE; i++ ) {
4420 /* printf( "%d : %s\n", i, splitStr[i] ); */
4421 ifType = strtol( splitStr[i], &endptr, 10 );
4424 if( strcmp( endptr, "/n" ) == 0 ) {
4428 /* printf( "\t%d : %s\n", ifType, enabled ? "yes" : "no" ); */
4429 adapter = addrbookctl_find_interface( ifType );
4431 newList = g_list_append( newList, adapter );
4438 /* printf( "i=%d\n", i ); */
4439 g_strfreev( splitStr );
4440 g_free( selectStr );
4442 /* Replace existing list */
4443 mgu_clear_list( _addressIFaceSelection_ );
4444 g_list_free( _addressIFaceSelection_ );
4445 _addressIFaceSelection_ = newList;
4449 /* ***********************************************************************
4450 * Add sender to address book.
4451 * ***********************************************************************
4455 * This function is used by the Add sender to address book function.
4457 gboolean addressbook_add_contact(
4458 const gchar *name, const gchar *address, const gchar *remarks )
4460 debug_print( "addressbook_add_contact: name/address: %s - %s\n", name, address );
4461 if( addressadd_selection( _addressIndex_, name, address, remarks ) ) {
4462 debug_print( "addressbook_add_contact - added\n" );
4463 addressbook_refresh();
4468 /* ***********************************************************************
4469 * Book/folder selection.
4470 * ***********************************************************************
4474 * This function is used by the matcher dialog to select a book/folder.
4476 gboolean addressbook_folder_selection( gchar **folderpath )
4478 AddressBookFile *book = NULL;
4479 ItemFolder *folder = NULL;
4482 g_return_val_if_fail( folderpath != NULL, FALSE);
4486 if ( addressbook_foldersel_selection( _addressIndex_, &book, &folder, path )
4488 if ( folder != NULL) {
4490 gchar *oldtmp = NULL;
4491 AddrItemObject *obj = NULL;
4493 /* walk thru folder->parent to build the full folder path */
4494 /* TODO: wwp: optimize this */
4496 tmp = g_strdup(obj->uid);
4497 while ( obj->parent ) {
4499 if ( obj->name != NULL ) {
4500 oldtmp = g_strdup(tmp);
4502 tmp = g_strdup_printf("%s/%s", obj->uid, oldtmp);
4506 *folderpath = g_strdup_printf("%s/%s", book->fileName, tmp);
4509 *folderpath = g_strdup_printf("%s", book->fileName);
4511 debug_print( "addressbook_foldersel: %s\n", *folderpath);
4512 return (*folderpath != NULL);
4517 /* ***********************************************************************
4518 * Book/folder checking.
4519 * ***********************************************************************
4522 static FolderInfo *addressbook_peek_subfolder_exists_create_folderinfo( AddressBookFile *abf, ItemFolder *folder )
4524 FolderInfo *fi = g_new0( FolderInfo, 1 );
4526 fi->folder = folder;
4530 static void addressbook_peek_subfolder_exists_load_folder( ItemFolder *parentFolder,
4531 FolderInfo *fiParent, FolderPathMatch *match )
4537 FolderPathMatch *nextmatch = NULL;
4539 list = parentFolder->listFolder;
4541 folder = list->data;
4542 fName = g_strdup( ADDRITEM_NAME(folder) );
4544 /* match folder name, match pointer will be set to NULL if next recursive call
4545 doesn't need to match subfolder name */
4546 if ( match != NULL &&
4547 match->matched == FALSE ) {
4548 if ( strcmp(match->folder_path[match->index], folder->obj.uid) == 0 ) {
4549 /* folder name matches, prepare next subfolder match */
4550 debug_print("matched folder name '%s'\n", fName);
4552 if ( match->folder_path[match->index] == NULL ) {
4553 /* we've matched all elements */
4554 match->matched = TRUE;
4555 match->folder = folder;
4556 debug_print("book/folder path matched!\n");
4558 /* keep on matching */
4566 fi = addressbook_peek_subfolder_exists_create_folderinfo( fiParent->book, folder );
4567 addressbook_peek_subfolder_exists_load_folder( folder, fi, nextmatch );
4568 list = g_list_next( list );
4573 * This function is used by to check if a matcher book/folder path corresponds to an
4574 existing addressbook book/folder ("" or "Any" are considered as valid, NULL invalid).
4577 gboolean addressbook_peek_folder_exists( gchar *folderpath,
4578 AddressDataSource **book,
4579 ItemFolder **folder )
4581 AddressDataSource *ds;
4582 GList *list, *nodeDS;
4583 ItemFolder *rootFolder;
4584 AddressBookFile *abf;
4586 FolderPathMatch folder_path_match = { NULL, FALSE, 0, NULL, NULL };
4587 FolderPathMatch *nextmatch;
4594 if ( folderpath == NULL )
4597 if ( strcasecmp(folderpath, _("Any")) == 0 || *folderpath == '\0' )
4600 /* split the folder path we've received, we'll try to match this path, subpath by
4601 subpath against the book/folder structure in order */
4602 folder_path_match.folder_path = g_strsplit( folderpath, "/", 256 );
4604 list = addrindex_get_interface_list( _addressIndex_ );
4606 AddressInterface *interface = list->data;
4607 if ( interface->type == ADDR_IF_BOOK ) {
4608 nodeDS = interface->listSource;
4612 /* Read address book */
4613 if( ! addrindex_ds_get_read_flag( ds ) ) {
4614 addrindex_ds_read_data( ds );
4617 /* Add node for address book */
4618 abf = ds->rawDataSource;
4620 /* try to match subfolders if this book is the right book
4621 (and if there's smth to match, and not yet matched) */
4623 if ( folder_path_match.folder_path != NULL &&
4624 folder_path_match.matched == FALSE &&
4625 strcmp(folder_path_match.folder_path[0], abf->fileName) == 0 ) {
4626 debug_print("matched book name '%s'\n", abf->fileName);
4627 folder_path_match.index = 1;
4628 if ( folder_path_match.folder_path[folder_path_match.index] == NULL ) {
4629 /* we've matched all elements */
4630 folder_path_match.matched = TRUE;
4631 folder_path_match.book = ds;
4632 debug_print("book path matched!\n");
4634 /* keep on matching */
4635 nextmatch = &folder_path_match;
4639 fi = addressbook_peek_subfolder_exists_create_folderinfo( abf, NULL );
4641 rootFolder = addrindex_ds_get_root_folder( ds );
4642 addressbook_peek_subfolder_exists_load_folder( rootFolder, fi, nextmatch );
4644 nodeDS = g_list_next( nodeDS );
4647 list = g_list_next( list );
4650 g_strfreev( folder_path_match.folder_path );
4653 *book = folder_path_match.book;
4655 *folder = folder_path_match.folder;
4656 return folder_path_match.matched;
4660 /* **********************************************************************
4662 * ***********************************************************************
4668 static void addressbook_import_ldif_cb( void ) {
4669 AddressDataSource *ds = NULL;
4670 AdapterDSource *ads = NULL;
4671 AddressBookFile *abf = NULL;
4672 AdapterInterface *adapter;
4673 GtkCTreeNode *newNode;
4675 adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
4677 if( adapter->treeNode ) {
4678 abf = addressbook_imp_ldif( _addressIndex_ );
4680 ds = addrindex_index_add_datasource(
4681 _addressIndex_, ADDR_IF_BOOK, abf );
4682 ads = addressbook_create_ds_adapter(
4683 ds, ADDR_BOOK, NULL );
4684 addressbook_ads_set_name(
4685 ads, addrbook_get_name( abf ) );
4686 newNode = addressbook_add_object(
4688 ADDRESS_OBJECT(ads) );
4690 gtk_sctree_select( GTK_SCTREE(addrbook.ctree),
4692 addrbook.treeSelected = newNode;
4695 /* Notify address completion */
4696 invalidate_address_completion();
4705 static void addressbook_import_mutt_cb( void ) {
4706 AddressDataSource *ds = NULL;
4707 AdapterDSource *ads = NULL;
4708 AddressBookFile *abf = NULL;
4709 AdapterInterface *adapter;
4710 GtkCTreeNode *newNode;
4712 adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
4714 if( adapter->treeNode ) {
4715 abf = addressbook_imp_mutt( _addressIndex_ );
4717 ds = addrindex_index_add_datasource(
4718 _addressIndex_, ADDR_IF_BOOK, abf );
4719 ads = addressbook_create_ds_adapter(
4720 ds, ADDR_BOOK, NULL );
4721 addressbook_ads_set_name(
4722 ads, addrbook_get_name( abf ) );
4723 newNode = addressbook_add_object(
4725 ADDRESS_OBJECT(ads) );
4727 gtk_sctree_select( GTK_SCTREE(addrbook.ctree),
4729 addrbook.treeSelected = newNode;
4732 /* Notify address completion */
4733 invalidate_address_completion();
4742 static void addressbook_import_pine_cb( void ) {
4743 AddressDataSource *ds = NULL;
4744 AdapterDSource *ads = NULL;
4745 AddressBookFile *abf = NULL;
4746 AdapterInterface *adapter;
4747 GtkCTreeNode *newNode;
4749 adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
4751 if( adapter->treeNode ) {
4752 abf = addressbook_imp_pine( _addressIndex_ );
4754 ds = addrindex_index_add_datasource(
4755 _addressIndex_, ADDR_IF_BOOK, abf );
4756 ads = addressbook_create_ds_adapter(
4757 ds, ADDR_BOOK, NULL );
4758 addressbook_ads_set_name(
4759 ads, addrbook_get_name( abf ) );
4760 newNode = addressbook_add_object(
4762 ADDRESS_OBJECT(ads) );
4764 gtk_sctree_select( GTK_SCTREE(addrbook.ctree),
4766 addrbook.treeSelected = newNode;
4769 /* Notify address completion */
4770 invalidate_address_completion();
4777 * Harvest addresses.
4778 * \param folderItem Folder to import.
4779 * \param sourceInd Source indicator: FALSE - Folder, TRUE - Messages.
4780 * \param msgList List of message numbers, or NULL to process folder.
4782 void addressbook_harvest(
4783 FolderItem *folderItem, gboolean sourceInd, GList *msgList )
4785 AddressDataSource *ds = NULL;
4786 AdapterDSource *ads = NULL;
4787 AddressBookFile *abf = NULL;
4788 AdapterInterface *adapter;
4789 GtkCTreeNode *newNode;
4791 abf = addrgather_dlg_execute(
4792 folderItem, _addressIndex_, sourceInd, msgList );
4794 ds = addrindex_index_add_datasource(
4795 _addressIndex_, ADDR_IF_BOOK, abf );
4797 adapter = addrbookctl_find_interface( ADDR_IF_BOOK );
4799 if( adapter->treeNode ) {
4800 ads = addressbook_create_ds_adapter(
4801 ds, ADDR_BOOK, addrbook_get_name( abf ) );
4802 newNode = addressbook_add_object(
4804 ADDRESS_OBJECT(ads) );
4808 /* Notify address completion */
4809 invalidate_address_completion();
4816 static void addressbook_export_html_cb( void ) {
4817 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
4819 AddressDataSource *ds = NULL;
4820 AddrBookBase *adbase;
4821 AddressCache *cache;
4822 GtkCTreeNode *node = NULL;
4824 if( ! addrbook.treeSelected ) return;
4825 node = addrbook.treeSelected;
4826 if( GTK_CTREE_ROW(node)->level == 1 ) return;
4827 obj = gtk_ctree_node_get_row_data( ctree, node );
4828 if( obj == NULL ) return;
4830 ds = addressbook_find_datasource( node );
4831 if( ds == NULL ) return;
4832 adbase = ( AddrBookBase * ) ds->rawDataSource;
4833 cache = adbase->addressCache;
4834 addressbook_exp_html( cache );
4840 static void addressbook_export_ldif_cb( void ) {
4841 GtkCTree *ctree = GTK_CTREE(addrbook.ctree);
4843 AddressDataSource *ds = NULL;
4844 AddrBookBase *adbase;
4845 AddressCache *cache;
4846 GtkCTreeNode *node = NULL;
4848 if( ! addrbook.treeSelected ) return;
4849 node = addrbook.treeSelected;
4850 if( GTK_CTREE_ROW(node)->level == 1 ) return;
4851 obj = gtk_ctree_node_get_row_data( ctree, node );
4852 if( obj == NULL ) return;
4854 ds = addressbook_find_datasource( node );
4855 if( ds == NULL ) return;
4856 adbase = ( AddrBookBase * ) ds->rawDataSource;
4857 cache = adbase->addressCache;
4858 addressbook_exp_ldif( cache );
4861 static void addressbook_start_drag(GtkWidget *widget, gint button,
4865 GdkDragContext *context;
4866 if (addressbook_target_list == NULL)
4867 addressbook_target_list = gtk_target_list_new(
4868 addressbook_drag_types, 1);
4869 context = gtk_drag_begin(widget, addressbook_target_list,
4870 GDK_ACTION_MOVE|GDK_ACTION_COPY|GDK_ACTION_DEFAULT, button, event);
4871 gtk_drag_set_icon_default(context);
4874 static void addressbook_drag_data_get(GtkWidget *widget,
4875 GdkDragContext *drag_context,
4876 GtkSelectionData *selection_data,
4881 AddrItemObject *aio = NULL;
4882 AddressObject *pobj = NULL;
4883 AdapterDSource *ads = NULL;
4884 AddressDataSource *ds = NULL;
4887 pobj = gtk_ctree_node_get_row_data( GTK_CTREE(addrbook.ctree), addrbook.treeSelected );
4889 if( pobj == NULL ) return;
4891 if( pobj->type == ADDR_DATASOURCE ) {
4892 ads = ADAPTER_DSOURCE(pobj);
4893 ds = ads->dataSource;
4894 } else if (pobj->type == ADDR_ITEM_GROUP) {
4899 else if( pobj->type != ADDR_INTERFACE ) {
4900 ds = addressbook_find_datasource( addrbook.treeSelected );
4906 for(cur = GTK_CLIST(addrbook.clist)->selection; cur; cur = cur->next) {
4907 aio = (AddrItemObject *)gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.clist),
4908 GTK_CTREE_NODE(cur->data));
4909 while (aio && aio->type != ADDR_ITEM_PERSON) {
4914 if (aio && aio->type == ADDR_ITEM_PERSON) {
4915 if( ds && ds->interface && ds->interface->readOnly)
4916 gtk_selection_data_set(selection_data,
4917 selection_data->target, 8,
4918 "Dummy_addr_copy", 15);
4920 gtk_selection_data_set(selection_data,
4921 selection_data->target, 8,
4922 "Dummy_addr_move", 15);
4926 static gboolean addressbook_drag_motion_cb(GtkWidget *widget,
4927 GdkDragContext *context,
4934 GtkCTreeNode *node = NULL;
4935 gboolean acceptable = FALSE;
4936 gint height = addrbook.ctree->allocation.height;
4937 gint total_height = addrbook.ctree->requisition.height;
4938 GtkAdjustment *pos = gtk_scrolled_window_get_vadjustment(
4939 GTK_SCROLLED_WINDOW(addrbook.ctree_swin));
4940 gfloat vpos = pos->value;
4942 if (gtk_clist_get_selection_info
4943 (GTK_CLIST(widget), x - 24, y - 24, &row, &column)) {
4945 if (y > height - 24 && height + vpos < total_height)
4946 gtk_adjustment_set_value(pos, (vpos+5 > height ? height : vpos+5));
4948 if (y < 24 && y > 0)
4949 gtk_adjustment_set_value(pos, (vpos-5 < 0 ? 0 : vpos-5));
4951 node = gtk_ctree_node_nth(GTK_CTREE(widget), row);
4954 AddressObject *obj = gtk_ctree_node_get_row_data(GTK_CTREE(widget), node );
4955 if( obj->type == ADDR_ITEM_FOLDER
4956 || obj->type == ADDR_ITEM_GROUP)
4959 AdapterDSource *ads = NULL;
4960 AddressDataSource *ds = NULL;
4961 ads = ADAPTER_DSOURCE(obj);
4962 if (ads == NULL ){ return FALSE;}
4963 ds = ads->dataSource;
4964 if (ds == NULL ) { return FALSE;}
4972 g_signal_handlers_block_by_func
4974 G_CALLBACK(addressbook_tree_selected), NULL);
4975 gtk_sctree_select( GTK_SCTREE(widget), node);
4976 g_signal_handlers_unblock_by_func
4978 G_CALLBACK(addressbook_tree_selected), NULL);
4979 gdk_drag_status(context,
4980 (context->actions == GDK_ACTION_COPY ?
4981 GDK_ACTION_COPY : GDK_ACTION_MOVE) , time);
4983 gdk_drag_status(context, 0, time);
4989 static void addressbook_drag_leave_cb(GtkWidget *widget,
4990 GdkDragContext *context,
4994 if (addrbook.treeSelected) {
4995 g_signal_handlers_block_by_func
4997 G_CALLBACK(addressbook_tree_selected), NULL);
4998 gtk_sctree_select( GTK_SCTREE(widget), addrbook.opened);
4999 g_signal_handlers_unblock_by_func
5001 G_CALLBACK(addressbook_tree_selected), NULL);
5006 static void addressbook_drag_received_cb(GtkWidget *widget,
5007 GdkDragContext *drag_context,
5010 GtkSelectionData *data,
5017 GtkCTreeNode *lastopened = addrbook.opened;
5019 if (!strncmp(data->data, "Dummy_addr", 10)) {
5020 if (gtk_clist_get_selection_info
5021 (GTK_CLIST(widget), x - 24, y - 24, &row, &column) == 0) {
5025 node = gtk_ctree_node_nth(GTK_CTREE(widget), row);
5026 if( !node || !gtk_ctree_node_get_row_data(GTK_CTREE(addrbook.ctree), node))
5029 gtk_clist_freeze(GTK_CLIST(addrbook.clist));
5030 if (drag_context->action == GDK_ACTION_COPY ||
5031 !strcmp(data->data, "Dummy_addr_copy"))
5032 addressbook_clip_copy_cb();
5034 addressbook_clip_cut_cb();
5035 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), node);
5036 addressbook_clip_paste_cb();
5037 gtk_sctree_select( GTK_SCTREE(addrbook.ctree), lastopened);
5038 gtk_clist_thaw(GTK_CLIST(addrbook.clist));
5039 gtk_drag_finish(drag_context, TRUE, TRUE, time);