sync with sylpheed 0.6.0cvs4
[claws.git] / src / compose.c
index 919400b2957927d3735b160ed201ce0ae68bbb9e..a56c70e3382ab00687dba903c36d1a422af87499 100644 (file)
@@ -98,6 +98,7 @@
 #include "gtkshruler.h"
 #include "folder.h"
 #include "addr_compl.h"
+#include "template_select.h"
 
 #if USE_GPGME
 #  include "rfc2015.h"
@@ -170,6 +171,7 @@ static gint compose_write_body_to_file              (Compose        *compose,
                                                 const gchar    *file);
 static gint compose_save_to_outbox             (Compose        *compose,
                                                 const gchar    *file);
+static gint compose_remove_reedit_target       (Compose        *compose);
 static gint compose_queue                      (Compose        *compose,
                                                 const gchar    *file);
 static void compose_write_attach               (Compose        *compose,
@@ -241,6 +243,10 @@ static void toolbar_linewrap_cb            (GtkWidget      *widget,
                                         gpointer        data);
 static void toolbar_address_cb         (GtkWidget      *widget,
                                         gpointer        data);
+static void template_select_cb         (gpointer        daat,
+                                        guint           action,
+                                        GtkWidget      *widget);
+
 
 static void select_account(Compose * compose, PrefsAccount * ac);
 static void account_activated          (GtkMenuItem    *menuitem,
@@ -448,6 +454,7 @@ static GtkItemFactoryEntry compose_entries[] =
        {N_("/_Tool"),                  NULL, NULL,     0, "<Branch>"},
        {N_("/_Tool/Show _ruler"),      NULL, compose_toggle_ruler_cb,  0, "<ToggleItem>"},
        {N_("/_Tool/_Address book"),    "<alt>A",       compose_address_cb, 0, NULL},
+       {N_("/_Tool/_Templates ..."),   NULL, template_select_cb, 0, NULL},
        {N_("/_Help"),                  NULL, NULL,     0, "<LastBranch>"},
        {N_("/_Help/_About"),           NULL,           about_show,     0, NULL}
 };
@@ -481,6 +488,7 @@ Compose * compose_generic_new(PrefsAccount *account, const gchar *to, FolderItem
 
        compose = compose_create(account);
        compose->mode = COMPOSE_NEW;
+       compose->replyinfo = NULL;
 
        if (prefs_common.auto_sig)
                compose_insert_sig(compose);
@@ -582,6 +590,14 @@ static void compose_generic_reply(MsgInfo *msginfo, gboolean quote,
                extract_address(to);
                account = account_find_from_address(to);
        }
+        if(!account&& prefs_common.reply_account_autosel) {
+                       gchar cc[BUFFSIZE];
+               if(!get_header_from_msginfo(msginfo,cc,sizeof(cc),"CC:")){ /* Found a CC header */
+                       extract_address(cc);
+                       account = account_find_from_address(cc);
+                }        
+       }
+
        if (!account) account = cur_account;
        g_return_if_fail(account != NULL);
 
@@ -602,6 +618,7 @@ static void compose_generic_reply(MsgInfo *msginfo, gboolean quote,
 
        compose = compose_create(reply_account);
        compose->mode = COMPOSE_REPLY;
+       compose->replyinfo = msginfo;
 
 
        if (followup_and_reply_to) {
@@ -611,6 +628,13 @@ static void compose_generic_reply(MsgInfo *msginfo, gboolean quote,
                compose->use_to = TRUE;
        }
 
+       if(msginfo->folder && msginfo->folder->prefs->request_return_receipt) {
+               GtkItemFactory *ifactory;
+       
+               ifactory = gtk_item_factory_from_widget(compose->menubar);
+               menu_set_toggle(ifactory, "/Message/Request Return Receipt", TRUE);
+       }
+
        if (compose_parse_header(compose, msginfo) < 0) return;
        compose_reply_set_entry(compose, msginfo, to_all, ignore_replyto,
                                followup_and_reply_to);
@@ -894,7 +918,23 @@ Compose * compose_forward(PrefsAccount * account, MsgInfo *msginfo,
        g_return_val_if_fail(msginfo != NULL, NULL);
        g_return_val_if_fail(msginfo->folder != NULL, NULL);
 
-       if (account == NULL) {
+       account = msginfo->folder->folder->account;
+        if (!account && msginfo->to && prefs_common.forward_account_autosel) {
+               gchar *to;
+               Xstrdup_a(to, msginfo->to, return);
+               extract_address(to);
+               account = account_find_from_address(to);
+       }
+
+        if(!account&& prefs_common.forward_account_autosel) {
+                       gchar cc[BUFFSIZE];
+               if(!get_header_from_msginfo(msginfo,cc,sizeof(cc),"CC:")){ /* Found a CC header */
+                       extract_address(cc);
+                       account = account_find_from_address(cc);
+                }
+       }
+
+        if (account == NULL) {
                account = cur_account;
                /*
                account = msginfo->folder->folder->account;
@@ -989,7 +1029,7 @@ Compose * compose_forward_multiple(PrefsAccount * account,
                if ( ((MsgInfo *)msginfo->data)->folder == NULL )
                        return NULL;
        }
-       
+
        if (account == NULL) {
                account = cur_account;
                /*
@@ -1040,12 +1080,21 @@ void compose_reedit(MsgInfo *msginfo)
        g_return_if_fail(msginfo != NULL);
        g_return_if_fail(msginfo->folder != NULL);
 
-       account = msginfo->folder->folder->account;
-       if (!account) account = cur_account;
+        account = msginfo->folder->folder->account;
+
+        if(!account&& prefs_common.reedit_account_autosel) {
+                       gchar from[BUFFSIZE];
+               if(!get_header_from_msginfo(msginfo,from,sizeof(from),"FROM:")){ /* Found a FROM header */
+                       extract_address(from);
+                       account = account_find_from_address(from);
+                }
+       }
+        if (!account) account = cur_account;
        g_return_if_fail(account != NULL);
 
        compose = compose_create(account);
-       compose->mode = COMPOSE_REEDIT_DRAFT;
+       compose->mode = COMPOSE_REEDIT;
+
        compose->targetinfo = procmsg_msginfo_copy(msginfo);
 
        if (compose_parse_header(compose, msginfo) < 0) return;
@@ -1141,7 +1190,7 @@ static gint compose_parse_header(Compose *compose, MsgInfo *msginfo)
                hentry[H_CC].body = NULL;
        }
        if (hentry[H_REFERENCES].body != NULL) {
-               if (compose->mode == COMPOSE_REEDIT_DRAFT)
+               if (compose->mode == COMPOSE_REEDIT)
                        compose->references = hentry[H_REFERENCES].body;
                else {
                        compose->references = compose_parse_references
@@ -1151,7 +1200,7 @@ static gint compose_parse_header(Compose *compose, MsgInfo *msginfo)
                hentry[H_REFERENCES].body = NULL;
        }
        if (hentry[H_BCC].body != NULL) {
-               if (compose->mode == COMPOSE_REEDIT_DRAFT) {
+               if (compose->mode == COMPOSE_REEDIT) {
                        conv_unmime_header_overwrite(hentry[H_BCC].body);
                        compose->bcc = hentry[H_BCC].body;
                } else
@@ -1168,9 +1217,9 @@ static gint compose_parse_header(Compose *compose, MsgInfo *msginfo)
                hentry[H_FOLLOWUP_TO].body = NULL;
        }
 
-       if (compose->mode == COMPOSE_REEDIT_DRAFT && msginfo->inreplyto)
+       if (compose->mode == COMPOSE_REEDIT && msginfo->inreplyto)
                compose->inreplyto = g_strdup(msginfo->inreplyto);
-       else if (compose->mode != COMPOSE_REEDIT_DRAFT &&
+       else if (compose->mode != COMPOSE_REEDIT &&
                 msginfo->msgid && *msginfo->msgid) {
                compose->inreplyto = g_strdup(msginfo->msgid);
 
@@ -1621,6 +1670,49 @@ static void compose_reedit_set_entry(Compose *compose, MsgInfo *msginfo)
 
 #undef SET_ENTRY
 
+static void compose_exec_sig(Compose *compose, gchar *sigfile)
+{
+       FILE *tmpfp;
+       pid_t thepid;
+       gchar *sigtext;
+       FILE  *sigprg;
+       gchar  *buf;
+       size_t buf_len = 128;
+       if (strlen(sigfile) < 2)
+         return;
+       sigprg = popen(sigfile+1, "r");
+       if (sigprg) {
+
+               buf = g_malloc(buf_len);
+
+               if (!buf) {
+                       gtk_stext_insert(GTK_STEXT(compose->text), NULL, NULL, NULL, \
+                       "Unable to insert signature (malloc failed)\n", -1);
+
+                       pclose(sigprg);
+                       return;
+               }
+
+               while (!feof(sigprg)) {
+                       bzero(buf, buf_len);
+                       fread(buf, buf_len-1, 1, sigprg);
+                       gtk_stext_insert(GTK_STEXT(compose->text), NULL, NULL, NULL, buf, -1);
+               }
+
+               g_free(buf);
+               pclose(sigprg);
+       }
+       else
+       {
+               gtk_stext_insert(GTK_STEXT(compose->text), NULL, NULL, NULL, \
+               "Can't exec file: ", -1);
+               gtk_stext_insert(GTK_STEXT(compose->text), NULL, NULL, NULL, \
+               sigfile+1, -1);
+       }
+}
+
 static void compose_insert_sig(Compose *compose)
 {
        gchar *sigfile;
@@ -1632,7 +1724,7 @@ static void compose_insert_sig(Compose *compose)
                                      DEFAULT_SIGNATURE, NULL);
        }
 
-       if (!is_file_or_fifo_exist(sigfile)) {
+       if (!is_file_or_fifo_exist(sigfile) & (sigfile[0] != '|')) {
                g_free(sigfile);
                return;
        }
@@ -1645,7 +1737,14 @@ static void compose_insert_sig(Compose *compose)
                                "\n", 1);
        }
 
-       compose_insert_file(compose, sigfile);
+       if (sigfile[0] == '|')
+       {
+               compose_exec_sig(compose, sigfile);
+       }
+       else
+       {
+               compose_insert_file(compose, sigfile);
+       }
        g_free(sigfile);
 }
 
@@ -2013,6 +2112,13 @@ gint compose_send(Compose *compose)
                        }
                } else
                        alertpanel_error(_("Error occurred while sending the message."));
+       } else {
+               if (compose->mode == COMPOSE_REEDIT) {
+                       compose_remove_reedit_target(compose);
+                       if (compose->targetinfo)
+                               folderview_update_item
+                                       (compose->targetinfo->folder, TRUE);
+               }
        }
 
        /* save message to outbox */
@@ -2228,6 +2334,29 @@ static gint compose_save_to_outbox(Compose *compose, const gchar *file)
        return 0;
 }
 
+static gint compose_remove_reedit_target(Compose *compose)
+{
+       FolderItem *item;
+       MsgInfo *msginfo = compose->targetinfo;
+
+       g_return_val_if_fail(compose->mode == COMPOSE_REEDIT, -1);
+       if (!msginfo) return -1;
+
+       item = msginfo->folder;
+       g_return_val_if_fail(item != NULL, -1);
+
+       folder_item_scan(item);
+       if (procmsg_msg_exist(msginfo) &&
+           (item->stype == F_DRAFT || item->stype == F_QUEUE)) {
+               if (folder_item_remove_msg(item, msginfo->msgnum) < 0) {
+                       g_warning(_("can't remove the old message\n"));
+                       return -1;
+               }
+       }
+
+       return 0;
+}
+
 static gint compose_queue(Compose *compose, const gchar *file)
 {
        FolderItem *queue;
@@ -2288,6 +2417,8 @@ static gint compose_queue(Compose *compose, const gchar *file)
        for (cur = compose->to_list->next; cur != NULL; cur = cur->next)
                fprintf(fp, ",<%s>", (gchar *)cur->data);
        fprintf(fp, "\n");
+       /* Sylpheed account ID */
+       fprintf(fp, "AID:%d\n", compose->account->account_id);
        fprintf(fp, "\n");
 
        while (fgets(buf, sizeof(buf), src_fp) != NULL) {
@@ -2310,6 +2441,7 @@ static gint compose_queue(Compose *compose, const gchar *file)
        }
 
        queue = folder_get_default_queue();
+
        folder_item_scan(queue);
        queue_path = folder_item_get_path(queue);
        if (!is_dir_exist(queue_path))
@@ -2323,6 +2455,14 @@ static gint compose_queue(Compose *compose, const gchar *file)
        }
        g_free(tmp);
 
+       if (compose->mode == COMPOSE_REEDIT) {
+               compose_remove_reedit_target(compose);
+               if (compose->targetinfo &&
+                   compose->targetinfo->folder != queue)
+                       folderview_update_item
+                               (compose->targetinfo->folder, TRUE);
+       }
+
        if ((fp = procmsg_open_mark_file(queue_path, TRUE)) == NULL)
                g_warning(_("can't open mark file\n"));
        else {
@@ -2522,12 +2662,9 @@ static gint compose_write_headers(Compose *compose, FILE *fp,
                        if (*str != '\0') {
                                compose->to_list = address_list_append
                                        (compose->to_list, str);
-                               if (is_draft) {
-                                       compose_convert_header
-                                               (buf, sizeof(buf), str,
-                                                strlen("Bcc: "));
-                                       fprintf(fp, "Bcc: %s\n", buf);
-                               }
+                               compose_convert_header(buf, sizeof(buf), str,
+                                                      strlen("Bcc: "));
+                               fprintf(fp, "Bcc: %s\n", buf);
                        }
                }
        }
@@ -2834,6 +2971,8 @@ static Compose *compose_create(PrefsAccount *account)
                           GTK_SIGNAL_FUNC(manage_window_focus_out), NULL);
        gtk_widget_realize(window);
 
+       gtkut_widget_set_composer_icon(window);
+
        vbox = gtk_vbox_new(FALSE, 0);
        gtk_container_add(GTK_CONTAINER(window), vbox);
 
@@ -3538,8 +3677,8 @@ static void compose_attach_property(Compose *compose)
        GtkCList *clist = GTK_CLIST(compose->attach_clist);
        AttachInfo *ainfo;
        gint row;
-       gboolean cancelled;
        GtkOptionMenu *optmenu;
+       static gboolean cancelled;
 
        if (!clist->selection) return;
        row = GPOINTER_TO_INT(clist->selection->data);
@@ -3574,7 +3713,8 @@ static void compose_attach_property(Compose *compose)
                GtkWidget *menu;
                GtkWidget *menuitem;
 
-               gtk_main();
+               cancelled = FALSE;
+                gtk_main();
 
                if (cancelled == TRUE) {
                        gtk_widget_hide(attach_prop.window);
@@ -4324,15 +4464,8 @@ static void compose_draft_cb(gpointer data, guint action, GtkWidget *widget)
        gchar *tmp;
 
        draft = folder_get_default_draft();
-       folder_item_scan(draft);
 
-       if (procmsg_msg_exist(compose->targetinfo) &&
-           compose->targetinfo->folder == draft) {
-               if (folder_item_remove_msg(draft,
-                                          compose->targetinfo->msgnum) < 0)
-                       g_warning(_("can't remove the old draft message\n"));
-       }
-       tmp = g_strdup_printf("%s%cdraft.%d", g_get_tmp_dir(),
+        tmp = g_strdup_printf("%s%cdraft.%d", g_get_tmp_dir(),
                              G_DIR_SEPARATOR, (gint)compose);
 
        if (compose_write_to_file(compose, tmp, TRUE) < 0) {
@@ -4340,15 +4473,24 @@ static void compose_draft_cb(gpointer data, guint action, GtkWidget *widget)
                return;
        }
 
+       folder_item_scan(draft);
        if (folder_item_add_msg(draft, tmp, TRUE) < 0) {
                unlink(tmp);
                g_free(tmp);
                return;
        }
-
        g_free(tmp);
 
-       /* folderview_scan_folder_a(DRAFT_DIR, TRUE); */
+       if (compose->mode == COMPOSE_REEDIT) {
+               compose_remove_reedit_target(compose);
+               if (compose->targetinfo &&
+                   compose->targetinfo->folder != draft)
+                       folderview_update_item(compose->targetinfo->folder,
+                                              TRUE);
+       }
+
+       folder_item_scan(draft);
+       folderview_update_item(draft, TRUE);
 
        gtk_widget_destroy(compose->window);
 }
@@ -4836,3 +4978,62 @@ static gchar *compose_quote_fmt          (Compose        *compose,
 
        return quote_fmt_get_buffer();
 }
+
+static void template_apply_cb(gchar *s, gpointer data)
+{
+       Compose *compose = (Compose*)data;
+       GtkSText *text = GTK_STEXT(compose->text);
+       gchar *quote_str;
+       gchar *qmark;
+       gchar *parsed_text;
+       gchar *tmpl;
+       gchar *old_tmpl = s;
+
+       if(!s) return;
+       
+       if(compose->replyinfo == NULL) {
+               gtk_stext_freeze(text);
+               gtk_stext_set_point(text, 0);
+               gtk_stext_forward_delete(text, gtk_stext_get_length(text));
+               gtk_stext_insert(text, NULL, NULL, NULL, s, -1);
+               gtk_stext_thaw(text);
+               g_free(old_tmpl);
+               return;
+       }
+
+       parsed_text = g_new(gchar, strlen(s)*2 + 1);
+       tmpl = parsed_text;
+       while(*s) {
+               if (*s == '\n') {
+                       *parsed_text++ = '\\';
+                       *parsed_text++ = 'n';
+                       s++;
+               } else {
+                       *parsed_text++ = *s++;
+               }
+       }
+       *parsed_text = '\0';
+
+       if (prefs_common.quotemark && *prefs_common.quotemark)
+               qmark = prefs_common.quotemark;
+       else
+               qmark = "> ";
+
+       quote_str = compose_quote_fmt(compose, compose->replyinfo, tmpl, qmark);
+       if (quote_str != NULL) {
+               gtk_stext_freeze(text);
+               gtk_stext_set_point(text, 0);
+               gtk_stext_forward_delete(text, gtk_stext_get_length(text));
+               gtk_stext_insert(text, NULL, NULL, NULL, quote_str, -1);
+               gtk_stext_thaw(text);
+       }
+
+       g_free(old_tmpl);
+       g_free(tmpl);
+}
+
+static void template_select_cb(gpointer data, guint action,
+                              GtkWidget *widget)
+{
+       template_select(&template_apply_cb, data);
+}