/* -*- 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@helixcode.com>
* Jeffrey Stedfast <fejj@helixcode.com>
* Peter Williams <peterw@helixcode.com>
*
* Copyright 2000 Helix Code, Inc. (http://www.helixcode.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
*/
#include <config.h>
#include <ctype.h>
#include <errno.h>
#include <gnome.h>
#include <libgnomeprint/gnome-print-master.h>
#include <libgnomeprint/gnome-print-master-preview.h>
#include "mail.h"
#include "mail-threads.h"
#include "folder-browser.h"
#include "e-util/e-setup.h"
#include "filter/filter-editor.h"
#include "filter/filter-driver.h"
#include "widgets/e-table/e-table.h"
#include "mail-local.h"
/* FIXME: is there another way to do this? */
#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;
const char *uid;
guint32 flags;
};
typedef struct rfm_s {
FolderBrowser *fb;
MailConfigService *source;
} rfm_t;
typedef struct rsm_s {
EMsgComposer *composer;
CamelTransport *transport;
CamelMimeMessage *message;
const char *subject;
char *from;
struct post_send_data *psd;
gboolean ok;
} rsm_t;
static void
real_fetch_mail( gpointer user_data );
static void
real_send_mail( gpointer user_data );
static void
cleanup_send_mail( gpointer userdata );
static void
mail_exception_dialog (char *head, CamelException *ex, gpointer widget)
{
char *msg;
GtkWindow *window =
GTK_WINDOW (gtk_widget_get_ancestor (widget, GTK_TYPE_WINDOW));
msg = g_strdup_printf ("%s:\n%s", head,
camel_exception_get_description (ex));
gnome_error_dialog_parented (msg, window);
g_free (msg);
}
#ifdef USE_BROKEN_THREADS
static void
async_mail_exception_dialog (char *head, CamelException *ex, gpointer unused )
{
mail_op_error( "%s: %s", head, camel_exception_get_description( ex ) );
}
#else
#define async_mail_exception_dialog mail_exception_dialog
#endif
static gboolean
check_configured (void)
{
if (mail_config_is_configured ())
return TRUE;
mail_config_druid ();
return mail_config_is_configured ();
}
static void
select_first_unread (CamelFolder *folder, int type, gpointer data)
{
FolderBrowser *fb = data;
ETable *table = E_TABLE_SCROLLED (fb->message_list->etable)->table;
int mrow;
mrow = e_table_view_to_model_row (table, 0);
message_list_select (fb->message_list, mrow, MESSAGE_LIST_SELECT_NEXT,
0, CAMEL_MESSAGE_SEEN);
}
static CamelFolder *
filter_get_folder(FilterDriver *fd, const char *uri, void *data)
{
return mail_uri_to_folder(uri);
}
static void
fetch_remote_mail (CamelFolder *source, CamelFolder *dest,
gboolean keep_on_server, FolderBrowser *fb,
CamelException *ex)
{
CamelUIDCache *cache;
GPtrArray *uids;
int i;
uids = camel_folder_get_uids (source);
if (keep_on_server) {
GPtrArray *new_uids;
char *url, *p, *filename;
url = camel_url_to_string (
CAMEL_SERVICE (source->parent_store)->url, FALSE);
for (p = url; *p; p++) {
if (!isascii ((unsigned char)*p) ||
strchr (" /'\"`&();|<>${}!", *p))
*p = '_';
}
filename = g_strdup_printf ("%s/config/cache-%s",
evolution_dir, url);
g_free (url);
cache = camel_uid_cache_new (filename);
g_free (filename);
if (cache) {
new_uids = camel_uid_cache_get_new_uids (cache, uids);
camel_folder_free_uids (source, uids);
uids = new_uids;
} else {
async_mail_exception_dialog ("Could not read UID "
"cache file. You may "
"receive duplicate "
"messages.", NULL, fb);
}
} else
cache = NULL;
printf ("got %d new messages in source\n", uids->len);
for (i = 0; i < uids->len; i++) {
CamelMimeMessage *msg;
msg = camel_folder_get_message (source, uids->pdata[i], ex);
if (camel_exception_is_set (ex)) {
async_mail_exception_dialog ("Unable to get message",
ex, fb);
goto done;
}
/* Append with flags = 0 since this is a new message */
camel_folder_append_message (dest, msg, 0, ex);
if (camel_exception_is_set (ex)) {
async_mail_exception_dialog ("Unable to write message",
ex, fb);
gtk_object_unref (GTK_OBJECT (msg));
goto done;
}
if (!cache)
camel_folder_delete_message (source, uids->pdata[i]);
gtk_object_unref (GTK_OBJECT (msg));
}
camel_folder_sync (source, TRUE, ex);
done:
if (cache) {
camel_uid_cache_free_uids (uids);
if (!camel_exception_is_set (ex))
camel_uid_cache_save (cache);
camel_uid_cache_destroy (cache);
} else
camel_folder_free_uids (source, uids);
}
void
real_fetch_mail (gpointer user_data)
{
rfm_t *info;
FolderBrowser *fb = NULL;
CamelException *ex;
CamelStore *store = NULL, *dest_store = NULL;
CamelFolder *folder = NULL, *dest_folder = NULL;
char *url = NULL, *dest_url;
FilterContext *fc = NULL;
FilterDriver *driver = NULL;
char *userrules, *systemrules;
char *tmp_mbox = NULL, *source;
guint handler_id = 0;
struct stat st;
gboolean keep;
info = (rfm_t *) user_data;
fb = info->fb;
url = info->source->url;
keep = info->source->keep_on_server;
/* If using IMAP, don't do anything... */
if (!strncmp (url, "imap:", 5))
return;
ex = camel_exception_new ();
dest_url = g_strdup_printf ("mbox://%s/local/Inbox", evolution_dir);
dest_store = camel_session_get_store (session, dest_url, ex);
g_free (dest_url);
if (!dest_store) {
async_mail_exception_dialog ("Unable to get new mail", ex, fb);
goto cleanup;
}
dest_folder = camel_store_get_folder (dest_store, "mbox", FALSE, ex);
if (!dest_folder) {
async_mail_exception_dialog ("Unable to get new mail", ex, fb);
goto cleanup;
}
tmp_mbox = g_strdup_printf ("%s/local/Inbox/movemail", evolution_dir);
/* If fetching mail from an mbox store, safely copy it to a
* temporary store first.
*/
if (!strncmp (url, "mbox:", 5)) {
int tmpfd;
tmpfd = open (tmp_mbox, O_RDWR | O_CREAT | O_APPEND, 0660);
if (tmpfd == -1) {
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
"Couldn't create temporary "
"mbox: %s", g_strerror (errno));
async_mail_exception_dialog ("Unable to move mail", ex, fb );
goto cleanup;
}
close (tmpfd);
/* Skip over "mbox:" plus host part (if any) of url. */
source = url + 5;
if (!strncmp (source, "//", 2))
source = strchr (source + 2, '/');
camel_movemail (source, tmp_mbox, ex);
if (camel_exception_is_set (ex)) {
async_mail_exception_dialog ("Unable to move mail",
ex, fb);
goto cleanup;
}
if (stat (tmp_mbox, &st) == -1 || st.st_size == 0) {
gnome_ok_dialog ("No new messages.");
goto cleanup;
}
folder = camel_store_get_folder (dest_store, "movemail",
FALSE, ex);
if (camel_exception_get_id (ex) != CAMEL_EXCEPTION_NONE) {
async_mail_exception_dialog ("Unable to move mail", ex, fb);
goto cleanup;
}
} else {
CamelFolder *sourcefolder;
store = camel_session_get_store(session, url, ex);
if (!store) {
async_mail_exception_dialog("Unable to get new mail", ex, fb);
goto cleanup;
}
camel_service_connect(CAMEL_SERVICE (store), ex);
if (camel_exception_get_id(ex) != CAMEL_EXCEPTION_NONE) {
if (camel_exception_get_id(ex) != CAMEL_EXCEPTION_USER_CANCEL)
async_mail_exception_dialog("Unable to get new mail", ex, fb);
goto cleanup;
}
sourcefolder = camel_store_get_folder(store, "inbox", FALSE, ex);
if (camel_exception_get_id(ex) != CAMEL_EXCEPTION_NONE) {
async_mail_exception_dialog("Unable to get new mail", ex, fb);
goto cleanup;
}
/* can we perform filtering on this source? */
if (!(sourcefolder->has_summary_capability
&& sourcefolder->has_search_capability)) {
folder = camel_store_get_folder (dest_store,
"movemail", TRUE, ex);
if (camel_exception_is_set (ex)) {
async_mail_exception_dialog ("Unable to move mail", ex, fb);
goto cleanup;
}
fetch_remote_mail (sourcefolder, folder, keep, fb, ex);
gtk_object_unref (GTK_OBJECT (sourcefolder));
if (camel_exception_is_set (ex))
goto cleanup;
} else {
folder = sourcefolder;
}
}
if (camel_folder_get_message_count (folder) == 0) {
gnome_ok_dialog ("No new messages.");
goto cleanup;
} else if (camel_exception_is_set (ex)) {
async_mail_exception_dialog ("Unable to get new mail", ex, fb);
goto cleanup;
}
folder_browser_clear_search (fb);
/* apply filtering rules to this inbox */
fc = filter_context_new();
userrules = g_strdup_printf("%s/filters.xml", evolution_dir);
systemrules = g_strdup_printf("%s/evolution/filtertypes.xml", EVOLUTION_DATADIR);
rule_context_load((RuleContext *)fc, systemrules, userrules);
g_free (userrules);
g_free (systemrules);
driver = filter_driver_new(fc, filter_get_folder, 0);
/* Attach a handler to the destination folder to select the first unread
* message iff it changes and iff it's the folder being viewed.
*/
if (dest_folder == fb->folder)
handler_id = gtk_signal_connect (GTK_OBJECT (dest_folder), "folder_changed",
GTK_SIGNAL_FUNC (select_first_unread), fb);
if (filter_driver_run(driver, folder, dest_folder) == -1) {
async_mail_exception_dialog ("Unable to get new mail", ex, fb);
goto cleanup;
}
if (dest_folder == fb->folder)
gtk_signal_disconnect (GTK_OBJECT (dest_folder), handler_id);
cleanup:
if (stat (tmp_mbox, &st) == 0 && st.st_size == 0)
unlink (tmp_mbox); /* FIXME: should use camel to do this */
g_free (tmp_mbox);
if (driver)
gtk_object_unref((GtkObject *)driver);
if (fc)
gtk_object_unref((GtkObject *)fc);
if (folder) {
camel_folder_sync (folder, TRUE, ex);
gtk_object_unref (GTK_OBJECT (folder));
}
if (dest_folder) {
camel_folder_sync (dest_folder, TRUE, ex);
gtk_object_unref (GTK_OBJECT (dest_folder));
}
if (store) {
camel_service_disconnect (CAMEL_SERVICE (store), ex);
gtk_object_unref (GTK_OBJECT (store));
}
if (dest_store && dest_store != fb->folder->parent_store) {
camel_service_disconnect (CAMEL_SERVICE (dest_store), ex);
gtk_object_unref (GTK_OBJECT (dest_store));
}
camel_exception_free (ex);
}
void
fetch_mail (GtkWidget *button, gpointer user_data)
{
MailConfigService *source;
rfm_t *info;
if (!check_configured ())
return;
source = mail_config_get_default_source ();
if (!source || !source->url) {
GtkWidget *win = gtk_widget_get_ancestor (GTK_WIDGET (user_data),
GTK_TYPE_WINDOW);
gnome_error_dialog_parented ("You have no remote mail source "
"configured", GTK_WINDOW (win));
return;
}
/* This must be dynamically allocated so as not to be clobbered
* when we return. Actually, making it static in the whole file
* would probably work.
*/
info = g_new (rfm_t, 1);
info->fb = FOLDER_BROWSER (user_data);
info->source = source;
#ifdef USE_BROKEN_THREADS
mail_operation_try (_("Fetching mail"), real_fetch_mail, NULL, info);
#else
real_fetch_mail (info);
#endif
}
static gboolean
ask_confirm_for_empty_subject (EMsgComposer *composer)
{
GtkWidget *message_box;
int button;
message_box = gnome_message_box_new (_("This message has no subject.\nReally send?"),
GNOME_MESSAGE_BOX_QUESTION,
GNOME_STOCK_BUTTON_YES, GNOME_STOCK_BUTTON_NO,
NULL);
button = gnome_dialog_run_and_close (GNOME_DIALOG (message_box));
if (button == 0)
return TRUE;
else
return FALSE;
}
static void
set_x_mailer_header (CamelMedium *medium)
{
char *mailer_string;
mailer_string = g_strdup_printf ("Evolution %s (Developer Preview)", VERSION);
camel_medium_add_header (medium, "X-Mailer", mailer_string);
g_free (mailer_string);
}
static void
real_send_mail (gpointer user_data)
{
rsm_t *info = (rsm_t *) user_data;
EMsgComposer *composer = NULL;
CamelTransport *transport = NULL;
CamelException *ex = NULL;
CamelMimeMessage *message = NULL;
const char *subject = NULL;
char *from = NULL;
struct post_send_data *psd = NULL;
#ifdef USE_BROKEN_THREADS
mail_op_hide_progressbar ();
mail_op_set_message ("Connecting to transport...");
#endif
ex = camel_exception_new ();
composer = info->composer;
transport = info->transport;
message = info->message;
subject = info->subject;
from = info->from;
psd = info->psd;
set_x_mailer_header (CAMEL_MEDIUM (message));
camel_mime_message_set_from (message, from);
camel_mime_message_set_date (message, CAMEL_MESSAGE_DATE_CURRENT, 0);
camel_service_connect (CAMEL_SERVICE (transport), ex);
#ifdef USE_BROKEN_THREADS
mail_op_set_message ("Connected. Sending...");
#endif
if (!camel_exception_is_set (ex))
camel_transport_send (transport, CAMEL_MEDIUM (message), ex);
if (!camel_exception_is_set (ex)) {
#ifdef USE_BROKEN_THREADS
mail_op_set_message ("Sent. Disconnecting...");
#endif
camel_service_disconnect (CAMEL_SERVICE (transport), ex);
}
if (camel_exception_is_set (ex)) {
async_mail_exception_dialog ("Could not send message", ex, composer);
info->ok = FALSE;
} else {
if (psd) {
camel_folder_set_message_flags (psd->folder, psd->uid,
psd->flags, psd->flags);
}
info->ok = TRUE;
}
camel_exception_free (ex);
}
static void
cleanup_send_mail (gpointer userdata)
{
rsm_t *info = (rsm_t *) userdata;
if (info->ok) {
gtk_object_destroy (GTK_OBJECT (info->composer));
}
gtk_object_unref (GTK_OBJECT (info->message));
g_free (info);
}
static void
composer_send_cb (EMsgComposer *composer, gpointer data)
{
const MailConfigIdentity *id = NULL;
static CamelTransport *transport = NULL;
struct post_send_data *psd = data;
rsm_t *info;
static char *from = NULL;
const char *subject;
CamelException *ex;
CamelMimeMessage *message;
char *name, *addr;
ex = camel_exception_new ();
id = mail_config_get_default_identity ();
if (!check_configured() || !id) {
GtkWidget *message;
message = gnome_warning_dialog_parented (_("You need to configure an identity\n"
"before you can send mail."),
GTK_WINDOW (gtk_widget_get_ancestor (GTK_WIDGET (composer),
GTK_TYPE_WINDOW)));
gnome_dialog_run_and_close (GNOME_DIALOG (message));
return;
}
if (!from) {
CamelInternetAddress *ciaddr;
g_assert (id);
name = id->name;
g_assert (name);
addr = id->address;
g_assert (addr);
ciaddr = camel_internet_address_new ();
camel_internet_address_add (ciaddr, name, addr);
from = camel_address_encode (CAMEL_ADDRESS (ciaddr));
}
if (!transport) {
MailConfigService *t;
char *url;
t = mail_config_get_transport ();
url = t->url;
g_assert (url);
transport = camel_session_get_transport (session, url, ex);
if (camel_exception_get_id (ex) != CAMEL_EXCEPTION_NONE) {
mail_exception_dialog ("Could not load mail transport",
ex, composer);
camel_exception_free (ex);
return;
}
}
message = e_msg_composer_get_message (composer);
subject = camel_mime_message_get_subject (message);
if (!subject || !*subject) {
if (!ask_confirm_for_empty_subject (composer)) {
gtk_object_unref (GTK_OBJECT (message));
return;
}
}
info = g_new0 (rsm_t, 1);
info->composer = composer;
info->transport = transport;
info->message = message;
info->subject = subject;
info->from = from;
info->psd = psd;
#ifdef USE_BROKEN_THREADS
mail_operation_try ("Send Message", real_send_mail, cleanup_send_mail, info);
#else
real_send_mail (info);
cleanup_send_mail (info);
#endif
}
static void
free_psd (GtkWidget *composer, gpointer user_data)
{
struct post_send_data *psd = user_data;
gtk_object_unref (GTK_OBJECT (psd->folder));
g_free (psd);
}
static GtkWidget *
create_msg_composer (const char *url)
{
MailConfigIdentity *id;
gboolean send_html;
gchar *sig_file = NULL;
GtkWidget *composer_widget;
id = mail_config_get_default_identity ();
send_html = mail_config_send_html ();
if (id) {
sig_file = id->sig;
}
if (url != NULL)
composer_widget = e_msg_composer_new_from_url (url);
else
composer_widget = e_msg_composer_new_with_sig_file (sig_file);
e_msg_composer_set_send_html (E_MSG_COMPOSER (composer_widget),
send_html);
return composer_widget;
}
void
compose_msg (GtkWidget *widget, gpointer user_data)
{
GtkWidget *composer;
if (!check_configured ())
return;
composer = create_msg_composer (NULL);
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), NULL);
gtk_widget_show (composer);
}
/* Send according to a mailto (RFC 2368) URL. */
void
send_to_url (const char *url)
{
GtkWidget *composer;
if (!check_configured ())
return;
composer = create_msg_composer (url);
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), NULL);
gtk_widget_show (composer);
}
static void
reply (FolderBrowser *fb, gboolean to_all)
{
EMsgComposer *composer;
struct post_send_data *psd;
if (!check_configured () || !fb->message_list->cursor_uid ||
!fb->mail_display->current_message)
return;
psd = g_new (struct post_send_data, 1);
psd->folder = fb->folder;
gtk_object_ref (GTK_OBJECT (psd->folder));
psd->uid = fb->message_list->cursor_uid;
psd->flags = CAMEL_MESSAGE_ANSWERED;
composer = mail_generate_reply (fb->mail_display->current_message, to_all);
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), psd);
gtk_signal_connect (GTK_OBJECT (composer), "destroy",
GTK_SIGNAL_FUNC (free_psd), psd);
gtk_widget_show (GTK_WIDGET (composer));
}
void
reply_to_sender (GtkWidget *widget, gpointer user_data)
{
reply (FOLDER_BROWSER (user_data), FALSE);
}
void
reply_to_all (GtkWidget *widget, gpointer user_data)
{
reply (FOLDER_BROWSER (user_data), TRUE);
}
static void
attach_msg (MessageList *ml, const char *uid, gpointer data)
{
EMsgComposer *composer = data;
CamelMimeMessage *message;
CamelMimePart *part;
const char *subject;
char *desc;
message = camel_folder_get_message (ml->folder, uid, NULL);
if (!message)
return;
subject = camel_mime_message_get_subject (message);
if (subject)
desc = g_strdup_printf ("Forwarded message - %s", subject);
else
desc = g_strdup ("Forwarded message");
part = camel_mime_part_new ();
camel_mime_part_set_disposition (part, "inline");
camel_mime_part_set_description (part, desc);
camel_medium_set_content_object (CAMEL_MEDIUM (part),
CAMEL_DATA_WRAPPER (message));
camel_mime_part_set_content_type (part, "message/rfc822");
e_msg_composer_attach (composer, part);
gtk_object_unref (GTK_OBJECT (part));
gtk_object_unref (GTK_OBJECT (message));
g_free (desc);
}
void
forward_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
EMsgComposer *composer;
CamelMimeMessage *cursor_msg;
const char *from, *subject;
char *fwd_subj;
cursor_msg = fb->mail_display->current_message;
if (!check_configured () || !cursor_msg)
return;
composer = E_MSG_COMPOSER (create_msg_composer (NULL));
message_list_foreach (fb->message_list, attach_msg, composer);
from = camel_mime_message_get_from (cursor_msg);
subject = camel_mime_message_get_subject (cursor_msg);
if (from) {
if (subject && *subject) {
fwd_subj = g_strdup_printf ("[%s] %s", from, subject);
} else {
fwd_subj = g_strdup_printf ("[%s] (forwarded message)",
from);
}
} else {
fwd_subj = NULL;
}
e_msg_composer_set_headers (composer, NULL, NULL, NULL, fwd_subj);
g_free (fwd_subj);
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), NULL);
gtk_widget_show (GTK_WIDGET (composer));
}
struct move_data {
CamelFolder *source, *dest;
CamelException *ex;
};
static void
real_move_msg (MessageList *ml, const char *uid, gpointer user_data)
{
struct move_data *rfd = user_data;
if (camel_exception_is_set (rfd->ex))
return;
camel_folder_move_message_to (rfd->source, uid, rfd->dest, rfd->ex);
}
void
move_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = user_data;
MessageList *ml = fb->message_list;
char *uri, *physical, *path;
struct move_data rfd;
const char *allowed_types[] = { "mail", NULL };
extern EvolutionShellClient *global_shell_client;
static char *last = NULL;
if (!last)
last = g_strdup ("");
evolution_shell_client_user_select_folder (global_shell_client,
_("Move message(s) to"),
last, allowed_types, &uri, &physical);
if (!uri)
return;
path = strchr (uri, '/');
if (path && strcmp (last, path) != 0) {
g_free (last);
last = g_strdup (path);
}
g_free (uri);
rfd.source = ml->folder;
rfd.dest = mail_uri_to_folder (physical);
g_free (physical);
if (!rfd.dest)
return;
rfd.ex = camel_exception_new ();
message_list_foreach (ml, real_move_msg, &rfd);
gtk_object_unref (GTK_OBJECT (rfd.dest));
if (camel_exception_is_set (rfd.ex))
mail_exception_dialog ("Could not move message", rfd.ex, fb);
camel_exception_free (rfd.ex);
}
void
mark_all_seen (BonoboUIHandler *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER(user_data);
MessageList *ml = fb->message_list;
GPtrArray *uids;
int i;
uids = camel_folder_get_uids (ml->folder);
for (i = 0; i < uids->len; i++) {
camel_folder_set_message_flags (ml->folder, uids->pdata[i],
CAMEL_MESSAGE_SEEN,
CAMEL_MESSAGE_SEEN);
}
}
static void
real_edit_msg (MessageList *ml, const char *uid, gpointer user_data)
{
CamelException *ex = user_data;
CamelMimeMessage *msg;
GtkWidget *composer;
if (camel_exception_is_set (ex))
return;
msg = camel_folder_get_message (ml->folder, uid, ex);
composer = e_msg_composer_new_with_message (msg);
gtk_signal_connect (GTK_OBJECT (composer), "send",
GTK_SIGNAL_FUNC (composer_send_cb), NULL);
gtk_widget_show (composer);
}
void
edit_msg (GtkWidget *widget, gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
MessageList *ml = fb->message_list;
CamelException ex;
extern CamelFolder *drafts_folder;
camel_exception_init (&ex);
if (fb->folder != drafts_folder) {
camel_exception_setv (&ex, CAMEL_EXCEPTION_FOLDER_INSUFFICIENT_PERMISSION,
"FIXME: some error message about not being in the Drafts folder...");
mail_exception_dialog ("Could not open message for editing", &ex, fb);
return;
}
message_list_foreach (ml, real_edit_msg, &ex);
if (camel_exception_is_set (&ex)) {
mail_exception_dialog ("Could not open message for editing", &ex, fb);
camel_exception_clear (&ex);
return;
}
}
void
edit_message (BonoboUIHandler *uih, void *user_data, const char *path)
{
edit_msg (NULL, user_data);
}
static void
real_delete_msg (MessageList *ml, const char *uid, gpointer user_data)
{
CamelException *ex = user_data;
guint32 flags;
if (camel_exception_is_set (ex))
return;
/* Toggle the deleted flag without touching other flags. */
flags = camel_folder_get_message_flags (ml->folder, uid);
camel_folder_set_message_flags (ml->folder, uid,
CAMEL_MESSAGE_DELETED, ~flags);
}
void
delete_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = user_data;
MessageList *ml = fb->message_list;
CamelException ex;
camel_exception_init (&ex);
message_list_foreach (ml, real_delete_msg, &ex);
if (camel_exception_is_set (&ex)) {
mail_exception_dialog ("Could not toggle deleted flag",
&ex, fb);
camel_exception_clear (&ex);
return;
}
}
static void real_expunge_folder (gpointer user_data)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
CamelException ex;
e_table_model_pre_change(fb->message_list->table_model);
#ifdef USE_BROKEN_THREADS
mail_op_hide_progressbar ();
mail_op_set_message ("Expunging %s...", fb->message_list->folder->full_name);
#endif
camel_exception_init (&ex);
camel_folder_expunge (fb->message_list->folder, &ex);
/* FIXME: is there a better way to force an update? */
/* FIXME: Folder should raise a signal to say its contents has changed ... */
e_table_model_changed (fb->message_list->table_model);
if (camel_exception_get_id (&ex) != CAMEL_EXCEPTION_NONE) {
async_mail_exception_dialog ("Unable to expunge deleted messages", &ex, fb);
}
}
void
expunge_folder (BonoboUIHandler *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER(user_data);
if (fb->message_list->folder) {
#ifdef USE_BROKEN_THREADS
mail_operation_try ("Expunge Folder", real_expunge_folder, NULL, fb);
#else
real_expunge_folder (fb);
#endif
}
}
static void
filter_druid_clicked(GtkWidget *w, int button, FolderBrowser *fb)
{
FilterContext *fc;
if (button == 0) {
char *user;
fc = gtk_object_get_data((GtkObject *)w, "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((GnomeDialog *)w);
}
}
void
filter_edit (BonoboUIHandler *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER (user_data);
FilterContext *fc;
char *user, *system;
GtkWidget *w;
fc = filter_context_new();
user = g_strdup_printf("%s/filters.xml", evolution_dir);
system = g_strdup_printf("%s/evolution/filtertypes.xml", EVOLUTION_DATADIR);
rule_context_load((RuleContext *)fc, system, user);
g_free(user);
g_free(system);
w = filter_editor_construct(fc);
gtk_object_set_data_full((GtkObject *)w, "context", fc, (GtkDestroyNotify)gtk_object_unref);
gtk_signal_connect((GtkObject *)w, "clicked", filter_druid_clicked, fb);
gtk_widget_show(w);
}
void
vfolder_edit_vfolders (BonoboUIHandler *uih, void *user_data, const char *path)
{
void vfolder_edit(void);
vfolder_edit();
}
void
providers_config (BonoboUIHandler *uih, void *user_data, const char *path)
{
mail_config();
}
void
print_msg (GtkWidget *button, gpointer user_data)
{
FolderBrowser *fb = user_data;
GnomePrintMaster *print_master;
GnomePrintContext *print_context;
GtkWidget *preview;
print_master = gnome_print_master_new ();
print_context = gnome_print_master_get_context (print_master);
gtk_html_print (fb->mail_display->html, print_context);
preview = GTK_WIDGET (gnome_print_master_preview_new (
print_master, "Mail Print Preview"));
gtk_widget_show (preview);
gtk_object_unref (GTK_OBJECT (print_master));
}
void
configure_folder(BonoboUIHandler *uih, void *user_data, const char *path)
{
FolderBrowser *fb = FOLDER_BROWSER(user_data);
local_reconfigure_folder(fb);
}
struct view_msg_data {
FolderBrowser *fb;
CamelException *ex;
};
static void
real_view_msg (MessageList *ml, const char *uid, gpointer user_data)
{
struct view_msg_data *data = user_data;
FolderBrowser *fb;
CamelMimeMessage *msg;
GtkWidget *view;
if (camel_exception_is_set (data->ex))
return;
msg = camel_folder_get_message (ml->folder, uid, data->ex);
fb = FOLDER_BROWSER (folder_browser_new ());
folder_browser_set_uri (fb, data->fb->uri);
fb->message_list->cursor_uid = uid;
fb->mail_display->current_message = msg;
view = mail_view_create (fb);
gtk_widget_show (view);
}
void
view_msg (GtkWidget *widget, gpointer user_data)
{
struct view_msg_data data;
FolderBrowser *fb = user_data;
FolderBrowser *folder_browser;
CamelException ex;
MessageList *ml;
camel_exception_init (&ex);
data.fb = fb;
data.ex = &ex;
ml = fb->message_list;
message_list_foreach (ml, real_view_msg, &data);
if (camel_exception_is_set (&ex)) {
mail_exception_dialog ("Could not open message for viewing", &ex, fb);
camel_exception_clear (&ex);
return;
}
}
void
view_message (BonoboUIHandler *uih, void *user_data, const char *path)
{
view_msg (NULL, user_data);
}