2007-10-09 [colin] 3.0.2cvs45
[claws.git] / src / headerview.c
index 8a6853b5fa7f45ed01e435beea20e77d15169cf4..4d8e2a1d36fdcb721b0d25bcdf9e03278859680b 100644 (file)
@@ -1,10 +1,10 @@
 /*
  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
- * Copyright (C) 1999-2001 Hiroyuki Yamamoto
+ * Copyright (C) 1999-2007 Hiroyuki Yamamoto
  *
  * This program is free software; you can redistribute it and/or modify
  * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
+ * the Free Software Foundation; either version 3 of the License, or
  * (at your option) any later version.
  *
  * This program is distributed in the hope that it will be useful,
@@ -13,8 +13,8 @@
  * GNU General Public License for more details.
  *
  * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ * 
  */
 
 #ifdef HAVE_CONFIG_H
 #include "defs.h"
 
 #include <glib.h>
+#include <glib/gi18n.h>
 #include <gtk/gtkwidget.h>
 #include <gtk/gtkstyle.h>
 #include <gtk/gtkscrolledwindow.h>
 #include <gtk/gtkhbox.h>
 #include <gtk/gtkvbox.h>
 #include <gtk/gtklabel.h>
-#include <gtk/gtkpixmap.h>
+#include <gtk/gtkimage.h>
 #include <stdio.h>
 #include <string.h>
 #include <time.h>
 #  include <compface.h>
 #endif
 
-#include "intl.h"
+#include <gdk-pixbuf/gdk-pixbuf.h>
+
 #include "headerview.h"
 #include "prefs_common.h"
+#include "codeconv.h"
 #include "gtkutils.h"
 #include "utils.h"
-
-static GdkFont *boldfont;
-
-#define TR(str)        (prefs_common.trans_hdr ? gettext(str) : str)
-
-#if 0
-       _("From:");
-       _("To:");
-       _("Newsgroups:");
-       _("Subject:");
-#endif
+#include "base64.h"
+#include "headers.h"
+#include "addrindex.h"
 
 #if HAVE_LIBCOMPFACE
 #define XPM_XFACE_HEIGHT       (HEIGHT + 3)  /* 3 = 1 header + 2 colors */
 
 static gchar *xpm_xface[XPM_XFACE_HEIGHT];
 
-static void headerview_show_xface      (HeaderView     *headerview,
+static gint headerview_show_xface      (HeaderView     *headerview,
                                         MsgInfo        *msginfo);
-static gint create_xpm_from_xface      (gchar          *xpm[],
-                                        const gchar    *xface);
 #endif
 
+static gint headerview_show_face       (HeaderView     *headerview,
+                                        MsgInfo        *msginfo);
+static gint headerview_show_contact_pic        (HeaderView     *headerview,
+                                        MsgInfo        *msginfo);
+static void headerview_save_contact_pic        (HeaderView     *headerview,
+                                        MsgInfo        *msginfo);
+
 HeaderView *headerview_create(void)
 {
        HeaderView *headerview;
@@ -74,6 +74,7 @@ HeaderView *headerview_create(void)
        GtkWidget *vbox;
        GtkWidget *hbox1;
        GtkWidget *hbox2;
+       GtkWidget *hbox3;
        GtkWidget *from_header_label;
        GtkWidget *from_body_label;
        GtkWidget *to_header_label;
@@ -82,37 +83,66 @@ HeaderView *headerview_create(void)
        GtkWidget *ng_body_label;
        GtkWidget *subject_header_label;
        GtkWidget *subject_body_label;
+       GtkWidget *tags_header_label;
+       GtkWidget *tags_body_label;
 
        debug_print("Creating header view...\n");
        headerview = g_new0(HeaderView, 1);
 
        hbox = gtk_hbox_new(FALSE, 0);
        gtk_container_set_border_width(GTK_CONTAINER(hbox), 2);
-       vbox = gtk_vbox_new(FALSE, 0);
+       vbox = gtk_vbox_new(FALSE, 2);
        gtk_box_pack_start(GTK_BOX(hbox), vbox, TRUE, TRUE, 0);
 
        hbox1 = gtk_hbox_new(FALSE, 4);
        gtk_box_pack_start(GTK_BOX(vbox), hbox1, FALSE, FALSE, 0);
        hbox2 = gtk_hbox_new(FALSE, 4);
        gtk_box_pack_start(GTK_BOX(vbox), hbox2, FALSE, FALSE, 0);
+       hbox3 = gtk_hbox_new(FALSE, 4);
+       gtk_box_pack_start(GTK_BOX(vbox), hbox3, FALSE, FALSE, 0);
 
-       from_header_label    = gtk_label_new(TR("From:"));
+       from_header_label    = gtk_label_new(prefs_common_translated_header_name("From:"));
        from_body_label      = gtk_label_new("");
-       to_header_label      = gtk_label_new(TR("To:"));
+       to_header_label      = gtk_label_new(prefs_common_translated_header_name("To:"));
        to_body_label        = gtk_label_new("");
-       ng_header_label      = gtk_label_new(TR("Newsgroups:"));
+       ng_header_label      = gtk_label_new(prefs_common_translated_header_name("Newsgroups:"));
        ng_body_label        = gtk_label_new("");
-       subject_header_label = gtk_label_new(TR("Subject:"));
+       subject_header_label = gtk_label_new(prefs_common_translated_header_name("Subject:"));
        subject_body_label   = gtk_label_new("");
+       tags_header_label = gtk_label_new(_("Tags:"));
+       tags_body_label   = gtk_label_new("");
+
+       gtk_label_set_selectable(GTK_LABEL(from_body_label), TRUE);
+       gtk_label_set_selectable(GTK_LABEL(to_body_label), TRUE);
+       gtk_label_set_selectable(GTK_LABEL(ng_body_label), TRUE);
+       gtk_label_set_selectable(GTK_LABEL(subject_body_label), TRUE);
+       gtk_label_set_selectable(GTK_LABEL(tags_body_label), TRUE);
+
+       GTK_WIDGET_UNSET_FLAGS(from_body_label, GTK_CAN_FOCUS);
+       GTK_WIDGET_UNSET_FLAGS(to_body_label, GTK_CAN_FOCUS);
+       GTK_WIDGET_UNSET_FLAGS(ng_body_label, GTK_CAN_FOCUS);
+       GTK_WIDGET_UNSET_FLAGS(subject_body_label, GTK_CAN_FOCUS);
+       GTK_WIDGET_UNSET_FLAGS(tags_body_label, GTK_CAN_FOCUS);
 
        gtk_box_pack_start(GTK_BOX(hbox1), from_header_label, FALSE, FALSE, 0);
        gtk_box_pack_start(GTK_BOX(hbox1), from_body_label, FALSE, FALSE, 0);
        gtk_box_pack_start(GTK_BOX(hbox1), to_header_label, FALSE, FALSE, 0);
-       gtk_box_pack_start(GTK_BOX(hbox1), to_body_label, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(hbox1), to_body_label, TRUE, TRUE, 0);
        gtk_box_pack_start(GTK_BOX(hbox1), ng_header_label, FALSE, FALSE, 0);
-       gtk_box_pack_start(GTK_BOX(hbox1), ng_body_label, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(hbox1), ng_body_label, TRUE, TRUE, 0);
        gtk_box_pack_start(GTK_BOX(hbox2), subject_header_label, FALSE, FALSE, 0);
-       gtk_box_pack_start(GTK_BOX(hbox2), subject_body_label, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(hbox2), subject_body_label, TRUE, TRUE, 0);
+       gtk_box_pack_start(GTK_BOX(hbox3), tags_header_label, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(hbox3), tags_body_label, TRUE, TRUE, 0);
+
+       gtk_misc_set_alignment(GTK_MISC(to_body_label), 0, 0.5);
+       gtk_misc_set_alignment(GTK_MISC(ng_body_label), 0, 0.5);
+       gtk_misc_set_alignment(GTK_MISC(subject_body_label), 0, 0.5);
+       gtk_misc_set_alignment(GTK_MISC(tags_body_label), 0, 0.5);
+       gtk_label_set_ellipsize(GTK_LABEL(to_body_label), PANGO_ELLIPSIZE_END);
+       gtk_label_set_ellipsize(GTK_LABEL(ng_body_label), PANGO_ELLIPSIZE_END);
+       gtk_label_set_ellipsize(GTK_LABEL(subject_body_label), PANGO_ELLIPSIZE_END);
+       gtk_label_set_ellipsize(GTK_LABEL(tags_body_label), PANGO_ELLIPSIZE_END);
 
        headerview->hbox = hbox;
        headerview->from_header_label    = from_header_label;
@@ -123,6 +153,8 @@ HeaderView *headerview_create(void)
        headerview->ng_body_label        = ng_body_label;
        headerview->subject_header_label = subject_header_label;
        headerview->subject_body_label   = subject_body_label;
+       headerview->tags_header_label = tags_header_label;
+       headerview->tags_body_label   = tags_body_label;
        headerview->image = NULL;
 
        gtk_widget_show_all(hbox);
@@ -130,26 +162,37 @@ HeaderView *headerview_create(void)
        return headerview;
 }
 
-void headerview_init(HeaderView *headerview)
+void headerview_set_font(HeaderView *headerview)
 {
-       if (!boldfont)
-               boldfont = gtkut_font_load(BOLD_FONT);
-
-#define SET_FONT_STYLE(wid) \
-{ \
-       GtkStyle *style; \
- \
-       style = gtk_style_copy(gtk_widget_get_style(headerview->wid)); \
-       if (boldfont) \
-               style->font = boldfont; \
-       gtk_widget_set_style(headerview->wid, style); \
-}
+       PangoFontDescription *boldfont = NULL;
+       PangoFontDescription *normalfont = NULL;
+       
+       if (!boldfont) {
+               normalfont = pango_font_description_from_string(NORMAL_FONT);
+               boldfont = pango_font_description_from_string(NORMAL_FONT);
+               pango_font_description_set_weight(boldfont, PANGO_WEIGHT_BOLD);
+       }
 
-       SET_FONT_STYLE(from_header_label);
-       SET_FONT_STYLE(to_header_label);
-       SET_FONT_STYLE(ng_header_label);
-       SET_FONT_STYLE(subject_header_label);
+       if (boldfont) {
+               gtk_widget_modify_font(headerview->from_header_label, boldfont);
+               gtk_widget_modify_font(headerview->to_header_label, boldfont);
+               gtk_widget_modify_font(headerview->ng_header_label, boldfont);
+               gtk_widget_modify_font(headerview->subject_header_label, boldfont);
+               gtk_widget_modify_font(headerview->tags_header_label, boldfont);
+               pango_font_description_free(boldfont);
+
+               gtk_widget_modify_font(headerview->from_body_label, normalfont);
+               gtk_widget_modify_font(headerview->to_body_label, normalfont);
+               gtk_widget_modify_font(headerview->ng_body_label, normalfont);
+               gtk_widget_modify_font(headerview->subject_body_label, normalfont);
+               gtk_widget_modify_font(headerview->tags_body_label, normalfont);
+               pango_font_description_free(normalfont);
+       }
+}
 
+void headerview_init(HeaderView *headerview)
+{
+       headerview_set_font(headerview);
        headerview_clear(headerview);
        headerview_set_visibility(headerview, prefs_common.display_header_pane);
 
@@ -167,6 +210,8 @@ void headerview_init(HeaderView *headerview)
 
 void headerview_show(HeaderView *headerview, MsgInfo *msginfo)
 {
+       gchar *tags = procmsg_msginfo_get_tags_str(msginfo);
+
        headerview_clear(headerview);
 
        gtk_label_set_text(GTK_LABEL(headerview->from_body_label),
@@ -186,70 +231,195 @@ void headerview_show(HeaderView *headerview, MsgInfo *msginfo)
        gtk_label_set_text(GTK_LABEL(headerview->subject_body_label),
                           msginfo->subject ? msginfo->subject :
                           _("(No Subject)"));
+       if (tags) {
+               gtk_label_set_text(GTK_LABEL(headerview->tags_body_label),
+                                  tags);
+               gtk_widget_show(headerview->tags_header_label);
+               gtk_widget_show(headerview->tags_body_label);
+               g_free(tags);
+       }
+       if (!headerview_show_face(headerview, msginfo))
+               return;
 
 #if HAVE_LIBCOMPFACE
-       headerview_show_xface(headerview, msginfo);
+       if (!headerview_show_xface(headerview, msginfo))
+               return;
 #endif
+
+       if (!headerview_show_contact_pic(headerview, msginfo))
+               return;
+
 }
 
 #if HAVE_LIBCOMPFACE
-static void headerview_show_xface(HeaderView *headerview, MsgInfo *msginfo)
+static gint headerview_show_xface(HeaderView *headerview, MsgInfo *msginfo)
 {
-       gchar xface[2048];
-       GdkPixmap *pixmap;
-       GdkBitmap *mask;
        GtkWidget *hbox = headerview->hbox;
+       GtkWidget *image;
 
-       if (!msginfo->xface || strlen(msginfo->xface) < 5) {
+       if (!msginfo->extradata || 
+           !msginfo->extradata->xface || 
+           strlen(msginfo->extradata->xface) < 5) {
                if (headerview->image &&
                    GTK_WIDGET_VISIBLE(headerview->image)) {
                        gtk_widget_hide(headerview->image);
                        gtk_widget_queue_resize(hbox);
                }
-               return;
+               return -1;
        }
-       if (!GTK_WIDGET_VISIBLE(headerview->hbox)) return;
+       if (!GTK_WIDGET_VISIBLE(headerview->hbox)) return -1;
+
+       if (headerview->image) {
+               gtk_widget_destroy(headerview->image);
+               headerview->image = NULL;
+       }
+       
+
+       image = xface_get_from_header(msginfo->extradata->xface, &hbox->style->white,
+                               hbox->window);
+
+       if (image) {
+               gtk_box_pack_start(GTK_BOX(hbox), image, FALSE, FALSE, 0);
+               gtk_widget_show(image);
+       }
+
+       headerview->image = image;
+       if (image) {
+               headerview_save_contact_pic(headerview, msginfo);
+       }
+       return 0;
+}
+#endif
 
-       strncpy(xface, msginfo->xface, sizeof(xface));
+static gint headerview_show_face (HeaderView *headerview, MsgInfo *msginfo)
+{
+       GtkWidget *hbox = headerview->hbox;
+       GtkWidget *image;
 
-       if (uncompface(xface) < 0) {
-               g_warning("uncompface failed\n");
-               if (headerview->image)
+       if (!msginfo->extradata || !msginfo->extradata->face) {
+               if (headerview->image &&
+                   GTK_WIDGET_VISIBLE(headerview->image)) {
                        gtk_widget_hide(headerview->image);
-               return;
+                       gtk_widget_queue_resize(hbox);
+               }
+               return -1;
        }
+       if (!GTK_WIDGET_VISIBLE(headerview->hbox)) return -1;
 
-       create_xpm_from_xface(xpm_xface, xface);
+       if (headerview->image) {
+               gtk_widget_destroy(headerview->image);
+               headerview->image = NULL;
+       }
+       
 
-       pixmap = gdk_pixmap_create_from_xpm_d
-               (hbox->window, &mask, &hbox->style->white, xpm_xface);
+       image = face_get_from_header(msginfo->extradata->face);
 
-       if (!headerview->image) {
-               GtkWidget *image;
+       if (image) {
+               gtk_box_pack_start(GTK_BOX(hbox), image, FALSE, FALSE, 0);
+               gtk_widget_show(image);
+       }
 
-               image = gtk_pixmap_new(pixmap, mask);
+       headerview->image = image;
+       if (image == NULL)
+               return -1;
+       else {
+               headerview_save_contact_pic(headerview, msginfo);
+               return 0;
+       }
+}
+
+static void headerview_save_contact_pic (HeaderView *headerview, MsgInfo *msginfo)
+{
+       gchar *filename = NULL;
+       GError *error = NULL;
+       GdkPixbuf *picture = NULL;
+
+       if (!GTK_WIDGET_VISIBLE(headerview->hbox)) return;
+
+       if (headerview->image) {
+               picture = gtk_image_get_pixbuf(GTK_IMAGE(headerview->image));
+       }
+       
+       filename = addrindex_get_picture_file(msginfo->from);
+       if (!filename)
+               return;
+       if (!is_file_exist(filename))
+               gdk_pixbuf_save(picture, filename, "png", &error, NULL);
+       g_free(filename);
+}      
+
+static gint headerview_show_contact_pic (HeaderView *headerview, MsgInfo *msginfo)
+{
+       GtkWidget *hbox = headerview->hbox;
+       GtkWidget *image;
+       gchar *filename = NULL;
+       GError *error = NULL;
+       GdkPixbuf *picture = NULL;
+       gint w, h;
+
+       if (!GTK_WIDGET_VISIBLE(headerview->hbox)) return -1;
+
+       if (headerview->image) {
+               gtk_widget_destroy(headerview->image);
+               headerview->image = NULL;
+       }
+       
+       filename = addrindex_get_picture_file(msginfo->from);
+       
+       if (!filename)
+               return -1;
+       if (!is_file_exist(filename)) {
+               g_free(filename);
+               return -1;
+       }
+       gdk_pixbuf_get_file_info(filename, &w, &h);
+       
+       if (w > 48 || h > 48)
+               picture = gdk_pixbuf_new_from_file_at_scale(filename, 
+                                               48, 48, TRUE, &error);
+       else
+               picture = gdk_pixbuf_new_from_file(filename, &error);
+
+       g_free(filename);
+       if (error) {
+               debug_print("Failed to import image: \n%s",
+                               error->message);
+               g_error_free(error);
+               return -1;
+       }
+       if (picture)
+               image = gtk_image_new_from_pixbuf(picture);
+       else 
+               return -1;
+
+       if (image) {
                gtk_box_pack_start(GTK_BOX(hbox), image, FALSE, FALSE, 0);
                gtk_widget_show(image);
-               headerview->image = image;
-       } else {
-               gtk_pixmap_set(GTK_PIXMAP(headerview->image), pixmap, mask);
-               gtk_widget_show(headerview->image);
        }
 
-       gdk_pixmap_unref(pixmap);
+       headerview->image = image;
+       if (image == NULL)
+               return -1;
+       else 
+               return 0;
 }
-#endif
 
 void headerview_clear(HeaderView *headerview)
 {
+       if (headerview == NULL)
+               return;
+
        gtk_label_set_text(GTK_LABEL(headerview->from_body_label), "");
        gtk_label_set_text(GTK_LABEL(headerview->to_body_label), "");
        gtk_label_set_text(GTK_LABEL(headerview->ng_body_label), "");
        gtk_label_set_text(GTK_LABEL(headerview->subject_body_label), "");
+       gtk_label_set_text(GTK_LABEL(headerview->tags_body_label), "");
        gtk_widget_hide(headerview->to_header_label);
        gtk_widget_hide(headerview->to_body_label);
        gtk_widget_hide(headerview->ng_header_label);
        gtk_widget_hide(headerview->ng_body_label);
+       gtk_widget_hide(headerview->tags_header_label);
+       gtk_widget_hide(headerview->tags_body_label);
 
        if (headerview->image && GTK_WIDGET_VISIBLE(headerview->image)) {
                gtk_widget_hide(headerview->image);
@@ -269,75 +439,3 @@ void headerview_destroy(HeaderView *headerview)
 {
        g_free(headerview);
 }
-
-#if HAVE_LIBCOMPFACE
-static gint create_xpm_from_xface(gchar *xpm[], const gchar *xface)
-{
-       static gchar *bit_pattern[] = {
-               "....",
-               "...#",
-               "..#.",
-               "..##",
-               ".#..",
-               ".#.#",
-               ".##.",
-               ".###",
-               "#...",
-               "#..#",
-               "#.#.",
-               "#.##",
-               "##..",
-               "##.#",
-               "###.",
-               "####"
-       };
-
-       static gchar *xface_header = "48 48 2 1";
-       static gchar *xface_black  = "# c #000000";
-       static gchar *xface_white  = ". c #ffffff";
-
-       gint i, line = 0;
-       const guchar *p;
-       gchar buf[WIDTH * 4 + 1];  /* 4 = strlen("0x0000") */
-
-       p = xface;
-
-       strcpy(xpm[line++], xface_header);
-       strcpy(xpm[line++], xface_black);
-       strcpy(xpm[line++], xface_white);
-
-       for (i = 0; i < HEIGHT; i++) {
-               gint col;
-
-               buf[0] = '\0';
-     
-               for (col = 0; col < 3; col++) {
-                       gint figure;
-
-                       p += 2;  /* skip '0x' */
-
-                       for (figure = 0; figure < 4; figure++) {
-                               gint n = 0;
-
-                               if ('0' <= *p && *p <= '9') {
-                                       n = *p - '0';
-                               } else if ('a' <= *p && *p <= 'f') {
-                                       n = *p - 'a' + 10;
-                               } else if ('A' <= *p && *p <= 'F') {
-                                       n = *p - 'A' + 10;
-                               }
-
-                               strcat(buf, bit_pattern[n]);
-                               p++;  /* skip ',' */
-                       }
-
-                       p++;  /* skip '\n' */
-               }
-
-               strcpy(xpm[line++], buf);
-               p++;
-       }
-
-       return 0;
-}
-#endif