3.7.2 unleashed
[claws.git] / src / addrduplicates.c
blob7dd97dd9035687ede474d26299e37ab029e8dabd
1 /* Claws Mail -- a GTK+ based, lightweight, and fast e-mail client
2 * Copyright (C) 2007-2009 Holger Berndt <hb@claws-mail.org>
3 * and the Claws Mail 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 3 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, see <http://www.gnu.org/licenses/>.
19 #ifdef HAVE_CONFIG_H
20 # include "config.h"
21 #endif
23 #include <gdk/gdk.h>
24 #include <gdk/gdkkeysyms.h>
25 #include <glib/gi18n.h>
26 #include <string.h>
28 #include "defs.h"
30 #ifdef USE_LDAP
31 #include "ldapserver.h"
32 #include "ldapupdate.h"
33 #endif
34 #include "addrduplicates.h"
35 #include "addrbook.h"
36 #include "addressbook.h"
37 #include "editaddress.h"
38 #include "alertpanel.h"
39 #include "gtkutils.h"
40 #include "inc.h"
41 #include "utils.h"
42 #include "prefs_common.h"
44 typedef struct
46 ItemPerson *person;
47 AddressDataSource *ds;
48 gchar *book_path;
50 AddrDupListEntry;
52 enum {
53 COL_BOOKPATH = 0,
54 COL_NAME,
55 COL_ITEM,
56 COL_DS,
57 NUM_COLS
60 static gboolean create_dialog();
61 static void refresh_addr_hash(void);
62 static void refresh_stores(gchar*,GSList*);
63 static void present_finder_results(GtkWindow*);
64 static void cb_finder_results_dialog_destroy(GtkWindow*, gpointer);
65 static gboolean cb_finder_results_dialog_key_pressed(GtkWidget*, GdkEventKey*,
66 gpointer);
67 static void destroy_addr_hash_val(gpointer);
68 static GSList* deep_copy_hash_val(GSList*);
69 static void fill_hash_table();
70 static gint collect_emails(ItemPerson*, AddressDataSource*);
71 static gboolean is_not_duplicate(gpointer, gpointer, gpointer);
72 static gint books_compare(gconstpointer, gconstpointer);
73 static GtkWidget* create_email_view(GtkListStore*);
74 static GtkWidget* create_detail_view(GtkListStore*);
75 static void append_to_email_store(gpointer,gpointer,gpointer);
76 static void email_selection_changed(GtkTreeSelection*,gpointer);
77 static void detail_selection_changed(GtkTreeSelection*,gpointer);
78 static void detail_row_activated(GtkTreeView*,GtkTreePath*,
79 GtkTreeViewColumn*,
80 gpointer);
81 static gboolean detail_focus_in(GtkWidget*,GdkEventFocus*,gpointer);
82 static gboolean detail_focus_out(GtkWidget*,GdkEventFocus*,gpointer);
84 static void cb_del_btn_clicked(GtkButton *, gpointer);
85 static void cb_edit_btn_clicked(GtkButton *, gpointer);
86 static gchar* get_bookpath(ItemPerson*,AddressDataSource*);
87 static gboolean is_editing_entry_only_selection(void);
88 static void edit_post_update_cb(ItemPerson*);
90 static GHashTable *addr_hash;
91 static gboolean include_same_book = TRUE;
92 static gboolean include_other_books = TRUE;
94 static GtkListStore *email_store;
95 static GtkListStore *detail_store;
96 static GtkWidget *email_view;
97 static GtkWidget *detail_view;
98 static GtkWidget *inline_edit_vbox;
100 static GtkWidget *del_btn;
101 static GtkWidget *edit_btn;
103 static GtkWidget *dialog;
104 static gchar *editing_uid;
105 static gboolean detail_view_has_focus;
107 void addrduplicates_find(GtkWindow *parent)
109 if(create_dialog()) {
110 refresh_addr_hash();
111 present_finder_results(parent);
115 static gboolean create_dialog()
117 gboolean want_search;
118 GtkWidget *vbox;
119 GtkWidget *check_same_book;
120 GtkWidget *check_other_book;
121 AlertValue val;
123 want_search = FALSE;
125 vbox = gtk_vbox_new(FALSE, 0);
126 check_same_book = gtk_check_button_new_with_label(_("Show duplicates in "
127 "the same book"));
128 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check_same_book),
129 include_same_book);
130 gtk_box_pack_start(GTK_BOX(vbox), check_same_book, FALSE, FALSE, 0);
131 gtk_widget_show(check_same_book);
132 check_other_book = gtk_check_button_new_with_label(_("Show duplicates in "
133 "different books"));
134 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check_other_book),
135 include_other_books);
136 gtk_box_pack_start(GTK_BOX(vbox), check_other_book, FALSE, FALSE, 0);
137 gtk_widget_show(check_other_book);
139 /* prevent checkboxes from being destroyed on dialog close */
140 g_object_ref(check_same_book);
141 g_object_ref(check_other_book);
143 val = alertpanel_full(_("Find address book email duplicates"),
144 _("Claws Mail will now search for duplicate email "
145 "addresses in the address book."),
146 GTK_STOCK_CANCEL,GTK_STOCK_FIND,NULL, FALSE, vbox, ALERT_NOTICE,
147 G_ALERTALTERNATE);
148 if(val == G_ALERTALTERNATE) {
149 want_search = TRUE;
151 /* save options */
152 include_same_book =
153 gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(check_same_book));
154 include_other_books =
155 gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(check_other_book));
159 g_object_unref(check_same_book);
160 g_object_unref(check_other_book);
161 return want_search;
164 static void refresh_addr_hash(void)
166 if(addr_hash)
167 g_hash_table_destroy(addr_hash);
168 addr_hash = g_hash_table_new_full(g_str_hash, g_str_equal,
169 g_free, destroy_addr_hash_val);
170 fill_hash_table();
173 static void destroy_addr_hash_val(gpointer value)
175 GSList *list = (GSList*) value;
176 GSList *walk;
178 for(walk = list; walk; walk = walk->next) {
179 AddrDupListEntry *entry = (AddrDupListEntry*) walk->data;
180 if(entry && entry->book_path)
181 g_free(entry->book_path);
182 if(entry)
183 g_free(entry);
185 if(list)
186 g_slist_free(list);
189 static GSList* deep_copy_hash_val(GSList *in)
191 GSList *walk;
192 GSList *out = NULL;
194 out = g_slist_copy(in);
195 for(walk = out; walk; walk = walk->next) {
196 AddrDupListEntry *out_entry;
197 AddrDupListEntry *in_entry = walk->data;
199 out_entry = g_new0(AddrDupListEntry,1);
200 out_entry->person = in_entry->person;
201 out_entry->ds = in_entry->ds;
202 out_entry->book_path = g_strdup(in_entry->book_path);
203 walk->data = out_entry;
206 return out;
209 static void fill_hash_table()
211 addrindex_load_person_ds(collect_emails);
212 g_hash_table_foreach_remove(addr_hash,is_not_duplicate, NULL);
215 static gboolean is_not_duplicate(gpointer key, gpointer value,
216 gpointer user_data)
218 gboolean is_in_same_book;
219 gboolean is_in_other_books;
220 GSList *books;
221 GSList *walk;
222 gboolean retval;
223 GSList *list = value;
225 /* remove everything that is just in one book */
226 if(g_slist_length(list) <= 1)
227 return TRUE;
229 /* work on a shallow copy */
230 books = g_slist_copy(list);
232 /* sorting the list makes it easier to check for books */
233 books = g_slist_sort(books, books_compare);
235 /* check if a book appears twice */
236 is_in_same_book = FALSE;
237 for(walk = books; walk && walk->next; walk = walk->next) {
238 if(books_compare(walk->data, walk->next->data) == 0) {
239 is_in_same_book = TRUE;
240 break;
244 /* check is at least two different books appear in the list */
245 is_in_other_books = FALSE;
246 if(books && books->next) {
247 for(walk = books->next; walk; walk = walk->next) {
248 if(books_compare(walk->data, books->data) != 0) {
249 is_in_other_books = TRUE;
250 break;
255 /* delete the shallow copy */
256 g_slist_free(books);
258 retval = FALSE;
259 if(is_in_same_book && include_same_book)
260 retval = TRUE;
261 if(is_in_other_books && include_other_books)
262 retval = TRUE;
263 retval = !retval;
265 return retval;
268 static gint collect_emails(ItemPerson *itemperson, AddressDataSource *ds)
270 gchar *addr;
271 GList *nodeM;
272 GSList *old_val;
273 GSList *new_val;
274 AddrDupListEntry *entry;
276 /* Process each E-Mail address */
277 nodeM = itemperson->listEMail;
278 while(nodeM) {
279 ItemEMail *email = nodeM->data;
281 addr = g_utf8_strdown(email->address, -1);
282 old_val = g_hash_table_lookup(addr_hash, addr);
283 if(old_val)
284 new_val = deep_copy_hash_val(old_val);
285 else
286 new_val = NULL;
288 entry = g_new0(AddrDupListEntry,1);
289 entry->person = itemperson;
290 entry->ds = ds;
291 entry->book_path = get_bookpath(itemperson, ds);
293 new_val = g_slist_prepend(new_val, entry);
294 g_hash_table_insert(addr_hash, addr, new_val);
296 nodeM = g_list_next(nodeM);
298 return 0;
301 static gint books_compare(gconstpointer a, gconstpointer b)
303 const AddrDupListEntry *entry1;
304 const AddrDupListEntry *entry2;
305 entry1 = a;
306 entry2 = b;
307 return strcmp(entry1->book_path, entry2->book_path);
310 static void present_finder_results(GtkWindow *parent)
312 GtkWidget *scrolled_win;
313 GtkWidget *vbox;
314 GtkWidget *hbox;
315 GtkWidget *hpaned;
316 GtkWidget *vpaned;
317 GtkWidget *close;
318 gint pos;
319 GtkTreeSelection *email_select;
320 GtkTreeSelection *detail_select;
321 static GdkGeometry geometry;
323 if(g_hash_table_size(addr_hash) == 0) {
324 alertpanel_notice(_("No duplicate email addresses found in the address book"));
325 return;
328 email_store = gtk_list_store_new(1, G_TYPE_STRING);
329 refresh_stores(NULL,NULL);
330 email_view = create_email_view(email_store);
331 email_select = gtk_tree_view_get_selection(GTK_TREE_VIEW(email_view));
332 gtk_tree_selection_set_mode(email_select,GTK_SELECTION_SINGLE);
334 g_signal_connect(email_select, "changed",
335 (GCallback)email_selection_changed, NULL);
337 detail_store = gtk_list_store_new(NUM_COLS, G_TYPE_STRING, G_TYPE_STRING,
338 G_TYPE_POINTER, G_TYPE_POINTER);
339 detail_view = create_detail_view(detail_store);
340 detail_select = gtk_tree_view_get_selection(GTK_TREE_VIEW(detail_view));
341 gtk_tree_selection_set_mode(detail_select,GTK_SELECTION_MULTIPLE);
343 g_signal_connect(detail_select, "changed",
344 (GCallback)detail_selection_changed, NULL);
346 dialog = gtkut_window_new(GTK_WINDOW_TOPLEVEL, "address_dupes_finder");
347 gtk_window_set_transient_for(GTK_WINDOW(dialog),parent);
348 gtk_window_set_modal(GTK_WINDOW(dialog),TRUE);
349 if(!geometry.min_height) {
350 geometry.min_width = 600;
351 geometry.min_height = 400;
353 gtk_window_set_geometry_hints(GTK_WINDOW(dialog), NULL, &geometry,
354 GDK_HINT_MIN_SIZE);
355 gtk_window_set_title(GTK_WINDOW(dialog), _("Duplicate email addresses"));
357 vbox = gtk_vbox_new(FALSE, 0);
358 gtk_container_add(GTK_CONTAINER(dialog), vbox);
360 hpaned = gtk_hpaned_new();
361 gtk_box_pack_start(GTK_BOX(vbox), hpaned, TRUE, TRUE, 0);
363 scrolled_win = gtk_scrolled_window_new(NULL,NULL);
364 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled_win),
365 GTK_POLICY_AUTOMATIC,
366 GTK_POLICY_AUTOMATIC);
367 gtk_container_add(GTK_CONTAINER(scrolled_win), email_view);
369 gtk_paned_add1(GTK_PANED(hpaned), scrolled_win);
371 scrolled_win = gtk_scrolled_window_new(NULL,NULL);
372 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled_win),
373 GTK_POLICY_AUTOMATIC,
374 GTK_POLICY_AUTOMATIC);
375 gtk_container_add(GTK_CONTAINER(scrolled_win), detail_view);
377 if (prefs_common.addressbook_use_editaddress_dialog) {
378 gtk_paned_add2(GTK_PANED(hpaned), scrolled_win);
379 inline_edit_vbox = NULL;
380 } else {
381 inline_edit_vbox = gtk_vbox_new(FALSE, 4);
382 vpaned = gtk_vpaned_new();
383 gtk_paned_pack1(GTK_PANED(vpaned), scrolled_win, FALSE, FALSE);
384 gtk_paned_pack2(GTK_PANED(vpaned), inline_edit_vbox, TRUE, FALSE);
385 gtk_paned_pack2(GTK_PANED(hpaned), vpaned, TRUE, FALSE);
388 g_object_get(G_OBJECT(hpaned),
389 "position", &pos, NULL);
390 if(pos < 200)
391 gtk_paned_set_position(GTK_PANED(hpaned), 200);
393 hbox = gtk_hbutton_box_new();
394 gtk_button_box_set_layout(GTK_BUTTON_BOX(hbox), GTK_BUTTONBOX_END);
395 gtk_box_set_spacing(GTK_BOX(hbox), 2);
396 gtk_container_set_border_width(GTK_CONTAINER(hbox), 4);
397 gtk_box_pack_end(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
399 edit_btn = gtk_button_new_from_stock(GTK_STOCK_EDIT);
400 gtk_box_pack_start(GTK_BOX(hbox), edit_btn, TRUE, TRUE, 0);
401 gtk_widget_set_sensitive(edit_btn, FALSE);
403 del_btn = gtk_button_new_from_stock(GTK_STOCK_DELETE);
404 gtk_box_pack_start(GTK_BOX(hbox), del_btn, TRUE, TRUE, 0);
405 gtk_widget_set_sensitive(del_btn, FALSE);
407 close = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
408 gtk_box_pack_start(GTK_BOX(hbox), close, TRUE, TRUE, 0);
410 g_signal_connect(dialog, "destroy",
411 G_CALLBACK(cb_finder_results_dialog_destroy), NULL);
412 g_signal_connect(G_OBJECT(dialog), "key-press-event",
413 G_CALLBACK(cb_finder_results_dialog_key_pressed), NULL);
414 g_signal_connect_swapped(close, "clicked",
415 G_CALLBACK(gtk_widget_destroy), dialog);
416 g_signal_connect(del_btn, "clicked",
417 G_CALLBACK(cb_del_btn_clicked), detail_view);
418 g_signal_connect(edit_btn, "clicked",
419 G_CALLBACK(cb_edit_btn_clicked), detail_view);
421 inc_lock();
422 gtk_widget_show_all(dialog);
425 static void cb_finder_results_dialog_destroy(GtkWindow *win, gpointer data)
427 email_store = NULL;
428 detail_store = NULL;
429 email_view = NULL;
430 inline_edit_vbox = NULL;
432 if(addr_hash) {
433 g_hash_table_destroy(addr_hash);
434 addr_hash = NULL;
436 dialog = NULL;
437 addressbook_refresh();
438 inc_unlock();
441 static GtkWidget* create_email_view(GtkListStore *store)
443 GtkWidget *view;
444 GtkCellRenderer *renderer;
446 view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
447 gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(view), prefs_common.use_stripes_everywhere);
448 renderer = gtk_cell_renderer_text_new();
449 gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(view),
451 _("Address"),
452 renderer,
453 "text", 0,
454 NULL);
455 g_object_unref(store);
456 return view;
459 static GtkWidget* create_detail_view(GtkListStore *store)
461 GtkWidget *view;
462 GtkCellRenderer *renderer;
463 GList *cols;
464 GList *walk;
466 view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
467 gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(view), prefs_common.use_stripes_everywhere);
468 renderer = gtk_cell_renderer_text_new();
470 /* col 1 */
471 gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(view),
473 _("Address book path"),
474 renderer,
475 "text", COL_BOOKPATH,
476 NULL);
477 /* col 2 */
478 gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(view),
480 _("Name"),
481 renderer,
482 "text", COL_NAME,
483 NULL);
485 cols = gtk_tree_view_get_columns(GTK_TREE_VIEW(view));
486 for(walk = cols; walk; walk = walk->next)
487 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(walk->data),
488 TRUE);
489 g_list_free(cols);
491 g_signal_connect(view, "row-activated",
492 G_CALLBACK(detail_row_activated), NULL);
494 g_signal_connect(view, "focus-in-event",
495 G_CALLBACK(detail_focus_in), NULL);
496 g_signal_connect(view, "focus-out-event",
497 G_CALLBACK(detail_focus_out), NULL);
500 return view;
503 static void append_to_email_store(gpointer key,gpointer value,gpointer data)
505 GtkTreeIter iter;
506 GtkListStore *store = (GtkListStore*) data;
508 gtk_list_store_append(store, &iter);
509 gtk_list_store_set(store, &iter, 0, (gchar*) key, -1);
512 static gboolean is_editing_entry_only_selection(void)
514 GtkTreeSelection *sel_detail;
515 GtkTreeIter iter;
516 GList *selected;
517 GtkTreeModel *model;
518 ItemPerson *item;
520 sel_detail = gtk_tree_view_get_selection(GTK_TREE_VIEW(detail_view));
522 if(gtk_tree_selection_count_selected_rows(sel_detail) > 1)
523 return FALSE;
525 selected = gtk_tree_selection_get_selected_rows(sel_detail,&model);
526 if(!selected)
527 return FALSE;
529 gtk_tree_model_get_iter(model, &iter, (GtkTreePath*)selected->data);
530 g_list_foreach(selected, (GFunc)gtk_tree_path_free, NULL);
531 g_list_free(selected);
533 gtk_tree_model_get(model, &iter, COL_ITEM, &item,-1);
534 if(ADDRITEM_ID(item) && editing_uid &&
535 strcmp(ADDRITEM_ID(item),editing_uid) == 0)
536 return TRUE;
537 else
538 return FALSE;
541 static void detail_selection_changed(GtkTreeSelection *selection, gpointer data)
543 gint num_selected;
544 num_selected = gtk_tree_selection_count_selected_rows(selection);
546 if(num_selected > 0)
547 gtk_widget_set_sensitive(del_btn,TRUE);
548 else
549 gtk_widget_set_sensitive(del_btn,FALSE);
551 if(num_selected == 1)
552 gtk_widget_set_sensitive(edit_btn,TRUE);
553 else
554 gtk_widget_set_sensitive(edit_btn,FALSE);
556 if(!is_editing_entry_only_selection())
557 addressbook_edit_person_widgetset_hide();
560 static void email_selection_changed(GtkTreeSelection *selection, gpointer data)
562 GtkTreeIter iter;
563 GtkTreeModel *model;
564 gchar *email;
566 if(gtk_tree_selection_get_selected(selection, &model, &iter)) {
567 GSList *hashval;
568 GSList *walk;
570 gtk_tree_model_get(model, &iter, 0, &email, -1);
572 hashval = g_hash_table_lookup(addr_hash, email);
573 gtk_list_store_clear(detail_store);
574 for(walk = hashval; walk; walk = walk->next) {
575 AddrDupListEntry *entry = walk->data;
576 if(!entry)
577 continue;
578 gtk_list_store_append(detail_store, &iter);
579 gtk_list_store_set(detail_store, &iter,
580 COL_BOOKPATH, entry->book_path,
581 COL_NAME, addressbook_set_col_name_guard(ADDRITEM_NAME(entry->person)),
582 COL_ITEM, entry->person,
583 COL_DS, entry->ds,
584 -1);
586 g_free(email);
590 static gchar* get_bookpath(ItemPerson *itemPerson, AddressDataSource *ds)
592 gchar *path;
593 gchar *tmp;
594 AddrItemObject *item;
596 item = (AddrItemObject*)itemPerson;
597 path = g_strdup("");
598 while((item = ADDRITEM_PARENT(item)) != NULL) {
600 if(ADDRITEM_TYPE(item) == ITEMTYPE_FOLDER) {
601 ItemFolder *folder = (ItemFolder*) item;
602 tmp = path;
603 path = g_strdup_printf("%s%s%s",
604 folder->isRoot ? addrindex_ds_get_name(ds) :
605 ADDRITEM_NAME(folder),
606 (*tmp == '\0') ? "" : "/", tmp);
607 g_free(tmp);
612 /* prepend bookpath */
613 if(ds && ds->interface && ds->interface->name) {
614 tmp = path;
615 path = g_strdup_printf("%s%s%s", ds->interface->name,
616 (*tmp == '\0') ? "" : "/", tmp);
617 g_free(tmp);
620 return path;
623 static void refresh_stores(gchar *email_to_select, GSList *detail_to_select)
625 refresh_addr_hash();
626 if(email_store)
627 gtk_list_store_clear(email_store);
628 if(detail_store)
629 gtk_list_store_clear(detail_store);
630 g_hash_table_foreach(addr_hash,append_to_email_store,email_store);
632 /* sort the email store */
633 gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(email_store),
634 0, GTK_SORT_ASCENDING);
636 /* try to select email address */
637 if(email_to_select) {
638 /* Search email in email store */
639 GtkTreeIter iter;
640 GtkTreeSelection *selection;
642 if(!gtk_tree_model_get_iter_first(GTK_TREE_MODEL(email_store), &iter))
643 return;
644 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(email_view));
646 do {
647 gint retVal;
648 gchar *email;
650 gtk_tree_model_get(GTK_TREE_MODEL(email_store), &iter, 0, &email, -1);
651 retVal = g_ascii_strncasecmp(email,email_to_select,strlen(email));
652 g_free(email);
653 if(retVal == 0) {
654 gtk_tree_selection_select_iter(selection,&iter);
655 break;
657 } while(gtk_tree_model_iter_next(GTK_TREE_MODEL(email_store), &iter));
661 /* try to select detail rows */
662 if(detail_to_select) {
663 GtkTreeIter iter;
664 GtkTreeSelection *sel;
665 if(!gtk_tree_model_get_iter_first(GTK_TREE_MODEL(detail_store), &iter))
666 return;
667 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(detail_view));
669 do {
670 GSList *walk;
671 ItemPerson *person;
672 gtk_tree_model_get(GTK_TREE_MODEL(detail_store), &iter,
673 COL_ITEM, &person, -1);
674 for(walk = detail_to_select; walk; walk = walk->next) {
675 gchar *uid = walk->data;
676 if(uid && ADDRITEM_ID(person) &&
677 (strcmp(uid,ADDRITEM_ID(person)) == 0))
678 gtk_tree_selection_select_iter(sel,&iter);
680 } while(gtk_tree_model_iter_next(GTK_TREE_MODEL(detail_store), &iter));
684 static void detail_row_activated(GtkTreeView *tree_view,
685 GtkTreePath *path,
686 GtkTreeViewColumn *column,
687 gpointer user_data)
689 GtkTreeIter iter;
690 ItemPerson *person;
691 AddressDataSource *ds;
692 GtkTreeModel *model;
693 AddressBookFile *abf;
695 model = gtk_tree_view_get_model(tree_view);
697 if(!gtk_tree_model_get_iter(model,&iter,path))
698 return;
700 gtk_tree_model_get(model, &iter, COL_ITEM, &person, COL_DS, &ds, -1);
703 if(!((ds->type == ADDR_IF_BOOK) || ds->type == ADDR_IF_LDAP)) {
704 debug_print("Unsupported address datasource type for editing\n");
705 return;
708 abf = ds->rawDataSource;
709 if(inline_edit_vbox)
710 gtk_widget_show_all(inline_edit_vbox);
711 if(editing_uid)
712 g_free(editing_uid);
713 editing_uid = g_strdup(ADDRITEM_ID(person));
714 addressbook_edit_person(abf,NULL,person,FALSE,inline_edit_vbox,
715 edit_post_update_cb,FALSE);
718 static void edit_post_update_cb(ItemPerson *item)
720 GtkTreeSelection *sel;
721 gchar *email;
722 GList *detail_sel;
723 GList *walk;
724 GSList *detail;
725 GtkTreeIter iter;
726 GtkTreeModel *model;
727 ItemPerson *person;
729 /* save selection for after the update */
731 /* email -> string of email address */
732 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(email_view));
733 if(gtk_tree_selection_get_selected(sel,NULL,&iter))
734 gtk_tree_model_get(GTK_TREE_MODEL(email_store), &iter, 0, &email, -1);
735 else
736 email = NULL;
738 /* detail -> GSList of ItemPerson UIDs */
739 detail = NULL;
740 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(detail_view));
741 detail_sel = gtk_tree_selection_get_selected_rows(sel, &model);
742 for(walk = detail_sel; walk; walk = walk->next) {
743 GtkTreePath *path = walk->data;
744 if(!gtk_tree_model_get_iter(model,&iter,path))
745 continue;
746 gtk_tree_model_get(model, &iter, COL_ITEM, &person,-1);
747 detail = g_slist_prepend(detail, g_strdup(ADDRITEM_ID(person)));
749 g_list_foreach(detail_sel, (GFunc)gtk_tree_path_free, NULL);
750 g_list_free(detail_sel);
752 /* now refresh the stores, trying to keep the selections active */
753 refresh_stores(email,detail);
755 /* cleanup */
756 if(email)
757 g_free(email);
758 g_slist_foreach(detail, (GFunc)g_free, NULL);
759 g_slist_free(detail);
762 static void cb_edit_btn_clicked(GtkButton *button, gpointer data)
764 GtkTreeSelection *selection;
765 GList *selected;
766 GtkTreeModel *model;
768 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(detail_view));
769 selected = gtk_tree_selection_get_selected_rows(selection,&model);
770 cm_return_if_fail(selected);
772 /* we are guaranteed to have exactly one row selected */
773 gtk_tree_view_row_activated(GTK_TREE_VIEW(detail_view),(GtkTreePath*)selected->data,
774 gtk_tree_view_get_column(GTK_TREE_VIEW(detail_view),0));
776 g_list_foreach(selected, (GFunc)gtk_tree_path_free, NULL);
777 g_list_free(selected);
780 static void cb_del_btn_clicked(GtkButton *button, gpointer data)
782 GtkTreeIter iter;
783 GtkTreeModel *model;
784 GtkTreeSelection *selection;
785 ItemPerson *item;
786 AddressDataSource *ds;
787 GList *list;
788 GList *ref_list;
789 GList *walk;
790 GtkTreeRowReference *ref;
791 AlertValue aval;
792 GtkTreeSelection *sel;
793 gchar *email;
795 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(detail_view));
797 list = gtk_tree_selection_get_selected_rows(selection, &model);
799 if(!list)
800 return;
802 aval = alertpanel(_("Delete address(es)"),
803 _("Really delete the address(es)?"),
804 GTK_STOCK_CANCEL, "+"GTK_STOCK_DELETE, NULL);
805 if(aval != G_ALERTALTERNATE)
806 return;
808 ref_list = NULL;
809 for(walk = list; walk; walk = walk->next) {
810 ref = gtk_tree_row_reference_new(model,(GtkTreePath*)(walk->data));
811 ref_list = g_list_prepend(ref_list, ref);
813 g_list_foreach(list, (GFunc)gtk_tree_path_free, NULL);
814 g_list_free(list);
816 for(walk = ref_list; walk; walk = walk->next) {
817 GtkTreePath *path;
818 ref = walk->data;
819 if(!gtk_tree_row_reference_valid(ref))
820 continue;
821 path = gtk_tree_row_reference_get_path(ref);
822 if(gtk_tree_model_get_iter(model, &iter, path)) {
823 gtk_tree_model_get(model, &iter, COL_ITEM, &item, COL_DS, &ds, -1);
824 addrduplicates_delete_item_person(item,ds);
826 gtk_tree_path_free(path);
829 g_list_foreach(ref_list, (GFunc)gtk_tree_row_reference_free, NULL);
830 g_list_free(ref_list);
832 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(email_view));
833 if(gtk_tree_selection_get_selected(sel,NULL,&iter))
834 gtk_tree_model_get(GTK_TREE_MODEL(email_store), &iter, 0, &email, -1);
835 else
836 email = NULL;
837 refresh_stores(email,NULL);
838 if(email)
839 g_free(email);
842 gboolean addrduplicates_delete_item_person(ItemPerson *item, AddressDataSource *ds)
844 AddressBookFile *abf;
845 AddressInterface *iface;
846 if (!ds)
847 return FALSE;
848 /* Test for read only */
849 iface = ds->interface;
850 if( iface && iface->readOnly ) {
851 alertpanel( _("Delete address"),
852 _("This address data is readonly and cannot be deleted."),
853 GTK_STOCK_CLOSE, NULL, NULL );
854 return FALSE;
857 if(!(abf = ds->rawDataSource))
858 return FALSE;
860 item->status = DELETE_ENTRY;
861 item = addrbook_remove_person(abf, item);
863 #ifdef USE_LDAP
865 if (ds && ds->type == ADDR_IF_LDAP) {
866 LdapServer *server = ds->rawDataSource;
867 ldapsvr_set_modified(server, TRUE);
868 ldapsvr_update_book(server, item);
871 #endif
873 if(item) {
874 gchar *filename = addritem_person_get_picture(item);
875 if (filename && is_file_exist(filename))
876 claws_unlink(filename);
877 g_free(filename);
878 addritem_free_item_person(item);
880 return TRUE;
883 static gboolean cb_finder_results_dialog_key_pressed(GtkWidget *widget,
884 GdkEventKey *event,
885 gpointer data)
887 if(event) {
888 if(event->keyval == GDK_Delete && detail_view_has_focus)
889 cb_del_btn_clicked(NULL,NULL);
890 else if(event->keyval == GDK_Escape)
891 gtk_widget_destroy(dialog);
894 return FALSE;
897 static gboolean detail_focus_in(GtkWidget *widget,
898 GdkEventFocus *event,gpointer data)
900 detail_view_has_focus = TRUE;
901 return FALSE;
904 static gboolean detail_focus_out(GtkWidget *widget,
905 GdkEventFocus *event,gpointer data)
907 detail_view_has_focus = FALSE;
908 return FALSE;