201 lines
8.1 KiB
C++
201 lines
8.1 KiB
C++
|
/*
|
||
|
* Portable Agile C++ Classes (PACC)
|
||
|
* Copyright (C) 2001-2003 by Marc Parizeau
|
||
|
* http://manitou.gel.ulaval.ca/~parizeau/PACC
|
||
|
*
|
||
|
* This library is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU Lesser General Public
|
||
|
* License as published by the Free Software Foundation; either
|
||
|
* version 2.1 of the License, or (at your option) any later version.
|
||
|
*
|
||
|
* This library 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
|
||
|
* Lesser General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Lesser General Public
|
||
|
* License along with this library; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||
|
*
|
||
|
* Contact:
|
||
|
* Laboratoire de Vision et Systemes Numeriques
|
||
|
* Departement de genie electrique et de genie informatique
|
||
|
* Universite Laval, Quebec, Canada, G1K 7P4
|
||
|
* http://vision.gel.ulaval.ca
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*!
|
||
|
* \file PACC/XML/Iterator.hpp
|
||
|
* \brief Class definition for the %XML node iterator.
|
||
|
* \author Marc Parizeau, Laboratoire de vision et systèmes numériques, Université Laval
|
||
|
* $Revision: 1.6.2.1 $
|
||
|
* $Date: 2007/09/10 18:24:10 $
|
||
|
*/
|
||
|
|
||
|
#ifndef PACC_XML_Iterator_hpp_
|
||
|
#define PACC_XML_Iterator_hpp_
|
||
|
|
||
|
#include "XML/Node.hpp"
|
||
|
#include "Util/Assert.hpp"
|
||
|
|
||
|
namespace PACC {
|
||
|
|
||
|
using namespace std;
|
||
|
|
||
|
namespace XML {
|
||
|
|
||
|
/*! \brief %Node iterator.
|
||
|
\author Marc Parizeau, Laboratoire de vision et systèmes numériques, Université Laval
|
||
|
\ingroup XML
|
||
|
|
||
|
A node iterator is used to iterate through sibling nodes, using an interface similar to STL bidirectional iterators. For example, to iterate through the childs of node \c lNode:
|
||
|
\code
|
||
|
Node lNode;
|
||
|
...
|
||
|
for(Iterator lChild = lNode.getFirstChild(); lChild; ++lChild) {
|
||
|
// process each child
|
||
|
...
|
||
|
}
|
||
|
\endcode
|
||
|
Operator* can be used to return a node reference, operator-> to return a node pointer, operator++ (both prefix and postfix) to skip to next sibling, and operator-- (both prefix and postfix) to skip to previous sibling.
|
||
|
|
||
|
A boolean casting operator is also available to test for end of sequence (see second term of for-statement above). An iterator pointing to a valid node always returns true while an invalid iterator returns false. An invalid iterator should never be dereferenced, incremented, or decremented.
|
||
|
*/
|
||
|
class Iterator {
|
||
|
public:
|
||
|
//! Construct an empty iterator.
|
||
|
Iterator(void) : mNode(0) {}
|
||
|
//! Construct an iterator for node pointer \c inNode.
|
||
|
Iterator(Node* inNode) : mNode(inNode) {}
|
||
|
|
||
|
//! Iterate to next sibling of current node (prefix version); return iterator pointing to next sibling.
|
||
|
Iterator& operator++(void) {
|
||
|
PACC_AssertM(mNode, "Cannot increment an invalid iterator!");
|
||
|
mNode = mNode->mNextSibling;
|
||
|
return *this;
|
||
|
}
|
||
|
//! Iterate to next sibling of current node (postfix version); return iterator pointing to current node.
|
||
|
Iterator operator++(int) {
|
||
|
PACC_AssertM(mNode, "Cannot increment an invalid iterator!");
|
||
|
Iterator lTmp = *this;
|
||
|
mNode = mNode->mNextSibling;
|
||
|
return lTmp;
|
||
|
}
|
||
|
//! Iterate to previous sibling of current node (prefix version); return iterator pointing to previous sibling.
|
||
|
Iterator& operator--(void) {
|
||
|
PACC_AssertM(mNode, "Cannot decrement an invalid iterator!");
|
||
|
if(mNode) mNode = mNode->mPrevSibling;
|
||
|
return *this;
|
||
|
}
|
||
|
//! Iterate to previous sibling of current node (postfix version); return iterator pointing to current node.
|
||
|
Iterator operator--(int) {
|
||
|
PACC_AssertM(mNode, "Cannot increment an invalid iterator!");
|
||
|
Iterator lTmp = *this;
|
||
|
mNode = mNode->mPrevSibling;
|
||
|
return lTmp;
|
||
|
}
|
||
|
//! Return reference to current node (const version).
|
||
|
Node& operator*(void) const {
|
||
|
PACC_AssertM(mNode, "Cannot dereference an invalid iterator!");
|
||
|
return *mNode;
|
||
|
}
|
||
|
//! Return pointer to current node (const version).
|
||
|
Node* operator->(void) const {
|
||
|
PACC_AssertM(mNode, "Invalid iterator!");
|
||
|
return mNode;
|
||
|
}
|
||
|
//! Return true if this iterator is the same as iterator \c inIter; false otherwise.
|
||
|
bool operator==(const Iterator& inIter) const {return mNode == inIter.mNode;}
|
||
|
//! Return true if this iterator is different from iterator \c inIter; false otherwise.
|
||
|
bool operator!=(const Iterator& inIter) const {return mNode != inIter.mNode;}
|
||
|
//! Cast Iterator into bool; return true for a valid iterator, false otherwise.
|
||
|
operator bool(void) const {return mNode != 0;}
|
||
|
|
||
|
protected:
|
||
|
Node* mNode; //!< Pointer to current node.
|
||
|
|
||
|
friend class ConstIterator;
|
||
|
|
||
|
};
|
||
|
|
||
|
/*! \brief %Node const iterator.
|
||
|
\author Marc Parizeau, Laboratoire de vision et systèmes numériques, Université Laval
|
||
|
\ingroup XML
|
||
|
|
||
|
A const node iterator is used to iterate through sibling nodes, using an interface similar to STL bidirectional const_iterators. For example, to iterate through the childs of node \c lNode:
|
||
|
\code
|
||
|
Node lNode;
|
||
|
...
|
||
|
for(ConstIterator lChild = lNode.getFirstChild(); lChild; ++lChild) {
|
||
|
// process each child
|
||
|
...
|
||
|
}
|
||
|
\endcode
|
||
|
Operator* can be used to return a node const reference, operator-> to return a node const pointer, operator++ (both prefix and postfix) to skip to next element, and operator-- (both prefix and postfix) to skip to next element.
|
||
|
|
||
|
A boolean casting operator is also available to test for end of sequence (see second term of for-statement above). An iterator pointing to a valid node always returns true while an invalid iterator returns false. An invalid iterator should never be dereferenced, incremented, or decremented.
|
||
|
*/
|
||
|
class ConstIterator {
|
||
|
public:
|
||
|
//! Construct an empty const iterator.
|
||
|
ConstIterator(void) : mNode(0) {}
|
||
|
//! Construct a const iterator for node pointer \c inNode.
|
||
|
ConstIterator(const Node* inNode) : mNode(inNode) {}
|
||
|
//! Construct a const iterator from a non const iterator \c inPos.
|
||
|
ConstIterator(const Iterator& inPos) : mNode(inPos.mNode) {}
|
||
|
|
||
|
//! Iterate to next sibling of current node (prefix version); return iterator pointing to next sibling.
|
||
|
ConstIterator& operator++(void) {
|
||
|
PACC_AssertM(mNode, "Cannot increment an invalid iterator!");
|
||
|
mNode = mNode->mNextSibling;
|
||
|
return *this;
|
||
|
}
|
||
|
//! Iterate to next sibling of current node (postfix version); return iterator pointing to current node.
|
||
|
ConstIterator operator++(int) {
|
||
|
PACC_AssertM(mNode, "Cannot increment an invalid iterator!");
|
||
|
ConstIterator lTmp = *this;
|
||
|
mNode = mNode->mNextSibling;
|
||
|
return lTmp;
|
||
|
}
|
||
|
//! Iterate to previous sibling of current node (prefix version); return iterator pointing to previous sibling.
|
||
|
ConstIterator& operator--(void) {
|
||
|
PACC_AssertM(mNode, "Cannot decrement an invalid iterator!");
|
||
|
mNode = mNode->mPrevSibling;
|
||
|
return *this;
|
||
|
}
|
||
|
//! Iterate to previous sibling of current node (postfix version); return iterator pointing to current node.
|
||
|
ConstIterator operator--(int) {
|
||
|
PACC_AssertM(mNode, "Cannot decrement an invalid iterator!");
|
||
|
ConstIterator lTmp = *this;
|
||
|
mNode = mNode->mPrevSibling;
|
||
|
return lTmp;
|
||
|
}
|
||
|
//! Return reference to current node (const version).
|
||
|
const Node& operator*(void) const {
|
||
|
PACC_AssertM(mNode, "Cannot dereference an invalid iterator!");
|
||
|
return *mNode;
|
||
|
}
|
||
|
//! Return pointer to current node (const version).
|
||
|
const Node* operator->(void) const {
|
||
|
PACC_AssertM(mNode, "Invalid iterator!");
|
||
|
return mNode;
|
||
|
}
|
||
|
//! Return true if this iterator is the same as iterator \c inIter; false otherwise.
|
||
|
bool operator==(const ConstIterator& inIter) const {return mNode == inIter.mNode;}
|
||
|
//! Return true if this iterator is different from iterator \c inIter; false otherwise.
|
||
|
bool operator!=(const ConstIterator& inIter) const {return mNode != inIter.mNode;}
|
||
|
//! Cast Iterator into bool; return true for a valid iterator, false otherwise.
|
||
|
operator bool(void) const {return mNode != 0;}
|
||
|
|
||
|
protected:
|
||
|
const Node* mNode; //!< Pointer to current node.
|
||
|
|
||
|
};
|
||
|
|
||
|
} // end of XML namespace
|
||
|
|
||
|
} // end of PACC namespace
|
||
|
|
||
|
#endif // PACC_XML_Iterator_hpp_
|