/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* mail-ops.c: callbacks for the mail toolbar/menus */
/*
* Authors:
* Dan Winship <danw@ximian.com>
* Peter Williams <peterw@ximian.com>
* Jeffrey Stedfast <fejj@ximian.com>
*
* Copyright 2000 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 Place, Suite 330, Boston, MA 02111-1307
* USA
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <errno.h>
#include <time.h>
#include <libgnome/gnome-paper.h>
#include <libgnomeui/gnome-dialog.h>
#include <libgnomeui/gnome-dialog-util.h>
#include <libgnomeui/gnome-stock.h>
#include <libgnome/gnome-paper.h>
#include <libgnomeprint/gnome-print-master.h>
#include <libgnomeprint/gnome-print-master-preview.h>
#include <bonobo/bonobo-widget.h>
#include <bonobo/bonobo-socket.h>
#include <gal/e-table/e-table.h>
#include <gal/widgets/e-gui-utils.h>
#include <gal/widgets/e-unicode.h>
#include <filter/filter-editor.h>
#include "mail.h"
#include "message-browser.h"
#include "mail-callbacks.h"
#include "mail-config.h"
#include "mail-accounts.h"
#include "mail-config-druid.h"
#include "mail-mt.h"
#include "mail-tools.h"
#include "mail-ops.h"
#include "mail-local.h"
#include "mail-search.h"
#include "mail-send-recv.h"
#include "mail-vfolder.h"
#include "mail-folder-cache.h"
#include "folder-browser.h"
#include "subscribe-dialog.h"
#include "e-messagebox.h"
#include "Evolution.h"
#include "evolution-storage.h"
#include "evolution-shell-client.h"
#ifndef HAVE_MKSTEMP
#include <fcntl.h>
#include <sys/stat.h>
#endif
struct post_send_data {
CamelFolder *folder;
gchar *uid;
guint32 flags;
};
static void
druid_destroyed (void)
{
gtk_main_quit ();
}
static gboolean
configure_mail (FolderBrowser *fb)
{
MailConfigDruid *druid;
if (fb) {
GtkWidget *dialog;
dialog = gnome_message_box_new (
_("You have not configured the mail client.\n"
"You need to do this before you can send,\n"
"receive or compose mail.\n"
"Would you like to configure it now?"),
GNOME_MESSAGE_BOX_QUESTION,
GNOME_STOCK_BUTTON_YES,
GNOME_STOCK_BUTTON_NO, NULL);
/*
* Focus YES
*/
gnome_dialog_set_default (GNOME_DIALOG (dialog), 0);
gtk_widget_grab_focus (GTK_WIDGET (GNOME_DIALOG (dialog)->buttons->data));
gnome_dialog_set_parent (GNOME_DIALOG (dialog),
GTK_WINDOW (gtk_widget_get_ancestor (GTK_WIDGET (fb), GTK_TYPE_WINDOW)));
switch (gnome_dialog_run_and_close (GNOME_DIALOG (dialog))) {
case 0:
druid = mail_config_druid_new (fb->shell);
gtk_signal_connect (GTK_OBJECT (druid), "destroy",
GTK_SIGNAL_FUNC (druid_destroyed), NULL);
gtk_widget_show (GTK_WIDGET (druid));
gtk_grab_add (GTK_WIDGET (druid));
gtk_main ();
break;
case 1:
default:
break;
}
}
return mail_config_is_configured ();
}
static gboolean
check_send_configuration (FolderBrowser *fb)
{
const MailConfigAccount *account;
/* Check general */
if (!mail_config_is_configured ()) {
if (!configure_mail (fb))
return FALSE;
}
/* Get the default account */
account = mail_config_get_default_account ();
/* Check for an identity */
if (!account) {
GtkWidget *message;
message = gnome_warning_dialog_parented (_("You need to configure an identity\n"
"before you can compose mail."),
GTK_WINDOW (gtk_widget_get_ancestor (GTK_WIDGET (fb),
GTK_TYPE_WINDOW)));
gnome_dialog_run_and_close (GNOME_DIALOG (message));
return FALSE;
}
/* Check for a transport */
if (!account->transport || !account->transport->url) {
GtkWidget *message;
message = gnome_warning_dialog_parented (_("You need to configure a mail transport\n"
"before you can compose mail."),
GTK_WINDOW (gtk_widget_get_ancestor (GTK_WIDGET (fb),
GTK_TYPE_WINDOW)));
gnome_dialog_run_and_close (GNOME_DIALOG (message));
return FALSE;
}
return TRUE;
}
void
send_receive_mail (GtkWidget *widget, gpointer user_data)
{
const MailConfigAccount *account;
if (!mail_config_is_configured ()) {
if (!configure_mail (FOLDER_BROWSER (user_data)))
return;
}
account = mail_config_get_default_account ();
if (!account || !account->transport) {
GtkWidget *win = gtk_widget_get_ancestor (GTK_WIDGET (user_data), GTK_TYPE_WINDOW);
gnome_error_dialog_parented (_("You have not set a mail transport method"), GTK_WINDOW (win));
return;
}
mail_send_receive ();
}
static void
msgbox_destroyed (GtkWidget *widget, gpointer data)
{
gboolean *show_again = data;
GtkWidget *checkbox;
checkbox = e_message_box_get_checkbox (E_MESSAGE_BOX (widget));
*show_again = !gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (checkbox));
}
static gboolean
ask_confirm_for_empty_subject (EMsgComposer *composer)
{
/* FIXME: EMessageBox should really handle this stuff
automagically. What Miguel thinks would be nice is to pass
in a unique id which could be used as a key in the config
file and the value would be an int. -1 for always show or
the button pressed otherwise. This probably means we'd have
to write e_messagebox_run () */
gboolean show_again = TRUE;
GtkWidget *mbox;
int button;
if (!mail_config_get_prompt_empty_subject ())
return TRUE;
mbox = e_message_box_new (_("This message has no subject.\nReally send?"),
E_MESSAGE_BOX_QUESTION,
GNOME_STOCK_BUTTON_YES,
GNOME_STOCK_BUTTON_NO,
NULL);
gtk_signal_connect (GTK_OBJECT (mbox), "destroy",
msgbox_destroyed, &show_again);
button = gnome_dialog_run_and_close (GNOME_DIALOG (mbox));
mail_config_set_prompt_empty_subject (show_again);
if (button == 0)
return TRUE;
else
return FALSE;
}
static gboolean
ask_confirm_for_only_bcc (EMsgComposer *composer)
{
/* FIXME: EMessageBox should really handle this stuff
automagically. What Miguel thinks would be nice is to pass
in a message-id which could be used as a key in the config
file and the value would be an int. -1 for always show or
the button pressed otherwise. This probably means we'd have
to write e_messagebox_run () */
gboolean show_again = TRUE;
GtkWidget *mbox;
int button;
if (!mail_config_get_prompt_only_bcc ())
return TRUE;
mbox = e_message_box_new (_("This message contains only Bcc recipients.\nIt is "
"possible that the mail server may reveal the recipients "
"by adding an Apparently-To header.\nSend anyway?"),
E_MESSAGE_BOX_QUESTION,
GNOME_STOCK_BUTTON_YES,
GNOME_STOCK_BUTTON_NO,
NULL);
gtk_signal_connect (GTK_OBJECT (mbox), "destroy",
msgbox_destroyed, &show_again);
button = gnome_dialog_run_and_close (GNOME_DIALOG (mbox));
mail_config_set_prompt_only_bcc (show_again);
if (button == 0)
return TRUE;
else
return FALSE;
}
static void
free_psd (GtkWidget *composer, gpointer user_data)
{
struct post_send_data *psd = user_data;
if (psd->folder)
camel_object_unref (CAMEL_OBJECT (psd->folder));
if (psd->uid)
g_free (psd->uid);
g_free (psd);
}
struct _send_data {
EMsgComposer *composer;
struct post_send_data *psd;
};
static void
composer_sent_cb (char *uri, CamelMimeMessage *message, gboolean sent, void *data)
{
struct _send_data *send = data;
if (sent) {
if (send->psd) {
camel_folder_set_message_flags (send->psd->folder, send->psd->uid,
send->psd->flags, send->psd->flags);
}
gtk_widget_destroy (GTK_WIDGET (send->composer));
} else {
gtk_widget_show (GTK_WIDGET (send->composer));
gtk_object_unref (GTK_OBJECT (send->composer));
}
g_free (send);
camel_object_unref (CAMEL_OBJECT (message));
}
static CamelMimeMessage *
composer_get_message (EMsgComposer *composer)
{
static char *recipient_type[] = {
CAMEL_RECIPIENT_TYPE_TO,
CAMEL_RECIPIENT_TYPE_CC,
CAMEL_RECIPIENT_TYPE_BCC
};
const CamelInternetAddress *iaddr;
const MailConfigAccount *account;
CamelMimeMessage *message;
const char *subject;
int num_addrs, i;
message = e_msg_composer_get_message (composer);
if (message == NULL)
return NULL;
/* Check for recipients */
for (num_addrs = 0, i = 0; i < 3; i++) {
iaddr = camel_mime_message_get_recipients (message, recipient_type[i]);
num_addrs += iaddr ? camel_address_length (CAMEL_ADDRESS (iaddr)) : 0;
}
/* I'm sensing a lack of love, er, I mean recipients. */
if (num_addrs == 0) {
GtkWidget *message_box;
message_box = gnome_message_box_new (_("You must specify recipients in order to "
"send this message."),
GNOME_MESSAGE_BOX_WARNING,
GNOME_STOCK_BUTTON_OK,
NULL);
gnome_dialog_run_and_close (GNOME_DIALOG (message_box));
camel_object_unref (CAMEL_OBJECT (message));
return NULL;
}
if (iaddr && num_addrs == camel_address_length (CAMEL_ADDRESS (iaddr))) {
/* this means that the only recipients are Bcc's */
if (!ask_confirm_for_only_bcc (composer)) {
camel_object_unref (CAMEL_OBJECT (message));
return NULL;
}
}
/* Check for no subject */
subject = camel_mime_message_get_subject (message);
if (subject == NULL || subject[0] == '\0') {
if (!ask_confirm_for_empty_subject (composer)) {
camel_object_unref (CAMEL_OBJECT (message));
return NULL;
}
}
/* Add info about the sending account */
account = e_msg_composer_get_preferred_account (composer);
if (account) {
camel_medium_set_header (CAMEL_MEDIUM (message), "X-Evolution-Account", account->name);
camel_medium_set_header (CAMEL_MEDIUM (message), "X-Evolution-Transport", account->transport->url);
camel_medium_set_header (CAMEL_MEDIUM (message), "X-Evolution-Fcc", account->sent_folder_uri);
}
return message;
}
void
composer_send_cb (EMsgComposer *composer, gpointer data)
{
const MailConfigService *transport;
CamelMimeMessage *message;
struct post_send_data *psd = data;
struct _send_data *send;
if (!mail_config_is_configured ()) {
GtkWidget *dialog;
dialog = gnome_ok_dialog_parented (_("You must configure an account before you "
"can send this email."),
GTK_WINDOW (composer));
gnome_dialog_run_and_close (GNOME_DIALOG (dialog));
return;
}
message = composer_get_message (composer);
if (!message)
return;
transport = mail_config_get_default_transport ();
if (!transport)
return;
send = g_malloc (sizeof (*send));
send->psd = psd;
send->composer = composer;
gtk_object_ref (GTK_OBJECT (composer));
gtk_widget_hide (GTK_WIDGET (composer));
mail_send_mail (transport->url, message, composer_sent_cb, send);
}
static void
append_mail_cleanup (CamelFolder *folder, CamelMimeMessage *msg, CamelMessageInfo *info, int ok, void *data)
{
camel_message_info_free (info);
}
void
composer_postpone_cb (EMsgComposer *composer, gpointer data)
{
extern CamelFolder *outbox_folder;
CamelMimeMessage *message;
CamelMessageInfo *info;
struct post_send_data *psd = data;
message = composer_get_message (composer);
if (message == NULL)
return;
info = camel_message_info_new ();
info->flags = CAMEL_MESSAGE_SEEN;
mail_append_mail (outbox_folder, message, info, append_mail_cleanup, NULL);
camel_object_unref (CAMEL_OBJECT (message));
if (psd)
camel_folder_set_message_flags (psd->folder, psd->uid, psd->flags, psd->flags);
gtk_widget_destroy (GTK_WIDGET (composer));
}
static GtkWidget *
create_msg_composer (const char *url)
{
const MailConfigAccount *account;
gboolean send_html;
EMsgComposer *composer;
account = mail_config_get_default_account ();
send_html = mail_config_get_send_html ();
composer = url ? e_msg_composer_new_from_url (url) : e_msg_composer_new ();
if (composer) {
e_msg_composer_hdrs_set_from_account (E_MSG_COMPOSER_HDRS (composer->hdrs), account->name);
e_msg_composer_set_send_html (composer, send_html);
e_msg_composer_show_sig_file (composer);
}
return GTK_WIDGET (composer);
}
void
compose_msg (GtkWidget *widget, gpointer user_data)
{
GtkWidget *composer;
if (!check_send_configuration (FOLDER_BROWSER (user_data)))
return;
composer = create_msg_composer (NULL);
if (!composer)
return;
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), NULL);
gtk_signal_connect (GTK_OBJECT (composer), "postpone",
GTK_SIGNAL_FUNC (composer_postpone_cb), NULL);
gtk_widget_show (composer);
}
/* Send according to a mailto (RFC 2368) URL. */
void
send_to_url (const char *url)
{
GtkWidget *composer;
/* FIXME: no way to get folder browser? Not without
* big pain in the ass, as far as I can tell */
if (!check_send_configuration (NULL))
return;
composer = create_msg_composer (url);
if (!composer)
return;
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), NULL);
gtk_signal_connect (GTK_OBJECT (composer), "postpone",
GTK_SIGNAL_FUNC (composer_postpone_cb), NULL);
gtk_widget_show (composer);
}
static GList *
list_add_addresses (GList *list, const CamelInternetAddress *cia, const GSList *accounts,
GHashTable *rcpt_hash, const MailConfigAccount **me,
const char *ignore_addr)
{
const char *name, *addr;
const GSList *l;
gboolean notme;
char *full;
int i;
for (i = 0; camel_internet_address_get (cia, i, &name, &addr); i++) {
/* Make sure we don't want to ignore this address */
if (!ignore_addr || g_strcasecmp (ignore_addr, addr)) {
/* now, we format this, as if for display, but does the composer
then use it as a real address? If so, very broken. */
/* we should probably pass around CamelAddresse's if thats what
we mean */
full = camel_internet_address_format_address (name, addr);
/* Here I'll check to see if the cc:'d address is the address
of the sender, and if so, don't add it to the cc: list; this
is to fix Bugzilla bug #455. */
notme = TRUE;
l = accounts;
while (l) {
const MailConfigAccount *acnt = l->data;
if (!strcmp (acnt->id->address, addr)) {
notme = FALSE;
if (me && !*me)
*me = acnt;
break;
}
l = l->next;
}
if (notme && !g_hash_table_lookup (rcpt_hash, addr)) {
g_hash_table_insert (rcpt_hash, (char *) addr, GINT_TO_POINTER (1));
list = g_list_append (list, full);
} else
g_free (full);
}
}
return list;
}
static const MailConfigAccount *
guess_me (const CamelInternetAddress *to, const CamelInternetAddress *cc, const GSList *accounts)
{
const char *name, *addr;
const GSList *l;
gboolean notme;
char *full;
int i;
if (to) {
for (i = 0; camel_internet_address_get (to, i, &name, &addr); i++) {
full = camel_internet_address_format_address (name, addr);
l = accounts;
while (l) {
const MailConfigAccount *acnt = l->data;
if (!g_strcasecmp (acnt->id->address, addr)) {
notme = FALSE;
return acnt;
}
l = l->next;
}
}
}
if (cc) {
for (i = 0; camel_internet_address_get (cc, i, &name, &addr); i++) {
full = camel_internet_address_format_address (name, addr);
l = accounts;
while (l) {
const MailConfigAccount *acnt = l->data;
if (!g_strcasecmp (acnt->id->address, addr)) {
notme = FALSE;
return acnt;
}
l = l->next;
}
}
}
return NULL;
}
static void
free_recipients (GList *list)
{
GList *l;
for (l = list; l; l = l->next)
g_free (l->data);
g_list_free (list);
}
static EMsgComposer *
mail_generate_reply (CamelFolder *folder, CamelMimeMessage *message, const char *uid, int mode)
{
const CamelInternetAddress *reply_to, *sender, *to_addrs, *cc_addrs;
const char *name = NULL, *address = NULL, *source = NULL;
const char *message_id, *references, *reply_addr = NULL;
char *text, *subject, date_str[100], *format;
const MailConfigAccount *me = NULL;
const GSList *accounts = NULL;
GList *to = NULL, *cc = NULL;
EMsgComposer *composer;
time_t date;
composer = e_msg_composer_new_with_sig_file ();
if (!composer)
return NULL;
sender = camel_mime_message_get_from (message);
camel_internet_address_get (sender, 0, &name, &address);
date = camel_mime_message_get_date (message, NULL);
strftime (date_str, sizeof (date_str), _("On %a, %Y-%m-%d at %H:%M, %%s wrote:"),
localtime (&date));
format = e_utf8_from_locale_string (date_str);
text = mail_tool_quote_message (message, date_str, name && *name ? name : address);
g_free (format);
if (text) {
e_msg_composer_set_body_text (composer, text);
g_free (text);
}
/* Set the recipients */
accounts = mail_config_get_accounts ();
to_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_TO);
cc_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_CC);
if (mode == REPLY_LIST) {
CamelMessageInfo *info;
const char *mlist;
int i, max, len;
info = camel_folder_get_message_info (folder, uid);
mlist = camel_message_info_mlist (info);
if (mlist) {
/* look through the recipients to find the *real* mailing list address */
len = strlen (mlist);
to_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_TO);
max = camel_address_length (CAMEL_ADDRESS (to_addrs));
for (i = 0; i < max; i++) {
camel_internet_address_get (to_addrs, i, NULL, &address);
if (!g_strncasecmp (address, mlist, len))
break;
}
if (i == max) {
cc_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_CC);
max = camel_address_length (CAMEL_ADDRESS (cc_addrs));
for (i = 0; i < max; i++) {
camel_internet_address_get (cc_addrs, i, NULL, &address);
if (!g_strncasecmp (address, mlist, len))
break;
}
}
/* We only want to reply to the list address - if it even exists */
to = address && i != max ? g_list_append (to, g_strdup (address)) : to;
}
me = guess_me (to_addrs, cc_addrs, accounts);
} else {
GHashTable *rcpt_hash;
rcpt_hash = g_hash_table_new (g_str_hash, g_str_equal);
reply_to = camel_mime_message_get_reply_to (message);
if (!reply_to)
reply_to = camel_mime_message_get_from (message);
if (reply_to) {
/* Get the Reply-To address so we can ignore references to it in the Cc: list */
if (camel_internet_address_get (reply_to, 0, NULL, &reply_addr)) {
g_hash_table_insert (rcpt_hash, (char *) reply_addr, GINT_TO_POINTER (1));
to = g_list_append (to, camel_address_format (CAMEL_ADDRESS (reply_to)));
}
}
if (mode == REPLY_ALL) {
cc = list_add_addresses (cc, to_addrs, accounts, rcpt_hash, &me, NULL);
cc = list_add_addresses (cc, cc_addrs, accounts, rcpt_hash, me ? NULL : &me, reply_addr);
} else {
me = guess_me (to_addrs, cc_addrs, accounts);
}
g_hash_table_destroy (rcpt_hash);
}
if (me == NULL) {
/* as a last resort, set the replying account (aka me)
to the account this was fetched from */
source = camel_mime_message_get_source (message);
me = mail_config_get_account_by_source_url (source);
}
/* Set the subject of the new message. */
subject = (char *)camel_mime_message_get_subject (message);
if (!subject)
subject = g_strdup ("");
else {
if (!g_strncasecmp (subject, "Re: ", 4))
subject = g_strdup (subject);
else
subject = g_strdup_printf ("Re: %s", subject);
}
e_msg_composer_set_headers (composer, me ? me->name : NULL, to, cc, NULL, subject);
free_recipients (to);
free_recipients (cc);
g_free (subject);
/* Add In-Reply-To and References. */
message_id = camel_medium_get_header (CAMEL_MEDIUM (message), "Message-Id");
references = camel_medium_get_header (CAMEL_MEDIUM (message), "References");
if (message_id) {
char *reply_refs;
e_msg_composer_add_header (composer, "In-Reply-To", message_id);
if (references)
reply_refs = g_strdup_printf ("%s %s", references, message_id);
else
reply_refs = g_strdup (message_id);
e_msg_composer_add_header (composer, "References", reply_refs);
g_free (reply_refs);
} else if (references) {
e_msg_composer_add_header (composer, "References", references);
}
return composer;
}
static void
requeue_mail_reply(CamelFolder *folder, char *uid, CamelMimeMessage *msg, void *data)
{
int mode = GPOINTER_TO_INT(data);
mail_reply (folder, msg, uid, mode);
}
void
mail_reply (CamelFolder *folder, CamelMimeMessage *msg, const char *uid, int mode)
{
EMsgComposer *composer;
struct post_send_data *psd;
g_return_if_fail (folder != NULL);
g_return_if_fail (uid != NULL);
if (!msg) {
mail_get_message (folder, uid, requeue_mail_reply,
GINT_TO_POINTER(mode), mail_thread_new);
return;
}
psd = g_new (struct post_send_data, 1);
psd->folder = folder;
camel_object_ref (CAMEL_OBJECT (psd->folder));
psd->uid = g_strdup (uid);
psd->flags = CAMEL_MESSAGE_ANSWERED;
composer = mail_generate_reply (folder, msg, uid, mode);
if (!composer)
return;
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), psd);
gtk_signal_connect (GTK_OBJECT (composer), "postpone",
GTK_SIGNAL_FUNC (composer_postpone_cb), psd);
gtk_signal_connect (GTK_OBJECT (composer), "destroy",
GTK_SIGNAL_FUNC (free_psd), psd);
gtk_widget_show (GTK_WIDGET (composer));
e_msg_composer_unset_changed (composer);
}
void
reply_to_sender (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
if (!check_send_configuration (fb))
return;
mail_reply (fb->folder, fb->mail_display->current_message,
fb->message_list->cursor_uid, REPLY_SENDER);
}
void
reply_to_list (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
if (!check_send_configuration (fb))
return;
mail_reply (fb->folder, fb->mail_display->current_message,
fb->message_list->cursor_uid, REPLY_LIST);
}
void
reply_to_all (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
if (!check_send_configuration (fb))
return;
mail_reply (fb->folder, fb->mail_display->current_message,
fb->message_list->cursor_uid, REPLY_ALL);
}
void
enumerate_msg (MessageList *ml, const char *uid, gpointer data)
{
g_ptr_array_add ((GPtrArray *) data, g_strdup (uid));
}
static EMsgComposer *
forward_get_composer (CamelMimeMessage *message, const char *subject)
{
const MailConfigAccount *account = NULL;
EMsgComposer *composer;
if (message) {
const CamelInternetAddress *to_addrs, *cc_addrs;
const GSList *accounts = NULL;
accounts = mail_config_get_accounts ();
to_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_TO);
cc_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_CC);
account = guess_me (to_addrs, cc_addrs, accounts);
if (!account) {
const char *source;
source = camel_mime_message_get_source (message);
account = mail_config_get_account_by_source_url (source);
}
}
if (!account)
account = mail_config_get_default_account ();
composer = e_msg_composer_new_with_sig_file ();
if (composer) {
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), NULL);
gtk_signal_connect (GTK_OBJECT (composer), "postpone",
GTK_SIGNAL_FUNC (composer_postpone_cb), NULL);
e_msg_composer_set_headers (composer, account->name, NULL, NULL, NULL, subject);
} else {
g_warning ("Could not create composer");
}
return composer;
}
static void
do_forward_non_attached (CamelFolder *folder, char *uid, CamelMimeMessage *message, void *data)
{
char *subject, *text, *title;
if (!message)
return;
subject = mail_tool_generate_forward_subject (message);
if (GPOINTER_TO_INT (data) == MAIL_CONFIG_FORWARD_INLINE) {
text = mail_tool_forward_message (message);
} else {
title = e_utf8_from_locale_string (_("Forwarded message:\n"));
text = mail_tool_quote_message (message, title);
g_free (title);
}
if (text) {
EMsgComposer *composer = forward_get_composer (message, subject);
if (composer) {
e_msg_composer_set_body_text (composer, text);
gtk_widget_show (GTK_WIDGET (composer));
e_msg_composer_unset_changed (composer);
}
g_free (text);
}
g_free (subject);
}
static void
forward_message (FolderBrowser *fb, MailConfigForwardStyle style)
{
if (!fb->message_list->cursor_uid)
return;
if (!check_send_configuration (fb))
return;
mail_get_message (fb->folder, fb->message_list->cursor_uid,
do_forward_non_attached, GINT_TO_POINTER (style),
mail_thread_new);
}
void
forward_inline (GtkWidget *widget, gpointer user_data)
{
forward_message (user_data, MAIL_CONFIG_FORWARD_INLINE);
}
void
forward_quoted (GtkWidget *widget, gpointer user_data)
{
forward_message (user_data, MAIL_CONFIG_FORWARD_QUOTED);
}
static void
do_forward_attach (CamelFolder *folder, GPtrArray *messages, CamelMimePart *part, char *subject, void *data)
{
CamelMimeMessage *message = data;
if (part) {
EMsgComposer *composer = forward_get_composer (message, subject);
if (composer) {
e_msg_composer_attach (composer, part);
gtk_widget_show (GTK_WIDGET (composer));
e_msg_composer_unset_changed (composer);
}
}
}
void
forward_attached (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = (FolderBrowser *)user_data;
GPtrArray *uids;
if (!check_send_configuration (fb))
return;
uids = g_ptr_array_new ();
message_list_foreach (fb->message_list, enumerate_msg, uids);
mail_build_attachment (fb->message_list->folder, uids, do_forward_attach,
uids->len == 1 ? fb->mail_display->current_message : NULL);
}
void
forward (GtkWidget *widget, gpointer user_data)
{
MailConfigForwardStyle style = mail_config_get_default_forward_style ();
if (style == MAIL_CONFIG_FORWARD_ATTACHED)
forward_attached (widget, user_data);
else
forward_message (user_data, style);
}
static void
transfer_msg (GtkWidget *widget, gpointer user_data, gboolean delete_from_source)
{
FolderBrowser *fb = user_data;
MessageList *ml = fb->message_list;
GPtrArray *uids;
char *uri, *physical, *path;
char *desc;
const char *allowed_types[] = { "mail", /*"vtrash",*/ NULL };
extern EvolutionShellClient *global_shell_client;
static char *last = NULL;
if (last == NULL)
last = g_strdup ("");
if (delete_from_source)
desc = _("Move message(s) to");
else
desc = _("Copy message(s) to");
evolution_shell_client_user_select_folder (global_shell_client, desc, last,
allowed_types, &uri, &physical);
if (!uri)
return;
path = strchr (uri, '/');
if (path && strcmp (last, path) != 0) {
g_free (last);
last = g_strdup_printf ("evolution:%s", path);
}
g_free (uri);
uids = g_ptr_array_new ();
message_list_foreach (ml, enumerate_msg, uids);
mail_transfer_messages (ml->folder, uids, delete_from_source,
physical, NULL, NULL);
}
void
move_msg (GtkWidget *widget, gpointer user_data)
{
transfer_msg (widget, user_data, TRUE);
}
void
copy_msg (GtkWidget *widget, gpointer user_data)
{
transfer_msg (widget, user_data, FALSE);
}
/* Copied from e-shell-view.c */
static GtkWidget *
find_socket (GtkContainer *container)
{
GList *children, *tmp;
children = gtk_container_children (container);
while (children) {
if (BONOBO_IS_SOCKET (children->data))
return children->data;
else if (GTK_IS_CONTAINER (children->data)) {
GtkWidget *socket = find_socket (children->data);
if (socket)
return socket;
}
tmp = children->next;
g_list_free_1 (children);
children = tmp;
}
return NULL;
}
void
addrbook_sender (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
CamelMimeMessage *msg = NULL;
const CamelInternetAddress *addr;
gchar *addr_str;
GtkWidget *win;
GtkWidget *control;
GtkWidget *socket;
if (fb && fb->mail_display)
msg = fb->mail_display->current_message;
if (msg == NULL)
return;
addr = camel_mime_message_get_from (msg);
if (addr == NULL)
return;
addr_str = camel_address_format (CAMEL_ADDRESS (addr));
win = gtk_window_new (GTK_WINDOW_TOPLEVEL);
gtk_window_set_title (GTK_WINDOW (win), _("Sender"));
control = bonobo_widget_new_control ("OAFIID:GNOME_Evolution_Addressbook_AddressPopup",
CORBA_OBJECT_NIL);
bonobo_widget_set_property (BONOBO_WIDGET (control),
"email", addr_str,
NULL);
socket = find_socket (GTK_CONTAINER (control));
gtk_signal_connect_object (GTK_OBJECT (socket),
"destroy",
GTK_SIGNAL_FUNC (gtk_widget_destroy),
GTK_OBJECT (win));
gtk_container_add (GTK_CONTAINER (win), control);
gtk_widget_show_all (win);
}
void
apply_filters (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
MessageList *ml = fb->message_list;
GPtrArray *uids;
uids = g_ptr_array_new ();
message_list_foreach (ml, enumerate_msg, uids);
mail_filter_on_demand(fb->folder, uids);
}
void
select_all (BonoboUIComponent *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
MessageList *ml = fb->message_list;
ESelectionModel *etsm = e_tree_get_selection_model (ml->tree);
if (ml->folder == NULL)
return;
e_selection_model_select_all (etsm);
}
/* Thread selection */
typedef struct thread_select_info {
MessageList *ml;
GPtrArray *paths;
} thread_select_info_t;
static gboolean
select_node (ETreeModel *tm, ETreePath path, gpointer user_data)
{
thread_select_info_t *tsi = (thread_select_info_t *) user_data;
g_ptr_array_add (tsi->paths, path);
return FALSE; /*not done yet*/
}
static void
thread_select_foreach (ETreePath path, gpointer user_data)
{
thread_select_info_t *tsi = (thread_select_info_t *) user_data;
ETreeModel *tm = tsi->ml->model;
ETreePath node;
/* @path part of the initial selection. If it has children,
* we select them as well. If it doesn't, we select its siblings and
* their children (ie, the current node must be inside the thread
* that the user wants to mark.
*/
if (e_tree_model_node_get_first_child (tm, path))
node = path;
else {
node = e_tree_model_node_get_parent (tm, path);
/* Let's make an exception: if no parent, then we're about
* to mark the whole tree. No. */
if (e_tree_model_node_is_root (tm, node))
node = path;
}
e_tree_model_node_traverse (tm, node, select_node, tsi);
}
void
select_thread (BonoboUIComponent *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
MessageList *ml = fb->message_list;
ETreeSelectionModel *selection_model;
thread_select_info_t tsi;
int i;
if (ml->folder == NULL)
return;
/* For every selected item, select the thread containing it.
* We can't alter the selection while iterating through it,
* so build up a list of paths.
*/
tsi.ml = ml;
tsi.paths = g_ptr_array_new ();
e_tree_selected_path_foreach (ml->tree, thread_select_foreach, &tsi);
selection_model = E_TREE_SELECTION_MODEL (e_tree_get_selection_model (ml->tree));
for (i = 0; i < tsi.paths->len; i++)
e_tree_selection_model_add_to_selection (selection_model,
tsi.paths->pdata[i]);
g_ptr_array_free (tsi.paths, TRUE);
}
void
invert_selection (BonoboUIComponent *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
MessageList *ml = fb->message_list;
ESelectionModel *etsm = e_tree_get_selection_model (ml->tree);
if (ml->folder == NULL)
return;
e_selection_model_invert_selection (etsm);
}
/* flag all selected messages. Return number flagged */
static int
flag_messages (FolderBrowser *fb, guint32 mask, guint32 set)
{
MessageList *ml = fb->message_list;
GPtrArray *uids;
int i;
if (ml->folder == NULL)
return 0;
/* could just use specific callback but i'm lazy */
uids = g_ptr_array_new ();
message_list_foreach (ml, enumerate_msg, uids);
camel_folder_freeze (ml->folder);
for (i = 0; i < uids->len; i++) {
camel_folder_set_message_flags (ml->folder, uids->pdata[i], mask, set);
g_free (uids->pdata[i]);
}
camel_folder_thaw (ml->folder);
g_ptr_array_free (uids, TRUE);
return i;
}
static int
toggle_flags (FolderBrowser *fb, guint32 mask)
{
MessageList *ml = fb->message_list;
GPtrArray *uids;
int i;
if (ml->folder == NULL)
return 0;
/* could just use specific callback but i'm lazy */
uids = g_ptr_array_new ();
message_list_foreach (ml, enumerate_msg, uids);
camel_folder_freeze (ml->folder);
for (i = 0; i < uids->len; i++) {
gint flags;
flags = camel_folder_get_message_flags (ml->folder, uids->pdata[i]);
if (flags & mask)
camel_folder_set_message_flags (ml->folder, uids->pdata[i], mask, 0);
else {
if ((mask & CAMEL_MESSAGE_FLAGGED) && (flags & CAMEL_MESSAGE_DELETED))
camel_folder_set_message_flags (ml->folder, uids->pdata[i], CAMEL_MESSAGE_DELETED, 0);
camel_folder_set_message_flags (ml->folder, uids->pdata[i], mask, mask);
}
g_free (uids->pdata[i]);
}
camel_folder_thaw (ml->folder);
g_ptr_array_free (uids, TRUE);
return i;
}
void
mark_as_seen (BonoboUIComponent *uih, void *user_data, const char *path)
{
flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN);
}
void
mark_as_unseen (BonoboUIComponent *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
/* Remove the automatic mark-as-read timer first */
if (fb->seen_id) {
gtk_timeout_remove (fb->seen_id);
fb->seen_id = 0;
}
flag_messages (fb, CAMEL_MESSAGE_SEEN, 0);
}
void
mark_all_as_seen (BonoboUIComponent *uih, void *user_data, const char *path)
{
select_all (uih, user_data, path);
flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN);
}
void
mark_as_important (BonoboUIComponent *uih, void *user_data, const char *path)
{
flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_DELETED, 0);
flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_FLAGGED, CAMEL_MESSAGE_FLAGGED);
}
void
mark_as_unimportant (BonoboUIComponent *uih, void *user_data, const char *path)
{
flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_FLAGGED, 0);
}
void
toggle_as_important (BonoboUIComponent *uih, void *user_data, const char *path)
{
toggle_flags (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_FLAGGED);
}
static void
do_edit_messages(CamelFolder *folder, GPtrArray *uids, GPtrArray *messages, void *data)
{
/*FolderBrowser *fb = data;*/
int i;
for (i = 0; i < messages->len; i++) {
EMsgComposer *composer;
composer = e_msg_composer_new_with_message (messages->pdata[i]);
if (composer) {
gtk_signal_connect (GTK_OBJECT (composer), "send",
composer_send_cb, NULL);
gtk_signal_connect (GTK_OBJECT (composer), "postpone",
composer_postpone_cb, NULL);
gtk_widget_show (GTK_WIDGET (composer));
}
}
}
static gboolean
are_you_sure (const char *msg, GPtrArray *uids, FolderBrowser *fb)
{
GtkWidget *window = gtk_widget_get_ancestor (GTK_WIDGET (fb), GTK_TYPE_WINDOW);
GtkWidget *dialog;
char *buf;
int button, i;
buf = g_strdup_printf (msg, uids->len);
dialog = gnome_ok_cancel_dialog_parented (buf, NULL, NULL, (GtkWindow *)window);
button = gnome_dialog_run_and_close (GNOME_DIALOG (dialog));
if (button != 0) {
for (i = 0; i < uids->len; i++)
g_free (uids->pdata[i]);
g_ptr_array_free (uids, TRUE);
}
return button == 0;
}
static void
edit_msg_internal (FolderBrowser *fb)
{
GPtrArray *uids;
if (!check_send_configuration (fb))
return;
uids = g_ptr_array_new ();
message_list_foreach (fb->message_list, enumerate_msg, uids);
if (uids->len > 10 && !are_you_sure (_("Are you sure you want to edit all %d messages?"), uids, fb)) {
int i;
for (i = 0; i < uids->len; i++)
g_free (uids->pdata[i]);
g_ptr_array_free (uids, TRUE);
return;
}
mail_get_messages (fb->folder, uids, do_edit_messages, fb);
}
void
edit_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
if (!folder_browser_is_drafts (fb)) {
GtkWidget *message;
message = gnome_warning_dialog (_("You may only edit messages saved\n"
"in the Drafts folder."));
gnome_dialog_run_and_close (GNOME_DIALOG (message));
return;
}
edit_msg_internal (fb);
}
static void
do_resend_messages (CamelFolder *folder, GPtrArray *uids, GPtrArray *messages, void *data)
{
int i;
for (i = 0; i < messages->len; i++) {
/* generate a new Message-Id because they need to be unique */
camel_mime_message_set_message_id (messages->pdata[i], NULL);
}
/* "Resend" should open up the composer to let the user edit the message */
do_edit_messages (folder, uids, messages, data);
}
void
resend_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
GPtrArray *uids;
if (!folder_browser_is_sent (fb)) {
GtkWidget *message;
message = gnome_warning_dialog (_("You may only resend messages\n"
"in the Sent folder."));
gnome_dialog_run_and_close (GNOME_DIALOG (message));
return;
}
if (!check_send_configuration (fb))
return;
uids = g_ptr_array_new ();
message_list_foreach (fb->message_list, enumerate_msg, uids);
if (uids->len > 10 && !are_you_sure (_("Are you sure you want to resend all %d messages?"), uids, fb)) {
int i;
for (i = 0; i < uids->len; i++)
g_free (uids->pdata[i]);
g_ptr_array_free (uids, TRUE);
return;
}
mail_get_messages (fb->folder, uids, do_resend_messages, fb);
}
void
search_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
GtkWidget *w;
if (fb->mail_display->current_message == NULL) {
gtk_widget_show_all (gnome_warning_dialog (_("No Message Selected")));
return;
}
w = mail_search_new (fb->mail_display);
gtk_widget_show_all (w);
}
void
load_images (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
mail_display_load_images (fb->mail_display);
}
static void
save_msg_ok (GtkWidget *widget, gpointer user_data)
{
CamelFolder *folder;
GPtrArray *uids;
const char *path;
int fd, ret = 0;
path = gtk_file_selection_get_filename (GTK_FILE_SELECTION (user_data));
if (path[0] == '\0')
return;
fd = open (path, O_RDONLY);
if (fd != -1) {
GtkWidget *dialog;
GtkWidget *text;
close (fd);
dialog = gnome_dialog_new (_("Overwrite file?"),
GNOME_STOCK_BUTTON_YES,
GNOME_STOCK_BUTTON_NO,
NULL);
gtk_widget_set_parent (dialog, GTK_WIDGET (user_data));
text = gtk_label_new (_("A file by that name already exists.\nOverwrite it?"));
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (dialog)->vbox), text, TRUE, TRUE, 4);
gtk_window_set_policy (GTK_WINDOW (dialog), FALSE, TRUE, FALSE);
gtk_widget_show (text);
ret = gnome_dialog_run_and_close (GNOME_DIALOG (dialog));
}
if (ret == 0) {
folder = gtk_object_get_data (GTK_OBJECT (user_data), "folder");
uids = gtk_object_get_data (GTK_OBJECT (user_data), "uids");
gtk_object_remove_no_notify (GTK_OBJECT (user_data), "uids");
mail_save_messages (folder, uids, path, NULL, NULL);
gtk_widget_destroy (GTK_WIDGET (user_data));
}
}
static void
save_msg_destroy (gpointer user_data)
{
GPtrArray *uids = user_data;
if (uids) {
int i;
for (i = 0; i < uids->len; i++)
g_free (uids->pdata[i]);
g_ptr_array_free (uids, TRUE);
}
}
void
save_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
GtkFileSelection *filesel;
GPtrArray *uids;
char *title, *path;
uids = g_ptr_array_new ();
message_list_foreach (fb->message_list, enumerate_msg, uids);
if (uids->len == 1)
title = _("Save Message As...");
else
title = _("Save Messages As...");
filesel = GTK_FILE_SELECTION (gtk_file_selection_new (title));
path = g_strdup_printf ("%s/", g_get_home_dir ());
gtk_file_selection_set_filename (filesel, path);
g_free (path);
gtk_object_set_data_full (GTK_OBJECT (filesel), "uids", uids, save_msg_destroy);
gtk_object_set_data (GTK_OBJECT (filesel), "folder", fb->folder);
gtk_signal_connect (GTK_OBJECT (filesel->ok_button),
"clicked", GTK_SIGNAL_FUNC (save_msg_ok), filesel);
gtk_signal_connect_object (GTK_OBJECT (filesel->cancel_button),
"clicked", GTK_SIGNAL_FUNC (gtk_widget_destroy),
GTK_OBJECT (filesel));
gtk_widget_show (GTK_WIDGET (filesel));
}
void
delete_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
int deleted, row;
deleted = flag_messages (fb, CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN,
CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN);
/* Select the next message if we are only deleting one message */
if (deleted) {
MessageListSelectDirection direction = MESSAGE_LIST_SELECT_NEXT;
ETableState *state;
state = e_tree_get_state_object (fb->message_list->tree);
if (e_table_sort_info_grouping_get_count(state->sort_info) == 0 &&
e_table_sort_info_sorting_get_count(state->sort_info) == 1) {
ETableSortColumn column;
column = e_table_sort_info_sorting_get_nth (state->sort_info, 0);
if ((column.column == COL_SENT ||
column.column == COL_RECEIVED) &&
column.ascending == FALSE) {
direction = MESSAGE_LIST_SELECT_PREVIOUS;
}
}
gtk_object_unref (GTK_OBJECT (state));
row = e_tree_row_of_node (fb->message_list->tree,
e_tree_get_cursor (fb->message_list->tree));
/* If this is the last message and deleted messages
are hidden, select the previous */
if (((direction == MESSAGE_LIST_SELECT_NEXT && (row+1 == e_tree_row_count (fb->message_list->tree))) ||
(direction == MESSAGE_LIST_SELECT_PREVIOUS && (row == 0)))
&& mail_config_get_hide_deleted ()) {
if (direction == MESSAGE_LIST_SELECT_NEXT)
direction = MESSAGE_LIST_SELECT_PREVIOUS;
else
direction = MESSAGE_LIST_SELECT_NEXT;
message_list_select (fb->message_list, row, direction,
0, CAMEL_MESSAGE_DELETED, FALSE);
} else
message_list_select (fb->message_list, row, direction,
0, 0, FALSE);
}
}
void
undelete_msg (GtkWidget *button, gpointer user_data)
{
flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_DELETED, 0);
}
void
next_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
int row;
row = e_tree_row_of_node (fb->message_list->tree, e_tree_get_cursor (fb->message_list->tree));
message_list_select (fb->message_list, row, MESSAGE_LIST_SELECT_NEXT, 0, 0, FALSE);
}
void
next_unread_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
int row;
row = e_tree_row_of_node (fb->message_list->tree, e_tree_get_cursor (fb->message_list->tree));
message_list_select (fb->message_list, row,
MESSAGE_LIST_SELECT_NEXT,
0, CAMEL_MESSAGE_SEEN, TRUE);
}
void
next_flagged_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
int row;
row = e_tree_row_of_node (fb->message_list->tree, e_tree_get_cursor (fb->message_list->tree));
message_list_select (fb->message_list, row,
MESSAGE_LIST_SELECT_NEXT,
CAMEL_MESSAGE_FLAGGED, CAMEL_MESSAGE_FLAGGED, FALSE);
}
void
previous_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
int row;
row = e_tree_row_of_node (fb->message_list->tree, e_tree_get_cursor (fb->message_list->tree));
message_list_select (fb->message_list, row,
MESSAGE_LIST_SELECT_PREVIOUS,
0, 0, FALSE);
}
void
previous_unread_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
int row;
row = e_tree_row_of_node (fb->message_list->tree, e_tree_get_cursor (fb->message_list->tree));
message_list_select (fb->message_list, row,
MESSAGE_LIST_SELECT_PREVIOUS,
0, CAMEL_MESSAGE_SEEN, FALSE);
}
void
previous_flagged_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
int row;
row = e_tree_row_of_node (fb->message_list->tree, e_tree_get_cursor (fb->message_list->tree));
message_list_select (fb->message_list, row,
MESSAGE_LIST_SELECT_PREVIOUS,
CAMEL_MESSAGE_FLAGGED, CAMEL_MESSAGE_FLAGGED, TRUE);
}
struct _expunged_folder_data {
FolderBrowser *fb;
gboolean hidedeleted;
};
static void
expunged_folder (CamelFolder *f, void *data)
{
FolderBrowser *fb = ((struct _expunged_folder_data *) data)->fb;
gboolean hidedeleted = ((struct _expunged_folder_data *) data)->hidedeleted;
fb->expunging = NULL;
message_list_set_hidedeleted (fb->message_list, hidedeleted);
g_free (data);
}
static gboolean
confirm_expunge (void)
{
GtkWidget *dialog, *label, *checkbox;
int button;
if (!mail_config_get_confirm_expunge ())
return TRUE;
dialog = gnome_dialog_new (_("Warning"),
GNOME_STOCK_BUTTON_YES,
GNOME_STOCK_BUTTON_NO,
NULL);
label = gtk_label_new (_("This operation will permanently erase all messages marked as deleted. If you continue, you will not be able to recover these messages.\n\nReally erase these messages?"));
gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_LEFT);
gtk_widget_show (label);
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (dialog)->vbox), label, TRUE, TRUE, 4);
checkbox = gtk_check_button_new_with_label (_("Do not ask me again."));
gtk_object_ref (GTK_OBJECT (checkbox));
gtk_widget_show (checkbox);
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (dialog)->vbox), checkbox, TRUE, TRUE, 4);
button = gnome_dialog_run_and_close (GNOME_DIALOG (dialog));
if (button == 0 && gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (checkbox)))
mail_config_set_confirm_expunge (FALSE);
gtk_object_unref (GTK_OBJECT (checkbox));
if (button == 0)
return TRUE;
else
return FALSE;
}
void
expunge_folder (BonoboUIComponent *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
if (fb->folder && (fb->expunging == NULL || fb->folder != fb->expunging) && confirm_expunge ()) {
struct _expunged_folder_data *data;
CamelMessageInfo *info;
data = g_malloc (sizeof (*data));
data->fb = fb;
data->hidedeleted = fb->message_list->hidedeleted;
/* hide the deleted messages so user can't click on them while we expunge */
message_list_set_hidedeleted (fb->message_list, TRUE);
/* Only blank the mail display if the message being
viewed is one of those to be expunged */
if (fb->loaded_uid) {
info = camel_folder_get_message_info (fb->folder, fb->loaded_uid);
if (!info || info->flags & CAMEL_MESSAGE_DELETED)
mail_display_set_message (fb->mail_display, NULL);
}
fb->expunging = fb->folder;
mail_expunge_folder (fb->folder, expunged_folder, data);
}
}
/********************** Begin Filter Editor ********************/
static GtkWidget *filter_editor = NULL;
static void
filter_editor_destroy (GtkWidget *dialog, gpointer user_data)
{
filter_editor = NULL;
}
static void
filter_editor_clicked (GtkWidget *dialog, int button, FolderBrowser *fb)
{
FilterContext *fc;
if (button == 0) {
char *user;
fc = gtk_object_get_data (GTK_OBJECT (dialog), "context");
user = g_strdup_printf ("%s/filters.xml", evolution_dir);
rule_context_save ((RuleContext *)fc, user);
g_free (user);
}
if (button != -1) {
gnome_dialog_close (GNOME_DIALOG (dialog));
}
}
static const char *filter_source_names[] = {
"incoming",
"outgoing",
NULL,
};
void
filter_edit (BonoboUIComponent *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
FilterContext *fc;
char *user, *system;
if (filter_editor) {
gdk_window_raise (GTK_WIDGET (filter_editor)->window);
return;
}
fc = filter_context_new ();
user = g_strdup_printf ("%s/filters.xml", evolution_dir);
system = EVOLUTION_DATADIR "/evolution/filtertypes.xml";
rule_context_load ((RuleContext *)fc, system, user);
g_free (user);
if (((RuleContext *)fc)->error) {
GtkWidget *dialog;
gchar *err;
err = g_strdup_printf (_("Error loading filter information:\n%s"),
((RuleContext *)fc)->error);
dialog = gnome_warning_dialog (err);
g_free (err);
gnome_dialog_run_and_close (GNOME_DIALOG (dialog));
return;
}
filter_editor = (GtkWidget *)filter_editor_new (fc, filter_source_names);
gtk_window_set_title (GTK_WINDOW (filter_editor), _("Filters"));
gtk_object_set_data_full (GTK_OBJECT (filter_editor), "context", fc, (GtkDestroyNotify)gtk_object_unref);
gtk_signal_connect (GTK_OBJECT (filter_editor), "clicked", filter_editor_clicked, fb);
gtk_signal_connect (GTK_OBJECT (filter_editor), "destroy", filter_editor_destroy, NULL);
gtk_widget_show (GTK_WIDGET (filter_editor));
}
/********************** End Filter Editor ********************/
void
vfolder_edit_vfolders (BonoboUIComponent *uih, void *user_data, const char *path)
{
vfolder_edit ();
}
void
providers_config (BonoboUIComponent *uih, void *user_data, const char *path)
{
static MailAccountsDialog *dialog = NULL;
if (!dialog) {
dialog = mail_accounts_dialog_new ((FOLDER_BROWSER (user_data))->shell);
gnome_dialog_run_and_close (GNOME_DIALOG (dialog));
dialog = NULL;
} else {
gdk_window_raise (GTK_WIDGET (dialog)->window);
}
}
/*
* FIXME: This routine could be made generic, by having a closure
* function plus data, and having the whole process be taken care
* of for you
*/
static void
do_mail_print (FolderBrowser *fb, gboolean preview)
{
GnomePrintContext *print_context;
GnomePrintMaster *print_master;
GnomePrintDialog *dialog;
GnomePrinter *printer = NULL;
int copies = 1;
int collate = FALSE;
if (!preview) {
dialog = GNOME_PRINT_DIALOG (gnome_print_dialog_new (_("Print Message"),
GNOME_PRINT_DIALOG_COPIES));
gnome_dialog_set_default (GNOME_DIALOG (dialog), GNOME_PRINT_PRINT);
gnome_dialog_set_parent (GNOME_DIALOG (dialog), GTK_WINDOW (fb));
switch (gnome_dialog_run (GNOME_DIALOG (dialog))) {
case GNOME_PRINT_PRINT:
break;
case GNOME_PRINT_PREVIEW:
preview = TRUE;
break;
case -1:
return;
default:
gnome_dialog_close (GNOME_DIALOG (dialog));
return;
}
gnome_print_dialog_get_copies (dialog, &copies, &collate);
printer = gnome_print_dialog_get_printer (dialog);
gnome_dialog_close (GNOME_DIALOG (dialog));
}
print_master = gnome_print_master_new ();
/* FIXME: set paper size gnome_print_master_set_paper (print_master, */
if (printer)
gnome_print_master_set_printer (print_master, printer);
gnome_print_master_set_copies (print_master, copies, collate);
print_context = gnome_print_master_get_context (print_master);
gtk_html_print (fb->mail_display->html, print_context);
gnome_print_master_close (print_master);
if (preview){
gboolean landscape = FALSE;
GnomePrintMasterPreview *preview;
preview = gnome_print_master_preview_new_with_orientation (
print_master, _("Print Preview"), landscape);
gtk_widget_show (GTK_WIDGET (preview));
} else {
int result = gnome_print_master_print (print_master);
if (result == -1){
e_notice (NULL, GNOME_MESSAGE_BOX_ERROR,
_("Printing of message failed"));
}
}
gtk_object_unref (GTK_OBJECT (print_master));
}
void
print_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = user_data;
do_mail_print (fb, FALSE);
}
void
print_preview_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = user_data;
do_mail_print (fb, TRUE);
}
/******************** Begin Subscription Dialog ***************************/
static GtkObject *subscribe_dialog = NULL;
static void
subscribe_dialog_destroy (GtkWidget *widget, gpointer user_data)
{
subscribe_dialog = NULL;
}
void
manage_subscriptions (BonoboUIComponent *uih, void *user_data, const char *path)
{
if (!subscribe_dialog) {
subscribe_dialog = subscribe_dialog_new ();
gtk_signal_connect (GTK_OBJECT (subscribe_dialog), "destroy",
subscribe_dialog_destroy, NULL);
subscribe_dialog_run_and_close (SUBSCRIBE_DIALOG (subscribe_dialog));
gtk_object_unref (GTK_OBJECT (subscribe_dialog));
} else {
gdk_window_raise (SUBSCRIBE_DIALOG (subscribe_dialog)->app->window);
}
}
/******************** End Subscription Dialog ***************************/
void
configure_folder (BonoboUIComponent *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER(user_data);
mail_local_reconfigure_folder(fb);
}
static void
do_view_message (CamelFolder *folder, char *uid, CamelMimeMessage *message, void *data)
{
FolderBrowser *fb = FOLDER_BROWSER (data);
if (message && fb) {
GtkWidget *mb;
camel_folder_set_message_flags (folder, uid, CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN);
mb = message_browser_new (fb->shell, fb->uri, uid);
gtk_widget_show (mb);
}
}
void
view_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
GPtrArray *uids;
int i;
if (!fb->folder)
return;
uids = g_ptr_array_new ();
message_list_foreach (fb->message_list, enumerate_msg, uids);
if (uids->len > 10 && !are_you_sure (_("Are you sure you want to open all %d messages in separate windows?"), uids, fb))
return;
for (i = 0; i < uids->len; i++) {
mail_get_message (fb->folder, uids->pdata [i], do_view_message, fb, mail_thread_queued);
g_free (uids->pdata [i]);
}
g_ptr_array_free (uids, TRUE);
}
void
open_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
extern CamelFolder *outbox_folder;
if (folder_browser_is_drafts (fb) || fb->folder == outbox_folder)
edit_msg_internal (fb);
else
view_msg (NULL, user_data);
}
void
open_message (BonoboUIComponent *uih, void *user_data, const char *path)
{
open_msg (NULL, user_data);
}
void
edit_message (BonoboUIComponent *uih, void *user_data, const char *path)
{
edit_msg (NULL, user_data);
}
void
stop_threads (BonoboUIComponent *uih, void *user_data, const char *path)
{
camel_operation_cancel (NULL);
}
static void
empty_trash_expunged_cb (CamelFolder *folder, void *data)
{
camel_object_unref (CAMEL_OBJECT (folder));
}
void
empty_trash (BonoboUIComponent *uih, void *user_data, const char *path)
{
MailConfigAccount *account;
CamelProvider *provider;
CamelFolder *vtrash;
const GSList *accounts;
CamelException ex;
gboolean async;
/* the only time all three args are NULL is for empty-on-exit */
async = !(uih == NULL && user_data == NULL && path == NULL);
camel_exception_init (&ex);
/* expunge all remote stores */
accounts = mail_config_get_accounts ();
while (accounts) {
account = accounts->data;
/* make sure this is a valid source */
if (account->source && account->source->url) {
provider = camel_session_get_provider (session, account->source->url, &ex);
if (provider) {
/* make sure this store is a remote store */
if (provider->flags & CAMEL_PROVIDER_IS_STORAGE &&
provider->flags & CAMEL_PROVIDER_IS_REMOTE) {
vtrash = mail_tool_get_trash (account->source->url, &ex);
if (vtrash) {
if (async)
mail_expunge_folder (vtrash, empty_trash_expunged_cb, NULL);
else
camel_folder_sync (vtrash, TRUE, NULL);
}
}
}
/* clear the exception for the next round */
camel_exception_clear (&ex);
}
accounts = accounts->next;
}
/* Now empty the local trash folder */
vtrash = mail_tool_get_trash ("file:/", &ex);
if (vtrash) {
if (async)
mail_expunge_folder (vtrash, empty_trash_expunged_cb, NULL);
else
camel_folder_sync (vtrash, TRUE, NULL);
}
camel_exception_clear (&ex);
}
static void
create_folders (EvolutionStorage *storage, const char *prefix, CamelFolderInfo *fi)
{
char *path;
mail_folder_cache_set_update_estorage (fi->url, storage);
mail_folder_cache_note_folderinfo (fi->url, fi);
path = g_strdup_printf ("%s/%s", prefix, fi->name);
if (!strncmp (fi->url, "vtrash:", 7))
evolution_storage_new_folder (storage, path, fi->name,
"vtrash", fi->url,
fi->name, /* description */
fi->unread_message_count > 0);
else
evolution_storage_new_folder (storage, path, fi->name,
"mail", fi->url,
fi->name, /* description */
fi->unread_message_count > 0);
if (fi->child)
create_folders (storage, path, fi->child);
g_free (path);
if (fi->sibling)
create_folders (storage, prefix, fi->sibling);
}
void
folder_created (CamelStore *store, const char *prefix, CamelFolderInfo *fi)
{
EvolutionStorage *storage;
if ((storage = mail_lookup_storage (store))) {
create_folders (storage, prefix, fi);
bonobo_object_unref (BONOBO_OBJECT (storage));
}
}
void
mail_storage_create_folder (EvolutionStorage *storage, CamelStore *store, CamelFolderInfo *fi)
{
gboolean unref = FALSE;
if (!storage && store) {
storage = mail_lookup_storage (store);
unref = TRUE;
}
if (storage) {
if (fi)
create_folders (storage, "", fi);
if (unref)
bonobo_object_unref (BONOBO_OBJECT (storage));
}
}
static void
delete_folders (EvolutionStorage *storage, CamelFolderInfo *fi)
{
char *path;
if (fi->child)
delete_folders (storage, fi->child);
path = g_strdup_printf ("/%s", fi->full_name);
evolution_storage_removed_folder (storage, path);
g_free (path);
if (fi->sibling)
delete_folders (storage, fi->sibling);
}
void
folder_deleted (CamelStore *store, CamelFolderInfo *fi)
{
EvolutionStorage *storage;
if ((storage = mail_lookup_storage (store))) {
if (fi)
delete_folders (storage, fi);
bonobo_object_unref (BONOBO_OBJECT (storage));
}
}