aboutsummaryrefslogtreecommitdiffstats
path: root/src/disposition.cpp
diff options
context:
space:
mode:
authorVincent Richard <[email protected]>2005-03-23 08:23:55 +0000
committerVincent Richard <[email protected]>2005-03-23 08:23:55 +0000
commitaeb5da43242ab1be2c9f955f855bf901ac43513a (patch)
tree789f58bbc9bd8d92d0c737928d4dda4adf2897c4 /src/disposition.cpp
parentFixed multiple bugs in POP3 protocol. (diff)
downloadvmime-aeb5da43242ab1be2c9f955f855bf901ac43513a.tar.gz
vmime-aeb5da43242ab1be2c9f955f855bf901ac43513a.zip
Basic support for 'Disposition' header field.
Diffstat (limited to 'src/disposition.cpp')
-rw-r--r--src/disposition.cpp314
1 files changed, 314 insertions, 0 deletions
diff --git a/src/disposition.cpp b/src/disposition.cpp
new file mode 100644
index 00000000..b894f85e
--- /dev/null
+++ b/src/disposition.cpp
@@ -0,0 +1,314 @@
+//
+// VMime library (http://www.vmime.org)
+// Copyright (C) 2002-2005 Vincent Richard <[email protected]>
+//
+// 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 "vmime/disposition.hpp"
+
+#include "vmime/parserHelpers.hpp"
+#include "vmime/utility/stringUtils.hpp"
+
+
+namespace vmime
+{
+
+
+disposition::disposition()
+{
+}
+
+
+disposition::disposition(const string& actionMode, const string& sendingMode,
+ const string& type, const string& modifier)
+ : m_actionMode(actionMode), m_sendingMode(sendingMode), m_type(type)
+{
+ m_modifiers.push_back(modifier);
+}
+
+
+disposition* disposition::clone() const
+{
+ disposition* disp = new disposition;
+
+ disp->m_actionMode = m_actionMode;
+ disp->m_sendingMode = m_sendingMode;
+ disp->m_type = m_type;
+ disp->m_modifiers.resize(m_modifiers.size());
+
+ std::copy(m_modifiers.begin(), m_modifiers.end(), disp->m_modifiers.begin());
+
+ return (disp);
+}
+
+
+void disposition::copyFrom(const component& other)
+{
+ const disposition& disp = dynamic_cast <const disposition&>(other);
+
+ m_actionMode = disp.m_actionMode;
+ m_sendingMode = disp.m_sendingMode;
+ m_type = disp.m_type;
+ m_modifiers.resize(disp.m_modifiers.size());
+
+ std::copy(disp.m_modifiers.begin(), disp.m_modifiers.end(), m_modifiers.begin());
+}
+
+
+disposition& disposition::operator=(const disposition& other)
+{
+ copyFrom(other);
+ return (*this);
+}
+
+
+const std::vector <const component*> disposition::getChildComponents() const
+{
+ return std::vector <const component*>();
+}
+
+
+void disposition::setActionMode(const string& mode)
+{
+ m_actionMode = mode;
+}
+
+
+const string& disposition::getActionMode() const
+{
+ return (m_actionMode);
+}
+
+
+void disposition::setSendingMode(const string& mode)
+{
+ m_sendingMode = mode;
+}
+
+
+const string& disposition::getSendingMode() const
+{
+ return (m_sendingMode);
+}
+
+
+void disposition::setType(const string& type)
+{
+ m_type = type;
+}
+
+
+const string& disposition::getType() const
+{
+ return (m_type);
+}
+
+
+void disposition::addModifier(const string& modifier)
+{
+ if (!hasModifier(modifier))
+ m_modifiers.push_back(utility::stringUtils::toLower(modifier));
+}
+
+
+void disposition::removeModifier(const string& modifier)
+{
+ const string modifierLC = utility::stringUtils::toLower(modifier);
+
+ for (std::vector <string>::iterator it = m_modifiers.begin() ;
+ it != m_modifiers.end() ; ++it)
+ {
+ if (*it == modifierLC)
+ {
+ m_modifiers.erase(it);
+ break;
+ }
+ }
+}
+
+
+void disposition::removeAllModifiers()
+{
+ m_modifiers.clear();
+}
+
+
+const bool disposition::hasModifier(const string& modifier) const
+{
+ const string modifierLC = utility::stringUtils::toLower(modifier);
+
+ for (std::vector <string>::const_iterator it = m_modifiers.begin() ;
+ it != m_modifiers.end() ; ++it)
+ {
+ if (*it == modifierLC)
+ return (true);
+ }
+
+ return (false);
+}
+
+
+const std::vector <string> disposition::getModifierList() const
+{
+ return (m_modifiers);
+}
+
+
+void disposition::parse(const string& buffer, const string::size_type position,
+ const string::size_type end, string::size_type* newPosition)
+{
+ // disposition-mode ";" disposition-type
+ // [ "/" disposition-modifier *( "," disposition-modifier ) ]
+ //
+ // disposition-mode = action-mode "/" sending-mode
+
+ string::size_type pos = position;
+
+ while (pos < end && parserHelpers::isSpace(buffer[pos]))
+ ++pos;
+
+ // -- disposition-mode
+ const string::size_type modeStart = pos;
+ string::size_type modeEnd = pos;
+
+ while (pos < end && buffer[pos] != ';')
+ {
+ ++modeEnd;
+ ++pos;
+ }
+
+ while (modeEnd > modeStart && parserHelpers::isSpace(buffer[modeEnd - 1]))
+ --modeEnd;
+
+ const string mode = string(buffer.begin() + modeStart, buffer.begin() + modeEnd);
+ const string::size_type slash = mode.find('/');
+
+ if (slash != string::npos)
+ {
+ m_actionMode = string(mode.begin(), mode.begin() + slash);
+ m_sendingMode = string(mode.begin() + slash + 1, mode.end());
+ }
+ else
+ {
+ m_actionMode = mode;
+ m_sendingMode.clear();
+ }
+
+ if (pos < end)
+ {
+ // Skip ';'
+ ++pos;
+ }
+
+ while (pos < end && parserHelpers::isSpace(buffer[pos]))
+ ++pos;
+
+ // -- disposition-type
+ const string::size_type typeStart = pos;
+ string::size_type typeEnd = pos;
+
+ while (pos < end && buffer[pos] != '/')
+ {
+ ++typeEnd;
+ ++pos;
+ }
+
+ while (typeEnd > typeStart && parserHelpers::isSpace(buffer[typeEnd - 1]))
+ --typeEnd;
+
+ m_type = string(buffer.begin() + typeStart, buffer.begin() + typeEnd);
+
+ m_modifiers.clear();
+
+ if (pos < end) // modifiers follow
+ {
+ // Skip '/'
+ ++pos;
+
+ while (pos < end)
+ {
+ while (pos < end && parserHelpers::isSpace(buffer[pos]))
+ ++pos;
+
+ const string::size_type modifierStart = pos;
+ string::size_type modifierEnd = pos;
+
+ while (pos < end && buffer[pos] != ',')
+ {
+ ++modifierEnd;
+ ++pos;
+ }
+
+ while (modifierEnd > modifierStart && parserHelpers::isSpace(buffer[modifierEnd - 1]))
+ --modifierEnd;
+
+ if (modifierEnd > modifierStart)
+ {
+ m_modifiers.push_back(string(buffer.begin() + modifierStart,
+ buffer.begin() + modifierEnd));
+ }
+
+ // Skip ','
+ if (pos < end)
+ ++pos;
+ }
+ }
+
+ if (newPosition)
+ *newPosition = pos;
+}
+
+
+void disposition::generate(utility::outputStream& os, const string::size_type maxLineLength,
+ const string::size_type curLinePos, string::size_type* newLinePos) const
+{
+ string::size_type pos = curLinePos;
+
+ const string actionMode = (m_actionMode.empty() ? "automatic-action" : m_actionMode);
+ const string sendingMode = (m_sendingMode.empty() ? "MDN-sent-automatically" : m_sendingMode);
+
+ os << actionMode << "/" << sendingMode << ";";
+ pos += actionMode.length() + 1 + sendingMode.length() + 1;
+
+ if (pos > maxLineLength)
+ {
+ os << NEW_LINE_SEQUENCE;
+ pos = NEW_LINE_SEQUENCE_LENGTH;
+ }
+
+ const string type = (m_type.empty() ? "displayed" : m_type);
+
+ os << type;
+ pos += type.length();
+
+ if (m_modifiers.size() >= 1)
+ {
+ os << "/" << m_modifiers[0];
+ pos += 1 + m_modifiers[0].length();
+
+ for (std::vector <string>::size_type i = 1 ; i < m_modifiers.size() ; ++i)
+ {
+ os << "," << m_modifiers[i];
+ pos += 1 + m_modifiers[i].length();
+ }
+ }
+
+ if (newLinePos)
+ *newLinePos = pos;
+}
+
+
+}