2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2007 Michael Rasmussen 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/>.
29 #include "addressbook.h"
30 #include "addressitem.h"
33 #include "manage_window.h"
36 #include "editaddress.h"
37 #include "editaddress_other_attributes_ldap.h"
38 #include "prefs_common.h"
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
47 PersonEditDlg
*personEditDlg
;
48 gboolean attrib_adding
= FALSE
, attrib_saving
= FALSE
;
50 int get_attribute_index(const gchar
*string_literal
) {
52 /*int count = sizeof(ATTRIBUTE) / sizeof(*ATTRIBUTE);*/
53 const gchar
**attribute
= ATTRIBUTE
;
55 g_return_val_if_fail(string_literal
!= NULL
, -1);
57 debug_print("Comparing %s to %s\n", *attribute
, string_literal
);
58 if (strcmp(*attribute
++, string_literal
) == 0)
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
);
69 gtk_statusbar_push(GTK_STATUSBAR(personEditDlg
->statusbar
), personEditDlg
->status_cid
, msg
);
74 static void edit_person_attrib_clear(gpointer data
) {
75 gtk_combo_box_set_active(GTK_COMBO_BOX(personEditDlg
->entry_atname
), 0);
76 gtk_entry_set_text(GTK_ENTRY(personEditDlg
->entry_atvalue
), "");
79 static gboolean
list_find_attribute(const gchar
*attr
)
81 GtkCList
*clist
= GTK_CLIST(personEditDlg
->clist_attrib
);
82 UserAttribute
*attrib
;
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);
95 * Comparison using cell contents (text in first column). Used for sort
96 * address index widget.
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
;
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
);
110 static void edit_person_combo_box_changed(GtkComboBox
*opt_menu
, gpointer data
)
112 GtkCList
*clist
= GTK_CLIST(data
);
113 gint row
= personEditDlg
->rowIndAttrib
;
114 UserAttribute
*attrib
= gtk_clist_get_row_data(clist
, row
);
115 gint option
= gtk_combo_box_get_active(opt_menu
);
116 const gchar
*str
= attrib
? attrib
->name
:"";
118 g_return_if_fail (option
< ATTRIBUTE_SIZE
);
119 /* A corresponding attribute in contact does not match selected option */
120 if (strcmp(ATTRIBUTE
[option
], str
) != 0) {
121 gtk_widget_set_sensitive(personEditDlg
->attrib_add
, TRUE
);
122 gtk_widget_set_sensitive(personEditDlg
->attrib_mod
, TRUE
);
123 gtk_widget_set_sensitive(personEditDlg
->attrib_del
, FALSE
);
124 gtk_entry_set_text(GTK_ENTRY(personEditDlg
->entry_atvalue
), "");
125 gtk_widget_grab_focus(personEditDlg
->entry_atvalue
);
126 edit_person_status_show(NULL
);
130 static void edit_person_attrib_list_selected(GtkCList
*clist
, gint row
, gint column
, GdkEvent
*event
, gpointer data
) {
131 UserAttribute
*attrib
= gtk_clist_get_row_data(clist
, row
);
132 if (attrib
&& !personEditDlg
->read_only
) {
133 int index
= get_attribute_index(attrib
->name
);
137 gtk_combo_box_set_active(GTK_COMBO_BOX(personEditDlg
->entry_atname
), index
);
138 gtk_entry_set_text( GTK_ENTRY(personEditDlg
->entry_atvalue
), attrib
->value
);
139 gtk_widget_set_sensitive(personEditDlg
->attrib_del
, TRUE
);
142 /*g_printerr("Row: %d -> empty attribute\n", row);*/
143 gtk_entry_set_text( GTK_ENTRY(personEditDlg
->entry_atvalue
), "");
144 gtk_widget_set_sensitive(personEditDlg
->attrib_del
, FALSE
);
146 personEditDlg
->rowIndAttrib
= row
;
147 edit_person_status_show(NULL
);
150 static void edit_person_attrib_delete(gpointer data
) {
151 GtkCList
*clist
= GTK_CLIST(personEditDlg
->clist_attrib
);
152 gint row
= personEditDlg
->rowIndAttrib
;
153 UserAttribute
*attrib
= gtk_clist_get_row_data(clist
, row
);
154 edit_person_attrib_clear(NULL
);
156 /* Remove list entry */
157 gtk_clist_remove(clist
, row
);
158 addritem_free_attribute(attrib
);
162 /* Position hilite bar */
163 attrib
= gtk_clist_get_row_data(clist
, row
);
165 personEditDlg
->rowIndAttrib
= -1 + row
;
168 if (!personEditDlg
->read_only
)
169 gtk_widget_set_sensitive(personEditDlg
->attrib_del
, gtk_clist_get_row_data(clist
, 0) != NULL
);
171 edit_person_status_show(NULL
);
174 static UserAttribute
*edit_person_attrib_edit(gboolean
*error
, UserAttribute
*attrib
) {
175 UserAttribute
*retVal
= NULL
;
176 gchar
*sName
, *sValue
, *sName_
, *sValue_
;
180 index
= gtk_combo_box_get_active(GTK_COMBO_BOX(personEditDlg
->entry_atname
));
181 sName_
= (gchar
*) ATTRIBUTE
[index
];
182 sValue_
= gtk_editable_get_chars(GTK_EDITABLE(personEditDlg
->entry_atvalue
), 0, -1);
183 sName
= mgu_email_check_empty(sName_
);
184 sValue
= mgu_email_check_empty(sValue_
);
187 if (sName
&& sValue
) {
188 if (attrib
== NULL
) {
189 attrib
= addritem_create_attribute();
191 addritem_attrib_set_name(attrib
, sName
);
192 addritem_attrib_set_value(attrib
, sValue
);
197 edit_person_status_show(N_( "A Name and Value must be supplied." ));
198 gtk_widget_grab_focus(personEditDlg
->entry_atvalue
);
207 static void edit_person_attrib_modify(gpointer data
) {
208 gboolean errFlg
= FALSE
;
209 GtkCList
*clist
= GTK_CLIST(personEditDlg
->clist_attrib
);
210 gint row
= personEditDlg
->rowIndAttrib
;
211 UserAttribute
*attrib
= gtk_clist_get_row_data(clist
, row
);
213 edit_person_attrib_edit(&errFlg
, attrib
);
215 gtk_clist_set_text(clist
, row
, ATTRIB_COL_NAME
, attrib
->name
);
216 gtk_clist_set_text(clist
, row
, ATTRIB_COL_VALUE
, attrib
->value
);
217 edit_person_attrib_clear(NULL
);
222 static void edit_person_attrib_add(gpointer data
) {
223 GtkCList
*clist
= GTK_CLIST(personEditDlg
->clist_attrib
);
224 gboolean errFlg
= FALSE
;
225 UserAttribute
*attrib
= NULL
;
226 gint row
= personEditDlg
->rowIndAttrib
;
227 if (gtk_clist_get_row_data(clist
, row
) == NULL
) row
= 0;
229 attrib
= edit_person_attrib_edit(&errFlg
, NULL
);
231 gchar
*text
[EMAIL_N_COLS
];
232 text
[ATTRIB_COL_NAME
] = attrib
->name
;
233 text
[ATTRIB_COL_VALUE
] = attrib
->value
;
235 row
= gtk_clist_insert(clist
, 1 + row
, text
);
236 gtk_clist_set_row_data(clist
, row
, attrib
);
237 gtk_clist_select_row(clist
, row
, 0);
238 edit_person_attrib_clear(NULL
);
242 static void edit_person_entry_att_changed (GtkWidget
*entry
, gpointer data
)
244 gboolean non_empty
= gtk_clist_get_row_data(GTK_CLIST(personEditDlg
->clist_attrib
), 0) != NULL
;
248 if (personEditDlg
->read_only
)
251 index
= gtk_combo_box_get_active(GTK_COMBO_BOX(personEditDlg
->entry_atname
));
252 sName
= ATTRIBUTE
[index
];
253 if (list_find_attribute(sName
)) {
254 gtk_widget_set_sensitive(personEditDlg
->attrib_add
,FALSE
);
255 gtk_widget_set_sensitive(personEditDlg
->attrib_mod
,non_empty
);
256 attrib_adding
= FALSE
;
257 attrib_saving
= non_empty
;
260 gtk_widget_set_sensitive(personEditDlg
->attrib_add
,TRUE
);
261 gtk_widget_set_sensitive(personEditDlg
->attrib_mod
,non_empty
);
262 attrib_adding
= TRUE
;
263 attrib_saving
= non_empty
;
267 static gboolean
edit_person_entry_att_pressed(GtkWidget
*widget
, GdkEventKey
*event
, gpointer data
)
269 if (event
&& (event
->keyval
== GDK_Return
|| event
->keyval
== GDK_KP_Enter
)) {
271 edit_person_attrib_modify(NULL
);
272 else if (attrib_adding
)
273 edit_person_attrib_add(NULL
);
278 void addressbook_edit_person_page_attrib_ldap(PersonEditDlg
*dialog
, gint pageNum
, gchar
*pageLbl
) {
279 GtkWidget
*combo_box
;
284 GtkWidget
*vbuttonbox
;
285 GtkWidget
*buttonDel
;
286 GtkWidget
*buttonMod
;
287 GtkWidget
*buttonAdd
;
291 GtkWidget
*clist_swin
;
293 GtkWidget
*entry_value
;
296 personEditDlg
= dialog
;
298 gchar
*titles
[ATTRIB_N_COLS
];
301 titles
[ATTRIB_COL_NAME
] = N_("Name");
302 titles
[ATTRIB_COL_VALUE
] = N_("Value");
304 vbox
= gtk_vbox_new(FALSE
, 8);
305 gtk_widget_show(vbox
);
306 gtk_container_add(GTK_CONTAINER(personEditDlg
->notebook
), vbox
);
307 gtk_container_set_border_width(GTK_CONTAINER(vbox
), BORDER_WIDTH
);
309 label
= gtk_label_new_with_mnemonic(pageLbl
);
310 gtk_widget_show(label
);
311 gtk_notebook_set_tab_label(
312 GTK_NOTEBOOK(personEditDlg
->notebook
),
313 gtk_notebook_get_nth_page(GTK_NOTEBOOK(personEditDlg
->notebook
), pageNum
), label
);
315 /* Split into two areas */
316 hbox
= gtk_hbox_new(FALSE
, 0);
317 gtk_container_add(GTK_CONTAINER(vbox
), hbox
);
320 vboxl
= gtk_vbox_new(FALSE
, 4);
321 gtk_container_add(GTK_CONTAINER(hbox
), vboxl
);
322 gtk_container_set_border_width(GTK_CONTAINER(vboxl
), 4);
324 clist_swin
= gtk_scrolled_window_new(NULL
, NULL
);
325 gtk_container_add(GTK_CONTAINER(vboxl
), clist_swin
);
326 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(clist_swin
),
327 GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
329 clist
= gtk_clist_new_with_titles(ATTRIB_N_COLS
, titles
);
330 gtk_container_add(GTK_CONTAINER(clist_swin
), clist
);
331 gtk_clist_set_selection_mode(GTK_CLIST(clist
), GTK_SELECTION_BROWSE
);
332 gtk_clist_set_column_width(GTK_CLIST(clist
), ATTRIB_COL_NAME
, ATTRIB_COL_WIDTH_NAME
);
333 gtk_clist_set_column_width(GTK_CLIST(clist
), ATTRIB_COL_VALUE
, ATTRIB_COL_WIDTH_VALUE
);
334 gtk_clist_set_compare_func(GTK_CLIST(clist
), edit_person_attrib_compare_func
);
335 gtk_clist_set_auto_sort(GTK_CLIST(clist
), TRUE
);
337 for (i
= 0; i
< ATTRIB_N_COLS
; i
++)
338 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist
)->column
[i
].button
, GTK_CAN_FOCUS
);
340 /* Data entry area */
341 table
= gtk_table_new(4, 2, FALSE
);
342 gtk_box_pack_start(GTK_BOX(vboxl
), table
, FALSE
, FALSE
, 0);
343 gtk_container_set_border_width(GTK_CONTAINER(table
), 4);
344 gtk_table_set_row_spacings(GTK_TABLE(table
), 4);
345 gtk_table_set_col_spacings(GTK_TABLE(table
), 4);
349 label
= gtk_label_new(N_("Name"));
350 gtk_table_attach(GTK_TABLE(table
), label
, 0, 1, top
, (top
+ 1), GTK_FILL
, 0, 0, 0);
351 gtk_misc_set_alignment(GTK_MISC(label
), 0, 0.5);
353 gchar
**attribute
= (gchar
**) ATTRIBUTE
;
355 combo_box
= gtk_combo_box_new_text();
358 gtk_combo_box_append_text(GTK_COMBO_BOX(combo_box
), *attribute
++);
360 gtk_combo_box_set_active(GTK_COMBO_BOX(combo_box
), 0);
362 gtk_table_attach(GTK_TABLE(table
), combo_box
, 1, 2, top
, (top
+ 1), GTK_EXPAND
|GTK_SHRINK
|GTK_FILL
, 0, 0, 0);
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);
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);
374 vboxb
= gtk_vbox_new(FALSE
, 4);
375 gtk_box_pack_start(GTK_BOX(hbox
), vboxb
, FALSE
, FALSE
, 2);
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
);
384 buttonDel
= gtk_button_new_from_stock(GTK_STOCK_DELETE
);
385 gtk_container_add(GTK_CONTAINER(vbuttonbox
), buttonDel
);
387 buttonMod
= gtk_button_new_from_stock(GTK_STOCK_SAVE
);
388 gtk_container_add(GTK_CONTAINER(vbuttonbox
), buttonMod
);
390 buttonAdd
= gtk_button_new_from_stock(GTK_STOCK_ADD
);
391 gtk_container_add(GTK_CONTAINER(vbuttonbox
), buttonAdd
);
393 gtk_widget_set_sensitive(buttonDel
,FALSE
);
394 gtk_widget_set_sensitive(buttonMod
,FALSE
);
395 gtk_widget_set_sensitive(buttonAdd
,FALSE
);
397 gtk_widget_show_all(vbox
);
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(combo_box
), "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(combo_box
), "changed",
413 G_CALLBACK(edit_person_combo_box_changed
), clist
);
415 personEditDlg
->clist_attrib
= clist
;
416 personEditDlg
->entry_atname
= combo_box
;
417 personEditDlg
->entry_atvalue
= entry_value
;
418 personEditDlg
->attrib_add
= buttonAdd
;
419 personEditDlg
->attrib_del
= buttonDel
;
420 personEditDlg
->attrib_mod
= buttonMod
;
423 #endif /* USE_LDAP */