aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--composer/e-msg-composer.c98
1 files changed, 92 insertions, 6 deletions
diff --git a/composer/e-msg-composer.c b/composer/e-msg-composer.c
index 6f4aee2ff7..89cf9dcfe9 100644
--- a/composer/e-msg-composer.c
+++ b/composer/e-msg-composer.c
@@ -64,6 +64,8 @@
#include <camel/camel-folder.h>
#include <camel/camel-gpg-context.h>
#include <camel/camel-iconv.h>
+#include <camel/camel-mime-filter-basic.h>
+#include <camel/camel-mime-filter-canon.h>
#include <camel/camel-mime-filter-charset.h>
#include <camel/camel-mime-filter-tohtml.h>
#include <camel/camel-multipart-encrypted.h>
@@ -249,6 +251,29 @@ destination_list_to_vector (GList *list)
#define LINE_LEN 72
+static gboolean
+text_requires_quoted_printable (const gchar *text, gsize len)
+{
+ const gchar *p;
+ gsize pos;
+
+ if (!text)
+ return FALSE;
+
+ if (len == -1)
+ len = strlen (text);
+
+ if (len >= 5 && strncmp (text, "From ", 5) == 0)
+ return TRUE;
+
+ for (p = text, pos = 0; pos + 6 <= len; pos++, p++) {
+ if (*p == '\n' && strncmp (p + 1, "From ", 5) == 0)
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
static CamelTransferEncoding
best_encoding (GByteArray *buf, const gchar *charset)
{
@@ -280,7 +305,7 @@ best_encoding (GByteArray *buf, const gchar *charset)
if (status == (gsize) -1 || status > 0)
return -1;
- if ((count == 0) && (buf->len < LINE_LEN))
+ if ((count == 0) && (buf->len < LINE_LEN) && !text_requires_quoted_printable ((const gchar *)buf->data, buf->len))
return CAMEL_TRANSFER_ENCODING_7BIT;
else if (count <= buf->len * 0.17)
return CAMEL_TRANSFER_ENCODING_QUOTEDPRINTABLE;
@@ -607,11 +632,15 @@ build_message (EMsgComposer *composer,
CAMEL_MEDIUM (new), "X-Priority", "1");
if (p->mime_body) {
- plain_encoding = CAMEL_TRANSFER_ENCODING_7BIT;
- for (i = 0; p->mime_body[i]; i++) {
- if ((guchar) p->mime_body[i] > 127) {
- plain_encoding = CAMEL_TRANSFER_ENCODING_QUOTEDPRINTABLE;
- break;
+ if (text_requires_quoted_printable (p->mime_body, -1)) {
+ plain_encoding = CAMEL_TRANSFER_ENCODING_QUOTEDPRINTABLE;
+ } else {
+ plain_encoding = CAMEL_TRANSFER_ENCODING_7BIT;
+ for (i = 0; p->mime_body[i]; i++) {
+ if ((guchar) p->mime_body[i] > 127) {
+ plain_encoding = CAMEL_TRANSFER_ENCODING_QUOTEDPRINTABLE;
+ break;
+ }
}
}
data = g_byte_array_new ();
@@ -652,17 +681,45 @@ build_message (EMsgComposer *composer,
camel_object_unref (filter);
}
+ if (plain_encoding == CAMEL_TRANSFER_ENCODING_QUOTEDPRINTABLE) {
+ /* encode to quoted-printable by ourself, together with
+ taking care of "\nFrom " text */
+ CamelStreamFilter *filter_stream;
+ CamelMimeFilter *mf, *qp;
+
+ if (!CAMEL_IS_STREAM_FILTER (stream)) {
+ filter_stream = camel_stream_filter_new_with_stream (stream);
+ camel_object_unref (stream);
+
+ stream = (CamelStream *) filter_stream;
+ }
+
+ qp = (CamelMimeFilter *) camel_mime_filter_basic_new_type (CAMEL_MIME_FILTER_BASIC_QP_ENC);
+ camel_stream_filter_add (CAMEL_STREAM_FILTER (stream), qp);
+ camel_object_unref (qp);
+
+ mf = camel_mime_filter_canon_new (CAMEL_MIME_FILTER_CANON_FROM);
+ camel_stream_filter_add (CAMEL_STREAM_FILTER (stream), mf);
+ camel_object_unref (mf);
+ }
+
/* construct the content object */
plain = camel_data_wrapper_new ();
camel_data_wrapper_construct_from_stream (plain, stream);
camel_object_unref (stream);
+ if (plain_encoding == CAMEL_TRANSFER_ENCODING_QUOTEDPRINTABLE) {
+ /* to not re-encode the data when pushing it to a part */
+ plain->encoding = plain_encoding;
+ }
+
camel_data_wrapper_set_mime_type_field (plain, type);
camel_content_type_unref (type);
if (html_content) {
gchar *text;
gsize length;
+ gboolean pre_encode;
clear_current_images (composer);
@@ -672,6 +729,7 @@ build_message (EMsgComposer *composer,
data = g_byte_array_new ();
text = gtkhtml_editor_get_text_html (editor, &length);
g_byte_array_append (data, (guint8 *) text, (guint) length);
+ pre_encode = text_requires_quoted_printable (text, length);
g_free (text);
if (save_html_object_data)
@@ -680,10 +738,38 @@ build_message (EMsgComposer *composer,
html = camel_data_wrapper_new ();
stream = camel_stream_mem_new_with_byte_array (data);
+
+ if (pre_encode) {
+ /* encode to quoted-printable by ourself, together with
+ taking care of "\nFrom " text */
+ CamelStreamFilter *filter_stream;
+ CamelMimeFilter *mf, *qp;
+
+ if (!CAMEL_IS_STREAM_FILTER (stream)) {
+ filter_stream = camel_stream_filter_new_with_stream (stream);
+ camel_object_unref (stream);
+
+ stream = (CamelStream *) filter_stream;
+ }
+
+ qp = (CamelMimeFilter *) camel_mime_filter_basic_new_type (CAMEL_MIME_FILTER_BASIC_QP_ENC);
+ camel_stream_filter_add (CAMEL_STREAM_FILTER (stream), qp);
+ camel_object_unref (qp);
+
+ mf = camel_mime_filter_canon_new (CAMEL_MIME_FILTER_CANON_FROM);
+ camel_stream_filter_add (CAMEL_STREAM_FILTER (stream), mf);
+ camel_object_unref (mf);
+ }
+
camel_data_wrapper_construct_from_stream (html, stream);
camel_object_unref (stream);
camel_data_wrapper_set_mime_type (html, "text/html; charset=utf-8");
+ if (pre_encode) {
+ /* to not re-encode the data when pushing it to a part */
+ html->encoding = CAMEL_TRANSFER_ENCODING_QUOTEDPRINTABLE;
+ }
+
/* Build the multipart/alternative */
body = camel_multipart_new ();
camel_data_wrapper_set_mime_type (CAMEL_DATA_WRAPPER (body),