vmime/src/text.cpp

387 lines
7.1 KiB
C++
Raw Normal View History

2004-10-05 10:28:21 +00:00
//
2005-03-18 21:33:11 +00:00
// VMime library (http://www.vmime.org)
// Copyright (C) 2002-2005 Vincent Richard <vincent@vincent-richard.net>
2004-10-05 10:28:21 +00:00
//
// 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.
//
2005-09-17 10:10:29 +00:00
// 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.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
//
// Linking this library statically or dynamically with other modules is making
// a combined work based on this library. Thus, the terms and conditions of
// the GNU General Public License cover the whole combination.
2004-10-05 10:28:21 +00:00
//
#include "vmime/text.hpp"
2004-10-05 10:28:21 +00:00
#include "vmime/parserHelpers.hpp"
2004-10-05 10:28:21 +00:00
namespace vmime
{
text::text()
{
}
text::text(const text& t)
2004-10-21 15:05:47 +00:00
: component()
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
copyFrom(t);
2004-10-05 10:28:21 +00:00
}
text::text(const string& t, const charset& ch)
{
2005-07-12 22:28:02 +00:00
createFromString(t, ch);
2004-10-05 10:28:21 +00:00
}
text::text(const string& t)
{
2005-07-12 22:28:02 +00:00
createFromString(t, charset::getLocaleCharset());
2004-10-05 10:28:21 +00:00
}
text::text(const word& w)
{
2005-07-12 22:28:02 +00:00
appendWord(vmime::create <word>(w));
2004-10-05 10:28:21 +00:00
}
text::~text()
{
2004-10-21 15:05:47 +00:00
removeAllWords();
}
void text::parse(const string& buffer, const string::size_type position,
const string::size_type end, string::size_type* newPosition)
{
removeAllWords();
string::size_type newPos;
2005-07-12 22:28:02 +00:00
const std::vector <ref <word> > words = word::parseMultiple(buffer, position, end, &newPos);
2004-10-21 15:05:47 +00:00
copy_vector(words, m_words);
setParsedBounds(position, newPos);
2004-12-15 20:28:09 +00:00
2004-10-21 15:05:47 +00:00
if (newPosition)
*newPosition = newPos;
2004-10-21 15:05:47 +00:00
}
void text::generate(utility::outputStream& os, const string::size_type maxLineLength,
const string::size_type curLinePos, string::size_type* newLinePos) const
{
encodeAndFold(os, maxLineLength, curLinePos, newLinePos, 0);
2004-10-05 10:28:21 +00:00
}
#if VMIME_WIDE_CHAR_SUPPORT
const wstring text::getDecodedText() const
{
wstring out;
2005-07-12 22:28:02 +00:00
for (std::vector <ref <word> >::const_iterator i = m_words.begin() ; i != m_words.end() ; ++i)
2004-10-05 10:28:21 +00:00
out += (*i)->getDecodedText();
return (out);
}
#endif
2004-10-21 15:05:47 +00:00
void text::copyFrom(const component& other)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
const text& t = dynamic_cast <const text&>(other);
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
removeAllWords();
2004-10-05 10:28:21 +00:00
2005-07-12 22:28:02 +00:00
for (std::vector <ref <word> >::const_iterator i = t.m_words.begin() ; i != t.m_words.end() ; ++i)
m_words.push_back(vmime::create <word>(**i));
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
text& text::operator=(const component& other)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
copyFrom(other);
return (*this);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
text& text::operator=(const text& other)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
copyFrom(other);
2004-10-05 10:28:21 +00:00
return (*this);
}
const bool text::operator==(const text& t) const
{
2004-10-21 15:05:47 +00:00
if (getWordCount() == t.getWordCount())
2004-10-05 10:28:21 +00:00
{
2004-10-26 20:17:28 +00:00
bool equal = true;
2004-10-05 10:28:21 +00:00
2005-07-12 22:28:02 +00:00
std::vector <ref <word> >::const_iterator i = m_words.begin();
std::vector <ref <word> >::const_iterator j = t.m_words.begin();
2004-10-05 10:28:21 +00:00
for ( ; equal && i != m_words.end() ; ++i, ++j)
2004-10-26 20:17:28 +00:00
equal = (**i == **j);
2004-10-05 10:28:21 +00:00
return (equal);
}
return (false);
}
const bool text::operator!=(const text& t) const
{
return !(*this == t);
}
const string text::getConvertedText(const charset& dest) const
{
string out;
2005-07-12 22:28:02 +00:00
for (std::vector <ref <word> >::const_iterator i = m_words.begin() ; i != m_words.end() ; ++i)
2004-10-05 10:28:21 +00:00
out += (*i)->getConvertedText(dest);
return (out);
}
2005-07-12 22:28:02 +00:00
void text::appendWord(ref <word> w)
2004-10-21 15:05:47 +00:00
{
m_words.push_back(w);
}
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
2005-07-12 22:28:02 +00:00
void text::insertWordBefore(const int pos, ref <word> w)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
m_words.insert(m_words.begin() + pos, w);
2004-10-05 10:28:21 +00:00
}
2005-07-12 22:28:02 +00:00
void text::insertWordAfter(const int pos, ref <word> w)
2004-10-21 15:05:47 +00:00
{
m_words.insert(m_words.begin() + pos + 1, w);
}
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
void text::removeWord(const int pos)
2004-10-05 10:28:21 +00:00
{
2005-07-12 22:28:02 +00:00
const std::vector <ref <word> >::iterator it = m_words.begin() + pos;
2004-10-21 15:05:47 +00:00
m_words.erase(it);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
void text::removeAllWords()
{
2005-07-12 22:28:02 +00:00
m_words.clear();
2004-10-21 15:05:47 +00:00
}
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
const int text::getWordCount() const
2004-10-05 10:28:21 +00:00
{
return (m_words.size());
}
2004-10-21 15:05:47 +00:00
const bool text::isEmpty() const
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
return (m_words.empty());
2004-10-05 10:28:21 +00:00
}
2005-07-12 22:28:02 +00:00
const ref <word> text::getWordAt(const int pos)
2004-10-21 15:05:47 +00:00
{
return (m_words[pos]);
}
2004-10-05 10:28:21 +00:00
2005-07-12 22:28:02 +00:00
const ref <const word> text::getWordAt(const int pos) const
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
return (m_words[pos]);
2004-10-05 10:28:21 +00:00
}
2005-07-12 22:28:02 +00:00
const std::vector <ref <const word> > text::getWordList() const
2004-10-21 15:05:47 +00:00
{
2005-07-12 22:28:02 +00:00
std::vector <ref <const word> > list;
2004-10-21 15:05:47 +00:00
list.reserve(m_words.size());
2005-07-12 22:28:02 +00:00
for (std::vector <ref <word> >::const_iterator it = m_words.begin() ;
2004-10-21 15:05:47 +00:00
it != m_words.end() ; ++it)
{
list.push_back(*it);
}
return (list);
}
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
2005-07-12 22:28:02 +00:00
const std::vector <ref <word> > text::getWordList()
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
return (m_words);
2004-10-05 10:28:21 +00:00
}
2005-07-12 22:28:02 +00:00
ref <component> text::clone() const
{
return vmime::create <text>(*this);
}
ref <text> text::newFromString(const string& in, const charset& ch)
2004-10-21 15:05:47 +00:00
{
2005-07-12 22:28:02 +00:00
ref <text> t = vmime::create <text>();
t->createFromString(in, ch);
return t;
2004-10-21 15:05:47 +00:00
}
2004-10-05 10:28:21 +00:00
2005-07-12 22:28:02 +00:00
void text::createFromString(const string& in, const charset& ch)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
const string::const_iterator end = in.end();
string::const_iterator p = in.begin();
string::const_iterator start = in.begin();
bool is8bit = false; // is the current word 8-bit?
bool prevIs8bit = false; // is previous word 8-bit?
unsigned int count = 0; // total number of words
2005-07-12 22:28:02 +00:00
removeAllWords();
2004-10-21 15:05:47 +00:00
for ( ; ; )
{
if (p == end || parserHelpers::isSpace(*p))
2004-10-21 15:05:47 +00:00
{
if (p != end)
++p;
if (is8bit)
{
if (count && prevIs8bit)
{
// No need to create a new encoded word, just append
// the current word to the previous one.
2005-07-12 22:28:02 +00:00
ref <word> w = getWordAt(getWordCount() - 1);
w->getBuffer() += string(start, p);
2004-10-21 15:05:47 +00:00
}
else
{
2005-07-12 22:28:02 +00:00
appendWord(vmime::create <word>(string(start, p), ch));
2004-10-21 15:05:47 +00:00
prevIs8bit = true;
++count;
}
}
else
{
if (count && !prevIs8bit)
{
2005-07-12 22:28:02 +00:00
ref <word> w = getWordAt(getWordCount() - 1);
w->getBuffer() += string(start, p);
2004-10-21 15:05:47 +00:00
}
else
{
2005-07-12 22:28:02 +00:00
appendWord(vmime::create <word>
2004-10-21 15:05:47 +00:00
(string(start, p), charset(charsets::US_ASCII)));
prevIs8bit = false;
++count;
}
}
if (p == end)
break;
is8bit = false;
start = p;
}
else if (!parserHelpers::isAscii(*p))
2004-10-21 15:05:47 +00:00
{
is8bit = true;
++p;
}
else
{
++p;
}
}
2004-10-05 10:28:21 +00:00
}
void text::encodeAndFold(utility::outputStream& os, const string::size_type maxLineLength,
const string::size_type firstLineOffset, string::size_type* lastLineLength, const int flags) const
{
string::size_type curLineLength = firstLineOffset;
for (int wi = 0 ; wi < getWordCount() ; ++wi)
{
getWordAt(wi)->generate(os, maxLineLength, curLineLength,
&curLineLength, flags, (wi == 0));
}
if (lastLineLength)
*lastLineLength = curLineLength;
}
2005-07-12 22:28:02 +00:00
ref <text> text::decodeAndUnfold(const string& in)
{
ref <text> t = vmime::create <text>();
decodeAndUnfold(in, t.get());
return t;
}
text* text::decodeAndUnfold(const string& in, text* generateInExisting)
{
text* out = (generateInExisting != NULL) ? generateInExisting : new text();
out->removeAllWords();
2005-07-12 22:28:02 +00:00
const std::vector <ref <word> > words = word::parseMultiple(in, 0, in.length(), NULL);
copy_vector(words, out->m_words);
return (out);
}
2005-07-12 22:28:02 +00:00
const std::vector <ref <const component> > text::getChildComponents() const
{
2005-07-12 22:28:02 +00:00
std::vector <ref <const component> > list;
copy_vector(m_words, list);
return (list);
}
2004-10-05 10:28:21 +00:00
} // vmime