sync with sylpheed 0.6.0cvs4
[claws.git] / src / compose.c
index f3afaea9fbf70ee874d86b827e33d90a26992655..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);
@@ -576,21 +584,27 @@ static void compose_generic_reply(MsgInfo *msginfo, gboolean quote,
        g_return_if_fail(msginfo->folder != NULL);
 
        account = msginfo->folder->folder->account;
-       if (!account && msginfo->to && prefs_common.reply_account_autoselect) {
+       if (!account && msginfo->to && prefs_common.reply_account_autosel) {
                gchar *to;
-
-               to = g_strdup(msginfo->to);
+               Xstrdup_a(to, msginfo->to, return);
                extract_address(to);
-               account = account_find_mail_from_address(to);
-               g_free(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);
 
        if (ignore_replyto && account->protocol == A_NNTP &&
            !followup_and_reply_to) {
                reply_account =
-                       account_find_mail_from_address(account->address);
+                       account_find_from_address(account->address);
                if (!reply_account)
                        reply_account = compose_current_mail_account();
                if (!reply_account)
@@ -604,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) {
@@ -613,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);
@@ -896,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;
@@ -991,7 +1029,7 @@ Compose * compose_forward_multiple(PrefsAccount * account,
                if ( ((MsgInfo *)msginfo->data)->folder == NULL )
                        return NULL;
        }
-       
+
        if (account == NULL) {
                account = cur_account;
                /*
@@ -1042,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;
@@ -1143,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
@@ -1153,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
@@ -1170,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);
 
@@ -1432,6 +1479,10 @@ static gchar *compose_quote_parse_fmt(Compose *compose, MsgInfo *msginfo,
                                str = msginfo->to;
                                sp++;
                                break;
+                       case 'c':
+                               str = msginfo->cc;
+                               sp++;
+                               break;
                        case 'i':
                                if (!msginfo->msgid) {sp++; break;}
                                Xalloca(str, strlen(msginfo->msgid) + 3,
@@ -1619,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;
@@ -1630,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;
        }
@@ -1643,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);
 }
 
@@ -2011,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 */
@@ -2226,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;
@@ -2286,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) {
@@ -2308,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))
@@ -2321,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 {
@@ -2520,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);
                        }
                }
        }
@@ -2832,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);
 
@@ -2989,6 +3130,7 @@ static Compose *compose_create(PrefsAccount *account)
                            (GTK_SCROLLED_WINDOW(scrolledwin)),
                            gtk_scrolled_window_get_vadjustment
                            (GTK_SCROLLED_WINDOW(scrolledwin)));
+       GTK_STEXT(text)->default_tab_width = 8;
        gtk_stext_set_editable(GTK_STEXT(text), TRUE);
        if (prefs_common.smart_wrapping) {      
                gtk_stext_set_word_wrap(GTK_STEXT(text), TRUE);
@@ -3275,12 +3417,12 @@ static Compose *compose_create(PrefsAccount *account)
 }
 
 #include "pixmaps/stock_mail_send.xpm"
+#include "pixmaps/stock_mail_send_queue.xpm"
 #include "pixmaps/stock_mail.xpm"
 #include "pixmaps/stock_paste.xpm"
 #include "pixmaps/stock_mail_attach.xpm"
 #include "pixmaps/stock_mail_compose.xpm"
 #include "pixmaps/linewrap.xpm"
-/*#include "pixmaps/tb_mail_queue_send.xpm"*/
 #include "pixmaps/tb_address_book.xpm"
 
 #define CREATE_TOOLBAR_ICON(xpm_d) \
@@ -3322,7 +3464,7 @@ static void compose_toolbar_create(Compose *compose, GtkWidget *container)
                                           "Send",
                                           icon_wid, toolbar_send_cb, compose);
 
-       CREATE_TOOLBAR_ICON(stock_mail_send_xpm);
+       CREATE_TOOLBAR_ICON(stock_mail_send_queue_xpm);
        /* CREATE_TOOLBAR_ICON(tb_mail_queue_send_xpm); */
        sendl_btn = gtk_toolbar_append_item(GTK_TOOLBAR(toolbar),
                                           _("Send later"),
@@ -3435,8 +3577,8 @@ static GtkWidget *compose_account_option_menu_create(Compose *compose)
 
                if (ac == compose->account) def_menu = num;
 
-               name = g_strdup_printf("%s <%s> (%s)",
-                                      ac->name, ac->address, ac->account_name);
+               name = g_strdup_printf("%s: %s <%s>",
+                                      ac->account_name, ac->name, ac->address);
                MENUITEM_ADD(menu, menuitem, name, ac);
                g_free(name);
                gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
@@ -3535,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);
@@ -3571,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);
@@ -4321,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) {
@@ -4337,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);
 }
@@ -4833,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);
+}