vmime/src/mailboxGroup.cpp

365 lines
6.8 KiB
C++
Raw Normal View History

2004-10-05 10:28:21 +00:00
//
// VMime library (http://vmime.sourceforge.net)
// 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.
//
// 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/mailboxGroup.hpp"
#include "vmime/parserHelpers.hpp"
#include "vmime/exception.hpp"
2004-10-05 10:28:21 +00:00
namespace vmime
{
mailboxGroup::mailboxGroup()
{
}
2004-10-21 15:05:47 +00:00
mailboxGroup::mailboxGroup(const mailboxGroup& mboxGroup)
2004-10-05 10:28:21 +00:00
: address()
{
2004-10-21 15:05:47 +00:00
copyFrom(mboxGroup);
2004-10-05 10:28:21 +00:00
}
mailboxGroup::mailboxGroup(const text& name)
: m_name(name)
{
}
mailboxGroup::~mailboxGroup()
{
2004-10-21 15:05:47 +00:00
removeAllMailboxes();
2004-10-05 10:28:21 +00:00
}
void mailboxGroup::parse(const string& buffer, const string::size_type position,
const string::size_type end, string::size_type* newPosition)
{
const string::value_type* const pend = buffer.data() + end;
const string::value_type* const pstart = buffer.data() + position;
const string::value_type* p = pstart;
while (p < pend && parserHelpers::isSpace(*p))
2004-10-05 10:28:21 +00:00
++p;
string name;
while (p < pend && *p != ':')
{
name += *p;
++p;
}
if (p < pend && *p == ':')
++p;
string::size_type pos = position + (p - pstart);
while (pos < end)
{
address* parsedAddress = address::parseNext(buffer, pos, end, &pos);
if (parsedAddress)
{
if (parsedAddress->isGroup())
{
mailboxGroup* group = static_cast <mailboxGroup*>(parsedAddress);
// Sub-groups are not allowed in mailbox groups: so, we add all
// the contents of the sub-group into this group...
2004-10-21 15:05:47 +00:00
for (int i = 0 ; i < group->getMailboxCount() ; ++i)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
m_list.push_back(group->getMailboxAt(i)->clone());
2004-10-05 10:28:21 +00:00
}
delete (parsedAddress);
}
else
{
m_list.push_back(static_cast <mailbox*>(parsedAddress));
}
}
}
text::decodeAndUnfold(name, &m_name);
2004-10-05 10:28:21 +00:00
2004-12-15 20:28:09 +00:00
setParsedBounds(position, end);
2004-10-05 10:28:21 +00:00
if (newPosition)
*newPosition = end;
}
void mailboxGroup::generate(utility::outputStream& os, const string::size_type maxLineLength,
const string::size_type curLinePos, string::size_type* newLinePos) const
{
// We have to encode the name:
// - if it contains characters in a charset different from "US-ASCII",
// - and/or if it contains one or more of these special chars:
// SPACE TAB " ; , < > ( ) @ / ? . = :
// Check whether there are words that are not "US-ASCII"
// and/or contain the special chars.
bool forceEncode = false;
2004-10-21 15:05:47 +00:00
for (int w = 0 ; !forceEncode && w < m_name.getWordCount() ; ++w)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
if (m_name.getWordAt(w)->getCharset() == charset(charsets::US_ASCII))
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
const string& buffer = m_name.getWordAt(w)->getBuffer();
2004-10-05 10:28:21 +00:00
for (string::const_iterator c = buffer.begin() ;
!forceEncode && c != buffer.end() ; ++c)
{
switch (*c)
{
case ' ':
case '\t':
case ';':
case ',':
case '<': case '>':
case '(': case ')':
case '@':
case '/':
case '?':
case '.':
case '=':
case ':':
forceEncode = true;
break;
}
}
}
}
string::size_type pos = curLinePos;
m_name.encodeAndFold(os, maxLineLength - 2, pos, &pos,
forceEncode ? text::FORCE_ENCODING : 0);
2004-10-05 10:28:21 +00:00
os << ":";
++pos;
2004-10-21 15:05:47 +00:00
for (std::vector <mailbox*>::const_iterator it = m_list.begin() ;
it != m_list.end() ; ++it)
2004-10-05 10:28:21 +00:00
{
if (it != m_list.begin())
{
os << ", ";
pos += 2;
}
else
{
os << " ";
++pos;
}
2004-10-21 15:05:47 +00:00
(*it)->generate(os, maxLineLength - 2, pos, &pos);
2004-10-05 10:28:21 +00:00
}
os << ";";
pos++;
if (newLinePos)
*newLinePos = pos;
}
2004-10-21 15:05:47 +00:00
void mailboxGroup::copyFrom(const component& other)
{
const mailboxGroup& source = dynamic_cast <const mailboxGroup&>(other);
m_name = source.m_name;
removeAllMailboxes();
for (std::vector <mailbox*>::const_iterator it = source.m_list.begin() ;
it != source.m_list.end() ; ++it)
{
m_list.push_back((*it)->clone());
}
}
mailboxGroup* mailboxGroup::clone() const
2004-10-05 10:28:21 +00:00
{
return new mailboxGroup(*this);
}
2004-10-21 15:05:47 +00:00
mailboxGroup& mailboxGroup::operator=(const component& other)
{
copyFrom(other);
return (*this);
}
const text& mailboxGroup::getName() const
{
return (m_name);
}
void mailboxGroup::setName(const text& name)
{
m_name = name;
}
const bool mailboxGroup::isGroup() const
{
return (true);
}
const bool mailboxGroup::isEmpty() const
{
return (m_list.empty());
}
void mailboxGroup::appendMailbox(mailbox* mbox)
{
m_list.push_back(mbox);
}
void mailboxGroup::insertMailboxBefore(mailbox* beforeMailbox, mailbox* mbox)
{
const std::vector <mailbox*>::iterator it = std::find
(m_list.begin(), m_list.end(), beforeMailbox);
if (it == m_list.end())
throw exceptions::no_such_mailbox();
m_list.insert(it, mbox);
}
void mailboxGroup::insertMailboxBefore(const int pos, mailbox* mbox)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
m_list.insert(m_list.begin() + pos, mbox);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
void mailboxGroup::insertMailboxAfter(mailbox* afterMailbox, mailbox* mbox)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
const std::vector <mailbox*>::iterator it = std::find
(m_list.begin(), m_list.end(), afterMailbox);
if (it == m_list.end())
throw exceptions::no_such_mailbox();
m_list.insert(it + 1, mbox);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
void mailboxGroup::insertMailboxAfter(const int pos, mailbox* mbox)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
m_list.insert(m_list.begin() + pos + 1, mbox);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
void mailboxGroup::removeMailbox(mailbox* mbox)
{
const std::vector <mailbox*>::iterator it = std::find
(m_list.begin(), m_list.end(), mbox);
if (it == m_list.end())
throw exceptions::no_such_mailbox();
delete (*it);
m_list.erase(it);
}
void mailboxGroup::removeMailbox(const int pos)
{
const std::vector <mailbox*>::iterator it = m_list.begin() + pos;
delete (*it);
m_list.erase(it);
}
void mailboxGroup::removeAllMailboxes()
2004-10-05 10:28:21 +00:00
{
free_container(m_list);
}
2004-10-21 15:05:47 +00:00
const int mailboxGroup::getMailboxCount() const
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
return (m_list.size());
}
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
mailbox* mailboxGroup::getMailboxAt(const int pos)
{
return (m_list[pos]);
}
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
const mailbox* mailboxGroup::getMailboxAt(const int pos) const
2004-10-21 15:05:47 +00:00
{
return (m_list[pos]);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
const std::vector <const mailbox*> mailboxGroup::getMailboxList() const
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
std::vector <const mailbox*> list;
list.reserve(m_list.size());
for (std::vector <mailbox*>::const_iterator it = m_list.begin() ;
it != m_list.end() ; ++it)
{
list.push_back(*it);
}
return (list);
}
const std::vector <mailbox*> mailboxGroup::getMailboxList()
{
return (m_list);
2004-10-05 10:28:21 +00:00
}
const std::vector <const component*> mailboxGroup::getChildComponents() const
{
std::vector <const component*> list;
copy_vector(m_list, list);
return (list);
}
2004-10-05 10:28:21 +00:00
} // vmime