2012-09-12 [mones] 3.8.1cvs54
[claws.git] / src / prefs_customheader.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2012 Hiroyuki Yamamoto 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 3 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, see <http://www.gnu.org/licenses/>.
17  * 
18  */
19
20 #ifdef HAVE_CONFIG_H
21 #  include "config.h"
22 #include "claws-features.h"
23 #endif
24
25 #include "defs.h"
26
27 #include <glib.h>
28 #include <glib/gi18n.h>
29 #include <gtk/gtk.h>
30 #include <gdk/gdkkeysyms.h>
31 #include <stdio.h>
32 #include <stdlib.h>
33 #include <string.h>
34 #include <errno.h>
35
36 #include "main.h"
37 #include "prefs_gtk.h"
38 #include "prefs_customheader.h"
39 #include "prefs_common.h"
40 #include "prefs_account.h"
41 #include "mainwindow.h"
42 #include "foldersel.h"
43 #include "manage_window.h"
44 #include "customheader.h"
45 #include "folder.h"
46 #include "utils.h"
47 #include "gtkutils.h"
48 #include "alertpanel.h"
49 #include "base64.h"
50 #include "filesel.h"
51 #include "combobox.h"
52
53 enum {
54         CUSTHDR_STRING,         /*!< display string managed by list store */
55         CUSTHDR_DATA,           /*!< string managed by us */
56         N_CUSTHDR_COLUMNS
57 };
58
59 static struct CustomHdr {
60         GtkWidget *window;
61
62         GtkWidget *ok_btn;
63         GtkWidget *cancel_btn;
64
65         GtkWidget *hdr_combo;
66         GtkWidget *hdr_entry;
67         GtkWidget *val_entry;
68         GtkWidget *preview;
69         GtkWidget *list_view;
70 } customhdr;
71
72 /* widget creating functions */
73 static void prefs_custom_header_create  (void);
74
75 static void prefs_custom_header_set_dialog              (PrefsAccount *ac);
76 static void prefs_custom_header_set_list                (PrefsAccount *ac);
77 static void prefs_custom_header_list_view_set_row       (PrefsAccount *ac);
78
79 /* callback functions */
80 static void prefs_custom_header_add_cb          (void);
81 static void prefs_custom_header_val_from_file_cb(void);
82 static void prefs_custom_header_delete_cb       (void);
83 static void prefs_custom_header_up              (void);
84 static void prefs_custom_header_down            (void);
85
86 static gboolean prefs_custom_header_key_pressed (GtkWidget      *widget,
87                                                  GdkEventKey    *event,
88                                                  gpointer        data);
89 static void prefs_custom_header_ok              (void);
90 static void prefs_custom_header_cancel          (void);
91 static gint prefs_custom_header_deleted         (GtkWidget      *widget,
92                                                  GdkEventAny    *event,
93                                                  gpointer        data);
94
95 static GtkListStore* prefs_custom_header_create_data_store      (void);
96
97 static void prefs_custom_header_list_view_insert_header (GtkWidget *list_view,
98                                                          GtkTreeIter *row_iter,
99                                                          gchar *header,
100                                                          gpointer data);
101
102 static GtkWidget *prefs_custom_header_list_view_create (void);
103
104 static void prefs_custom_header_create_list_view_columns        (GtkWidget *list_view);
105
106 static gboolean prefs_custom_header_selected    (GtkTreeSelection *selector,
107                                                  GtkTreeModel *model, 
108                                                  GtkTreePath *path,
109                                                  gboolean currently_selected,
110                                                  gpointer data);
111
112
113 static PrefsAccount *cur_ac = NULL;
114
115 void prefs_custom_header_open(PrefsAccount *ac)
116 {
117         if (!customhdr.window) {
118                 prefs_custom_header_create();
119         }
120
121         manage_window_set_transient(GTK_WINDOW(customhdr.window));
122         gtk_widget_grab_focus(customhdr.ok_btn);
123
124         prefs_custom_header_set_dialog(ac);
125
126         cur_ac = ac;
127
128         gtk_widget_show(customhdr.window);
129         gtk_window_set_modal(GTK_WINDOW(customhdr.window), TRUE);
130 }
131
132 static void prefs_custom_header_create(void)
133 {
134         GtkWidget *window;
135         GtkWidget *vbox;
136
137         GtkWidget *ok_btn;
138         GtkWidget *cancel_btn;
139
140         GtkWidget *confirm_area;
141
142         GtkWidget *vbox1;
143
144         GtkWidget *table1;
145         GtkWidget *hdr_label;
146         GtkWidget *hdr_combo;
147         GtkWidget *val_label;
148         GtkWidget *val_entry;
149         GtkWidget *val_btn;
150
151         GtkWidget *reg_hbox;
152         GtkWidget *btn_hbox;
153         GtkWidget *arrow;
154         GtkWidget *add_btn;
155         GtkWidget *del_btn;
156         GtkWidget *preview;
157
158         GtkWidget *ch_hbox;
159         GtkWidget *ch_scrolledwin;
160         GtkWidget *list_view;
161
162         GtkWidget *btn_vbox;
163         GtkWidget *up_btn;
164         GtkWidget *down_btn;
165
166         debug_print("Creating custom header setting window...\n");
167
168         window = gtkut_window_new(GTK_WINDOW_TOPLEVEL, "prefs_customheader");
169         gtk_container_set_border_width (GTK_CONTAINER (window), 8);
170         gtk_window_set_position (GTK_WINDOW (window), GTK_WIN_POS_CENTER);
171         gtk_window_set_resizable(GTK_WINDOW (window), TRUE);
172
173         vbox = gtk_vbox_new (FALSE, 6);
174         gtk_widget_show (vbox);
175         gtk_container_add (GTK_CONTAINER (window), vbox);
176
177         gtkut_stock_button_set_create(&confirm_area, &cancel_btn, GTK_STOCK_CANCEL,
178                                       &ok_btn, GTK_STOCK_OK,
179                                       NULL, NULL);
180         gtk_widget_show (confirm_area);
181         gtk_box_pack_end (GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
182         gtk_widget_grab_default (ok_btn);
183
184         gtk_window_set_title (GTK_WINDOW(window), _("Custom header configuration"));
185         MANAGE_WINDOW_SIGNALS_CONNECT (window);
186         g_signal_connect (G_OBJECT(window), "delete_event",
187                           G_CALLBACK(prefs_custom_header_deleted),
188                           NULL);
189         g_signal_connect (G_OBJECT(window), "key_press_event",
190                           G_CALLBACK(prefs_custom_header_key_pressed),
191                           NULL);
192         g_signal_connect (G_OBJECT(ok_btn), "clicked",
193                           G_CALLBACK(prefs_custom_header_ok), NULL);
194         g_signal_connect (G_OBJECT(cancel_btn), "clicked",
195                           G_CALLBACK(prefs_custom_header_cancel), NULL);
196
197         vbox1 = gtk_vbox_new (FALSE, VSPACING);
198         gtk_widget_show (vbox1);
199         gtk_box_pack_start (GTK_BOX (vbox), vbox1, TRUE, TRUE, 0);
200         gtk_container_set_border_width (GTK_CONTAINER (vbox1), 2);
201
202         table1 = gtk_table_new (3, 2, FALSE);
203         gtk_widget_show (table1);
204         gtk_box_pack_start (GTK_BOX (vbox1), table1,
205                             FALSE, FALSE, 0);
206         gtk_table_set_row_spacings (GTK_TABLE (table1), 8);
207         gtk_table_set_col_spacings (GTK_TABLE (table1), 8);
208
209         hdr_label = gtk_label_new (_("Header"));
210         gtk_widget_show (hdr_label);
211         gtk_table_attach (GTK_TABLE (table1), hdr_label, 0, 1, 0, 1,
212                           GTK_EXPAND | GTK_SHRINK | GTK_FILL,
213                           0, 0, 0);
214         gtk_misc_set_alignment (GTK_MISC (hdr_label), 0, 0.5);
215         
216         hdr_combo = combobox_text_new(TRUE, "User-Agent", "Face", "X-Face",
217                                       "X-Operating-System", NULL);
218         gtk_table_attach (GTK_TABLE (table1), hdr_combo, 0, 1, 1, 2,
219                           GTK_EXPAND | GTK_SHRINK | GTK_FILL,
220                           0, 0, 0);
221         gtk_widget_set_size_request (hdr_combo, 150, -1);
222
223         val_label = gtk_label_new (_("Value"));
224         gtk_widget_show (val_label);
225         gtk_table_attach (GTK_TABLE (table1), val_label, 1, 2, 0, 1,
226                           GTK_EXPAND | GTK_SHRINK | GTK_FILL,
227                           0, 0, 0);
228         gtk_misc_set_alignment (GTK_MISC (val_label), 0, 0.5);
229         
230         val_entry = gtk_entry_new ();
231         gtk_widget_show (val_entry);
232         gtk_table_attach (GTK_TABLE (table1), val_entry, 1, 2, 1, 2,
233                           GTK_EXPAND | GTK_SHRINK | GTK_FILL,
234                           0, 0, 0);
235         gtk_widget_set_size_request (val_entry, 200, -1);
236
237         val_btn = gtkut_get_browse_file_btn(_("Bro_wse"));
238         gtk_widget_show (val_btn);
239         gtk_table_attach (GTK_TABLE (table1), val_btn, 2, 3, 1, 2,
240                           GTK_EXPAND | GTK_SHRINK | GTK_FILL,
241                           0, 0, 0);
242         g_signal_connect (G_OBJECT (val_btn), "clicked",
243                           G_CALLBACK (prefs_custom_header_val_from_file_cb),
244                           NULL);
245
246         /* add / delete */
247
248         reg_hbox = gtk_hbox_new (FALSE, 4);
249         gtk_widget_show (reg_hbox);
250         gtk_box_pack_start (GTK_BOX (vbox1), reg_hbox, FALSE, FALSE, 0);
251
252         arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_OUT);
253         gtk_widget_show (arrow);
254         gtk_box_pack_start (GTK_BOX (reg_hbox), arrow, FALSE, FALSE, 0);
255         gtk_widget_set_size_request (arrow, -1, 16);
256
257         btn_hbox = gtk_hbox_new (TRUE, 4);
258         gtk_widget_show (btn_hbox);
259         gtk_box_pack_start (GTK_BOX (reg_hbox), btn_hbox, FALSE, FALSE, 0);
260
261         add_btn = gtk_button_new_from_stock (GTK_STOCK_ADD);
262         gtk_widget_show (add_btn);
263         gtk_box_pack_start (GTK_BOX (btn_hbox), add_btn, FALSE, TRUE, 0);
264         g_signal_connect (G_OBJECT (add_btn), "clicked",
265                           G_CALLBACK (prefs_custom_header_add_cb),
266                           NULL);
267
268         del_btn = gtk_button_new_from_stock (GTK_STOCK_DELETE);
269         gtk_widget_show (del_btn);
270         gtk_box_pack_start (GTK_BOX (btn_hbox), del_btn, FALSE, TRUE, 0);
271         g_signal_connect (G_OBJECT (del_btn), "clicked",
272                           G_CALLBACK (prefs_custom_header_delete_cb),
273                           NULL);
274
275
276         ch_hbox = gtk_hbox_new (FALSE, 8);
277         gtk_widget_show (ch_hbox);
278         gtk_box_pack_start (GTK_BOX (vbox1), ch_hbox, TRUE, TRUE, 0);
279
280         ch_scrolledwin = gtk_scrolled_window_new (NULL, NULL);
281         gtk_widget_set_size_request (ch_scrolledwin, -1, 200);
282         gtk_widget_show (ch_scrolledwin);
283         gtk_box_pack_start (GTK_BOX (ch_hbox), ch_scrolledwin, TRUE, TRUE, 0);
284         gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (ch_scrolledwin),
285                                         GTK_POLICY_AUTOMATIC,
286                                         GTK_POLICY_AUTOMATIC);
287
288         list_view = prefs_custom_header_list_view_create();
289         gtk_widget_show (list_view);
290         gtk_container_add (GTK_CONTAINER (ch_scrolledwin), list_view);
291
292         btn_vbox = gtk_vbox_new (FALSE, 8);
293         gtk_widget_show (btn_vbox);
294         gtk_box_pack_start (GTK_BOX (ch_hbox), btn_vbox, FALSE, FALSE, 0);
295
296         up_btn = gtk_button_new_from_stock (GTK_STOCK_GO_UP);
297         gtk_widget_show (up_btn);
298         gtk_box_pack_start (GTK_BOX (btn_vbox), up_btn, FALSE, FALSE, 0);
299         g_signal_connect (G_OBJECT (up_btn), "clicked",
300                           G_CALLBACK (prefs_custom_header_up), NULL);
301
302         down_btn = gtk_button_new_from_stock (GTK_STOCK_GO_DOWN);
303         gtk_widget_show (down_btn);
304         gtk_box_pack_start (GTK_BOX (btn_vbox), down_btn, FALSE, FALSE, 0);
305         g_signal_connect (G_OBJECT (down_btn), "clicked",
306                           G_CALLBACK (prefs_custom_header_down), NULL);
307
308         preview = gtk_image_new ();
309         gtk_widget_show (preview);
310         gtk_box_pack_start (GTK_BOX (btn_vbox), preview, FALSE, FALSE, 0);
311
312         gtk_widget_show_all(window);
313
314         customhdr.window     = window;
315         customhdr.ok_btn     = ok_btn;
316         customhdr.cancel_btn = cancel_btn;
317         customhdr.preview = preview;
318
319         customhdr.hdr_combo  = hdr_combo;
320         customhdr.hdr_entry  = gtk_bin_get_child(GTK_BIN((hdr_combo)));
321         customhdr.val_entry  = val_entry;
322
323         customhdr.list_view   = list_view;
324 }
325
326 void prefs_custom_header_read_config(PrefsAccount *ac)
327 {
328         gchar *rcpath;
329         FILE *fp;
330         gchar buf[PREFSBUFSIZE];
331         CustomHeader *ch;
332
333         debug_print("Reading custom header configuration...\n");
334
335         rcpath = g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S,
336                              CUSTOM_HEADER_RC, NULL);
337         if ((fp = g_fopen(rcpath, "rb")) == NULL) {
338                 if (ENOENT != errno) FILE_OP_ERROR(rcpath, "fopen");
339                 g_free(rcpath);
340                 ac->customhdr_list = NULL;
341                 return;
342         }
343         g_free(rcpath);
344
345         /* remove all previous headers list */
346         while (ac->customhdr_list != NULL) {
347                 ch = (CustomHeader *)ac->customhdr_list->data;
348                 ac->customhdr_list = g_slist_remove(ac->customhdr_list, ch);
349                 custom_header_free(ch);
350         }
351
352         while (fgets(buf, sizeof(buf), fp) != NULL) {
353                 ch = custom_header_read_str(buf);
354                 if (ch) {
355                         if (ch->account_id == ac->account_id) {
356                                 ac->customhdr_list =
357                                         g_slist_append(ac->customhdr_list, ch);
358                         } else
359                                 custom_header_free(ch);
360                 }
361         }
362
363         fclose(fp);
364 }
365
366 static void prefs_custom_header_write_config(PrefsAccount *ac)
367 {
368         gchar *rcpath;
369         PrefFile *pfile;
370         GSList *cur;
371         gchar buf[PREFSBUFSIZE];
372         FILE * fp;
373         CustomHeader *ch;
374
375         GSList *all_hdrs = NULL;
376
377         debug_print("Writing custom header configuration...\n");
378
379         rcpath = g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S,
380                              CUSTOM_HEADER_RC, NULL);
381
382         if ((fp = g_fopen(rcpath, "rb")) == NULL) {
383                 if (ENOENT != errno) FILE_OP_ERROR(rcpath, "fopen");
384         } else {
385                 all_hdrs = NULL;
386
387                 while (fgets(buf, sizeof(buf), fp) != NULL) {
388                         ch = custom_header_read_str(buf);
389                         if (ch) {
390                                 if (ch->account_id != ac->account_id)
391                                         all_hdrs =
392                                                 g_slist_append(all_hdrs, ch);
393                                 else
394                                         custom_header_free(ch);
395                         }
396                 }
397
398                 fclose(fp);
399         }
400
401         if ((pfile = prefs_write_open(rcpath)) == NULL) {
402                 g_warning("failed to write configuration to file\n");
403                 g_free(rcpath);
404                 return;
405         }
406
407         for (cur = all_hdrs; cur != NULL; cur = cur->next) {
408                 CustomHeader *hdr = (CustomHeader *)cur->data;
409                 gchar *chstr;
410
411                 chstr = custom_header_get_str(hdr);
412                 if (fputs(chstr, pfile->fp) == EOF ||
413                     fputc('\n', pfile->fp) == EOF) {
414                         FILE_OP_ERROR(rcpath, "fputs || fputc");
415                         prefs_file_close_revert(pfile);
416                         g_free(rcpath);
417                         g_free(chstr);
418                         return;
419                 }
420                 g_free(chstr);
421         }
422
423         for (cur = ac->customhdr_list; cur != NULL; cur = cur->next) {
424                 CustomHeader *hdr = (CustomHeader *)cur->data;
425                 gchar *chstr;
426
427                 chstr = custom_header_get_str(hdr);
428                 if (fputs(chstr, pfile->fp) == EOF ||
429                     fputc('\n', pfile->fp) == EOF) {
430                         FILE_OP_ERROR(rcpath, "fputs || fputc");
431                         prefs_file_close_revert(pfile);
432                         g_free(rcpath);
433                         g_free(chstr);
434                         return;
435                 }
436                 g_free(chstr);
437         }
438
439         g_free(rcpath);
440
441         while (all_hdrs != NULL) {
442                 ch = (CustomHeader *)all_hdrs->data;
443                 all_hdrs = g_slist_remove(all_hdrs, ch);
444                 custom_header_free(ch);
445         }
446
447         if (prefs_file_close(pfile) < 0) {
448                 g_warning("failed to write configuration to file\n");
449                 return;
450         }
451 }
452
453 static void prefs_custom_header_set_dialog(PrefsAccount *ac)
454 {
455         GtkListStore *store;
456         GSList *cur;
457         
458         store = GTK_LIST_STORE(gtk_tree_view_get_model
459                                 (GTK_TREE_VIEW(customhdr.list_view)));
460         gtk_list_store_clear(store);
461
462         for (cur = ac->customhdr_list; cur != NULL; cur = cur->next) {
463                 CustomHeader *ch = (CustomHeader *)cur->data;
464                 gchar *ch_str;
465
466                 ch_str = g_strdup_printf("%s: %s", ch->name,
467                                          ch->value ? ch->value : "");
468
469                 prefs_custom_header_list_view_insert_header
470                         (customhdr.list_view, NULL, ch_str, ch);                                                 
471
472                 g_free(ch_str);
473         }
474 }
475
476 static void prefs_custom_header_set_list(PrefsAccount *ac)
477 {
478         CustomHeader *ch;
479         GtkTreeIter iter;
480         GtkListStore *store;
481
482         g_slist_free(ac->customhdr_list);
483         ac->customhdr_list = NULL;
484
485         store = GTK_LIST_STORE(gtk_tree_view_get_model
486                                 (GTK_TREE_VIEW(customhdr.list_view)));
487
488         if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store), &iter)) {
489                 do {
490                         gtk_tree_model_get(GTK_TREE_MODEL(store), &iter,
491                                            CUSTHDR_DATA, &ch,
492                                            -1);
493                         ac->customhdr_list = g_slist_append(ac->customhdr_list, ch);
494                 } while (gtk_tree_model_iter_next(GTK_TREE_MODEL(store),
495                                                   &iter));
496         }
497 }
498
499 static void prefs_custom_header_list_view_set_row(PrefsAccount *ac)
500 {
501         CustomHeader *ch;
502         const gchar *entry_text;
503         gchar *ch_str;
504         GtkListStore *store;
505
506         store = GTK_LIST_STORE(gtk_tree_view_get_model
507                                 (GTK_TREE_VIEW(customhdr.list_view)));
508
509         entry_text = gtk_entry_get_text(GTK_ENTRY(customhdr.hdr_entry));
510         if (entry_text[0] == '\0') {
511                 alertpanel_error(_("Header name is not set."));
512                 return;
513         }
514         
515         while (*entry_text && 
516                (*entry_text == '\n' || *entry_text == '\r' || 
517                 *entry_text == '\t' || *entry_text == ' '))
518                 entry_text++;
519         
520         if (!custom_header_is_allowed(entry_text)) {
521                 alertpanel_error(_("This Header name is not allowed as a custom header."));
522                 return;
523         }
524
525         ch = g_new0(CustomHeader, 1);
526
527         ch->account_id = ac->account_id;
528
529         ch->name = g_strdup(entry_text);
530         unfold_line(ch->name);
531         g_strstrip(ch->name);
532         gtk_entry_set_text(GTK_ENTRY(customhdr.hdr_entry), ch->name);
533
534         entry_text = gtk_entry_get_text(GTK_ENTRY(customhdr.val_entry));
535         while (*entry_text && 
536                (*entry_text == '\n' || *entry_text == '\r' || 
537                 *entry_text == '\t' || *entry_text == ' '))
538                 entry_text++;
539         
540         if (entry_text[0] != '\0') {
541                 ch->value = g_strdup(entry_text);
542                 unfold_line(ch->value);
543                 g_strstrip(ch->value);
544                 gtk_entry_set_text(GTK_ENTRY(customhdr.val_entry), ch->value);
545         }
546
547         ch_str = g_strdup_printf("%s: %s", ch->name,
548                                  ch->value ? ch->value : "");
549         
550         prefs_custom_header_list_view_insert_header
551                 (customhdr.list_view, NULL, ch_str, ch);
552         
553         g_free(ch_str);
554
555         prefs_custom_header_set_list(cur_ac);
556
557 }
558
559 #define B64_LINE_SIZE           57
560 #define B64_BUFFSIZE            77
561 static void prefs_custom_header_val_from_file_cb(void)
562 {
563         gchar *filename = NULL;
564         gchar *contents = NULL;
565         const gchar *hdr = gtk_entry_get_text(GTK_ENTRY(customhdr.hdr_entry));
566         
567         if (!strcmp(hdr, "Face"))
568                 filename = filesel_select_file_open(_("Choose a PNG file"), NULL);
569         else if (!strcmp(hdr, "X-Face"))
570                 filename = filesel_select_file_open(_("Choose an XBM file"), NULL);
571         else
572                 filename = filesel_select_file_open(_("Choose a text file"), NULL);
573
574         if (!strcmp(hdr, "Face") || !strcmp(hdr, "X-Face")) {
575                 if (filename && is_file_exist(filename)) {
576                         FILE *fp = NULL;
577                         gint len;
578                         gchar inbuf[B64_LINE_SIZE], outbuf[B64_BUFFSIZE];
579                         gchar *tmp = NULL;
580                         gint w, h;
581                         GdkPixbufFormat *format = gdk_pixbuf_get_file_info(
582                                                         filename, &w, &h);
583                         
584                         if (format == NULL) {
585                                 alertpanel_error(_("This file isn't an image."));
586                                 g_free(filename);
587                                 return;
588                         }
589                         if (w != 48 || h != 48) {
590                                 alertpanel_error(_("The chosen image isn't the correct size (48x48)."));
591                                 g_free(filename);
592                                 return; 
593                         }
594                         if (!strcmp(hdr, "Face")) {
595                                 if (get_file_size(filename) > 725) {
596                                         alertpanel_error(_("The image is too big; it must be maximum 725 bytes."));
597                                         g_free(filename);
598                                         return;
599                                 }
600                                 if (g_ascii_strcasecmp("png", gdk_pixbuf_format_get_name(format))) {
601                                         alertpanel_error(_("The image isn't in the correct format (PNG)."));
602                                         g_print("%s\n", gdk_pixbuf_format_get_name(format));
603                                         g_free(filename);
604                                         return;
605                                 }
606                         } else if (!strcmp(hdr, "X-Face")) {
607                                 gchar *tmp = NULL, *cmd = NULL;
608                                 int i = 0;
609                                 if (g_ascii_strcasecmp("xbm", gdk_pixbuf_format_get_name(format))) {
610                                         alertpanel_error(_("The image isn't in the correct format (XBM)."));
611                                         g_print("%s\n", gdk_pixbuf_format_get_name(format));
612                                         g_free(filename);
613                                         return;
614                                 }
615                                 cmd = g_strdup_printf("compface %s", filename);
616                                 tmp = get_command_output(cmd);
617                                 g_free(cmd);
618                                 if (tmp == NULL || *tmp == '\0') {
619                                         alertpanel_error(_("Couldn't call `compface`. Make sure it's in your $PATH."));
620                                         g_free(filename);
621                                         g_free(tmp);
622                                         return;
623                                 }
624                                 if (strstr(tmp, "compface:")) {
625                                         alertpanel_error(_("Compface error: %s"), tmp);
626                                         g_free(filename);
627                                         g_free(tmp);
628                                         return;
629                                 }
630                                 while (tmp[i]) {
631                                         gchar *tmp2 = NULL;
632                                         if (tmp[i] == ' ') {
633                                                 i++; continue;
634                                         } 
635                                         if (tmp[i] == '\r' || tmp[i] == '\n') {
636                                                 i++; continue;
637                                         }
638                                         tmp2 = contents;
639                                         contents = g_strdup_printf("%s%c",tmp2?tmp2:"", tmp[i]);
640                                         g_free(tmp2);
641                                         i++;
642                                 }
643                                 g_free(tmp);
644                                 goto settext;
645                         }
646
647                         fp = g_fopen(filename, "rb");
648                         if (!fp) {
649                                 g_free(filename);
650                                 return; 
651                         }
652
653                         while ((len = fread(inbuf, sizeof(gchar),
654                                             B64_LINE_SIZE, fp))
655                                == B64_LINE_SIZE) {
656                                 base64_encode(outbuf, inbuf, B64_LINE_SIZE);
657
658                                 tmp = contents;
659                                 contents = g_strconcat(tmp?tmp:"",outbuf, NULL);
660                                 g_free(tmp);
661                         }
662                         if (len > 0 && feof(fp)) {
663                                 tmp = contents;
664                                 base64_encode(outbuf, inbuf, len);
665                                 contents = g_strconcat(tmp?tmp:"",outbuf, NULL);
666                                 g_free(tmp);
667                         }
668                         fclose(fp);
669                 }
670         } else {
671                 if (!filename)
672                         return;
673
674                 contents = file_read_to_str(filename);
675                 if (strchr(contents, '\n') || strchr(contents,'\r')) {
676                         alertpanel_error(_("This file contains newlines."));
677                         g_free(contents);
678                         g_free(filename);
679                         return;
680                 }
681         }
682 settext:
683         if (contents && strlen(contents))
684                 gtk_entry_set_text(GTK_ENTRY(customhdr.val_entry), contents);
685         
686         g_free(contents);
687         g_free(filename);
688 }
689
690 static void prefs_custom_header_add_cb(void)
691 {
692         prefs_custom_header_list_view_set_row(cur_ac);
693 }
694
695 static void prefs_custom_header_delete_cb(void)
696 {
697         GtkTreeIter sel;
698         GtkTreeModel *model;
699         CustomHeader *ch;
700
701         if (!gtk_tree_selection_get_selected(gtk_tree_view_get_selection
702                                 (GTK_TREE_VIEW(customhdr.list_view)),
703                                 &model, &sel))
704                 return; 
705
706         if (alertpanel(_("Delete header"),
707                        _("Do you really want to delete this header?"),
708                        GTK_STOCK_CANCEL, GTK_STOCK_DELETE, NULL) != G_ALERTALTERNATE)
709                 return;
710
711         gtk_tree_model_get(model, &sel,
712                            CUSTHDR_DATA, &ch,
713                            -1);
714         gtk_list_store_remove(GTK_LIST_STORE(model), &sel);
715
716         cur_ac->customhdr_list = g_slist_remove(cur_ac->customhdr_list, ch);
717         
718         custom_header_free(ch);
719 }
720
721 static void prefs_custom_header_up(void)
722 {
723         GtkTreePath *prev, *sel;
724         GtkTreeIter isel;
725         GtkListStore *store = NULL;
726         GtkTreeModel *model = NULL;
727         GtkTreeIter iprev;
728         
729         if (!gtk_tree_selection_get_selected
730                 (gtk_tree_view_get_selection
731                         (GTK_TREE_VIEW(customhdr.list_view)),
732                  &model,
733                  &isel))
734                 return;
735         store = (GtkListStore *)model;
736         sel = gtk_tree_model_get_path(GTK_TREE_MODEL(store), &isel);
737         if (!sel)
738                 return;
739         
740         /* no move if we're at row 0... */
741         prev = gtk_tree_path_copy(sel);
742         if (!gtk_tree_path_prev(prev)) {
743                 gtk_tree_path_free(prev);
744                 gtk_tree_path_free(sel);
745                 return;
746         }
747
748         gtk_tree_model_get_iter(GTK_TREE_MODEL(store),
749                                 &iprev, prev);
750         gtk_tree_path_free(sel);
751         gtk_tree_path_free(prev);
752
753         gtk_list_store_swap(store, &iprev, &isel);
754         prefs_custom_header_set_list(cur_ac);
755 }
756
757 static void prefs_custom_header_down(void)
758 {
759         GtkListStore *store = NULL;
760         GtkTreeModel *model = NULL;
761         GtkTreeIter next, sel;
762         
763         if (!gtk_tree_selection_get_selected
764                 (gtk_tree_view_get_selection
765                         (GTK_TREE_VIEW(customhdr.list_view)),
766                  &model,
767                  &sel))
768                 return;
769         store = (GtkListStore *)model;
770         next = sel;
771         if (!gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &next)) 
772                 return;
773
774         gtk_list_store_swap(store, &next, &sel);
775         prefs_custom_header_set_list(cur_ac);
776 }
777
778 static gboolean prefs_custom_header_key_pressed(GtkWidget *widget,
779                                                 GdkEventKey *event,
780                                                 gpointer data)
781 {
782         if (event && event->keyval == GDK_KEY_Escape)
783                 prefs_custom_header_cancel();
784         return FALSE;
785 }
786
787 static void prefs_custom_header_ok(void)
788 {
789         prefs_custom_header_write_config(cur_ac);
790         gtk_widget_hide(customhdr.window);
791         gtk_window_set_modal(GTK_WINDOW(customhdr.window), FALSE);
792 }
793
794 static void prefs_custom_header_cancel(void)
795 {
796         prefs_custom_header_read_config(cur_ac); 
797         gtk_widget_hide(customhdr.window);
798         gtk_window_set_modal(GTK_WINDOW(customhdr.window), FALSE);
799 }
800
801 static gint prefs_custom_header_deleted(GtkWidget *widget, GdkEventAny *event,
802                                         gpointer data)
803 {
804         prefs_custom_header_cancel();
805         return TRUE;
806 }
807
808 static GtkListStore* prefs_custom_header_create_data_store(void)
809 {
810         return gtk_list_store_new(N_CUSTHDR_COLUMNS,
811                                   G_TYPE_STRING,        
812                                   G_TYPE_POINTER,
813                                   -1);
814 }
815
816 static void prefs_custom_header_list_view_insert_header(GtkWidget *list_view,
817                                                         GtkTreeIter *row_iter,
818                                                         gchar *header,
819                                                         gpointer data)
820 {
821         GtkTreeIter iter;
822         GtkListStore *list_store = GTK_LIST_STORE(gtk_tree_view_get_model
823                                         (GTK_TREE_VIEW(list_view)));
824
825         if (row_iter == NULL) {
826                 /* append new */
827                 gtk_list_store_append(list_store, &iter);
828                 gtk_list_store_set(list_store, &iter,
829                                    CUSTHDR_STRING, header,
830                                    CUSTHDR_DATA,   data,
831                                    -1);
832         } else {
833                 /* change existing */
834                 CustomHeader *old_data;
835
836                 gtk_tree_model_get(GTK_TREE_MODEL(list_store), row_iter,
837                                    CUSTHDR_DATA, &old_data,
838                                    -1);
839
840                 custom_header_free(old_data);
841                 
842                 gtk_list_store_set(list_store, row_iter,
843                                    CUSTHDR_STRING, header,
844                                    CUSTHDR_DATA, data,
845                                    -1);
846         }
847 }
848
849 static GtkWidget *prefs_custom_header_list_view_create(void)
850 {
851         GtkTreeView *list_view;
852         GtkTreeSelection *selector;
853         GtkTreeModel *model;
854
855         model = GTK_TREE_MODEL(prefs_custom_header_create_data_store());
856         list_view = GTK_TREE_VIEW(gtk_tree_view_new_with_model(model));
857         g_object_unref(model);  
858         
859         gtk_tree_view_set_rules_hint(list_view, prefs_common.use_stripes_everywhere);
860         gtk_tree_view_set_reorderable(list_view, TRUE);
861         
862         selector = gtk_tree_view_get_selection(list_view);
863         gtk_tree_selection_set_mode(selector, GTK_SELECTION_BROWSE);
864         gtk_tree_selection_set_select_function(selector, prefs_custom_header_selected,
865                                                NULL, NULL);
866
867         /* create the columns */
868         prefs_custom_header_create_list_view_columns(GTK_WIDGET(list_view));
869
870         return GTK_WIDGET(list_view);
871 }
872
873 static void prefs_custom_header_create_list_view_columns(GtkWidget *list_view)
874 {
875         GtkTreeViewColumn *column;
876         GtkCellRenderer *renderer;
877
878         renderer = gtk_cell_renderer_text_new();
879         column = gtk_tree_view_column_new_with_attributes
880                 (_("Current custom headers"),
881                  renderer,
882                  "text", CUSTHDR_STRING,
883                  NULL);
884         gtk_tree_view_append_column(GTK_TREE_VIEW(list_view), column);          
885 }
886
887 #define ENTRY_SET_TEXT(entry, str) \
888         gtk_entry_set_text(GTK_ENTRY(entry), str ? str : "")
889
890 static gboolean prefs_custom_header_selected(GtkTreeSelection *selector,
891                                              GtkTreeModel *model, 
892                                              GtkTreePath *path,
893                                              gboolean currently_selected,
894                                              gpointer data)
895 {
896         GtkTreeIter iter;
897         CustomHeader *ch;
898         GtkImage *preview;
899         GdkPixbuf *pixbuf;
900         CustomHeader default_ch = { 0, "", NULL };
901
902         if (currently_selected)
903                 return TRUE;
904
905         if (!gtk_tree_model_get_iter(model, &iter, path))
906                 return TRUE;
907
908         gtk_tree_model_get(model, &iter, 
909                            CUSTHDR_DATA, &ch,
910                            -1);
911         
912         if (!ch) ch = &default_ch;
913
914         ENTRY_SET_TEXT(customhdr.hdr_entry, ch->name);
915         ENTRY_SET_TEXT(customhdr.val_entry, ch->value);
916         if (!strcmp2("Face",ch->name)) {
917                 preview = GTK_IMAGE(face_get_from_header (ch->value));
918                 pixbuf = gtk_image_get_pixbuf(preview);
919                 gtk_image_set_from_pixbuf (GTK_IMAGE(customhdr.preview), pixbuf);
920                 gtk_widget_show(customhdr.preview);
921 #if GLIB_CHECK_VERSION(2,10,0)
922                 g_object_ref_sink (G_OBJECT(preview));
923 #else
924                 gtk_object_ref (G_OBJECT(preview));
925                 gtk_object_sink (G_OBJECT(preview));
926 #endif
927         } 
928 #if HAVE_LIBCOMPFACE
929 else if (!strcmp2("X-Face", ch->name)) {
930                 GdkColor color;
931                 color.pixel = 0;
932                 preview = GTK_IMAGE(xface_get_from_header(ch->value, 
933                                           &color, 
934                                           mainwindow_get_mainwindow()->window->window));        
935                 pixbuf = gtk_image_get_pixbuf(preview);
936                 gtk_image_set_from_pixbuf (GTK_IMAGE(customhdr.preview), pixbuf);
937                 gtk_widget_show(customhdr.preview);
938 #if GLIB_CHECK_VERSION(2,10,0)
939                 g_object_ref_sink (G_OBJECT(preview));
940 #else
941                 gtk_object_ref (G_OBJECT(preview));
942                 gtk_object_sink (G_OBJECT(preview));
943 #endif
944         } 
945 #endif
946 else {
947                 gtk_widget_hide(customhdr.preview);
948         }
949         return TRUE;
950 }
951
952 #undef ENTRY_SET_TEXT