2007-07-09 [colin] 2.10.0cvs12
[claws.git] / src / editaddress_other_attributes_ldap.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2007 Michael Rasmussen and the Claws Mail team
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2 of the License, or
8  * (at your option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18  */
19
20 #ifdef HAVE_CONFIG_H
21 #  include "config.h"
22 #endif
23
24 #ifdef USE_LDAP
25
26 #include "defs.h"
27
28 #include "mgutils.h"
29 #include "addressbook.h"
30 #include "addressitem.h"
31 #include "addritem.h"
32 #include "addrbook.h"
33 #include "manage_window.h"
34 #include "gtkutils.h"
35 #include "codeconv.h"
36 #include "editaddress.h"
37 #include "editaddress_other_attributes_ldap.h"
38 #include "prefs_common.h"
39
40 #define ATTRIB_COL_NAME                 0
41 #define ATTRIB_COL_VALUE                1
42 #define ATTRIB_N_COLS                   2
43 #define EMAIL_N_COLS                    3
44 #define ATTRIB_COL_WIDTH_NAME   120
45 #define ATTRIB_COL_WIDTH_VALUE  180
46
47 PersonEditDlg *personEditDlg;
48 gboolean attrib_adding = FALSE, attrib_saving = FALSE;
49
50 int get_attribute_index(const gchar *string_literal) {
51         int i = 0;
52         /*int count = sizeof(ATTRIBUTE) / sizeof(*ATTRIBUTE);*/
53         const gchar **attribute = ATTRIBUTE;
54
55         g_return_val_if_fail(string_literal != NULL, -1);
56         while (*attribute) {
57                 debug_print("Comparing %s to %s\n", *attribute, string_literal);
58                 if (strcmp(*attribute++, string_literal) == 0)
59                         return i;
60                 i++;
61         }
62         return -1;
63 }
64
65 static void edit_person_status_show(gchar *msg) {
66         if (personEditDlg->statusbar != NULL) {
67                 gtk_statusbar_pop(GTK_STATUSBAR(personEditDlg->statusbar), personEditDlg->status_cid);
68                 if(msg) {
69                         gtk_statusbar_push(GTK_STATUSBAR(personEditDlg->statusbar), personEditDlg->status_cid, msg);
70                 }
71         }
72 }
73
74 static void edit_person_attrib_clear(gpointer data) {
75         gtk_option_menu_set_history(GTK_OPTION_MENU(personEditDlg->entry_atname), 0);
76         gtk_entry_set_text(GTK_ENTRY(personEditDlg->entry_atvalue), "");
77 }
78
79 static gboolean list_find_attribute(const gchar *attr)
80 {
81         GtkCList *clist = GTK_CLIST(personEditDlg->clist_attrib);
82         UserAttribute *attrib;
83         gint row = 0;
84         while((attrib = gtk_clist_get_row_data(clist, row))) {
85                 if (!g_ascii_strcasecmp(attrib->name, attr)) {
86                         gtk_clist_select_row(clist, row, 0);
87                         return TRUE;
88                 }
89                 row++;
90         }
91         return FALSE;
92 }
93
94 /*
95 * Comparison using cell contents (text in first column). Used for sort
96 * address index widget.
97 */
98 static gint edit_person_attrib_compare_func(GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2) {
99         GtkCell *cell1 = ((GtkCListRow *)ptr1)->cell;
100         GtkCell *cell2 = ((GtkCListRow *)ptr2)->cell;
101         gchar *name1 = NULL, *name2 = NULL;
102
103         if (cell1) name1 = cell1->u.text;
104         if (cell2) name2 = cell2->u.text;
105         if (!name1) return (name2 != NULL);
106         if (!name2) return -1;
107         return g_utf8_collate(name1, name2);
108 }
109
110 static void edit_person_option_menu_changed(GtkOptionMenu *opt_menu, gpointer data) {
111         GtkCList *clist = GTK_CLIST(data);
112         gint row = personEditDlg->rowIndAttrib;
113         UserAttribute *attrib = gtk_clist_get_row_data(clist, row);
114         gint option = gtk_option_menu_get_history(opt_menu);
115
116         /* A corresponding attribute in contact does not match selected option */ 
117         if (strcmp(ATTRIBUTE[option], attrib->name) != 0) {
118                 gtk_widget_set_sensitive(personEditDlg->attrib_add, TRUE);
119                 gtk_widget_set_sensitive(personEditDlg->attrib_mod, TRUE);
120                 gtk_widget_set_sensitive(personEditDlg->attrib_del, FALSE);
121                 gtk_entry_set_text(GTK_ENTRY(personEditDlg->entry_atvalue), "");
122                 gtk_widget_grab_focus(personEditDlg->entry_atvalue);
123                 edit_person_status_show(NULL);
124         }
125 }
126
127 static void edit_person_attrib_list_selected(GtkCList *clist, gint row, gint column, GdkEvent *event, gpointer data) {
128         UserAttribute *attrib = gtk_clist_get_row_data(clist, row);
129         if (attrib && !personEditDlg->read_only) {
130                 int index = get_attribute_index(attrib->name);
131                 /*fprintf(stderr, "Row: %d ->  %s = %s\n", index, attrib->name, attrib->value);*/
132                 if (index == -1)
133                         index = 0;
134                 gtk_option_menu_set_history(GTK_OPTION_MENU(personEditDlg->entry_atname), index);
135                 gtk_entry_set_text( GTK_ENTRY(personEditDlg->entry_atvalue), attrib->value );
136                 gtk_widget_set_sensitive(personEditDlg->attrib_del, TRUE);
137         }
138         else {
139                 /*fprintf(stderr, "Row: %d -> empty attribute\n", row);*/
140                 gtk_entry_set_text( GTK_ENTRY(personEditDlg->entry_atvalue), "");       
141                 gtk_widget_set_sensitive(personEditDlg->attrib_del, FALSE);
142         }
143         personEditDlg->rowIndAttrib = row;
144         edit_person_status_show(NULL);
145 }
146
147 static void edit_person_attrib_delete(gpointer data) {
148         GtkCList *clist = GTK_CLIST(personEditDlg->clist_attrib);
149         gint row = personEditDlg->rowIndAttrib;
150         UserAttribute *attrib = gtk_clist_get_row_data(clist, row);
151         edit_person_attrib_clear(NULL);
152         if (attrib) {
153                 /* Remove list entry */
154                 gtk_clist_remove(clist, row);
155                 addritem_free_attribute(attrib);
156                 attrib = NULL;
157         }
158
159         /* Position hilite bar */
160         attrib = gtk_clist_get_row_data(clist, row);
161         if (!attrib) {
162                 personEditDlg->rowIndAttrib = -1 + row;
163         } 
164         
165         if (!personEditDlg->read_only)
166                 gtk_widget_set_sensitive(personEditDlg->attrib_del, gtk_clist_get_row_data(clist, 0) != NULL);
167         
168         edit_person_status_show(NULL);
169 }
170
171 static UserAttribute *edit_person_attrib_edit(gboolean *error, UserAttribute *attrib) {
172         UserAttribute *retVal = NULL;
173         gchar *sName, *sValue, *sName_, *sValue_;
174         gint index;
175
176         *error = TRUE;
177         index = gtk_option_menu_get_history(GTK_OPTION_MENU(personEditDlg->entry_atname));
178         sName_ = (gchar *) ATTRIBUTE[index];
179         sValue_ = gtk_editable_get_chars(GTK_EDITABLE(personEditDlg->entry_atvalue), 0, -1);
180         sName = mgu_email_check_empty(sName_);
181         sValue = mgu_email_check_empty(sValue_);
182         g_free(sValue_);
183
184         if (sName && sValue) {
185                 if (attrib == NULL) {
186                         attrib = addritem_create_attribute();
187                 }
188                 addritem_attrib_set_name(attrib, sName);
189                 addritem_attrib_set_value(attrib, sValue);
190                 retVal = attrib;
191                 *error = FALSE;
192         }
193         else {
194                 edit_person_status_show(N_( "A Name and Value must be supplied." ));
195                 gtk_widget_grab_focus(personEditDlg->entry_atvalue);            
196         }
197
198         g_free(sName);
199         g_free(sValue);
200
201         return retVal;
202 }
203
204 static void edit_person_attrib_modify(gpointer data) {
205         gboolean errFlg = FALSE;
206         GtkCList *clist = GTK_CLIST(personEditDlg->clist_attrib);
207         gint row = personEditDlg->rowIndAttrib;
208         UserAttribute *attrib = gtk_clist_get_row_data(clist, row);
209         if (attrib) {
210                 edit_person_attrib_edit(&errFlg, attrib);
211                 if (!errFlg) {
212                         gtk_clist_set_text(clist, row, ATTRIB_COL_NAME, attrib->name);
213                         gtk_clist_set_text(clist, row, ATTRIB_COL_VALUE, attrib->value);
214                         edit_person_attrib_clear(NULL);
215                 }
216         }
217 }
218
219 static void edit_person_attrib_add(gpointer data) {
220         GtkCList *clist = GTK_CLIST(personEditDlg->clist_attrib);
221         gboolean errFlg = FALSE;
222         UserAttribute *attrib = NULL;
223         gint row = personEditDlg->rowIndAttrib;
224         if (gtk_clist_get_row_data(clist, row) == NULL) row = 0;
225
226         attrib = edit_person_attrib_edit(&errFlg, NULL);
227         if (!errFlg) {
228                 gchar *text[EMAIL_N_COLS];
229                 text[ATTRIB_COL_NAME] = attrib->name;
230                 text[ATTRIB_COL_VALUE] = attrib->value;
231
232                 row = gtk_clist_insert(clist, 1 + row, text);
233                 gtk_clist_set_row_data(clist, row, attrib);
234                 gtk_clist_select_row(clist, row, 0);
235                 edit_person_attrib_clear(NULL);
236         }
237 }
238
239 static void edit_person_entry_att_changed (GtkWidget *entry, gpointer data)
240 {
241         gboolean non_empty = gtk_clist_get_row_data(GTK_CLIST(personEditDlg->clist_attrib), 0) != NULL;
242         const gchar *sName;
243         int index;
244
245         if (personEditDlg->read_only)
246                 return;
247
248         index = gtk_option_menu_get_history(GTK_OPTION_MENU(personEditDlg->entry_atname));
249         sName = ATTRIBUTE[index];
250         if (list_find_attribute(sName)) {
251                 gtk_widget_set_sensitive(personEditDlg->attrib_add,FALSE);
252                 gtk_widget_set_sensitive(personEditDlg->attrib_mod,non_empty);
253                 attrib_adding = FALSE;
254                 attrib_saving = non_empty;
255         } 
256         else {
257                 gtk_widget_set_sensitive(personEditDlg->attrib_add,TRUE);
258                 gtk_widget_set_sensitive(personEditDlg->attrib_mod,non_empty);
259                 attrib_adding = TRUE;
260                 attrib_saving = non_empty;
261         }
262 }
263
264 static gboolean edit_person_entry_att_pressed(GtkWidget *widget, GdkEventKey *event, gpointer data)
265 {
266         if (event && (event->keyval == GDK_Return || event->keyval == GDK_KP_Enter)) {
267                 if (attrib_saving)
268                         edit_person_attrib_modify(NULL);
269                 else if (attrib_adding)
270                         edit_person_attrib_add(NULL);
271         }
272         return FALSE;
273 }
274
275 void addressbook_edit_person_page_attrib_ldap(PersonEditDlg *dialog, gint pageNum, gchar *pageLbl) {
276         GtkWidget *option_menu;
277         GtkMenu *names_menu;
278
279         GtkWidget *vbox;
280         GtkWidget *hbox;
281         GtkWidget *vboxl;
282         GtkWidget *vboxb;
283         GtkWidget *vbuttonbox;
284         GtkWidget *buttonDel;
285         GtkWidget *buttonMod;
286         GtkWidget *buttonAdd;
287
288         GtkWidget *table;
289         GtkWidget *label;
290         GtkWidget *clist_swin;
291         GtkWidget *clist;
292         GtkWidget *entry_value;
293         gint top;
294
295         personEditDlg = dialog;
296
297         gchar *titles[ATTRIB_N_COLS];
298         gint i;
299
300         titles[ATTRIB_COL_NAME] = N_("Name");
301         titles[ATTRIB_COL_VALUE] = N_("Value");
302
303         vbox = gtk_vbox_new(FALSE, 8);
304         gtk_widget_show(vbox);
305         gtk_container_add(GTK_CONTAINER(personEditDlg->notebook), vbox);
306         gtk_container_set_border_width(GTK_CONTAINER(vbox), BORDER_WIDTH);
307
308         label = gtk_label_new_with_mnemonic(pageLbl);
309         gtk_widget_show(label);
310         gtk_notebook_set_tab_label(
311                 GTK_NOTEBOOK(personEditDlg->notebook),
312                 gtk_notebook_get_nth_page(GTK_NOTEBOOK(personEditDlg->notebook), pageNum), label);
313
314         /* Split into two areas */
315         hbox = gtk_hbox_new(FALSE, 0);
316         gtk_container_add(GTK_CONTAINER(vbox), hbox);
317
318         /* Attribute list */
319         vboxl = gtk_vbox_new(FALSE, 4);
320         gtk_container_add(GTK_CONTAINER(hbox), vboxl);
321         gtk_container_set_border_width(GTK_CONTAINER(vboxl), 4);
322
323         clist_swin = gtk_scrolled_window_new(NULL, NULL);
324         gtk_container_add(GTK_CONTAINER(vboxl), clist_swin);
325         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(clist_swin),
326                                        GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
327
328         clist = gtk_clist_new_with_titles(ATTRIB_N_COLS, titles);
329         gtk_container_add(GTK_CONTAINER(clist_swin), clist);
330         gtk_clist_set_selection_mode(GTK_CLIST(clist), GTK_SELECTION_BROWSE);
331         gtk_clist_set_column_width(GTK_CLIST(clist), ATTRIB_COL_NAME, ATTRIB_COL_WIDTH_NAME);
332         gtk_clist_set_column_width(GTK_CLIST(clist), ATTRIB_COL_VALUE, ATTRIB_COL_WIDTH_VALUE);
333         gtk_clist_set_compare_func(GTK_CLIST(clist), edit_person_attrib_compare_func);
334         gtk_clist_set_auto_sort(GTK_CLIST(clist), TRUE);
335
336         for (i = 0; i < ATTRIB_N_COLS; i++)
337                 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist)->column[i].button, GTK_CAN_FOCUS);
338
339         /* Data entry area */
340         table = gtk_table_new(4, 2, FALSE);
341         gtk_box_pack_start(GTK_BOX(vboxl), table, FALSE, FALSE, 0);
342         gtk_container_set_border_width(GTK_CONTAINER(table), 4);
343         gtk_table_set_row_spacings(GTK_TABLE(table), 4);
344         gtk_table_set_col_spacings(GTK_TABLE(table), 4);
345
346         /* First row */
347         top = 0;
348         label = gtk_label_new(N_("Name"));
349         gtk_table_attach(GTK_TABLE(table), label, 0, 1, top, (top + 1), GTK_FILL, 0, 0, 0);
350         gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
351
352         names_menu = g_object_new(GTK_TYPE_MENU, NULL);
353         gchar **attribute = (gchar **) ATTRIBUTE;
354
355         while (*attribute) {
356                 gtk_menu_shell_append(GTK_MENU_SHELL(names_menu),
357                         gtk_menu_item_new_with_label(*attribute++));
358         }
359         option_menu = GTK_WIDGET(g_object_new(GTK_TYPE_OPTION_MENU, "menu", names_menu, NULL));
360         gtk_option_menu_set_history(GTK_OPTION_MENU(option_menu), 0);
361
362         gtk_table_attach(GTK_TABLE(table), option_menu, 1, 2, top, (top + 1), GTK_EXPAND|GTK_SHRINK|GTK_FILL, 0, 0, 0);
363
364         /* Next row */
365         ++top;
366         label = gtk_label_new(N_("Value"));
367         gtk_table_attach(GTK_TABLE(table), label, 0, 1, top, (top + 1), GTK_FILL, 0, 0, 0);
368         gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
369
370         entry_value = gtk_entry_new();
371         gtk_table_attach(GTK_TABLE(table), entry_value, 1, 2, top, (top + 1), GTK_EXPAND|GTK_SHRINK|GTK_FILL, 0, 0, 0);
372
373         /* Button box */
374         vboxb = gtk_vbox_new(FALSE, 4);
375         gtk_box_pack_start(GTK_BOX(hbox), vboxb, FALSE, FALSE, 2);
376
377         vbuttonbox = gtk_vbutton_box_new();
378         gtk_button_box_set_layout(GTK_BUTTON_BOX(vbuttonbox), GTK_BUTTONBOX_START);
379         gtk_box_set_spacing(GTK_BOX(vbuttonbox), 8);
380         gtk_container_set_border_width(GTK_CONTAINER(vbuttonbox), 4);
381         gtk_container_add(GTK_CONTAINER(vboxb), vbuttonbox);
382
383         /* Buttons */
384         buttonDel = gtk_button_new_from_stock(GTK_STOCK_DELETE);
385         gtk_container_add(GTK_CONTAINER(vbuttonbox), buttonDel);
386
387         buttonMod = gtk_button_new_from_stock(GTK_STOCK_SAVE);
388         gtk_container_add(GTK_CONTAINER(vbuttonbox), buttonMod);
389
390         buttonAdd = gtk_button_new_from_stock(GTK_STOCK_ADD);
391         gtk_container_add(GTK_CONTAINER(vbuttonbox), buttonAdd);
392         
393         gtk_widget_set_sensitive(buttonDel,FALSE);
394         gtk_widget_set_sensitive(buttonMod,FALSE);
395         gtk_widget_set_sensitive(buttonAdd,FALSE);
396
397         gtk_widget_show_all(vbox);
398         
399         /* Event handlers */
400         g_signal_connect(G_OBJECT(clist), "select_row",
401                           G_CALLBACK( edit_person_attrib_list_selected), NULL);
402         g_signal_connect(G_OBJECT(buttonDel), "clicked",
403                           G_CALLBACK(edit_person_attrib_delete), NULL);
404         g_signal_connect(G_OBJECT(buttonMod), "clicked",
405                           G_CALLBACK(edit_person_attrib_modify), NULL);
406         g_signal_connect(G_OBJECT(buttonAdd), "clicked",
407                           G_CALLBACK(edit_person_attrib_add), NULL);
408         g_signal_connect(G_OBJECT(option_menu), "changed",
409                          G_CALLBACK(edit_person_entry_att_changed), NULL);
410         g_signal_connect(G_OBJECT(entry_value), "key_press_event",
411                          G_CALLBACK(edit_person_entry_att_pressed), NULL);
412         g_signal_connect(G_OBJECT(option_menu), "changed",
413                          G_CALLBACK(edit_person_option_menu_changed), clist);
414
415         personEditDlg->clist_attrib  = clist;
416         personEditDlg->entry_atname  = option_menu;
417         personEditDlg->entry_atvalue = entry_value;
418         personEditDlg->attrib_add = buttonAdd;
419         personEditDlg->attrib_del = buttonDel;
420         personEditDlg->attrib_mod = buttonMod;
421 }
422
423 #endif  /* USE_LDAP */
424
425