fixed cr/lf and ">" in html pages
[claws.git] / src / editgroup.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2001 Hiroyuki Yamamoto
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2 of the License, or
8  * (at your option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software
17  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18  */
19
20 #ifdef HAVE_CONFIG_H
21 #  include "config.h"
22 #endif
23
24 #include "defs.h"
25
26 #include <glib.h>
27 #include <gdk/gdkkeysyms.h>
28
29 #include "intl.h"
30 #include "addressbook.h"
31 #include "addressitem.h"
32 #include "addrbook.h"
33 #include "addritem.h"
34
35 #include "mgutils.h"
36
37 #include "prefs_common.h"
38
39 #include "alertpanel.h"
40 #include "inputdialog.h"
41 #include "manage_window.h"
42 #include "gtkutils.h"
43
44 #define ADDRESSBOOK_GUESS_FOLDER_NAME   "NewFolder"
45 #define ADDRESSBOOK_GUESS_GROUP_NAME    "NewGroup"
46
47 #define EDITGROUP_WIDTH      580
48 #define EDITGROUP_HEIGHT     340
49
50 typedef enum {
51         GROUP_COL_NAME    = 0,
52         GROUP_COL_EMAIL   = 1,
53         GROUP_COL_REMARKS = 2
54 } GroupEditEMailColumnPos;
55
56 #define GROUP_N_COLS          3
57 #define GROUP_COL_WIDTH_NAME  140
58 #define GROUP_COL_WIDTH_EMAIL 120
59
60 static struct _GroupEdit_dlg {
61         GtkWidget *window;
62         GtkWidget *ok_btn;
63         GtkWidget *cancel_btn;
64         GtkWidget *statusbar;
65         gint status_cid;
66
67         /* Basic data tab */
68         GtkWidget *entry_name;
69         GtkCList *clist_group;
70         GtkCList *clist_avail;
71
72         GHashTable *hashEMail;
73         gint rowIndGroup;
74         gint rowIndAvail;
75
76 } groupeditdlg;
77
78
79 static gchar *_edit_group_dfl_message_ = NULL;
80
81 static void edit_group_status_show( gchar *msg ) {
82         if( groupeditdlg.statusbar != NULL ) {
83                 gtk_statusbar_pop( GTK_STATUSBAR(groupeditdlg.statusbar), groupeditdlg.status_cid );
84                 if( msg ) {
85                         gtk_statusbar_push( GTK_STATUSBAR(groupeditdlg.statusbar), groupeditdlg.status_cid, msg );
86                 }
87         }
88 }
89
90 static void edit_group_ok(GtkWidget *widget, gboolean *cancelled) {
91         gchar *sName = g_strdup( gtk_editable_get_chars( GTK_EDITABLE(groupeditdlg.entry_name), 0, -1 ) );
92         gboolean errFlag = TRUE;
93         if( sName ) {
94                 g_strstrip( sName );
95                 if( *sName != '\0' ) {
96                         gtk_entry_set_text(GTK_ENTRY(groupeditdlg.entry_name), sName );
97                         *cancelled = FALSE;
98                         gtk_main_quit();
99                         errFlag = FALSE;
100                 }
101         }
102         if( errFlag ) {
103                 edit_group_status_show( _( "A Group Name must be supplied." ) );
104         }
105         g_free( sName );
106 }
107         
108 static void edit_group_cancel(GtkWidget *widget, gboolean *cancelled) {
109         *cancelled = TRUE;
110         gtk_main_quit();
111 }
112
113 static gint edit_group_delete_event(GtkWidget *widget, GdkEventAny *event, gboolean *cancelled) {
114         *cancelled = TRUE;
115         gtk_main_quit();
116         return TRUE;
117 }
118
119 static void edit_group_key_pressed(GtkWidget *widget, GdkEventKey *event, gboolean *cancelled) {
120         if (event && event->keyval == GDK_Escape) {
121                 *cancelled = TRUE;
122                 gtk_main_quit();
123         }
124 }
125
126 static gchar *edit_group_format_item_clist( ItemPerson *person, ItemEMail *email ) {
127         gchar *str = NULL;
128         gchar *aName = ADDRITEM_NAME(email);
129         if( aName == NULL || *aName == '\0' ) return str;
130         if( person ) {
131                 str = g_strdup_printf( "%s - %s", ADDRITEM_NAME(person), aName );
132         }
133         else {
134                 str = g_strdup( aName );
135         }
136         return str;
137 }
138
139 static gint edit_group_clist_add_email( GtkCList *clist, ItemEMail *email ) {
140         ItemPerson *person = ( ItemPerson * ) ADDRITEM_PARENT(email);
141         gchar *str = edit_group_format_item_clist( person, email );
142         gchar *text[ GROUP_N_COLS ];
143         gint row;
144         if( str ) {
145                 text[ GROUP_COL_NAME ] = str;
146         }
147         else {
148                 text[ GROUP_COL_NAME ] = ADDRITEM_NAME(person);
149         }
150         text[ GROUP_COL_EMAIL   ] = email->address;
151         text[ GROUP_COL_REMARKS ] = email->remarks;
152         row = gtk_clist_append( clist, text );
153         gtk_clist_set_row_data( clist, row, email );
154         return row;
155 }
156
157 static void edit_group_load_clist( GtkCList *clist, GList *listEMail ) {
158         GList *node = listEMail;
159         while( node ) {
160                 ItemEMail *email = node->data;
161                 edit_group_clist_add_email( clist, email );
162                 node = g_list_next( node );
163         }
164 }
165
166 static void edit_group_group_selected( GtkCList *clist, gint row, gint column, GdkEvent *event, gpointer data ) {
167         groupeditdlg.rowIndGroup = row;
168 }
169
170 static void edit_group_avail_selected( GtkCList *clist, gint row, gint column, GdkEvent *event, gpointer data ) {
171         groupeditdlg.rowIndAvail = row;
172 }
173
174 static gint edit_group_move_email( GtkCList *clist_from, GtkCList *clist_to, gint row ) {
175         ItemEMail *email = gtk_clist_get_row_data( clist_from, row );
176         gint rrow = -1;
177         if( email ) {
178                 gtk_clist_remove( clist_from, row );
179                 rrow = edit_group_clist_add_email( clist_to, email );
180                 gtk_clist_select_row( clist_to, rrow, 0 );
181         }
182         return rrow;
183 }
184
185 static void edit_group_to_group( GtkWidget *widget, gpointer data ) {
186         groupeditdlg.rowIndGroup = edit_group_move_email( groupeditdlg.clist_avail,
187                                         groupeditdlg.clist_group, groupeditdlg.rowIndAvail );
188 }
189
190 static void edit_group_to_avail( GtkWidget *widget, gpointer data ) {
191         groupeditdlg.rowIndAvail = edit_group_move_email( groupeditdlg.clist_group,
192                                         groupeditdlg.clist_avail, groupeditdlg.rowIndGroup );
193 }
194
195 static void edit_group_list_group_button( GtkCList *clist, GdkEventButton *event, gpointer data ) {
196         if( ! event ) return;
197         if( event->button == 1 ) {
198                 if( event->type == GDK_2BUTTON_PRESS ) {
199                         edit_group_to_avail( NULL, NULL );
200                 }
201         }
202 }
203
204 static void edit_group_list_avail_button( GtkCList *clist, GdkEventButton *event, gpointer data ) {
205         if( ! event ) return;
206         if( event->button == 1 ) {
207                 if( event->type == GDK_2BUTTON_PRESS ) {
208                         edit_group_to_group( NULL, NULL );
209                 }
210         }
211 }
212
213 static gint edit_group_list_compare_func( GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2 ) {
214         GtkCell *cell1 = ((GtkCListRow *)ptr1)->cell;
215         GtkCell *cell2 = ((GtkCListRow *)ptr2)->cell;
216         gchar *name1 = NULL, *name2 = NULL;
217         if( cell1 ) name1 = cell1->u.text;
218         if( cell2 ) name2 = cell2->u.text;
219         if( ! name1 ) return ( name2 != NULL );
220         if( ! name2 ) return -1;
221         return strcasecmp( name1, name2 );
222 }
223
224 static void addressbook_edit_group_create( gboolean *cancelled ) {
225         GtkWidget *window;
226         GtkWidget *vbox;
227         GtkWidget *hbbox;
228         GtkWidget *ok_btn;
229         GtkWidget *cancel_btn;
230         GtkWidget *hsbox;
231         GtkWidget *statusbar;
232
233         GtkWidget *hboxg;
234         GtkWidget *table;
235         GtkWidget *label;
236         GtkWidget *entry_name;
237         GtkWidget *hboxl;
238         GtkWidget *vboxl;
239         GtkWidget *hboxh;
240
241         GtkWidget *clist_swin;
242         GtkWidget *clist_group;
243         GtkWidget *clist_avail;
244
245         GtkWidget *buttonGroup;
246         GtkWidget *buttonAvail;
247         gint top;
248
249         gchar *titles[ GROUP_N_COLS ];
250         gint i;
251
252         titles[ GROUP_COL_NAME    ] = _( "Name" );
253         titles[ GROUP_COL_EMAIL   ] = _("E-Mail Address");
254         titles[ GROUP_COL_REMARKS ] = _("Remarks");
255
256         window = gtk_window_new(GTK_WINDOW_DIALOG);
257         gtk_widget_set_usize(window, EDITGROUP_WIDTH, EDITGROUP_HEIGHT );
258         gtk_container_set_border_width(GTK_CONTAINER(window), 0);
259         gtk_window_set_title(GTK_WINDOW(window), _("Edit Group Data"));
260         gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
261         gtk_window_set_modal(GTK_WINDOW(window), TRUE); 
262         gtk_signal_connect(GTK_OBJECT(window), "delete_event",
263                            GTK_SIGNAL_FUNC(edit_group_delete_event),
264                            cancelled);
265         gtk_signal_connect(GTK_OBJECT(window), "key_press_event",
266                            GTK_SIGNAL_FUNC(edit_group_key_pressed),
267                            cancelled);
268
269         vbox = gtk_vbox_new( FALSE, 6 );
270         gtk_container_set_border_width(GTK_CONTAINER(vbox), BORDER_WIDTH);
271         gtk_widget_show( vbox );
272         gtk_container_add( GTK_CONTAINER( window ), vbox );
273
274         /* Group area */
275         hboxg = gtk_hbox_new( FALSE, 0 );
276         gtk_box_pack_start(GTK_BOX(vbox), hboxg, FALSE, FALSE, 0);
277
278         /* Data entry area */
279         table = gtk_table_new( 1, 3, FALSE);
280         gtk_box_pack_start(GTK_BOX(hboxg), table, TRUE, TRUE, 0);
281         gtk_container_set_border_width( GTK_CONTAINER(table), 4 );
282         gtk_table_set_row_spacings(GTK_TABLE(table), 0);
283         gtk_table_set_col_spacings(GTK_TABLE(table), 4);
284
285         /* First row */
286         top = 0;
287         label = gtk_label_new(_("Group Name"));
288         gtk_table_attach(GTK_TABLE(table), label, 0, 1, top, (top + 1), GTK_FILL, 0, 0, 0);
289         gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
290
291         entry_name = gtk_entry_new();
292         gtk_table_attach(GTK_TABLE(table), entry_name, 1, 2, top, (top + 1), GTK_EXPAND|GTK_SHRINK|GTK_FILL, 0, 0, 0);
293
294         /* List area */
295         hboxl = gtk_hbox_new( FALSE, 6 );
296         gtk_container_set_border_width( GTK_CONTAINER(hboxl), 8 );
297         gtk_box_pack_start(GTK_BOX(vbox), hboxl, TRUE, TRUE, 0);
298
299         /* Group list */
300         vboxl = gtk_vbox_new( FALSE, 0 );
301         gtk_box_pack_start(GTK_BOX(hboxl), vboxl, TRUE, TRUE, 0);
302
303         hboxh = gtk_hbox_new( FALSE, 0 );
304         gtk_container_set_border_width( GTK_CONTAINER(hboxh), 4 );
305         gtk_box_pack_start(GTK_BOX(vboxl), hboxh, FALSE, FALSE, 0);
306         label = gtk_label_new(_("Addresses in Group"));
307         gtk_box_pack_start(GTK_BOX(hboxh), label, TRUE, TRUE, 0);
308         buttonAvail = gtk_button_new_with_label( _( " -> " ) );
309         gtk_box_pack_end(GTK_BOX(hboxh), buttonAvail, FALSE, FALSE, 0);
310
311         clist_swin = gtk_scrolled_window_new( NULL, NULL );
312         gtk_box_pack_start(GTK_BOX(vboxl), clist_swin, TRUE, TRUE, 0);
313         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(clist_swin),
314                                        GTK_POLICY_AUTOMATIC,
315                                        GTK_POLICY_ALWAYS);
316
317         clist_group = gtk_clist_new_with_titles( GROUP_N_COLS, titles );
318         gtk_container_add( GTK_CONTAINER(clist_swin), clist_group );
319         gtk_clist_set_selection_mode( GTK_CLIST(clist_group), GTK_SELECTION_BROWSE );
320         gtk_clist_set_column_width( GTK_CLIST(clist_group), GROUP_COL_NAME, GROUP_COL_WIDTH_NAME );
321         gtk_clist_set_column_width( GTK_CLIST(clist_group), GROUP_COL_EMAIL, GROUP_COL_WIDTH_EMAIL );
322         gtk_clist_set_compare_func( GTK_CLIST(clist_group), edit_group_list_compare_func );
323         gtk_clist_set_auto_sort( GTK_CLIST(clist_group), TRUE );
324
325         for( i = 0; i < GROUP_N_COLS; i++ )
326                 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist_group)->column[i].button, GTK_CAN_FOCUS);
327
328         /* Available list */
329         vboxl = gtk_vbox_new( FALSE, 0 );
330         gtk_box_pack_start(GTK_BOX(hboxl), vboxl, TRUE, TRUE, 0);
331
332         hboxh = gtk_hbox_new( FALSE, 0 );
333         gtk_container_set_border_width( GTK_CONTAINER(hboxh), 4 );
334         gtk_box_pack_start(GTK_BOX(vboxl), hboxh, FALSE, FALSE, 0);
335         buttonGroup = gtk_button_new_with_label( _( " <- " ) );
336         gtk_box_pack_start(GTK_BOX(hboxh), buttonGroup, FALSE, FALSE, 0);
337         label = gtk_label_new(_("Available Addresses"));
338         gtk_box_pack_end(GTK_BOX(hboxh), label, TRUE, TRUE, 0);
339
340         clist_swin = gtk_scrolled_window_new( NULL, NULL );
341         gtk_box_pack_start(GTK_BOX(vboxl), clist_swin, TRUE, TRUE, 0);
342         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(clist_swin),
343                                        GTK_POLICY_AUTOMATIC,
344                                        GTK_POLICY_ALWAYS);
345
346         clist_avail = gtk_clist_new_with_titles( GROUP_N_COLS, titles );
347         gtk_container_add( GTK_CONTAINER(clist_swin), clist_avail );
348         gtk_clist_set_selection_mode( GTK_CLIST(clist_avail), GTK_SELECTION_BROWSE );
349         gtk_clist_set_column_width( GTK_CLIST(clist_avail), GROUP_COL_NAME, GROUP_COL_WIDTH_NAME );
350         gtk_clist_set_column_width( GTK_CLIST(clist_avail), GROUP_COL_EMAIL, GROUP_COL_WIDTH_EMAIL );
351         gtk_clist_set_compare_func( GTK_CLIST(clist_avail), edit_group_list_compare_func );
352         gtk_clist_set_auto_sort( GTK_CLIST(clist_avail), TRUE );
353
354         for( i = 0; i < GROUP_N_COLS; i++ )
355                 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist_avail)->column[i].button, GTK_CAN_FOCUS);
356
357         /* Status line */
358         hsbox = gtk_hbox_new(FALSE, 0);
359         gtk_box_pack_end(GTK_BOX(vbox), hsbox, FALSE, FALSE, BORDER_WIDTH);
360         statusbar = gtk_statusbar_new();
361         gtk_box_pack_start(GTK_BOX(hsbox), statusbar, TRUE, TRUE, BORDER_WIDTH);
362
363         /* Button panel */
364         gtkut_button_set_create(&hbbox, &ok_btn, _("OK"),
365                                 &cancel_btn, _("Cancel"), NULL, NULL);
366         gtk_box_pack_end(GTK_BOX(vbox), hbbox, FALSE, FALSE, 0);
367         gtk_widget_grab_default(ok_btn);
368
369         gtk_signal_connect(GTK_OBJECT(ok_btn), "clicked",
370                            GTK_SIGNAL_FUNC(edit_group_ok), cancelled);
371         gtk_signal_connect(GTK_OBJECT(cancel_btn), "clicked",
372                            GTK_SIGNAL_FUNC(edit_group_cancel), cancelled);
373
374         gtk_widget_show_all(vbox);
375
376         /* Event handlers */
377         gtk_signal_connect( GTK_OBJECT(clist_group), "select_row",
378                         GTK_SIGNAL_FUNC( edit_group_group_selected), NULL );
379         gtk_signal_connect( GTK_OBJECT(clist_avail), "select_row",
380                         GTK_SIGNAL_FUNC( edit_group_avail_selected), NULL );
381         gtk_signal_connect( GTK_OBJECT(buttonGroup), "clicked",
382                         GTK_SIGNAL_FUNC( edit_group_to_group ), NULL );
383         gtk_signal_connect( GTK_OBJECT(buttonAvail), "clicked",
384                         GTK_SIGNAL_FUNC( edit_group_to_avail ), NULL );
385         gtk_signal_connect(GTK_OBJECT(clist_avail), "button_press_event",
386                            GTK_SIGNAL_FUNC(edit_group_list_avail_button), NULL);
387         gtk_signal_connect(GTK_OBJECT(clist_group), "button_press_event",
388                            GTK_SIGNAL_FUNC(edit_group_list_group_button), NULL);
389
390         groupeditdlg.window     = window;
391         groupeditdlg.ok_btn     = ok_btn;
392         groupeditdlg.cancel_btn = cancel_btn;
393         groupeditdlg.statusbar  = statusbar;
394         groupeditdlg.status_cid = gtk_statusbar_get_context_id( GTK_STATUSBAR(statusbar), "Edit Group Dialog" );
395
396         groupeditdlg.entry_name  = entry_name;
397         groupeditdlg.clist_group = GTK_CLIST( clist_group );
398         groupeditdlg.clist_avail = GTK_CLIST( clist_avail );
399
400         if( ! _edit_group_dfl_message_ ) {
401                 _edit_group_dfl_message_ = _( "Move E-Mail Addresses to or from Group with arrow buttons" );
402         }
403 }
404
405 /*
406 * Return list of email items.
407 */
408 static GList *edit_group_build_email_list() {
409         GtkCList *clist = GTK_CLIST(groupeditdlg.clist_group);
410         GList *listEMail = NULL;
411         ItemEMail *email;
412         gint row = 0;
413         while( (email = gtk_clist_get_row_data( clist, row )) ) {
414                 listEMail = g_list_append( listEMail, email );
415                 row++;
416         }
417         return listEMail;
418 }
419
420 /*
421 * Edit group.
422 * Enter: abf    Address book.
423 *        folder Parent folder for group (or NULL if adding to root folder). Argument is
424 *               only required for new objects).
425 *        group  Group to edit, or NULL for a new group object.
426 * Return: Edited object, or NULL if cancelled.
427 */
428 ItemGroup *addressbook_edit_group( AddressBookFile *abf, ItemFolder *parent, ItemGroup *group ) {
429         static gboolean cancelled;
430         GList *listEMail = NULL;
431
432         if (!groupeditdlg.window)
433                 addressbook_edit_group_create(&cancelled);
434         gtk_widget_grab_focus(groupeditdlg.ok_btn);
435         gtk_widget_grab_focus(groupeditdlg.entry_name);
436         gtk_widget_show(groupeditdlg.window);
437         manage_window_set_transient(GTK_WINDOW(groupeditdlg.window));
438
439         /* Clear all fields */
440         groupeditdlg.rowIndGroup = -1;
441         groupeditdlg.rowIndAvail = -1;
442         edit_group_status_show( "" );
443         gtk_clist_clear( GTK_CLIST(groupeditdlg.clist_group) );
444         gtk_clist_clear( GTK_CLIST(groupeditdlg.clist_avail) );
445
446         if( group ) {
447                 if( ADDRITEM_NAME(group) )
448                         gtk_entry_set_text(GTK_ENTRY(groupeditdlg.entry_name), ADDRITEM_NAME(group) );
449                 edit_group_load_clist( groupeditdlg.clist_group, group->listEMail );
450                 gtk_window_set_title( GTK_WINDOW(groupeditdlg.window), _("Edit Group Details"));
451         }
452         else {
453                 gtk_window_set_title( GTK_WINDOW(groupeditdlg.window), _("Add New Group"));
454                 gtk_entry_set_text(GTK_ENTRY(groupeditdlg.entry_name), ADDRESSBOOK_GUESS_GROUP_NAME );
455         }
456
457         listEMail = addrbook_get_available_email_list( abf, group );
458         edit_group_load_clist( groupeditdlg.clist_avail, listEMail );
459         mgu_clear_list( listEMail );
460         listEMail = NULL;
461         gtk_clist_select_row( groupeditdlg.clist_group, 0, 0 );
462         gtk_clist_select_row( groupeditdlg.clist_avail, 0, 0 );
463
464         edit_group_status_show( _edit_group_dfl_message_ );
465
466         gtk_main();
467         gtk_widget_hide( groupeditdlg.window );
468
469         if( cancelled ) {
470                 return NULL;
471         }
472
473         listEMail = edit_group_build_email_list();
474         if( group ) {
475                 /* Update email list */
476                 addrbook_update_group_list( abf, group, listEMail );
477         }
478         else {
479                 /* Create new person and email list */
480                 group = addrbook_add_group_list( abf, parent, listEMail );
481         }
482         addritem_group_set_name( group, gtk_editable_get_chars( GTK_EDITABLE(groupeditdlg.entry_name), 0, -1 ) );
483
484         listEMail = NULL;
485         return group;
486 }
487
488 /*
489 * Edit folder.
490 * Enter: abf    Address book.
491 *        parent Parent folder for folder (or NULL if adding to root folder). Argument is
492 *               only required for new objects).
493 *        folder Folder to edit, or NULL for a new folder object.
494 * Return: Edited object, or NULL if cancelled.
495 */
496 ItemFolder *addressbook_edit_folder( AddressBookFile *abf, ItemFolder *parent, ItemFolder *folder ) {
497         gchar *name = NULL;
498
499         if( folder ) {
500                 name = g_strdup( ADDRITEM_NAME(folder) );
501                 name = input_dialog( _("Edit folder"), _("Input the new name of folder:"), name );
502         }
503         else {
504                 name = input_dialog( _("New folder"),
505                                 _("Input the name of new folder:"),
506                                 _(ADDRESSBOOK_GUESS_FOLDER_NAME) );
507         }
508         if( ! name ) return NULL;
509         g_strstrip( name );
510         if( *name == '\0' ) {
511                 g_free( name );
512                 return NULL;
513         }
514         if( folder ) {
515                 if( strcasecmp( name, ADDRITEM_NAME(folder) ) == 0 ) {
516                         g_free( name );
517                         return NULL;
518                 }
519         }
520
521         if( ! folder ) {
522                 folder = addrbook_add_new_folder( abf, parent );
523         }
524         addritem_folder_set_name( folder, name );
525         g_free( name );
526         return folder;
527 }
528
529 /*
530 * End of Source.
531 */
532