diff options
author | Not Zed <NotZed@Ximian.com> | 2003-09-18 05:19:04 +0800 |
---|---|---|
committer | Michael Zucci <zucchi@src.gnome.org> | 2003-09-18 05:19:04 +0800 |
commit | 81a0ff5bc44a3bd11399e6b3c985735737606c8c (patch) | |
tree | 54b5ed4342a6843c1db4c7e75f2e1b1fe9b82dff /mail/em-folder-view.c | |
parent | a36a1bb70b6ebcb51ac39304370c89bda63e11b9 (diff) | |
download | gsoc2013-evolution-81a0ff5bc44a3bd11399e6b3c985735737606c8c.tar.gz gsoc2013-evolution-81a0ff5bc44a3bd11399e6b3c985735737606c8c.tar.zst gsoc2013-evolution-81a0ff5bc44a3bd11399e6b3c985735737606c8c.zip |
cvs removed.
2003-09-17 Not Zed <NotZed@Ximian.com>
* folder-browser.c, folder-browser.h, folder-browser-ui.c
folder-browser-ui.h, mail-callbacks.c, mail-callbacks.h
mail-display.c, mail-display.h, mail-display-stream.c
mail-display-stream.h, mail-format.c, mail-format.h
mail-identify.c, mail-search.c, mail-search.h
message-browser.c, message-browser.h, subscribe-dialog.c
subscribe-dialog.h, mail-font-prefs.c, mail-font-prefs.h: cvs
removed.
* Makefile.am: Removed mail-font-prefs.[ch], hasn't been built for
ages.
* em-*.c: killed a bunch of printfs.
* em-format-html-display.c (efhd_html_button_press_event): update
for html object api chagnes.
** Merge in mail-refactor-2 branch.
svn path=/trunk/; revision=22602
Diffstat (limited to 'mail/em-folder-view.c')
-rw-r--r-- | mail/em-folder-view.c | 1866 |
1 files changed, 1866 insertions, 0 deletions
diff --git a/mail/em-folder-view.c b/mail/em-folder-view.c new file mode 100644 index 0000000000..7122f3da36 --- /dev/null +++ b/mail/em-folder-view.c @@ -0,0 +1,1866 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ +/* + * Authors: Michael Zucchi <notzed@ximian.com> + * + * Copyright 2003 Ximian, Inc. (www.ximian.com) + * + * 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 + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * 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 Street #330, Boston, MA 02111-1307, USA. + * + */ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include <string.h> + +#include <gtk/gtkvbox.h> +#include <gtk/gtkbutton.h> +#include <gtk/gtkvpaned.h> +#include <gdk/gdkkeysyms.h> + +#include <gtkhtml/gtkhtml.h> + +#include <libgnome/gnome-url.h> + +#include <libgnomeprintui/gnome-print-dialog.h> + +#include <camel/camel-mime-message.h> +#include <camel/camel-stream.h> +#include <camel/camel-stream-filter.h> +#include <camel/camel-mime-filter.h> +#include <camel/camel-mime-filter-tohtml.h> +#include <camel/camel-mime-filter-enriched.h> +#include <camel/camel-multipart.h> +#include <camel/camel-stream-mem.h> +#include <camel/camel-url.h> + +#include <bonobo/bonobo-main.h> +#include <bonobo/bonobo-object.h> +#include <bonobo/bonobo-generic-factory.h> +#include <bonobo/bonobo-control.h> +#include <bonobo/bonobo-ui-component.h> +#include <bonobo/bonobo-ui-util.h> + +#include "widgets/misc/e-charset-picker.h" + +#include <e-util/e-dialog-utils.h> + +#include "em-format-html-display.h" +#include "em-format-html-print.h" +#include "em-folder-view.h" +#include "em-message-browser.h" +#include "message-list.h" +#include "em-utils.h" + +#include <gtkhtml/gtkhtml.h> +#include <gtkhtml/htmlobject.h> +#include <gtkhtml/htmlengine.h> +#include <gtkhtml/htmlengine-save.h> + +#include "mail-mt.h" +#include "mail-ops.h" +#include "mail-config.h" /* hrm, pity we need this ... */ +#include "mail-autofilter.h" +#include "mail-vfolder.h" + +#include "evolution-shell-component-utils.h" /* Pixmap stuff, sigh */ + +static void emfv_folder_changed(CamelFolder *folder, CamelFolderChangeInfo *changes, EMFolderView *emfv); + +static void emfv_list_message_selected(MessageList *ml, const char *uid, EMFolderView *emfv); +static int emfv_list_right_click(ETree *tree, gint row, ETreePath path, gint col, GdkEvent *event, EMFolderView *emfv); +static void emfv_list_double_click(ETree *tree, gint row, ETreePath path, gint col, GdkEvent *event, EMFolderView *emfv); +static int emfv_list_key_press(ETree *tree, int row, ETreePath path, int col, GdkEvent *ev, EMFolderView *emfv); + +static void emfv_format_link_clicked(EMFormatHTMLDisplay *efhd, const char *uri, EMFolderView *); +static int emfv_format_popup_event(EMFormatHTMLDisplay *efhd, GdkEventButton *event, const char *uri, CamelMimePart *part, EMFolderView *); + +static void emfv_enable_menus(EMFolderView *emfv); + +static void emfv_set_folder(EMFolderView *emfv, CamelFolder *folder, const char *uri); +static void emfv_set_folder_uri(EMFolderView *emfv, const char *uri); +static void emfv_set_message(EMFolderView *emfv, const char *uid); +static void emfv_activate(EMFolderView *emfv, BonoboUIComponent *uic, int state); + +static void emfv_message_reply(EMFolderView *emfv, int mode); +static void vfolder_type_current (EMFolderView *emfv, int type); +static void filter_type_current (EMFolderView *emfv, int type); + +static void emfv_setting_setup(EMFolderView *emfv); + +static const EMFolderViewEnable emfv_enable_map[]; + +struct _EMFolderViewPrivate { + guint seen_id; + guint setting_notify_id; + + CamelObjectHookID folder_changed_id; + + GtkWidget *invisible; + char *selection_uri; +}; + +static GtkVBoxClass *emfv_parent; + +static void emfv_selection_get(GtkWidget *widget, GtkSelectionData *data, guint info, guint time_stamp, EMFolderView *emfv); +static void emfv_selection_clear_event(GtkWidget *widget, GdkEventSelection *event, EMFolderView *emfv); + +static void +emfv_init(GObject *o) +{ + EMFolderView *emfv = (EMFolderView *)o; + struct _EMFolderViewPrivate *p; + + gtk_box_set_homogeneous (GTK_BOX (emfv), FALSE); + + p = emfv->priv = g_malloc0(sizeof(struct _EMFolderViewPrivate)); + + emfv->ui_files = g_slist_append(NULL, EVOLUTION_UIDIR "/evolution-mail-message.xml"); + emfv->ui_app_name = "evolution-mail"; + + emfv->enable_map = g_slist_prepend(NULL, (void *)emfv_enable_map); + + emfv->list = (MessageList *)message_list_new(); + g_signal_connect(emfv->list, "message_selected", G_CALLBACK(emfv_list_message_selected), emfv); + + /* FIXME: should this hang off message-list instead? */ + g_signal_connect(emfv->list->tree, "right_click", G_CALLBACK(emfv_list_right_click), emfv); + g_signal_connect(emfv->list->tree, "double_click", G_CALLBACK(emfv_list_double_click), emfv); + g_signal_connect(emfv->list->tree, "key_press", G_CALLBACK(emfv_list_key_press), emfv); + + emfv->preview = (EMFormatHTMLDisplay *)em_format_html_display_new(); + g_signal_connect(emfv->preview, "link_clicked", G_CALLBACK(emfv_format_link_clicked), emfv); + g_signal_connect(emfv->preview, "popup_event", G_CALLBACK(emfv_format_popup_event), emfv); + + p->invisible = gtk_invisible_new(); + g_object_ref(p->invisible); + gtk_object_sink((GtkObject *)p->invisible); + g_signal_connect(p->invisible, "selection_get", G_CALLBACK(emfv_selection_get), emfv); + g_signal_connect(p->invisible, "selection_clear_event", G_CALLBACK(emfv_selection_clear_event), emfv); + gtk_selection_add_target(p->invisible, GDK_SELECTION_PRIMARY, GDK_SELECTION_TYPE_STRING, 0); + gtk_selection_add_target(p->invisible, GDK_SELECTION_PRIMARY, GDK_SELECTION_TYPE_STRING, 1); + + emfv->async = mail_async_event_new(); + + emfv_setting_setup(emfv); +} + +static void +emfv_finalise(GObject *o) +{ + EMFolderView *emfv = (EMFolderView *)o; + struct _EMFolderViewPrivate *p = emfv->priv; + + if (emfv->async) + mail_async_event_destroy(emfv->async); + + if (emfv->folder) { + if (p->folder_changed_id) + camel_object_remove_event(emfv->folder, p->folder_changed_id); + camel_object_unref(emfv->folder); + g_free(emfv->folder_uri); + } + + g_slist_free(emfv->ui_files); + g_slist_free(emfv->enable_map); + + g_free(p); + + ((GObjectClass *)emfv_parent)->finalize(o); +} + +static void +emfv_destroy (GtkObject *o) +{ + EMFolderView *emfv = (EMFolderView *) o; + struct _EMFolderViewPrivate *p = emfv->priv; + + if (p->seen_id) { + g_source_remove(p->seen_id); + p->seen_id = 0; + } + + if (p->setting_notify_id) { + GConfClient *gconf = gconf_client_get_default(); + + gconf_client_notify_remove(gconf, p->setting_notify_id); + p->setting_notify_id = 0; + g_object_unref(gconf); + } + + if (p->invisible) { + g_object_unref(p->invisible); + p->invisible = NULL; + } + + emfv->preview = NULL; + emfv->list = NULL; + emfv->preview_active = FALSE; + emfv->uic = NULL; + + ((GtkObjectClass *) emfv_parent)->destroy (o); +} + +static void +emfv_class_init(GObjectClass *klass) +{ + klass->finalize = emfv_finalise; + + ((GtkObjectClass *) klass)->destroy = emfv_destroy; + + ((EMFolderViewClass *)klass)->set_folder = emfv_set_folder; + ((EMFolderViewClass *)klass)->set_folder_uri = emfv_set_folder_uri; + ((EMFolderViewClass *)klass)->set_message = emfv_set_message; + ((EMFolderViewClass *)klass)->activate = emfv_activate; +} + +GType +em_folder_view_get_type(void) +{ + static GType type = 0; + + if (type == 0) { + static const GTypeInfo info = { + sizeof(EMFolderViewClass), + NULL, NULL, + (GClassInitFunc)emfv_class_init, + NULL, NULL, + sizeof(EMFolderView), 0, + (GInstanceInitFunc)emfv_init + }; + emfv_parent = g_type_class_ref(gtk_vbox_get_type()); + type = g_type_register_static(gtk_vbox_get_type(), "EMFolderView", &info, 0); + } + + return type; +} + +GtkWidget *em_folder_view_new(void) +{ + EMFolderView *emfv = g_object_new(em_folder_view_get_type(), 0); + + return (GtkWidget *)emfv; +} + +/* flag all selected messages. Return number flagged */ +/* FIXME: Should this be part of message-list instead? */ +int +em_folder_view_mark_selected(EMFolderView *emfv, guint32 mask, guint32 set) +{ + GPtrArray *uids; + int i; + + if (emfv->folder == NULL) + return 0; + + uids = message_list_get_selected(emfv->list); + camel_folder_freeze(emfv->folder); + + for (i=0; i<uids->len; i++) + camel_folder_set_message_flags(emfv->folder, uids->pdata[i], mask, set); + + message_list_free_uids(emfv->list, uids); + camel_folder_thaw(emfv->folder); + + return i; +} + +/* should this be elsewhere/take a uid list? */ +int +em_folder_view_open_selected(EMFolderView *emfv) +{ + GPtrArray *uids; + int i = 0; + + /* FIXME: handle editing message? Should be a different method? editing handled by 'Resend' method already */ + + uids = message_list_get_selected(emfv->list); + + if (em_utils_folder_is_drafts(emfv->folder, emfv->folder_uri) + || em_utils_folder_is_outbox(emfv->folder, emfv->folder_uri)) { + em_utils_edit_messages((GtkWidget *)emfv, emfv->folder, uids); + } else { + /* TODO: have an em_utils_open_messages call? */ + + /* FIXME: 'are you sure' for > 10 messages; is this even necessary? */ + + for (i=0; i<uids->len; i++) { + EMMessageBrowser *emmb; + + emmb = (EMMessageBrowser *)em_message_browser_window_new(); + /* FIXME: session needs to be passed easier than this */ + em_format_set_session((EMFormat *)((EMFolderView *)emmb)->preview, ((EMFormat *)emfv->preview)->session); + em_folder_view_set_folder((EMFolderView *)emmb, emfv->folder, emfv->folder_uri); + em_folder_view_set_message((EMFolderView *)emmb, uids->pdata[i]); + gtk_widget_show(emmb->window); + } + + message_list_free_uids(emfv->list, uids); + } + + return i; +} + +/* ********************************************************************** */ + +static void +emfv_set_folder(EMFolderView *emfv, CamelFolder *folder, const char *uri) +{ + int isout = (folder && uri + && (em_utils_folder_is_drafts(folder, uri) + || em_utils_folder_is_sent(folder, uri) + || em_utils_folder_is_outbox(folder, uri))); + + message_list_set_folder(emfv->list, folder, uri, isout); + g_free(emfv->folder_uri); + emfv->folder_uri = g_strdup(uri); + if (folder != emfv->folder) { + if (emfv->folder) { + if (emfv->priv->folder_changed_id) + camel_object_remove_event(emfv->folder, emfv->priv->folder_changed_id); + camel_object_unref(emfv->folder); + } + emfv->folder = folder; + if (folder) { + emfv->priv->folder_changed_id = camel_object_hook_event(folder, "folder_changed", + (CamelObjectEventHookFunc)emfv_folder_changed, emfv); + camel_object_ref(folder); + } + } + + emfv_enable_menus(emfv); +} + +static void +emfv_got_folder(char *uri, CamelFolder *folder, void *data) +{ + EMFolderView *emfv = data; + + em_folder_view_set_folder(emfv, folder, uri); +} + +static void +emfv_set_folder_uri(EMFolderView *emfv, const char *uri) +{ + if (emfv->preview) + em_format_format((EMFormat *)emfv->preview, NULL); + + mail_get_folder(uri, 0, emfv_got_folder, emfv, mail_thread_new); +} + +static void +emfv_set_message(EMFolderView *emfv, const char *uid) +{ + message_list_select_uid(emfv->list, uid); +} + +/* ********************************************************************** */ + +static void +emfv_selection_get(GtkWidget *widget, GtkSelectionData *data, guint info, guint time_stamp, EMFolderView *emfv) +{ + struct _EMFolderViewPrivate *p = emfv->priv; + + if (p->selection_uri == NULL) + return; + + gtk_selection_data_set(data, data->target, 8, p->selection_uri, strlen(p->selection_uri)); +} + +static void +emfv_selection_clear_event(GtkWidget *widget, GdkEventSelection *event, EMFolderView *emfv) +{ +#if 0 /* do i care? */ + struct _EMFolderViewPrivate *p = emfv->priv; + + g_free(p->selection_uri); + p->selection_uri = NULL; +#endif +} + +/* ********************************************************************** */ + +/* Popup menu + In many cases these are the functions called by the bonobo callbacks too */ + +struct _emfv_label_item { + EMPopupItem item; + + EMFolderView *emfv; + const char *label; +}; + +static void +emfv_popup_open(GtkWidget *w, EMFolderView *emfv) +{ + em_folder_view_open_selected(emfv); +} + +static void +emfv_popup_resend(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids; + + if (!em_utils_check_user_can_send_mail((GtkWidget *)emfv)) + return; + + uids = message_list_get_selected(emfv->list); + em_utils_edit_messages((GtkWidget *)emfv, emfv->folder, uids); +} + +static void +emfv_popup_saveas(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids; + + uids = message_list_get_selected(emfv->list); + em_utils_save_messages((GtkWidget *)emfv, emfv->folder, uids); +} + +static void +emfv_popup_print(GtkWidget *w, EMFolderView *emfv) +{ + em_folder_view_print(emfv, FALSE); +} + +static void +emfv_popup_reply_sender(GtkWidget *w, EMFolderView *emfv) +{ + emfv_message_reply(emfv, REPLY_MODE_SENDER); +} + +static void +emfv_popup_reply_list(GtkWidget *w, EMFolderView *emfv) +{ + emfv_message_reply(emfv, REPLY_MODE_LIST); +} + +static void +emfv_popup_reply_all(GtkWidget *w, EMFolderView *emfv) +{ + emfv_message_reply(emfv, REPLY_MODE_ALL); +} + +static void +emfv_popup_forward(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids; + + if (!em_utils_check_user_can_send_mail((GtkWidget *)emfv)) + return; + + uids = message_list_get_selected(emfv->list); + em_utils_forward_messages((GtkWidget *)emfv, emfv->folder, uids); +} + +static void +emfv_popup_flag_followup(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids = message_list_get_selected(emfv->list); + + em_utils_flag_for_followup((GtkWidget *)emfv, emfv->folder, uids); +} + +static void +emfv_popup_flag_completed(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids; + + uids = message_list_get_selected(emfv->list); + em_utils_flag_for_followup_completed((GtkWidget *)emfv, emfv->folder, uids); +} + +static void +emfv_popup_flag_clear(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids = message_list_get_selected(emfv->list); + + em_utils_flag_for_followup_clear((GtkWidget *)emfv, emfv->folder, uids); +} + +static void +emfv_popup_mark_read(GtkWidget *w, EMFolderView *emfv) +{ + em_folder_view_mark_selected(emfv, CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN); +} + +static void +emfv_popup_mark_unread(GtkWidget *w, EMFolderView *emfv) +{ + em_folder_view_mark_selected(emfv, CAMEL_MESSAGE_SEEN|CAMEL_MESSAGE_DELETED, 0); + + if (emfv->priv->seen_id) { + g_source_remove(emfv->priv->seen_id); + emfv->priv->seen_id = 0; + } +} + +static void +emfv_popup_mark_important(GtkWidget *w, EMFolderView *emfv) +{ + em_folder_view_mark_selected(emfv, CAMEL_MESSAGE_FLAGGED|CAMEL_MESSAGE_DELETED, CAMEL_MESSAGE_FLAGGED); +} + +static void +emfv_popup_mark_unimportant(GtkWidget *w, EMFolderView *emfv) +{ + em_folder_view_mark_selected(emfv, CAMEL_MESSAGE_FLAGGED, 0); +} + +static void +emfv_popup_delete(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids; + + uids = message_list_get_selected (emfv->list); + em_folder_view_mark_selected (emfv, CAMEL_MESSAGE_SEEN|CAMEL_MESSAGE_DELETED, CAMEL_MESSAGE_SEEN|CAMEL_MESSAGE_DELETED); + + if (uids->len == 1) + message_list_select (emfv->list, MESSAGE_LIST_SELECT_NEXT, 0, 0, FALSE); + + em_utils_uids_free (uids); +} + +static void +emfv_popup_undelete(GtkWidget *w, EMFolderView *emfv) +{ + em_folder_view_mark_selected(emfv, CAMEL_MESSAGE_DELETED, 0); +} + +static void +emfv_popup_move(GtkWidget *w, EMFolderView *emfv) +{ + /* FIXME */ +} + +static void +emfv_popup_copy(GtkWidget *w, EMFolderView *emfv) +{ + /* FIXME */ +} + +static void +emfv_set_label(EMFolderView *emfv, const char *label) +{ + GPtrArray *uids = message_list_get_selected(emfv->list); + int i; + + for (i=0;i<uids->len;i++) + camel_folder_set_message_user_tag(emfv->folder, uids->pdata[i], "label", label); + + message_list_free_uids(emfv->list, uids); +} + +static void +emfv_popup_label_clear(GtkWidget *w, EMFolderView *emfv) +{ + emfv_set_label(emfv, NULL); +} + +static void +emfv_popup_label_set(GtkWidget *w, struct _emfv_label_item *item) +{ + emfv_set_label(item->emfv, item->label); +} + +static void +emfv_popup_add_sender(GtkWidget *w, EMFolderView *emfv) +{ + /* FIXME */ + printf("UNIMPLEMENTED: add sender to addressbook\n"); +} + +static void +emfv_popup_apply_filters(GtkWidget *w, EMFolderView *emfv) +{ + GPtrArray *uids = message_list_get_selected(emfv->list); + + mail_filter_on_demand(emfv->folder, uids); +} + +/* filter callbacks, this will eventually be a wizard, see + filter_type_current/vfolder_type_current for implementation */ + +#define EMFV_POPUP_AUTO_TYPE(autotype, name, type) \ +static void \ +name(GtkWidget *w, EMFolderView *emfv) \ +{ \ + autotype(emfv, type); \ +} + +EMFV_POPUP_AUTO_TYPE(vfolder_type_current, emfv_popup_vfolder_subject, AUTO_SUBJECT) +EMFV_POPUP_AUTO_TYPE(vfolder_type_current, emfv_popup_vfolder_sender, AUTO_FROM) +EMFV_POPUP_AUTO_TYPE(vfolder_type_current, emfv_popup_vfolder_recipients, AUTO_TO) +EMFV_POPUP_AUTO_TYPE(vfolder_type_current, emfv_popup_vfolder_mlist, AUTO_MLIST) + +EMFV_POPUP_AUTO_TYPE(filter_type_current, emfv_popup_filter_subject, AUTO_SUBJECT) +EMFV_POPUP_AUTO_TYPE(filter_type_current, emfv_popup_filter_sender, AUTO_FROM) +EMFV_POPUP_AUTO_TYPE(filter_type_current, emfv_popup_filter_recipients, AUTO_TO) +EMFV_POPUP_AUTO_TYPE(filter_type_current, emfv_popup_filter_mlist, AUTO_MLIST) + +/* TODO: Move some of these to be 'standard' menu's */ + +static EMPopupItem emfv_popup_menu[] = { + { EM_POPUP_ITEM, "00.emfv.00", N_("_Open"), G_CALLBACK(emfv_popup_open), NULL, NULL, 0 }, + { EM_POPUP_ITEM, "00.emfv.01", N_("_Edit as New Message..."), G_CALLBACK(emfv_popup_resend), NULL, NULL, EM_POPUP_SELECT_RESEND }, + { EM_POPUP_ITEM, "00.emfv.02", N_("_Save As..."), G_CALLBACK(emfv_popup_saveas), NULL, "save-as-16.png", 0 }, + { EM_POPUP_ITEM, "00.emfv.03", N_("_Print"), G_CALLBACK(emfv_popup_print), NULL, "print.xpm", 0 }, + + { EM_POPUP_BAR, "10.emfv" }, + { EM_POPUP_ITEM, "10.emfv.00", N_("_Reply to Sender"), G_CALLBACK(emfv_popup_reply_sender), NULL, "reply.xpm", EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "10.emfv.01", N_("Reply to _List"), G_CALLBACK(emfv_popup_reply_list), NULL, NULL, EM_POPUP_SELECT_ONE|EM_POPUP_SELECT_MAILING_LIST }, + { EM_POPUP_ITEM, "10.emfv.02", N_("Reply to _All"), G_CALLBACK(emfv_popup_reply_all), NULL, "reply_to_all.xpm", EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "10.emfv.03", N_("_Forward"), G_CALLBACK(emfv_popup_forward), NULL, "forward.xpm", EM_POPUP_SELECT_MANY }, + + { EM_POPUP_BAR, "20.emfv", NULL, NULL, NULL, NULL, EM_POPUP_SELECT_FLAG_FOLLOWUP|EM_POPUP_SELECT_FLAG_COMPLETED|EM_POPUP_SELECT_FLAG_CLEAR }, + { EM_POPUP_ITEM, "20.emfv.00", N_("Follo_w Up..."), G_CALLBACK(emfv_popup_flag_followup), NULL, "flag-for-followup-16.png", EM_POPUP_SELECT_FLAG_FOLLOWUP }, + { EM_POPUP_ITEM, "20.emfv.01", N_("Fla_g Completed"), G_CALLBACK(emfv_popup_flag_completed), NULL, NULL, EM_POPUP_SELECT_FLAG_COMPLETED }, + { EM_POPUP_ITEM, "20.emfv.02", N_("Cl_ear Flag"), G_CALLBACK(emfv_popup_flag_clear), NULL, NULL, EM_POPUP_SELECT_FLAG_CLEAR }, + + { EM_POPUP_BAR, "30.emfv" }, + { EM_POPUP_ITEM, "30.emfv.00", N_("Mar_k as Read"), G_CALLBACK(emfv_popup_mark_read), NULL, "mail-read.xpm", EM_POPUP_SELECT_MARK_READ }, + { EM_POPUP_ITEM, "30.emfv.01", N_("Mark as _Unread"), G_CALLBACK(emfv_popup_mark_unread), NULL, "mail-new.xpm", EM_POPUP_SELECT_MARK_UNREAD }, + { EM_POPUP_ITEM, "30.emfv.02", N_("Mark as _Important"), G_CALLBACK(emfv_popup_mark_important), NULL, "priority-high.xpm", EM_POPUP_SELECT_MARK_IMPORTANT }, + { EM_POPUP_ITEM, "30.emfv.03", N_("_Mark as Unimportant"), G_CALLBACK(emfv_popup_mark_unimportant), NULL, NULL, EM_POPUP_SELECT_MARK_UNIMPORTANT }, + + { EM_POPUP_BAR, "40.emfv" }, + { EM_POPUP_ITEM, "40.emfv.00", N_("_Delete"), G_CALLBACK(emfv_popup_delete), NULL, "evolution-trash-mini.png", EM_POPUP_SELECT_DELETE }, + { EM_POPUP_ITEM, "40.emfv.01", N_("U_ndelete"), G_CALLBACK(emfv_popup_undelete), NULL, "undelete_message-16.png", EM_POPUP_SELECT_UNDELETE }, + + { EM_POPUP_BAR, "50.emfv" }, + { EM_POPUP_ITEM, "50.emfv.00", N_("Mo_ve to Folder..."), G_CALLBACK(emfv_popup_move) }, + { EM_POPUP_ITEM, "50.emfv.01", N_("_Copy to Folder..."), G_CALLBACK(emfv_popup_copy) }, + + { EM_POPUP_BAR, "60.label" }, + { EM_POPUP_SUBMENU, "60.label.00", N_("Label") }, + { EM_POPUP_IMAGE, "60.label.00/00.label", N_("None"), G_CALLBACK(emfv_popup_label_clear) }, + { EM_POPUP_BAR, "60.label.00/00.label.00" }, + + { EM_POPUP_BAR, "70.emfv", NULL, NULL, NULL, NULL, EM_POPUP_SELECT_ONE|EM_POPUP_SELECT_ADD_SENDER }, + { EM_POPUP_ITEM, "70.emfv.00", N_("Add Sender to Address_book"), G_CALLBACK(emfv_popup_add_sender), NULL, NULL, EM_POPUP_SELECT_ONE|EM_POPUP_SELECT_ADD_SENDER }, + + { EM_POPUP_BAR, "80.emfv" }, + { EM_POPUP_ITEM, "80.emfv.00", N_("Appl_y Filters"), G_CALLBACK(emfv_popup_apply_filters) }, + + { EM_POPUP_BAR, "90.filter" }, + { EM_POPUP_SUBMENU, "90.filter.00", N_("Crea_te Rule From Message"), NULL, NULL, NULL, EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "90.filter.00/00.00", N_("VFolder on _Subject"), G_CALLBACK(emfv_popup_vfolder_subject), NULL, NULL, EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "90.filter.00/00.01", N_("VFolder on Se_nder"), G_CALLBACK(emfv_popup_vfolder_sender), NULL, NULL, EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "90.filter.00/00.02", N_("VFolder on _Recipients"), G_CALLBACK(emfv_popup_vfolder_recipients), NULL, NULL, EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "90.filter.00/00.03", N_("VFolder on Mailing _List"), + G_CALLBACK(emfv_popup_vfolder_mlist), NULL, NULL, EM_POPUP_SELECT_ONE|EM_POPUP_SELECT_MAILING_LIST }, + + { EM_POPUP_BAR, "90.filter.00/10" }, + { EM_POPUP_ITEM, "90.filter.00/10.00", N_("Filter on Sub_ject"), G_CALLBACK(emfv_popup_filter_subject), NULL, NULL, EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "90.filter.00/10.01", N_("Filter on Sen_der"), G_CALLBACK(emfv_popup_filter_sender), NULL, NULL, EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "90.filter.00/10.02", N_("Filter on Re_cipients"), G_CALLBACK(emfv_popup_filter_recipients), NULL, NULL, EM_POPUP_SELECT_ONE }, + { EM_POPUP_ITEM, "90.filter.00/10.03", N_("Filter on _Mailing List"), + G_CALLBACK(emfv_popup_filter_mlist), NULL, NULL, EM_POPUP_SELECT_ONE|EM_POPUP_SELECT_MAILING_LIST }, +}; + +static void +emfv_popup_labels_free(void *data) +{ + GSList *l = data; + + while (l) { + GSList *n = l->next; + struct _emfv_label_item *item = l->data; + + g_free(item->item.path); + g_free(item); + + g_slist_free_1(l); + l = n; + } +} + +static void +emfv_popup(EMFolderView *emfv, GdkEvent *event) +{ + GSList *menus = NULL, *l, *label_list = NULL; + GtkMenu *menu; + EMPopup *emp; + EMPopupTarget *target; + int i; + + emp = em_popup_new("com.ximian.mail.folderview.popup.select"); + target = em_folder_view_get_popup_target(emfv); + + for (i=0;i<sizeof(emfv_popup_menu)/sizeof(emfv_popup_menu[0]);i++) { + EMPopupItem *item = &emfv_popup_menu[i]; + + item->activate_data = emfv; + menus = g_slist_prepend(menus, item); + } + + em_popup_add_items(emp, menus, (GDestroyNotify)g_slist_free); + + i = 1; + for (l = mail_config_get_labels(); l; l = l->next) { + struct _emfv_label_item *item; + MailConfigLabel *label = l->data; + GdkPixmap *pixmap; + GdkColor colour; + GdkGC *gc; + + item = g_malloc0(sizeof(*item)); + item->item.type = EM_POPUP_IMAGE; + item->item.path = g_strdup_printf("60.label.00/00.label.%02d", i++); + item->item.label = label->name; + item->item.activate = G_CALLBACK(emfv_popup_label_set); + item->item.activate_data = item; + item->emfv = emfv; + item->label = label->tag; + + gdk_color_parse(label->colour, &colour); + gdk_color_alloc(gdk_colormap_get_system(), &colour); + + pixmap = gdk_pixmap_new(((GtkWidget *)emfv)->window, 16, 16, -1); + gc = gdk_gc_new(((GtkWidget *)emfv)->window); + gdk_gc_set_foreground(gc, &colour); + gdk_draw_rectangle(pixmap, gc, TRUE, 0, 0, 16, 16); + gdk_gc_unref(gc); + + item->item.image = gtk_image_new_from_pixmap(pixmap, NULL); + gtk_widget_show(item->item.image); + + label_list = g_slist_prepend(label_list, item); + } + + em_popup_add_items(emp, label_list, emfv_popup_labels_free); + + menu = em_popup_create_menu_once(emp, target, target->mask, target->mask); + + if (event == NULL || event->type == GDK_KEY_PRESS) { + /* FIXME: menu pos function */ + gtk_menu_popup(menu, NULL, NULL, NULL, NULL, 0, event->key.time); + } else { + gtk_menu_popup(menu, NULL, NULL, NULL, NULL, event->button.button, event->button.time); + } +} + +/* ********************************************************************** */ + +/* Bonobo menu's */ + +/* a lot of stuff maps directly to the popup menu equivalent */ +#define EMFV_MAP_CALLBACK(from, to) \ +static void \ +from(BonoboUIComponent *uid, void *data, const char *path) \ +{ \ + to(NULL, (EMFolderView *)data); \ +} + +EMFV_MAP_CALLBACK(emfv_message_apply_filters, emfv_popup_apply_filters) +EMFV_MAP_CALLBACK(emfv_message_copy, emfv_popup_copy) +EMFV_MAP_CALLBACK(emfv_message_move, emfv_popup_move) +EMFV_MAP_CALLBACK(emfv_message_forward, emfv_popup_forward) +EMFV_MAP_CALLBACK(emfv_message_reply_all, emfv_popup_reply_all) +EMFV_MAP_CALLBACK(emfv_message_reply_list, emfv_popup_reply_list) +EMFV_MAP_CALLBACK(emfv_message_reply_sender, emfv_popup_reply_sender) +EMFV_MAP_CALLBACK(emfv_message_mark_read, emfv_popup_mark_read) +EMFV_MAP_CALLBACK(emfv_message_mark_unread, emfv_popup_mark_unread) +EMFV_MAP_CALLBACK(emfv_message_mark_important, emfv_popup_mark_important) +EMFV_MAP_CALLBACK(emfv_message_mark_unimportant, emfv_popup_mark_unimportant) +EMFV_MAP_CALLBACK(emfv_message_delete, emfv_popup_delete) +EMFV_MAP_CALLBACK(emfv_message_undelete, emfv_popup_undelete) +EMFV_MAP_CALLBACK(emfv_message_followup_flag, emfv_popup_flag_followup) +/*EMFV_MAP_CALLBACK(emfv_message_followup_clear, emfv_popup_flag_clear) + EMFV_MAP_CALLBACK(emfv_message_followup_completed, emfv_popup_flag_completed)*/ +EMFV_MAP_CALLBACK(emfv_message_open, emfv_popup_open) +EMFV_MAP_CALLBACK(emfv_message_resend, emfv_popup_resend) +EMFV_MAP_CALLBACK(emfv_message_saveas, emfv_popup_saveas) +EMFV_MAP_CALLBACK(emfv_print_message, emfv_popup_print) + +static void +emfv_edit_cut(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (message_list_has_primary_selection(emfv->list)) + message_list_copy(emfv->list, TRUE); + else if (emfv->preview_active) + em_format_html_display_cut(emfv->preview); +} + +static void +emfv_edit_copy(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (message_list_has_primary_selection(emfv->list)) + message_list_copy(emfv->list, FALSE); + else if (emfv->preview_active) + em_format_html_display_copy(emfv->preview); +} + +static void +emfv_edit_paste(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_paste(emfv->list); +} + +static void +emfv_mail_next(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_select(emfv->list, MESSAGE_LIST_SELECT_NEXT, 0, 0, FALSE); +} + +static void +emfv_mail_next_flagged(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_select(emfv->list, MESSAGE_LIST_SELECT_NEXT, CAMEL_MESSAGE_FLAGGED, CAMEL_MESSAGE_FLAGGED, FALSE); +} + +static void +emfv_mail_next_unread(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_select(emfv->list, MESSAGE_LIST_SELECT_NEXT, 0, CAMEL_MESSAGE_SEEN, TRUE); +} + +static void +emfv_mail_next_thread(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_select_next_thread(emfv->list); +} + +static void +emfv_mail_previous(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_select(emfv->list, MESSAGE_LIST_SELECT_PREVIOUS, 0, 0, FALSE); +} + +static void +emfv_mail_previous_flagged(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_select(emfv->list, MESSAGE_LIST_SELECT_PREVIOUS, CAMEL_MESSAGE_FLAGGED, CAMEL_MESSAGE_FLAGGED, TRUE); +} + +static void +emfv_mail_previous_unread(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + message_list_select(emfv->list, MESSAGE_LIST_SELECT_PREVIOUS, 0, CAMEL_MESSAGE_SEEN, TRUE); +} + +static void +emfv_add_sender_addressbook(BonoboUIComponent *uid, void *data, const char *path) +{ + EMFolderView *emfv = data; + + emfv = emfv; + /* FIXME: need to find out what the new addressbook API is for this... */ +} + +static void +emfv_message_forward_attached (BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + GPtrArray *uids; + + if (!em_utils_check_user_can_send_mail ((GtkWidget *) emfv)) + return; + + uids = message_list_get_selected (emfv->list); + em_utils_forward_attached ((GtkWidget *) emfv, emfv->folder, uids); +} + +static void +emfv_message_forward_inline (BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + GPtrArray *uids; + + if (!em_utils_check_user_can_send_mail ((GtkWidget *) emfv)) + return; + + uids = message_list_get_selected (emfv->list); + em_utils_forward_inline ((GtkWidget *) emfv, emfv->folder, uids); +} + +static void +emfv_message_forward_quoted (BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + GPtrArray *uids; + + if (!em_utils_check_user_can_send_mail ((GtkWidget *) emfv)) + return; + + uids = message_list_get_selected (emfv->list); + em_utils_forward_quoted ((GtkWidget *) emfv, emfv->folder, uids); +} + +static void +emfv_message_redirect (BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (emfv->list->cursor_uid == NULL) + return; + + if (!em_utils_check_user_can_send_mail ((GtkWidget *) emfv)) + return; + + em_utils_redirect_message_by_uid ((GtkWidget *) emfv, emfv->folder, emfv->list->cursor_uid); +} + +static void +emfv_message_post_reply (BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (emfv->list->cursor_uid == NULL) + return; + + if (!em_utils_check_user_can_send_mail ((GtkWidget *) emfv)) + return; + + em_utils_post_reply_to_message_by_uid ((GtkWidget *) emfv, emfv->folder, emfv->list->cursor_uid); +} + +static void +emfv_message_reply(EMFolderView *emfv, int mode) +{ + /* GtkClipboard *clip; */ + + if (emfv->list->cursor_uid == NULL) + return; + + if (!em_utils_check_user_can_send_mail ((GtkWidget *) emfv)) + return; + + /* Look away! Look away! */ + + /* HACK: Nasty internal gtkhtml poking going on here */ + + /* Disabled since there's no simple way to find out if + gtkhtml has the primary selection right now */ + + /* Ugh, to use the clipboard we need to request the selection + and have an async callback - painful to deal with */ + + /*clip = gtk_clipboard_get(GDK_SELECTION_PRIMARY);*/ + if (FALSE /*gtk_clipboard_get_owner(clip) == (GObject *)emfv->preview*/ + && ((EMFormatHTML *)emfv->preview)->html->engine->primary) { + CamelMimeMessage *msg, *src; + struct _header_raw *header; + HTMLEngineSaveState *state; + + src = (CamelMimeMessage *)((EMFormat *)emfv->preview)->message; + msg = camel_mime_message_new(); + + header = ((CamelMimePart *)src)->headers; + while (header) { + /* FIXME: shouldn't we strip out *all* Content-* headers? */ + if (g_ascii_strcasecmp(header->name, "content-type") != 0) + camel_medium_add_header((CamelMedium *)msg, header->name, header->value); + header = header->next; + } + + state = html_engine_save_buffer_new(((EMFormatHTML *)emfv->preview)->html->engine, TRUE); + html_object_save(((EMFormatHTML *)emfv->preview)->html->engine->primary, state); + camel_mime_part_set_content((CamelMimePart *)msg, + ((GString *)state->user_data)->str, + ((GString *)state->user_data)->len, + "text/html"); + + html_engine_save_buffer_free(state); + + em_utils_reply_to_message((GtkWidget *)emfv, msg, mode); + camel_object_unref(msg); + } else { + em_utils_reply_to_message_by_uid ((GtkWidget *) emfv, emfv->folder, emfv->list->cursor_uid, mode); + } + + /*g_object_unref(clip);*/ +} + +static void +emfv_message_search(BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + em_format_html_display_search(emfv->preview); +} + +static void +emfv_print_preview_message(BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + em_folder_view_print(emfv, TRUE); +} + +static void +emfv_text_zoom_in(BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (emfv->preview) + em_format_html_display_zoom_in(emfv->preview); +} + +static void +emfv_text_zoom_out(BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (emfv->preview) + em_format_html_display_zoom_out(emfv->preview); +} + +static void +emfv_text_zoom_reset(BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (emfv->preview) + em_format_html_display_zoom_reset(emfv->preview); +} + +/* ********************************************************************** */ + +struct _filter_data { + CamelFolder *folder; + const char *source; + char *uid; + int type; + char *uri; + char *mlist; +}; + +static void +filter_data_free (struct _filter_data *fdata) +{ + g_free (fdata->uid); + g_free (fdata->uri); + if (fdata->folder) + camel_object_unref (fdata->folder); + g_free (fdata->mlist); + g_free (fdata); +} + +static void +filter_type_got_message (CamelFolder *folder, const char *uid, CamelMimeMessage *msg, void *user_data) +{ + struct _filter_data *data = user_data; + + if (msg) + filter_gui_add_from_message (msg, data->source, data->type); + + filter_data_free (data); +} + +static void +filter_type_uid (CamelFolder *folder, const char *uid, const char *source, int type) +{ + struct _filter_data *data; + + data = g_malloc0 (sizeof (*data)); + data->type = type; + data->source = source; + + mail_get_message (folder, uid, filter_type_got_message, data, mail_thread_new); +} + +static void +filter_type_current (EMFolderView *emfv, int type) +{ + const char *source; + GPtrArray *uids; + + if (em_utils_folder_is_sent (emfv->folder, emfv->folder_uri) + || em_utils_folder_is_outbox (emfv->folder, emfv->folder_uri)) + source = FILTER_SOURCE_OUTGOING; + else + source = FILTER_SOURCE_INCOMING; + + uids = message_list_get_selected (emfv->list); + + if (uids->len == 1) + filter_type_uid (emfv->folder, (char *) uids->pdata[0], source, type); + + em_utils_uids_free (uids); +} + +EMFV_MAP_CALLBACK(emfv_tools_filter_subject, emfv_popup_filter_subject) +EMFV_MAP_CALLBACK(emfv_tools_filter_sender, emfv_popup_filter_sender) +EMFV_MAP_CALLBACK(emfv_tools_filter_recipient, emfv_popup_filter_recipients) +EMFV_MAP_CALLBACK(emfv_tools_filter_mlist, emfv_popup_filter_mlist) + +static void +vfolder_type_got_message (CamelFolder *folder, const char *uid, CamelMimeMessage *msg, void *user_data) +{ + struct _filter_data *data = user_data; + + if (msg) + vfolder_gui_add_from_message (msg, data->type, data->uri); + + filter_data_free (data); +} + +static void +vfolder_type_uid (CamelFolder *folder, const char *uid, const char *uri, int type) +{ + struct _filter_data *data; + + data = g_malloc0 (sizeof (*data)); + data->type = type; + data->uri = g_strdup (uri); + + mail_get_message (folder, uid, vfolder_type_got_message, data, mail_thread_new); +} + +static void +vfolder_type_current (EMFolderView *emfv, int type) +{ + GPtrArray *uids; + + uids = message_list_get_selected (emfv->list); + + if (uids->len == 1) + vfolder_type_uid (emfv->folder, (char *) uids->pdata[0], emfv->folder_uri, type); + + em_utils_uids_free (uids); +} + +EMFV_MAP_CALLBACK(emfv_tools_vfolder_subject, emfv_popup_vfolder_subject) +EMFV_MAP_CALLBACK(emfv_tools_vfolder_sender, emfv_popup_vfolder_sender) +EMFV_MAP_CALLBACK(emfv_tools_vfolder_recipient, emfv_popup_vfolder_recipients) +EMFV_MAP_CALLBACK(emfv_tools_vfolder_mlist, emfv_popup_vfolder_mlist) + +/* ********************************************************************** */ + +static void +emfv_view_load_images(BonoboUIComponent *uic, void *data, const char *path) +{ + EMFolderView *emfv = data; + + if (emfv->preview) + em_format_html_load_http((EMFormatHTML *)emfv->preview); +} + +static BonoboUIVerb emfv_message_verbs[] = { + BONOBO_UI_UNSAFE_VERB ("EditCut", emfv_edit_cut), + BONOBO_UI_UNSAFE_VERB ("EditCopy", emfv_edit_copy), + BONOBO_UI_UNSAFE_VERB ("EditPaste", emfv_edit_paste), + + BONOBO_UI_UNSAFE_VERB ("MailNext", emfv_mail_next), + BONOBO_UI_UNSAFE_VERB ("MailNextFlagged", emfv_mail_next_flagged), + BONOBO_UI_UNSAFE_VERB ("MailNextUnread", emfv_mail_next_unread), + BONOBO_UI_UNSAFE_VERB ("MailNextThread", emfv_mail_next_thread), + BONOBO_UI_UNSAFE_VERB ("MailPrevious", emfv_mail_previous), + BONOBO_UI_UNSAFE_VERB ("MailPreviousFlagged", emfv_mail_previous_flagged), + BONOBO_UI_UNSAFE_VERB ("MailPreviousUnread", emfv_mail_previous_unread), + + BONOBO_UI_UNSAFE_VERB ("AddSenderToAddressbook", emfv_add_sender_addressbook), + + BONOBO_UI_UNSAFE_VERB ("MessageApplyFilters", emfv_message_apply_filters), + BONOBO_UI_UNSAFE_VERB ("MessageCopy", emfv_message_copy), + BONOBO_UI_UNSAFE_VERB ("MessageDelete", emfv_message_delete), + BONOBO_UI_UNSAFE_VERB ("MessageForward", emfv_message_forward), + BONOBO_UI_UNSAFE_VERB ("MessageForwardAttached", emfv_message_forward_attached), + BONOBO_UI_UNSAFE_VERB ("MessageForwardInline", emfv_message_forward_inline), + BONOBO_UI_UNSAFE_VERB ("MessageForwardQuoted", emfv_message_forward_quoted), + BONOBO_UI_UNSAFE_VERB ("MessageRedirect", emfv_message_redirect), + BONOBO_UI_UNSAFE_VERB ("MessageMarkAsRead", emfv_message_mark_read), + BONOBO_UI_UNSAFE_VERB ("MessageMarkAsUnRead", emfv_message_mark_unread), + BONOBO_UI_UNSAFE_VERB ("MessageMarkAsImportant", emfv_message_mark_important), + BONOBO_UI_UNSAFE_VERB ("MessageMarkAsUnimportant", emfv_message_mark_unimportant), + BONOBO_UI_UNSAFE_VERB ("MessageFollowUpFlag", emfv_message_followup_flag), + BONOBO_UI_UNSAFE_VERB ("MessageMove", emfv_message_move), + BONOBO_UI_UNSAFE_VERB ("MessageOpen", emfv_message_open), + BONOBO_UI_UNSAFE_VERB ("MessagePostReply", emfv_message_post_reply), + BONOBO_UI_UNSAFE_VERB ("MessageReplyAll", emfv_message_reply_all), + BONOBO_UI_UNSAFE_VERB ("MessageReplyList", emfv_message_reply_list), + BONOBO_UI_UNSAFE_VERB ("MessageReplySender", emfv_message_reply_sender), + BONOBO_UI_UNSAFE_VERB ("MessageResend", emfv_message_resend), + BONOBO_UI_UNSAFE_VERB ("MessageSaveAs", emfv_message_saveas), + BONOBO_UI_UNSAFE_VERB ("MessageSearch", emfv_message_search), + BONOBO_UI_UNSAFE_VERB ("MessageUndelete", emfv_message_undelete), + + BONOBO_UI_UNSAFE_VERB ("PrintMessage", emfv_print_message), + BONOBO_UI_UNSAFE_VERB ("PrintPreviewMessage", emfv_print_preview_message), + + BONOBO_UI_UNSAFE_VERB ("TextZoomIn", emfv_text_zoom_in), + BONOBO_UI_UNSAFE_VERB ("TextZoomOut", emfv_text_zoom_out), + BONOBO_UI_UNSAFE_VERB ("TextZoomReset", emfv_text_zoom_reset), + + /* TODO: This stuff should just be 1 item that runs a wizard */ + BONOBO_UI_UNSAFE_VERB ("ToolsFilterMailingList", emfv_tools_filter_mlist), + BONOBO_UI_UNSAFE_VERB ("ToolsFilterRecipient", emfv_tools_filter_recipient), + BONOBO_UI_UNSAFE_VERB ("ToolsFilterSender", emfv_tools_filter_sender), + BONOBO_UI_UNSAFE_VERB ("ToolsFilterSubject", emfv_tools_filter_subject), + BONOBO_UI_UNSAFE_VERB ("ToolsVFolderMailingList", emfv_tools_vfolder_mlist), + BONOBO_UI_UNSAFE_VERB ("ToolsVFolderRecipient", emfv_tools_vfolder_recipient), + BONOBO_UI_UNSAFE_VERB ("ToolsVFolderSender", emfv_tools_vfolder_sender), + BONOBO_UI_UNSAFE_VERB ("ToolsVFolderSubject", emfv_tools_vfolder_subject), + + BONOBO_UI_UNSAFE_VERB ("ViewLoadImages", emfv_view_load_images), + /* ViewHeaders stuff is a radio */ + /* CaretMode is a toggle */ + + BONOBO_UI_VERB_END +}; +static EPixmap emfv_message_pixmaps[] = { + E_PIXMAP ("/commands/EditCut", "16_cut.png"), + E_PIXMAP ("/commands/EditCopy", "16_copy.png"), + E_PIXMAP ("/commands/EditPaste", "16_paste.png"), + + E_PIXMAP ("/commands/PrintMessage", "print.xpm"), + E_PIXMAP ("/commands/PrintPreviewMessage", "print-preview.xpm"), + E_PIXMAP ("/commands/MessageDelete", "evolution-trash-mini.png"), + E_PIXMAP ("/commands/MessageUndelete", "undelete_message-16.png"), + E_PIXMAP ("/commands/MessageCopy", "copy_16_message.xpm"), + E_PIXMAP ("/commands/MessageMove", "move_message.xpm"), + E_PIXMAP ("/commands/MessageReplyAll", "reply_to_all.xpm"), + E_PIXMAP ("/commands/MessageReplySender", "reply.xpm"), + E_PIXMAP ("/commands/MessageForward", "forward.xpm"), + E_PIXMAP ("/commands/MessageApplyFilters", "apply-filters-16.xpm"), + E_PIXMAP ("/commands/MessageSearch", "search-16.png"), + E_PIXMAP ("/commands/MessageSaveAs", "save-as-16.png"), + E_PIXMAP ("/commands/MessageMarkAsRead", "mail-read.xpm"), + E_PIXMAP ("/commands/MessageMarkAsUnRead", "mail-new.xpm"), + E_PIXMAP ("/commands/MessageMarkAsImportant", "priority-high.xpm"), + E_PIXMAP ("/commands/MessageFollowUpFlag", "flag-for-followup-16.png"), + + E_PIXMAP ("/Toolbar/MailMessageToolbar/MessageReplySender", "buttons/reply.png"), + E_PIXMAP ("/Toolbar/MailMessageToolbar/MessageReplyAll", "buttons/reply-to-all.png"), + E_PIXMAP ("/Toolbar/MailMessageToolbar/MessageForward", "buttons/forward.png"), + E_PIXMAP ("/Toolbar/MailMessageToolbar/PrintMessage", "buttons/print.png"), + E_PIXMAP ("/Toolbar/MailMessageToolbar/MessageMove", "buttons/move-message.png"), + E_PIXMAP ("/Toolbar/MailMessageToolbar/MessageCopy", "buttons/copy-message.png"), + E_PIXMAP ("/Toolbar/MailMessageToolbar/MessageDelete", "buttons/delete-message.png"), + + E_PIXMAP ("/Toolbar/MailNextButtons/MailNext", "buttons/next-message.png"), + E_PIXMAP ("/Toolbar/MailNextButtons/MailPrevious", "buttons/previous-message.png"), + + E_PIXMAP_END +}; + +/* this is added to emfv->enable_map in :init() */ +static const EMFolderViewEnable emfv_enable_map[] = { + { "EditCut", EM_POPUP_SELECT_MANY }, + { "EditCopy", EM_POPUP_SELECT_MANY }, + { "EditPaste", 0 }, + + /* FIXME: should these be single-selection? */ + { "MailNext", EM_POPUP_SELECT_MANY }, + { "MailNextFlagged", EM_POPUP_SELECT_MANY }, + { "MailNextUnread", EM_POPUP_SELECT_MANY }, + { "MailNextThread", EM_POPUP_SELECT_MANY }, + { "MailPrevious", EM_POPUP_SELECT_MANY }, + { "MailPreviousFlagged", EM_POPUP_SELECT_MANY }, + { "MailPreviousUnread", EM_POPUP_SELECT_MANY }, + + { "AddSenderToAddressbook", EM_POPUP_SELECT_ADD_SENDER }, + + { "MessageApplyFilters", EM_POPUP_SELECT_MANY }, + { "MessageCopy", EM_POPUP_SELECT_MANY }, + { "MessageDelete", EM_POPUP_SELECT_MANY|EM_POPUP_SELECT_DELETE }, + { "MessageForward", EM_POPUP_SELECT_MANY }, + { "MessageForwardAttached", EM_POPUP_SELECT_MANY }, + { "MessageForwardInline", EM_POPUP_SELECT_ONE }, + { "MessageForwardQuoted", EM_POPUP_SELECT_ONE }, + { "MessageRedirect", EM_POPUP_SELECT_ONE }, + { "MessageMarkAsRead", EM_POPUP_SELECT_MANY|EM_POPUP_SELECT_MARK_READ }, + { "MessageMarkAsUnRead", EM_POPUP_SELECT_MANY|EM_POPUP_SELECT_MARK_UNREAD }, + { "MessageMarkAsImportant", EM_POPUP_SELECT_MANY|EM_POPUP_SELECT_MARK_IMPORTANT }, + { "MessageMarkAsUnimportant", EM_POPUP_SELECT_MANY|EM_POPUP_SELECT_MARK_UNIMPORTANT }, + { "MessageFollowUpFlag", EM_POPUP_SELECT_MANY }, + { "MessageMove", EM_POPUP_SELECT_MANY }, + { "MessageOpen", EM_POPUP_SELECT_MANY }, + { "MessagePostReply", EM_POPUP_SELECT_ONE }, + { "MessageReplyAll", EM_POPUP_SELECT_ONE }, + { "MessageReplyList", EM_POPUP_SELECT_ONE|EM_POPUP_SELECT_MAILING_LIST }, + { "MessageReplySender", EM_POPUP_SELECT_ONE }, + { "MessageResend", EM_POPUP_SELECT_RESEND }, + { "MessageSaveAs", EM_POPUP_SELECT_MANY }, + { "MessageSearch", EM_POPUP_SELECT_ONE }, + { "MessageUndelete", EM_POPUP_SELECT_MANY|EM_POPUP_SELECT_UNDELETE }, + { "PrintMessage", EM_POPUP_SELECT_ONE }, + { "PrintPreviewMessage", EM_POPUP_SELECT_ONE }, + + { "TextZoomIn", EM_POPUP_SELECT_ONE }, + { "TextZoomOut", EM_POPUP_SELECT_ONE }, + { "TextZoomReset", EM_POPUP_SELECT_ONE }, + + { "ToolsFilterMailingList", EM_POPUP_SELECT_ONE }, + { "ToolsFilterRecipient", EM_POPUP_SELECT_ONE }, + { "ToolsFilterSender", EM_POPUP_SELECT_ONE }, + { "ToolsFilterSubject", EM_POPUP_SELECT_ONE }, + { "ToolsVFolderMailingList", EM_POPUP_SELECT_ONE }, + { "ToolsVFolderRecipient", EM_POPUP_SELECT_ONE }, + { "ToolsVFolderSender", EM_POPUP_SELECT_ONE }, + { "ToolsVFolderSubject", EM_POPUP_SELECT_ONE }, + + { "ViewLoadImages", EM_POPUP_SELECT_ONE }, + + { NULL }, + + /* always enabled + + { "ViewFullHeaders", IS_0MESSAGE, 0 }, + { "ViewNormal", IS_0MESSAGE, 0 }, + { "ViewSource", IS_0MESSAGE, 0 }, + { "CaretMode", IS_0MESSAGE, 0 }, */ +}; + +static void +emfv_enable_menus(EMFolderView *emfv) +{ + guint32 disable_mask; + GString *name; + GSList *l; + EMPopupTarget *t; + + if (emfv->uic == NULL) + return; + + if (emfv->folder) { + t = em_folder_view_get_popup_target(emfv); + disable_mask = t->mask; + em_popup_target_free(t); + } else { + disable_mask = ~0; + } + + name = g_string_new(""); + for (l = emfv->enable_map; l; l = l->next) { + EMFolderViewEnable *map = l->data; + int i; + + for (i=0;map[i].name;i++) { + int state = (map[i].mask & disable_mask) == 0; + + g_string_printf(name, "/commands/%s", map[i].name); + bonobo_ui_component_set_prop(emfv->uic, name->str, "sensitive", state?"1":"0", NULL); + } + } + + g_string_free(name, TRUE); +} + +/* must match em_format_mode_t order */ +static const char * const emfv_display_styles[] = { + "/commands/ViewNormal", + "/commands/ViewFullHeaders", + "/commands/ViewSource" +}; + +static void +emfv_view_mode(BonoboUIComponent *uic, const char *path, Bonobo_UIComponent_EventType type, const char *state, void *data) +{ + EMFolderView *emfv = data; + int i; + + if (type != Bonobo_UIComponent_STATE_CHANGED + || state[0] == '0') + return; + + /* TODO: I don't like this stuff much, is there any way we can move listening for such events + elsehwere? Probably not I guess, unless there's a EMFolderViewContainer for bonobo usage + of a folder view */ + + for (i=0;i<= EM_FORMAT_SOURCE;i++) { + if (strcmp(emfv_display_styles[i]+strlen("/commands/"), path) == 0) { + em_format_set_mode((EMFormat *)emfv->preview, i); + + if (TRUE /* set preferences but not for EMMessageBrowser? */) { + GConfClient *gconf = mail_config_get_gconf_client (); + + gconf_client_set_int (gconf, "/apps/evolution/mail/display/message_style", i, NULL); + } + break; + } + } +} + +static void +emfv_caret_mode(BonoboUIComponent *uic, const char *path, Bonobo_UIComponent_EventType type, const char *state, void *data) +{ + EMFolderView *emfv = data; + + if (type != Bonobo_UIComponent_STATE_CHANGED) + return; + + em_format_html_display_set_caret_mode(emfv->preview, state[0] != '0'); + + gconf_client_set_bool(mail_config_get_gconf_client(), "/apps/evolution/mail/display/caret_mode", state[0] != '0', NULL); +} + +static void +emfv_charset_changed(BonoboUIComponent *uic, const char *path, Bonobo_UIComponent_EventType type, const char *state, void *data) +{ + EMFolderView *emfv = data; + + if (type != Bonobo_UIComponent_STATE_CHANGED) + return; + + /* menu items begin with "Charset-" = 8 characters */ + if (state[0] != '0' && strlen(path) > 8) { + path += 8; + /* default charset used in mail view */ + if (!strcmp(path, _("Default"))) + path = NULL; + + em_format_set_charset((EMFormat *)emfv->preview, path); + } +} + +static void +emfv_activate(EMFolderView *emfv, BonoboUIComponent *uic, int act) +{ + if (act) { + em_format_mode_t style; + gboolean state; + GSList *l; + + emfv->uic = uic; + + for (l = emfv->ui_files;l;l = l->next) + bonobo_ui_util_set_ui(uic, PREFIX, (char *)l->data, emfv->ui_app_name, NULL); + + bonobo_ui_component_add_verb_list_with_data(uic, emfv_message_verbs, emfv); + e_pixmaps_update(uic, emfv_message_pixmaps); + + state = emfv->preview->caret_mode; + bonobo_ui_component_set_prop(uic, "/commands/CaretMode", "state", state?"1":"0", NULL); + bonobo_ui_component_add_listener(uic, "CaretMode", emfv_caret_mode, emfv); + + style = ((EMFormat *)emfv->preview)->mode; + bonobo_ui_component_set_prop(uic, emfv_display_styles[style], "state", "1", NULL); + bonobo_ui_component_add_listener(uic, "ViewNormal", emfv_view_mode, emfv); + bonobo_ui_component_add_listener(uic, "ViewFullHeaders", emfv_view_mode, emfv); + bonobo_ui_component_add_listener(uic, "ViewSource", emfv_view_mode, emfv); + em_format_set_mode((EMFormat *)emfv->preview, style); + + if (emfv->folder && !em_utils_folder_is_sent(emfv->folder, emfv->folder_uri)) + bonobo_ui_component_set_prop(uic, "/commands/MessageResend", "sensitive", "0", NULL); + + /* default charset used in mail view */ + e_charset_picker_bonobo_ui_populate (uic, "/menu/View", _("Default"), emfv_charset_changed, emfv); + + emfv_enable_menus(emfv); + } else { + const BonoboUIVerb *v; + + /* TODO: Should this just rm /? */ + for (v = &emfv_message_verbs[0]; v->cname; v++) + bonobo_ui_component_remove_verb(uic, v->cname); + + if (emfv->folder) + mail_sync_folder(emfv->folder, NULL, NULL); + + emfv->uic = NULL; + } +} + +int em_folder_view_print(EMFolderView *emfv, int preview) +{ + /*struct _EMFolderViewPrivate *p = emfv->priv;*/ + EMFormatHTMLPrint *print; + GnomePrintConfig *config = NULL; + int res; + struct _CamelMedium *msg; + + /* FIXME: need to load the message first */ + if (!emfv->preview_active) + return 0; + + msg = emfv->preview->formathtml.format.message; + if (msg == NULL) + return 0; + + if (!preview) { + GtkDialog *dialog = (GtkDialog *)gnome_print_dialog_new(NULL, _("Print Message"), GNOME_PRINT_DIALOG_COPIES); + + gtk_dialog_set_default_response(dialog, GNOME_PRINT_DIALOG_RESPONSE_PRINT); + e_dialog_set_transient_for ((GtkWindow *) dialog, (GtkWidget *) emfv); + + switch (gtk_dialog_run(dialog)) { + case GNOME_PRINT_DIALOG_RESPONSE_PRINT: + break; + case GNOME_PRINT_DIALOG_RESPONSE_PREVIEW: + preview = TRUE; + break; + default: + gtk_widget_destroy((GtkWidget *)dialog); + return 0; + } + + config = gnome_print_dialog_get_config((GnomePrintDialog *)dialog); + gtk_widget_destroy((GtkWidget *)dialog); + } + + print = em_format_html_print_new(); + res = em_format_html_print_print(print, msg, (EMFormatHTML *)emfv->preview, config, preview); + g_object_unref(print); + if (config) + g_object_unref(config); + + return res; +} + +EMPopupTarget * +em_folder_view_get_popup_target(EMFolderView *emfv) +{ + EMPopupTarget *t; + + t = em_popup_target_new_select(emfv->folder, emfv->folder_uri, message_list_get_selected(emfv->list)); + t->widget = (GtkWidget *)emfv; + + if (emfv->list->threaded) + t->mask &= ~EM_FOLDER_VIEW_SELECT_THREADED; + + if (message_list_hidden(emfv->list) != 0) + t->mask &= ~EM_FOLDER_VIEW_SELECT_HIDDEN; + + return t; +} + +/* ********************************************************************** */ + +struct mst_t { + EMFolderView *emfv; + char *uid; +}; + +static void +mst_free (struct mst_t *mst) +{ + mst->emfv->priv->seen_id = 0; + + g_free (mst->uid); + g_free (mst); +} + +static int +do_mark_seen (gpointer user_data) +{ + struct mst_t *mst = user_data; + EMFolderView *emfv = mst->emfv; + MessageList *list = emfv->list; + + if (mst->uid && list->cursor_uid && !strcmp (mst->uid, list->cursor_uid)) + camel_folder_set_message_flags (emfv->folder, mst->uid, CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN); + + return FALSE; +} + +static void +emfv_list_done_message_selected(CamelFolder *folder, const char *uid, CamelMimeMessage *msg, void *data) +{ + EMFolderView *emfv = data; + + em_format_format((EMFormat *) emfv->preview, (struct _CamelMedium *)msg); + + if (emfv->priv->seen_id) + g_source_remove(emfv->priv->seen_id); + + if (msg && emfv->mark_seen) { + if (emfv->mark_seen_timeout > 0) { + struct mst_t *mst; + + mst = g_new (struct mst_t, 1); + mst->emfv = emfv; + mst->uid = g_strdup (uid); + + emfv->priv->seen_id = g_timeout_add_full(G_PRIORITY_DEFAULT_IDLE, emfv->mark_seen_timeout, + (GSourceFunc)do_mark_seen, mst, (GDestroyNotify)mst_free); + } else { + camel_folder_set_message_flags(emfv->folder, uid, CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN); + } + } +} + +static void +emfv_list_message_selected(MessageList *ml, const char *uid, EMFolderView *emfv) +{ + /* FIXME: ui stuff based on messageinfo, if available */ + + if (emfv->preview_active) { + if (uid) + mail_get_message(emfv->folder, uid, emfv_list_done_message_selected, emfv, mail_thread_new); + else + em_format_format((EMFormat *)emfv->preview, NULL); + } + + emfv_enable_menus(emfv); +} + +static void +emfv_list_double_click(ETree *tree, gint row, ETreePath path, gint col, GdkEvent *event, EMFolderView *emfv) +{ + /* Ignore double-clicks on columns that handle thier own state */ + if (MESSAGE_LIST_COLUMN_IS_ACTIVE (col)) + return; + + em_folder_view_open_selected(emfv); +} + +static int +emfv_list_right_click(ETree *tree, gint row, ETreePath path, gint col, GdkEvent *event, EMFolderView *emfv) +{ + emfv_popup(emfv, event); + + return TRUE; +} + +static int +emfv_list_key_press(ETree *tree, int row, ETreePath path, int col, GdkEvent *ev, EMFolderView *emfv) +{ + GPtrArray *uids; + int i; + guint32 flags; + + if ((ev->key.state & GDK_CONTROL_MASK) != 0) + return FALSE; + + switch (ev->key.keyval) { + case GDK_Return: + case GDK_KP_Enter: + case GDK_ISO_Enter: + em_folder_view_open_selected(emfv); + break; + case GDK_Delete: + case GDK_KP_Delete: + /* If any messages are undeleted, run delete, if all are deleted, run undelete */ + flags = 0; + uids = message_list_get_selected(emfv->list); + for (i = 0; i < uids->len; i++) { + if ((camel_folder_get_message_flags(emfv->folder, uids->pdata[i]) & CAMEL_MESSAGE_DELETED) == 0) + break; + } + message_list_free_uids(emfv->list, uids); + if (i == uids->len) + emfv_popup_undelete(NULL, emfv); + else + emfv_popup_delete(NULL, emfv); + break; + case GDK_Menu: + /* FIXME: location of popup */ + emfv_popup(emfv, NULL); + break; + case '!': + uids = message_list_get_selected(emfv->list); + + camel_folder_freeze(emfv->folder); + for (i = 0; i < uids->len; i++) { + flags = camel_folder_get_message_flags(emfv->folder, uids->pdata[i]) ^ CAMEL_MESSAGE_FLAGGED; + if (flags & CAMEL_MESSAGE_FLAGGED) + flags &= ~CAMEL_MESSAGE_DELETED; + camel_folder_set_message_flags(emfv->folder, uids->pdata[i], + CAMEL_MESSAGE_FLAGGED|CAMEL_MESSAGE_DELETED, flags); + } + camel_folder_thaw(emfv->folder); + + message_list_free_uids(emfv->list, uids); + break; + default: + return FALSE; + } + + return TRUE; +} + +static void +emfv_format_link_clicked(EMFormatHTMLDisplay *efhd, const char *uri, EMFolderView *emfv) +{ + if (!strncasecmp (uri, "mailto:", 7)) { + em_utils_compose_new_message_with_mailto ((GtkWidget *) efhd, uri); + } else if (*uri == '#') { + gtk_html_jump_to_anchor (((EMFormatHTML *) efhd)->html, uri + 1); + } else if (!strncasecmp (uri, "thismessage:", 12)) { + /* ignore */ + } else if (!strncasecmp (uri, "cid:", 4)) { + /* ignore */ + } else { + GError *err = NULL; + + gnome_url_show (uri, &err); + + if (err) { + g_warning ("gnome_url_show: %s", err->message); + g_error_free (err); + } + } +} + +static int +emfv_format_popup_event(EMFormatHTMLDisplay *efhd, GdkEventButton *event, const char *uri, CamelMimePart *part, EMFolderView *emfv) +{ + EMPopup *emp; + EMPopupTarget *target; + GtkMenu *menu; + + /* FIXME: this maybe should just fit on em-html-display, it has access to the + snooped part type */ + + emp = em_popup_new("com.ximian.mail.folderview.popup.uri"); + if (part) + target = em_popup_target_new_part(part, NULL); + else + target = em_popup_target_new_uri(uri); + + menu = em_popup_create_menu_once(emp, target, target->mask, target->mask); + gtk_menu_popup(menu, NULL, NULL, NULL, NULL, event->button, event->time); + + return TRUE; +} + +static void +emfv_gui_folder_changed(CamelFolder *folder, void *dummy, EMFolderView *emfv) +{ + emfv_enable_menus(emfv); + g_object_unref(emfv); +} + +static void +emfv_folder_changed(CamelFolder *folder, CamelFolderChangeInfo *changes, EMFolderView *emfv) +{ + g_object_ref(emfv); + mail_async_event_emit(emfv->async, MAIL_ASYNC_GUI, (MailAsyncFunc)emfv_gui_folder_changed, folder, NULL, emfv); +} + +/* keep these two tables in sync */ +enum { + EMFV_ANIMATE_IMAGES = 1, + EMFV_CITATION_COLOUR, + EMFV_CITATION_MARK, + EMFV_CARET_MODE, + EMFV_MESSAGE_STYLE, + EMFV_MARK_SEEN, + EMFV_MARK_SEEN_TIMEOUT, + EMFV_SETTINGS /* last, for loop count */ +}; + +/* IF these get too long, update key field */ +static const char * const emfv_display_keys[] = { + "animate_images", + "citation_colour", + "mark_citations", + "caret_mode", + "message_style", + "mark_seen", + "mark_seen_timeout" +}; + +static GHashTable *emfv_setting_key; + +static void +emfv_setting_notify(GConfClient *gconf, guint cnxn_id, GConfEntry *entry, EMFolderView *emfv) +{ + char *tkey; + + g_return_if_fail (gconf_entry_get_key (entry) != NULL); + g_return_if_fail (gconf_entry_get_value (entry) != NULL); + + tkey = strrchr(entry->key, '/'); + g_return_if_fail (tkey != NULL); + + switch(GPOINTER_TO_INT(g_hash_table_lookup(emfv_setting_key, tkey+1))) { + case EMFV_ANIMATE_IMAGES: + em_format_html_display_set_animate(emfv->preview, gconf_value_get_bool(gconf_entry_get_value(entry))); + break; + case EMFV_CITATION_COLOUR: { + const char *s; + GdkColor colour; + guint32 rgb; + + s = gconf_value_get_string(gconf_entry_get_value(entry)); + gdk_color_parse(s?s:"#737373", &colour); + rgb = ((colour.red & 0xff00) << 8) | (colour.green & 0xff00) | ((colour.blue & 0xff00) >> 8); + em_format_html_set_mark_citations((EMFormatHTML *)emfv->preview, + ((EMFormatHTML *)emfv->preview)->mark_citations, rgb); + break; } + case EMFV_CITATION_MARK: + em_format_html_set_mark_citations((EMFormatHTML *)emfv->preview, + gconf_value_get_bool(gconf_entry_get_value(entry)), + ((EMFormatHTML *)emfv->preview)->citation_colour); + break; + case EMFV_CARET_MODE: + em_format_html_display_set_caret_mode(emfv->preview, gconf_value_get_bool(gconf_entry_get_value(entry))); + break; + case EMFV_MESSAGE_STYLE: { + int style = gconf_value_get_int(gconf_entry_get_value(entry)); + + if (style < EM_FORMAT_NORMAL || style > EM_FORMAT_SOURCE) + style = EM_FORMAT_NORMAL; + em_format_set_mode((EMFormat *)emfv->preview, style); + break; } + case EMFV_MARK_SEEN: + emfv->mark_seen = gconf_value_get_bool(gconf_entry_get_value(entry)); + break; + case EMFV_MARK_SEEN_TIMEOUT: + emfv->mark_seen_timeout = gconf_value_get_int(gconf_entry_get_value(entry)); + break; + } +} + +static void +emfv_setting_setup(EMFolderView *emfv) +{ + GConfClient *gconf = gconf_client_get_default(); + GConfEntry *entry; + GError *err = NULL; + int i; + char key[64]; + + if (emfv_setting_key == NULL) { + emfv_setting_key = g_hash_table_new(g_str_hash, g_str_equal); + for (i=1;i<EMFV_SETTINGS;i++) + g_hash_table_insert(emfv_setting_key, (void *)emfv_display_keys[i-1], GINT_TO_POINTER(i)); + } + + gconf_client_add_dir(gconf, "/apps/evolution/mail/display", GCONF_CLIENT_PRELOAD_NONE, NULL); + + for (i=1;err == NULL && i<EMFV_SETTINGS;i++) { + sprintf(key, "/apps/evolution/mail/display/%s", emfv_display_keys[i-1]); + entry = gconf_client_get_entry(gconf, key, NULL, TRUE, &err); + if (entry) { + emfv_setting_notify(gconf, 0, entry, emfv); + gconf_entry_free(entry); + } + } + + if (err) { + g_warning("Could not load display settings: %s", err->message); + g_error_free(err); + } + + emfv->priv->setting_notify_id = gconf_client_notify_add(gconf, "/apps/evolution/mail/display", + (GConfClientNotifyFunc)emfv_setting_notify, + emfv, NULL, NULL); + g_object_unref(gconf); +} |