// // VMime library (http://vmime.sourceforge.net) // Copyright (C) 2002-2004 Vincent Richard // // 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. // // 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., 675 Mass Ave, Cambridge, MA 02139, USA. // #include "encoding.hpp" #include "encoderFactory.hpp" #include "contentHandler.hpp" #include namespace vmime { encoding::encoding() : m_name(encodingTypes::SEVEN_BIT) { } encoding::encoding(const string& name) : m_name(stringUtils::toLower(name)) { } encoding::encoding(const encoding& enc) : component(), m_name(enc.m_name) { } void encoding::parse(const string& buffer, const string::size_type position, const string::size_type end, string::size_type* newPosition) { m_name = stringUtils::toLower(string(buffer.begin() + position, buffer.begin() + end)); setParsedBounds(position, end); if (newPosition) *newPosition = end; } void encoding::generate(utility::outputStream& os, const string::size_type /* maxLineLength */, const string::size_type curLinePos, string::size_type* newLinePos) const { os << m_name; if (newLinePos) *newLinePos = curLinePos + m_name.length(); } encoder* encoding::getEncoder() const { return (encoderFactory::getInstance()->create(generate())); } encoding& encoding::operator=(const encoding& other) { copyFrom(other); return (*this); } encoding& encoding::operator=(const string& name) { m_name = stringUtils::toLower(name); return (*this); } const bool encoding::operator==(const encoding& value) const { return (stringUtils::toLower(m_name) == value.m_name); } const bool encoding::operator!=(const encoding& value) const { return !(*this == value); } const encoding encoding::decide (const string::const_iterator begin, const string::const_iterator end) { const string::difference_type length = end - begin; const string::difference_type count = std::count_if (begin, end, std::bind2nd(std::less(), 127)); // All is in 7-bit US-ASCII --> 7-bit (or Quoted-Printable...) if (length == count) { // Now, we check if there is any line with more than // "lineLengthLimits::convenient" characters (7-bit requires that) string::const_iterator p = begin; const string::size_type maxLen = lineLengthLimits::convenient; string::size_type len = 0; for ( ; p != end && len <= maxLen ; ) { if (*p == '\n') { len = 0; ++p; // May or may not need to be encoded, we don't take // any risk (avoid problems with SMTP) if (p != end && *p == '.') len = maxLen + 1; } else { ++len; ++p; } } if (len > maxLen) return (encoding(encodingTypes::QUOTED_PRINTABLE)); else return (encoding(encodingTypes::SEVEN_BIT)); } // Less than 20% non US-ASCII --> Quoted-Printable else if ((length - count) <= length / 5) { return (encoding(encodingTypes::QUOTED_PRINTABLE)); } // Otherwise --> Base64 else { return (encoding(encodingTypes::BASE64)); } } const encoding encoding::decide(const contentHandler& /* data */) { // TODO: a better solution to do that? return (encoding(encodingTypes::BASE64)); } encoding* encoding::clone() const { return new encoding(*this); } void encoding::copyFrom(const component& other) { const encoding& e = dynamic_cast (other); m_name = e.m_name; } const string& encoding::getName() const { return (m_name); } void encoding::setName(const string& name) { m_name = name; } } // vmime