2004-09-21 02:31:50 +04:00
|
|
|
/* (Text)Component - message component base class and plain text
|
|
|
|
**
|
|
|
|
** Copyright 2001 Dr. Zoidberg Enterprises. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <String.h>
|
|
|
|
#include <Mime.h>
|
|
|
|
|
|
|
|
#include <ctype.h>
|
2007-10-30 20:32:27 +03:00
|
|
|
#include <stdlib.h>
|
2004-09-21 02:31:50 +04:00
|
|
|
|
|
|
|
class _EXPORT BMailComponent;
|
|
|
|
class _EXPORT BTextMailComponent;
|
|
|
|
|
|
|
|
#include <MailComponent.h>
|
|
|
|
#include <MailAttachment.h>
|
|
|
|
#include <MailContainer.h>
|
|
|
|
#include <mail_util.h>
|
|
|
|
|
2004-11-11 16:11:30 +03:00
|
|
|
#include <CharacterSet.h>
|
|
|
|
#include <CharacterSetRoster.h>
|
|
|
|
|
|
|
|
using namespace BPrivate ;
|
|
|
|
|
2004-09-21 02:31:50 +04:00
|
|
|
struct CharsetConversionEntry
|
|
|
|
{
|
|
|
|
const char *charset;
|
|
|
|
uint32 flavor;
|
|
|
|
};
|
|
|
|
|
|
|
|
extern const CharsetConversionEntry mail_charsets[];
|
|
|
|
|
|
|
|
|
|
|
|
extern const char *kHeaderCharsetString = "header-charset";
|
|
|
|
extern const char *kHeaderEncodingString = "header-encoding";
|
|
|
|
// Special field names in the headers which specify the character set (int32)
|
|
|
|
// and encoding (int8) to use when converting the headers from UTF-8 to the
|
|
|
|
// output e-mail format (rfc2047). Since they are numbers, not strings, the
|
|
|
|
// extra fields won't be output.
|
|
|
|
|
|
|
|
|
|
|
|
BMailComponent::BMailComponent(uint32 defaultCharSet)
|
|
|
|
: _charSetForTextDecoding (defaultCharSet)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
BMailComponent::~BMailComponent()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 BMailComponent::ComponentType()
|
|
|
|
{
|
|
|
|
if (NULL != dynamic_cast<BAttributedMailAttachment *> (this))
|
|
|
|
return B_MAIL_ATTRIBUTED_ATTACHMENT;
|
|
|
|
|
|
|
|
BMimeType type, super;
|
|
|
|
MIMEType(&type);
|
|
|
|
type.GetSupertype(&super);
|
|
|
|
|
|
|
|
//---------ATT-This code *desperately* needs to be improved
|
|
|
|
if (super == "multipart") {
|
|
|
|
if (type == "multipart/x-bfile") // Not likely, they have the MIME
|
|
|
|
return B_MAIL_ATTRIBUTED_ATTACHMENT; // of their data contents.
|
|
|
|
else
|
|
|
|
return B_MAIL_MULTIPART_CONTAINER;
|
|
|
|
} else if (!IsAttachment() && (super == "text" || type.Type() == NULL))
|
|
|
|
return B_MAIL_PLAIN_TEXT_BODY;
|
|
|
|
else
|
|
|
|
return B_MAIL_SIMPLE_ATTACHMENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
BMailComponent *BMailComponent::WhatIsThis() {
|
|
|
|
switch (ComponentType())
|
|
|
|
{
|
|
|
|
case B_MAIL_SIMPLE_ATTACHMENT:
|
|
|
|
return new BSimpleMailAttachment;
|
|
|
|
case B_MAIL_ATTRIBUTED_ATTACHMENT:
|
|
|
|
return new BAttributedMailAttachment;
|
|
|
|
case B_MAIL_MULTIPART_CONTAINER:
|
|
|
|
return new BMIMEMultipartMailContainer (NULL, NULL, _charSetForTextDecoding);
|
|
|
|
case B_MAIL_PLAIN_TEXT_BODY:
|
|
|
|
default:
|
|
|
|
return new BTextMailComponent (NULL, _charSetForTextDecoding);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BMailComponent::IsAttachment() {
|
|
|
|
const char *disposition = HeaderField("Content-Disposition");
|
|
|
|
if ((disposition != NULL) && (strncasecmp(disposition,"Attachment",strlen("Attachment")) == 0))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
BMessage header;
|
|
|
|
HeaderField("Content-Type",&header);
|
|
|
|
if (header.HasString("name"))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (HeaderField("Content-Location",&header) == B_OK)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
BMimeType type;
|
|
|
|
MIMEType(&type);
|
|
|
|
if (type == "multipart/x-bfile")
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BMailComponent::SetHeaderField(const char *key, const char *value, uint32 charset, mail_encoding encoding, bool replace_existing) {
|
|
|
|
if (replace_existing)
|
|
|
|
headers.RemoveName(key);
|
|
|
|
if (value != NULL && value[0] != 0) // Empty or NULL strings mean delete header.
|
|
|
|
headers.AddString(key,value);
|
|
|
|
|
|
|
|
// Latest setting of the character set and encoding to use when outputting
|
|
|
|
// the headers is the one which affects all the headers. There used to be
|
|
|
|
// separate settings for each item in the headers, but it never actually
|
|
|
|
// worked (can't store multiple items of different types in a BMessage).
|
|
|
|
if (charset != B_MAIL_NULL_CONVERSION &&
|
|
|
|
headers.ReplaceInt32 (kHeaderCharsetString, charset) != B_OK)
|
|
|
|
headers.AddInt32 (kHeaderCharsetString, charset);
|
|
|
|
if (encoding != null_encoding &&
|
|
|
|
headers.ReplaceInt8 (kHeaderEncodingString, encoding) != B_OK)
|
|
|
|
headers.AddInt8 (kHeaderEncodingString, encoding);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BMailComponent::SetHeaderField(const char *key, BMessage *structure, bool replace_existing) {
|
|
|
|
int32 charset = B_MAIL_NULL_CONVERSION;
|
|
|
|
int8 encoding = null_encoding;
|
|
|
|
const char *unlabeled = "unlabeled";
|
|
|
|
|
|
|
|
if (replace_existing)
|
|
|
|
headers.RemoveName(key);
|
|
|
|
|
|
|
|
BString value;
|
|
|
|
if (structure->HasString(unlabeled))
|
|
|
|
value << structure->FindString(unlabeled) << "; ";
|
|
|
|
|
|
|
|
const char *name, *sub_val;
|
|
|
|
type_code type;
|
|
|
|
for (int32 i = 0; structure->GetInfo(B_STRING_TYPE,i,
|
2006-01-02 07:37:47 +03:00
|
|
|
#if !defined(HAIKU_TARGET_PLATFORM_DANO)
|
2004-09-21 02:31:50 +04:00
|
|
|
(char**)
|
|
|
|
#endif
|
|
|
|
&name,&type) == B_OK; i++)
|
|
|
|
{
|
|
|
|
if (strcasecmp(name, unlabeled) == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
structure->FindString(name, &sub_val);
|
|
|
|
value << name << '=';
|
|
|
|
if (BString(sub_val).FindFirst(' ') > 0)
|
|
|
|
value << '\"' << sub_val << "\"; ";
|
|
|
|
else
|
|
|
|
value << sub_val << "; ";
|
|
|
|
}
|
|
|
|
|
|
|
|
value.Truncate(value.Length() - 2); //-----Remove the last "; "
|
|
|
|
|
|
|
|
if (structure->HasInt32(kHeaderCharsetString))
|
|
|
|
structure->FindInt32(kHeaderCharsetString, &charset);
|
|
|
|
if (structure->HasInt8(kHeaderEncodingString))
|
|
|
|
structure->FindInt8(kHeaderEncodingString, &encoding);
|
|
|
|
|
|
|
|
SetHeaderField(key,value.String(),(uint32) charset, (mail_encoding) encoding);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *BMailComponent::HeaderField(const char *key, int32 index) {
|
|
|
|
const char *string = NULL;
|
|
|
|
|
|
|
|
headers.FindString(key,index,&string);
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
|
|
|
|
status_t BMailComponent::HeaderField(const char *key, BMessage *structure, int32 index) {
|
|
|
|
BString string = HeaderField(key,index);
|
|
|
|
if (string == "")
|
|
|
|
return B_NAME_NOT_FOUND;
|
|
|
|
|
|
|
|
BString sub_cat,end_piece;
|
|
|
|
int32 i = 0, end = 0;
|
|
|
|
|
|
|
|
// Break the header into parts, they're separated by semicolons, like this:
|
|
|
|
// Content-Type: multipart/mixed;boundary= "----=_NextPart_000_00AA_354DB459.5977A1CA"
|
|
|
|
// There's also white space and quotes to be removed, and even comments in
|
|
|
|
// parenthesis like this, which can appear anywhere white space is: (header comment)
|
|
|
|
|
|
|
|
while (end < string.Length()) {
|
|
|
|
end = string.FindFirst(';',i);
|
|
|
|
if (end < 0)
|
|
|
|
end = string.Length();
|
|
|
|
|
|
|
|
string.CopyInto(sub_cat,i,end - i);
|
|
|
|
i = end + 1;
|
|
|
|
|
|
|
|
//-------Trim spaces off of beginning and end of text
|
|
|
|
for (int32 h = 0; h < sub_cat.Length(); h++) {
|
|
|
|
if (!isspace(sub_cat.ByteAt(h))) {
|
|
|
|
sub_cat.Remove(0,h);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int32 h = sub_cat.Length()-1; h >= 0; h--) {
|
|
|
|
if (!isspace(sub_cat.ByteAt(h))) {
|
|
|
|
sub_cat.Truncate(h+1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//--------Split along '='
|
|
|
|
int32 first_equal = sub_cat.FindFirst('=');
|
|
|
|
if (first_equal >= 0) {
|
|
|
|
sub_cat.CopyInto(end_piece,first_equal+1,sub_cat.Length() - first_equal - 1);
|
|
|
|
sub_cat.Truncate(first_equal);
|
|
|
|
// Remove leading spaces from part after the equals sign.
|
|
|
|
while (isspace (end_piece.ByteAt(0)))
|
|
|
|
end_piece.Remove (0 /* index */, 1 /* number of chars */);
|
|
|
|
// Remove quote marks.
|
|
|
|
if (end_piece.ByteAt(0) == '\"') {
|
|
|
|
end_piece.Remove(0,1);
|
|
|
|
end_piece.Truncate(end_piece.Length() - 1);
|
|
|
|
}
|
|
|
|
sub_cat.ToLower();
|
|
|
|
structure->AddString(sub_cat.String(),end_piece.String());
|
|
|
|
} else {
|
|
|
|
structure->AddString("unlabeled",sub_cat.String());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
status_t BMailComponent::RemoveHeader(const char *key) {
|
|
|
|
return headers.RemoveName(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *BMailComponent::HeaderAt(int32 index) {
|
2006-01-02 07:37:47 +03:00
|
|
|
#if defined(HAIKU_TARGET_PLATFORM_DANO)
|
2004-09-21 02:31:50 +04:00
|
|
|
const
|
|
|
|
#endif
|
|
|
|
char *name = NULL;
|
|
|
|
type_code type;
|
|
|
|
|
|
|
|
headers.GetInfo(B_STRING_TYPE,index,&name,&type);
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
status_t BMailComponent::GetDecodedData(BPositionIO *) {return B_OK;}
|
|
|
|
status_t BMailComponent::SetDecodedData(BPositionIO *) {return B_OK;}
|
|
|
|
|
|
|
|
status_t
|
|
|
|
BMailComponent::SetToRFC822(BPositionIO *data, size_t /*length*/, bool /*parse_now*/)
|
|
|
|
{
|
|
|
|
headers.MakeEmpty();
|
|
|
|
|
|
|
|
// Only parse the header here
|
|
|
|
return parse_header(headers, *data);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status_t
|
|
|
|
BMailComponent::RenderToRFC822(BPositionIO *render_to) {
|
2004-11-11 16:11:30 +03:00
|
|
|
int32 charset = B_ISO15_CONVERSION;
|
2004-09-21 02:31:50 +04:00
|
|
|
int8 encoding = quoted_printable;
|
|
|
|
const char *key, *value;
|
|
|
|
char *allocd;
|
|
|
|
ssize_t amountWritten;
|
|
|
|
BString concat;
|
|
|
|
type_code stupidity_personified = B_STRING_TYPE;
|
|
|
|
int32 count = 0;
|
|
|
|
|
|
|
|
if (headers.HasInt32 (kHeaderCharsetString))
|
|
|
|
headers.FindInt32 (kHeaderCharsetString, &charset);
|
|
|
|
if (headers.HasInt8 (kHeaderEncodingString))
|
|
|
|
headers.FindInt8 (kHeaderEncodingString, &encoding);
|
|
|
|
|
|
|
|
for (int32 index = 0; headers.GetInfo(B_STRING_TYPE,index,
|
2006-01-02 07:37:47 +03:00
|
|
|
#if !defined(HAIKU_TARGET_PLATFORM_DANO)
|
|
|
|
(char**)
|
2004-09-21 02:31:50 +04:00
|
|
|
#endif
|
|
|
|
&key,&stupidity_personified,&count) == B_OK; index++) {
|
|
|
|
for (int32 g = 0; g < count; g++) {
|
|
|
|
headers.FindString(key,g,(const char **)&value);
|
|
|
|
allocd = (char *)malloc(strlen(value) + 1);
|
|
|
|
strcpy(allocd,value);
|
|
|
|
|
|
|
|
concat << key << ": ";
|
|
|
|
concat.CapitalizeEachWord();
|
|
|
|
|
|
|
|
concat.Append(allocd,utf8_to_rfc2047(&allocd, strlen(value), charset, encoding));
|
|
|
|
free(allocd);
|
|
|
|
FoldLineAtWhiteSpaceAndAddCRLF (concat);
|
|
|
|
|
|
|
|
amountWritten = render_to->Write(concat.String(), concat.Length());
|
|
|
|
if (amountWritten < 0)
|
|
|
|
return amountWritten; // IO error happened, usually disk full.
|
|
|
|
concat = "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render_to->Write("\r\n", 2);
|
|
|
|
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status_t BMailComponent::MIMEType(BMimeType *mime) {
|
|
|
|
bool foundBestHeader;
|
|
|
|
const char *boundaryString;
|
|
|
|
unsigned int i;
|
|
|
|
BMessage msg;
|
|
|
|
const char *typeAsString = NULL;
|
|
|
|
char typeAsLowerCaseString [B_MIME_TYPE_LENGTH];
|
|
|
|
|
|
|
|
// Find the best Content-Type header to use. There should really be just
|
|
|
|
// one, but evil spammers sneakily insert one for multipart (with no
|
|
|
|
// boundary string), then one for text/plain. We'll scan through them and
|
|
|
|
// only use the multipart one if there are no others, and it has a
|
|
|
|
// boundary.
|
|
|
|
|
|
|
|
foundBestHeader = false;
|
|
|
|
for (i = 0; msg.MakeEmpty(), HeaderField("Content-Type", &msg, i) == B_OK; i++) {
|
|
|
|
typeAsString = msg.FindString("unlabeled");
|
|
|
|
if (typeAsString != NULL && strncasecmp (typeAsString, "multipart", 9) != 0) {
|
|
|
|
foundBestHeader = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!foundBestHeader) {
|
|
|
|
for (i = 0; msg.MakeEmpty(), HeaderField("Content-Type", &msg, i) == B_OK; i++) {
|
|
|
|
typeAsString = msg.FindString("unlabeled");
|
|
|
|
if (typeAsString != NULL && strncasecmp (typeAsString, "multipart", 9) == 0) {
|
|
|
|
boundaryString = msg.FindString("boundary");
|
|
|
|
if (boundaryString != NULL && strlen (boundaryString) > 0) {
|
|
|
|
foundBestHeader = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// At this point we have the good MIME type in typeAsString, but only if
|
|
|
|
// foundBestHeader is true.
|
|
|
|
|
|
|
|
if (!foundBestHeader) {
|
|
|
|
strcpy (typeAsLowerCaseString, "text/plain"); // Hope this is an OK default.
|
|
|
|
} else {
|
|
|
|
// Some extra processing to convert mixed or upper case MIME types into
|
|
|
|
// lower case, since the BeOS R5 BMimeType is case sensitive (but OpenBeOS
|
|
|
|
// isn't). Also truncate the string if it is too long.
|
|
|
|
for (i = 0; i < sizeof (typeAsLowerCaseString) - 1 && typeAsString[i] != 0; i++)
|
|
|
|
typeAsLowerCaseString[i] = tolower (typeAsString[i]);
|
|
|
|
typeAsLowerCaseString[i] = 0;
|
|
|
|
|
|
|
|
// Some old e-mail programs saved the type as just "TEXT", which we need to
|
|
|
|
// convert to "text/plain" since the rest of the code looks for that.
|
|
|
|
if (strcmp (typeAsLowerCaseString, "text") == 0)
|
|
|
|
strcpy (typeAsLowerCaseString, "text/plain");
|
|
|
|
}
|
|
|
|
mime->SetTo(typeAsLowerCaseString);
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BMailComponent::_ReservedComponent1() {}
|
|
|
|
void BMailComponent::_ReservedComponent2() {}
|
|
|
|
void BMailComponent::_ReservedComponent3() {}
|
|
|
|
void BMailComponent::_ReservedComponent4() {}
|
|
|
|
void BMailComponent::_ReservedComponent5() {}
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
BTextMailComponent::BTextMailComponent(const char *text, uint32 defaultCharSet)
|
|
|
|
: BMailComponent(defaultCharSet),
|
|
|
|
encoding(quoted_printable),
|
2004-11-11 16:11:30 +03:00
|
|
|
charset(B_ISO15_CONVERSION),
|
2004-09-21 02:31:50 +04:00
|
|
|
raw_data(NULL)
|
|
|
|
{
|
|
|
|
if (text != NULL)
|
|
|
|
SetText(text);
|
|
|
|
|
|
|
|
SetHeaderField("MIME-Version","1.0");
|
|
|
|
}
|
|
|
|
|
|
|
|
BTextMailComponent::~BTextMailComponent()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void BTextMailComponent::SetEncoding(mail_encoding encoding, int32 charset) {
|
|
|
|
this->encoding = encoding;
|
|
|
|
this->charset = charset;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BTextMailComponent::SetText(const char *text) {
|
|
|
|
this->text.SetTo(text);
|
|
|
|
|
|
|
|
raw_data = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BTextMailComponent::AppendText(const char *text) {
|
|
|
|
ParseRaw();
|
|
|
|
|
|
|
|
this->text << text;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *BTextMailComponent::Text() {
|
|
|
|
ParseRaw();
|
|
|
|
|
|
|
|
return text.String();
|
|
|
|
}
|
|
|
|
|
|
|
|
BString *BTextMailComponent::BStringText() {
|
|
|
|
ParseRaw();
|
|
|
|
|
|
|
|
return &text;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BTextMailComponent::Quote(const char *message, const char *quote_style) {
|
|
|
|
ParseRaw();
|
|
|
|
|
|
|
|
BString string;
|
|
|
|
string << '\n' << quote_style;
|
|
|
|
text.ReplaceAll("\n",string.String());
|
|
|
|
|
|
|
|
string = message;
|
|
|
|
string << '\n';
|
|
|
|
text.Prepend(string.String());
|
|
|
|
}
|
|
|
|
|
|
|
|
status_t BTextMailComponent::GetDecodedData(BPositionIO *data) {
|
|
|
|
ParseRaw();
|
2004-10-30 06:45:37 +04:00
|
|
|
|
|
|
|
if (data == NULL)
|
|
|
|
return B_IO_ERROR;
|
2004-09-21 02:31:50 +04:00
|
|
|
|
|
|
|
BMimeType type;
|
2005-10-16 22:33:20 +04:00
|
|
|
BMimeType textAny ("text");
|
2004-09-21 02:31:50 +04:00
|
|
|
ssize_t written;
|
2005-10-16 22:33:20 +04:00
|
|
|
if (MIMEType(&type) == B_OK && textAny.Contains (&type))
|
|
|
|
// Write out the string which has been both decoded from quoted
|
|
|
|
// printable or base64 etc, and then converted to UTF-8 from whatever
|
|
|
|
// character set the message specified. Do it for text/html,
|
|
|
|
// text/plain and all other text datatypes. Of course, if the message
|
|
|
|
// is HTML and specifies a META tag for a character set, it will now be
|
|
|
|
// wrong. But then we don't display HTML in BeMail, yet.
|
2004-09-21 02:31:50 +04:00
|
|
|
written = data->Write(text.String(),text.Length());
|
|
|
|
else
|
2005-10-16 22:33:20 +04:00
|
|
|
// Just write out whatever the binary contents are, only decoded from
|
|
|
|
// the quoted printable etc format.
|
2004-09-21 02:31:50 +04:00
|
|
|
written = data->Write(decoded.String(), decoded.Length());
|
|
|
|
|
|
|
|
return written >= 0 ? B_OK : written;
|
|
|
|
}
|
|
|
|
|
|
|
|
status_t BTextMailComponent::SetDecodedData(BPositionIO *data) {
|
|
|
|
char buffer[255];
|
|
|
|
size_t buf_len;
|
|
|
|
|
|
|
|
while ((buf_len = data->Read(buffer,254)) > 0) {
|
|
|
|
buffer[buf_len] = 0;
|
|
|
|
this->text << buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
raw_data = NULL;
|
|
|
|
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status_t
|
|
|
|
BTextMailComponent::SetToRFC822(BPositionIO *data, size_t length, bool parseNow)
|
|
|
|
{
|
|
|
|
off_t position = data->Position();
|
|
|
|
BMailComponent::SetToRFC822(data, length);
|
|
|
|
|
|
|
|
// Some malformed MIME headers can have the header running into the
|
|
|
|
// boundary of the next MIME chunk, resulting in a negative length.
|
|
|
|
length -= data->Position() - position;
|
|
|
|
if ((ssize_t) length < 0)
|
|
|
|
length = 0;
|
|
|
|
|
|
|
|
raw_data = data;
|
|
|
|
raw_length = length;
|
|
|
|
raw_offset = data->Position();
|
|
|
|
|
|
|
|
if (parseNow) {
|
|
|
|
// copies the data stream and sets the raw_data variable to NULL
|
|
|
|
return ParseRaw();
|
|
|
|
}
|
|
|
|
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status_t
|
|
|
|
BTextMailComponent::ParseRaw()
|
|
|
|
{
|
|
|
|
if (raw_data == NULL)
|
|
|
|
return B_OK;
|
|
|
|
|
|
|
|
raw_data->Seek(raw_offset, SEEK_SET);
|
|
|
|
|
|
|
|
BMessage content_type;
|
|
|
|
HeaderField("Content-Type", &content_type);
|
|
|
|
|
|
|
|
charset = _charSetForTextDecoding;
|
|
|
|
if (charset == B_MAIL_NULL_CONVERSION && content_type.HasString("charset")) {
|
2004-11-11 16:11:30 +03:00
|
|
|
const char * charset_string = content_type.FindString("charset");
|
|
|
|
if (strcasecmp(charset_string, "us-ascii") == 0) {
|
|
|
|
charset = B_MAIL_US_ASCII_CONVERSION;
|
|
|
|
} else if (strcasecmp(charset_string, "utf-8") == 0) {
|
|
|
|
charset = B_MAIL_UTF8_CONVERSION;
|
|
|
|
} else {
|
|
|
|
const BCharacterSet * cs = BCharacterSetRoster::FindCharacterSetByName(charset_string);
|
|
|
|
if (cs != NULL) {
|
|
|
|
charset = cs->GetConversionID();
|
2004-09-21 02:31:50 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
encoding = encoding_for_cte(HeaderField("Content-Transfer-Encoding"));
|
|
|
|
|
|
|
|
char *buffer = (char *)malloc(raw_length + 1);
|
|
|
|
if (buffer == NULL)
|
|
|
|
return B_NO_MEMORY;
|
|
|
|
|
2004-11-12 21:58:55 +03:00
|
|
|
int32 bytes;
|
2004-09-21 02:31:50 +04:00
|
|
|
if ((bytes = raw_data->Read(buffer, raw_length)) < 0)
|
|
|
|
return B_IO_ERROR;
|
|
|
|
|
|
|
|
char *string = decoded.LockBuffer(bytes + 1);
|
|
|
|
bytes = decode(encoding, string, buffer, bytes, 0);
|
|
|
|
free (buffer);
|
|
|
|
buffer = NULL;
|
|
|
|
|
|
|
|
// Change line ends from \r\n to just \n. Though this won't work properly
|
|
|
|
// for UTF-16 because \r takes up two bytes rather than one.
|
|
|
|
char *dest, *src;
|
|
|
|
char *end = string + bytes;
|
|
|
|
for (dest = src = string; src < end; src++) {
|
|
|
|
if (*src != '\r')
|
|
|
|
*dest++ = *src;
|
|
|
|
}
|
|
|
|
decoded.UnlockBuffer(dest - string);
|
|
|
|
bytes = decoded.Length(); // Might have shrunk a bit.
|
|
|
|
|
|
|
|
// If the character set wasn't specified, try to guess. ISO-2022-JP
|
|
|
|
// contains the escape sequences ESC $ B or ESC $ @ to turn on 2 byte
|
|
|
|
// Japanese, and ESC ( J to switch to Roman, or sometimes ESC ( B for
|
|
|
|
// ASCII. We'll just try looking for the two switch to Japanese sequences.
|
|
|
|
|
|
|
|
if (charset == B_MAIL_NULL_CONVERSION) {
|
|
|
|
if (decoded.FindFirst ("\e$B") >= 0 || decoded.FindFirst ("\e$@") >= 0)
|
|
|
|
charset = B_JIS_CONVERSION;
|
2004-11-11 16:11:30 +03:00
|
|
|
else // Just assume the usual Latin-9 character set.
|
|
|
|
charset = B_ISO15_CONVERSION;
|
2004-09-21 02:31:50 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int32 state = 0;
|
|
|
|
int32 destLength = bytes * 3 /* in case it grows */ + 1 /* +1 so it isn't zero which crashes */;
|
|
|
|
string = text.LockBuffer(destLength);
|
|
|
|
mail_convert_to_utf8(charset, decoded.String(), &bytes, string, &destLength, &state);
|
|
|
|
if (destLength > 0)
|
|
|
|
text.UnlockBuffer(destLength);
|
|
|
|
else {
|
|
|
|
text.UnlockBuffer(0);
|
|
|
|
text.SetTo(decoded);
|
|
|
|
}
|
|
|
|
|
|
|
|
raw_data = NULL;
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status_t
|
|
|
|
BTextMailComponent::RenderToRFC822(BPositionIO *render_to)
|
|
|
|
{
|
|
|
|
status_t status = ParseRaw();
|
|
|
|
if (status < B_OK)
|
|
|
|
return status;
|
2004-10-30 21:43:01 +04:00
|
|
|
|
|
|
|
BMimeType type;
|
|
|
|
MIMEType(&type);
|
2004-09-21 02:31:50 +04:00
|
|
|
BString content_type;
|
2004-10-30 21:43:01 +04:00
|
|
|
content_type << type.Type(); // Preserve MIME type (e.g. text/html
|
2004-09-21 02:31:50 +04:00
|
|
|
|
|
|
|
for (uint32 i = 0; mail_charsets[i].charset != NULL; i++) {
|
|
|
|
if (mail_charsets[i].flavor == charset) {
|
|
|
|
content_type << "; charset=\"" << mail_charsets[i].charset << "\"";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SetHeaderField("Content-Type", content_type.String());
|
|
|
|
|
|
|
|
const char *transfer_encoding = NULL;
|
|
|
|
switch (encoding) {
|
|
|
|
case base64:
|
|
|
|
transfer_encoding = "base64";
|
|
|
|
break;
|
|
|
|
case quoted_printable:
|
|
|
|
transfer_encoding = "quoted-printable";
|
|
|
|
break;
|
|
|
|
case eight_bit:
|
|
|
|
transfer_encoding = "8bit";
|
|
|
|
break;
|
|
|
|
case seven_bit:
|
|
|
|
default:
|
|
|
|
transfer_encoding = "7bit";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
SetHeaderField("Content-Transfer-Encoding",transfer_encoding);
|
|
|
|
|
|
|
|
BMailComponent::RenderToRFC822(render_to);
|
|
|
|
|
|
|
|
BString modified = this->text;
|
|
|
|
BString alt;
|
|
|
|
|
|
|
|
int32 len = this->text.Length();
|
|
|
|
if (len > 0) {
|
|
|
|
int32 dest_len = len * 5;
|
|
|
|
// Shift-JIS can have a 3 byte escape sequence and a 2 byte code for
|
|
|
|
// each character (which could just be 2 bytes in UTF-8, or even 1 byte
|
|
|
|
// if it's regular ASCII), so it can get quite a bit larger than the
|
|
|
|
// original text. Multiplying by 5 should make more than enough space.
|
|
|
|
char *raw = alt.LockBuffer(dest_len);
|
|
|
|
int32 state = 0;
|
|
|
|
mail_convert_from_utf8(charset,this->text.String(),&len,raw,&dest_len,&state);
|
|
|
|
alt.UnlockBuffer(dest_len);
|
|
|
|
|
|
|
|
raw = modified.LockBuffer((alt.Length()*3)+1);
|
|
|
|
switch (encoding) {
|
|
|
|
case base64:
|
|
|
|
len = encode_base64(raw,alt.String(),alt.Length(),false);
|
|
|
|
raw[len] = 0;
|
|
|
|
break;
|
|
|
|
case quoted_printable:
|
|
|
|
len = encode_qp(raw,alt.String(),alt.Length(),false);
|
|
|
|
raw[len] = 0;
|
|
|
|
break;
|
|
|
|
case eight_bit:
|
|
|
|
case seven_bit:
|
|
|
|
default:
|
|
|
|
len = alt.Length();
|
|
|
|
strcpy(raw,alt.String());
|
|
|
|
}
|
|
|
|
modified.UnlockBuffer(len);
|
|
|
|
|
|
|
|
if (encoding != base64) // encode_base64 already does CRLF line endings.
|
|
|
|
modified.ReplaceAll("\n","\r\n");
|
|
|
|
|
|
|
|
// There seem to be a possibility of NULL bytes in the text, so lets
|
|
|
|
// filter them out, shouldn't be any after the encoding stage.
|
|
|
|
|
|
|
|
char *string = modified.LockBuffer(modified.Length());
|
|
|
|
for (int32 i = modified.Length();i-- > 0;)
|
|
|
|
{
|
|
|
|
if (string[i] != '\0')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
puts("BTextMailComponent::RenderToRFC822: NULL byte in text!!");
|
|
|
|
string[i] = ' ';
|
|
|
|
}
|
|
|
|
modified.UnlockBuffer();
|
|
|
|
|
|
|
|
// word wrapping is already done by BeMail (user-configurable)
|
|
|
|
// and it does it *MUCH* nicer.
|
|
|
|
|
|
|
|
// //------Desperate bid to wrap lines
|
|
|
|
// int32 curr_line_length = 0;
|
|
|
|
// int32 last_space = 0;
|
|
|
|
//
|
|
|
|
// for (int32 i = 0; i < modified.Length(); i++) {
|
|
|
|
// if (isspace(modified.ByteAt(i)))
|
|
|
|
// last_space = i;
|
|
|
|
//
|
|
|
|
// if ((modified.ByteAt(i) == '\r') && (modified.ByteAt(i+1) == '\n'))
|
|
|
|
// curr_line_length = 0;
|
|
|
|
// else
|
|
|
|
// curr_line_length++;
|
|
|
|
//
|
|
|
|
// if (curr_line_length > 80) {
|
|
|
|
// if (last_space >= 0) {
|
|
|
|
// modified.Insert("\r\n",last_space);
|
|
|
|
// last_space = -1;
|
|
|
|
// curr_line_length = 0;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
}
|
|
|
|
modified << "\r\n";
|
|
|
|
|
|
|
|
render_to->Write(modified.String(),modified.Length());
|
|
|
|
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BTextMailComponent::_ReservedText1() {}
|
|
|
|
void BTextMailComponent::_ReservedText2() {}
|