Fix segfault with invalid bold font.
[claws.git] / src / headerview.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999,2000 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 <gtk/gtkwidget.h>
28 #include <gtk/gtkstyle.h>
29 #include <gtk/gtkscrolledwindow.h>
30 #include <gtk/gtkhbox.h>
31 #include <gtk/gtkvbox.h>
32 #include <gtk/gtklabel.h>
33 #include <gtk/gtkpixmap.h>
34 #include <stdio.h>
35 #include <string.h>
36 #include <time.h>
37
38 #if HAVE_LIBCOMPFACE
39 #  include <compface.h>
40 #endif
41
42 #include "intl.h"
43 #include "main.h"
44 #include "headerview.h"
45 #include "prefs_common.h"
46 #include "gtkutils.h"
47 #include "utils.h"
48
49 static GdkFont *boldfont;
50
51 #define TR(str) (prefs_common.trans_hdr ? gettext(str) : str)
52
53 #if 0
54         _("From:");
55         _("To:");
56         _("Newsgroups:");
57         _("Subject:");
58 #endif
59
60 #if HAVE_LIBCOMPFACE
61 #define XPM_XFACE_HEIGHT        (HEIGHT + 3)  /* 3 = 1 header + 2 colors */
62
63 static gchar *xpm_xface[XPM_XFACE_HEIGHT];
64
65 static void headerview_show_xface       (HeaderView     *headerview,
66                                          MsgInfo        *msginfo);
67 static gint create_xpm_from_xface       (gchar          *xpm[],
68                                          const gchar    *xface);
69 #endif
70
71 HeaderView *headerview_create(void)
72 {
73         HeaderView *headerview;
74         GtkWidget *hbox;
75         GtkWidget *vbox;
76         GtkWidget *hbox1;
77         GtkWidget *hbox2;
78         GtkWidget *from_header_label;
79         GtkWidget *from_body_label;
80         GtkWidget *to_header_label;
81         GtkWidget *to_body_label;
82         GtkWidget *ng_header_label;
83         GtkWidget *ng_body_label;
84         GtkWidget *subject_header_label;
85         GtkWidget *subject_body_label;
86
87         debug_print(_("Creating header view...\n"));
88         headerview = g_new0(HeaderView, 1);
89
90         hbox = gtk_hbox_new(FALSE, 0);
91         gtk_container_set_border_width(GTK_CONTAINER(hbox), 2);
92         vbox = gtk_vbox_new(FALSE, 0);
93         gtk_box_pack_start(GTK_BOX(hbox), vbox, TRUE, TRUE, 0);
94
95         hbox1 = gtk_hbox_new(FALSE, 4);
96         gtk_box_pack_start(GTK_BOX(vbox), hbox1, FALSE, FALSE, 0);
97         hbox2 = gtk_hbox_new(FALSE, 4);
98         gtk_box_pack_start(GTK_BOX(vbox), hbox2, FALSE, FALSE, 0);
99
100         from_header_label    = gtk_label_new(TR("From:"));
101         from_body_label      = gtk_label_new("");
102         to_header_label      = gtk_label_new(TR("To:"));
103         to_body_label        = gtk_label_new("");
104         ng_header_label      = gtk_label_new(TR("Newsgroups:"));
105         ng_body_label        = gtk_label_new("");
106         subject_header_label = gtk_label_new(TR("Subject:"));
107         subject_body_label   = gtk_label_new("");
108
109         gtk_box_pack_start(GTK_BOX(hbox1), from_header_label, FALSE, FALSE, 0);
110         gtk_box_pack_start(GTK_BOX(hbox1), from_body_label, FALSE, FALSE, 0);
111         gtk_box_pack_start(GTK_BOX(hbox1), to_header_label, FALSE, FALSE, 0);
112         gtk_box_pack_start(GTK_BOX(hbox1), to_body_label, FALSE, FALSE, 0);
113         gtk_box_pack_start(GTK_BOX(hbox1), ng_header_label, FALSE, FALSE, 0);
114         gtk_box_pack_start(GTK_BOX(hbox1), ng_body_label, FALSE, FALSE, 0);
115         gtk_box_pack_start(GTK_BOX(hbox2), subject_header_label, FALSE, FALSE, 0);
116         gtk_box_pack_start(GTK_BOX(hbox2), subject_body_label, FALSE, FALSE, 0);
117
118         headerview->hbox = hbox;
119         headerview->from_header_label    = from_header_label;
120         headerview->from_body_label      = from_body_label;
121         headerview->to_header_label      = to_header_label;
122         headerview->to_body_label        = to_body_label;
123         headerview->ng_header_label      = ng_header_label;
124         headerview->ng_body_label        = ng_body_label;
125         headerview->subject_header_label = subject_header_label;
126         headerview->subject_body_label   = subject_body_label;
127         headerview->image = NULL;
128
129         gtk_widget_show_all(hbox);
130
131         return headerview;
132 }
133
134 void headerview_init(HeaderView *headerview)
135 {
136         if (!boldfont)
137                 boldfont = gdk_fontset_load(BOLD_FONT);
138
139 #define SET_FONT_STYLE(wid) \
140 { \
141         GtkStyle *style; \
142  \
143         style = gtk_style_copy(gtk_widget_get_style(headerview->wid)); \
144         if (boldfont) { \
145                 style->font = boldfont; \
146         } \
147         gtk_widget_set_style(headerview->wid, style); \
148 }
149
150         SET_FONT_STYLE(from_header_label);
151         SET_FONT_STYLE(to_header_label);
152         SET_FONT_STYLE(ng_header_label);
153         SET_FONT_STYLE(subject_header_label);
154
155         headerview_clear(headerview);
156         headerview_set_visibility(headerview, prefs_common.display_header_pane);
157
158 #if HAVE_LIBCOMPFACE
159         {
160                 gint i;
161
162                 for (i = 0; i < XPM_XFACE_HEIGHT; i++) {
163                         xpm_xface[i] = g_malloc(WIDTH + 1);
164                         *xpm_xface[i] = '\0';
165                 }
166         }
167 #endif
168 }
169
170 void headerview_show(HeaderView *headerview, MsgInfo *msginfo)
171 {
172         headerview_clear(headerview);
173
174         gtk_label_set_text(GTK_LABEL(headerview->from_body_label),
175                            msginfo->from ? msginfo->from : _("(No From)"));
176         if (msginfo->to) {
177                 gtk_label_set_text(GTK_LABEL(headerview->to_body_label),
178                                    msginfo->to);
179                 gtk_widget_show(headerview->to_header_label);
180                 gtk_widget_show(headerview->to_body_label);
181         }
182         if (msginfo->newsgroups) {
183                 gtk_label_set_text(GTK_LABEL(headerview->ng_body_label),
184                                    msginfo->newsgroups);
185                 gtk_widget_show(headerview->ng_header_label);
186                 gtk_widget_show(headerview->ng_body_label);
187         }
188         gtk_label_set_text(GTK_LABEL(headerview->subject_body_label),
189                            msginfo->subject ? msginfo->subject :
190                            _("(No Subject)"));
191
192 #if HAVE_LIBCOMPFACE
193         headerview_show_xface(headerview, msginfo);
194 #endif
195 }
196
197 #if HAVE_LIBCOMPFACE
198 static void headerview_show_xface(HeaderView *headerview, MsgInfo *msginfo)
199 {
200         gchar xface[2048];
201         GdkPixmap *pixmap;
202         GdkBitmap *mask;
203         GtkWidget *hbox = headerview->hbox;
204
205         if (!msginfo->xface || strlen(msginfo->xface) < 5) {
206                 if (headerview->image &&
207                     GTK_WIDGET_VISIBLE(headerview->image)) {
208                         gtk_widget_hide(headerview->image);
209                         gtk_widget_queue_resize(hbox);
210                 }
211                 return;
212         }
213         if (!GTK_WIDGET_VISIBLE(headerview->hbox)) return;
214
215         strncpy(xface, msginfo->xface, sizeof(xface));
216
217         if (uncompface(xface) < 0) {
218                 g_warning("uncompface failed\n");
219                 if (headerview->image)
220                         gtk_widget_hide(headerview->image);
221                 return;
222         }
223
224         create_xpm_from_xface(xpm_xface, xface);
225
226         pixmap = gdk_pixmap_create_from_xpm_d
227                 (hbox->window, &mask, &hbox->style->white, xpm_xface);
228
229         if (!headerview->image) {
230                 GtkWidget *image;
231
232                 image = gtk_pixmap_new(pixmap, mask);
233                 gtk_box_pack_start(GTK_BOX(hbox), image, FALSE, FALSE, 0);
234                 gtk_widget_show(image);
235                 headerview->image = image;
236         } else {
237                 gtk_pixmap_set(GTK_PIXMAP(headerview->image), pixmap, mask);
238                 gtk_widget_show(headerview->image);
239         }
240
241         gdk_pixmap_unref(pixmap);
242 }
243 #endif
244
245 void headerview_clear(HeaderView *headerview)
246 {
247         gtk_label_set_text(GTK_LABEL(headerview->from_body_label), "");
248         gtk_label_set_text(GTK_LABEL(headerview->to_body_label), "");
249         gtk_label_set_text(GTK_LABEL(headerview->ng_body_label), "");
250         gtk_label_set_text(GTK_LABEL(headerview->subject_body_label), "");
251         gtk_widget_hide(headerview->to_header_label);
252         gtk_widget_hide(headerview->to_body_label);
253         gtk_widget_hide(headerview->ng_header_label);
254         gtk_widget_hide(headerview->ng_body_label);
255
256         if (headerview->image && GTK_WIDGET_VISIBLE(headerview->image)) {
257                 gtk_widget_hide(headerview->image);
258                 gtk_widget_queue_resize(headerview->hbox);
259         }
260 }
261
262 void headerview_set_visibility(HeaderView *headerview, gboolean visibility)
263 {
264         if (visibility)
265                 gtk_widget_show(headerview->hbox);
266         else
267                 gtk_widget_hide(headerview->hbox);
268 }
269
270 void headerview_destroy(HeaderView *headerview)
271 {
272         g_free(headerview);
273 }
274
275 #if HAVE_LIBCOMPFACE
276 static gint create_xpm_from_xface(gchar *xpm[], const gchar *xface)
277 {
278         static gchar *bit_pattern[] = {
279                 "....",
280                 "...#",
281                 "..#.",
282                 "..##",
283                 ".#..",
284                 ".#.#",
285                 ".##.",
286                 ".###",
287                 "#...",
288                 "#..#",
289                 "#.#.",
290                 "#.##",
291                 "##..",
292                 "##.#",
293                 "###.",
294                 "####"
295         };
296
297         static gchar *xface_header = "48 48 2 1";
298         static gchar *xface_black  = "# c #000000";
299         static gchar *xface_white  = ". c #ffffff";
300
301         gint i, line = 0;
302         const guchar *p;
303         gchar buf[WIDTH * 4 + 1];  /* 4 = strlen("0x0000") */
304
305         p = xface;
306
307         strcpy(xpm[line++], xface_header);
308         strcpy(xpm[line++], xface_black);
309         strcpy(xpm[line++], xface_white);
310
311         for (i = 0; i < HEIGHT; i++) {
312                 gint col;
313
314                 buf[0] = '\0';
315      
316                 for (col = 0; col < 3; col++) {
317                         gint figure;
318
319                         p += 2;  /* skip '0x' */
320
321                         for (figure = 0; figure < 4; figure++) {
322                                 gint n = 0;
323
324                                 if ('0' <= *p && *p <= '9') {
325                                         n = *p - '0';
326                                 } else if ('a' <= *p && *p <= 'f') {
327                                         n = *p - 'a' + 10;
328                                 } else if ('A' <= *p && *p <= 'F') {
329                                         n = *p - 'A' + 10;
330                                 }
331
332                                 strcat(buf, bit_pattern[n]);
333                                 p++;  /* skip ',' */
334                         }
335
336                         p++;  /* skip '\n' */
337                 }
338
339                 strcpy(xpm[line++], buf);
340                 p++;
341         }
342
343         return 0;
344 }
345 #endif