arabica/include/DOM/Simple/NamedNodeMapImpl.hpp

238 lines
6.5 KiB
C++
Raw Normal View History

2002-06-21 11:16:28 +00:00
#ifndef JEZUK_SimpleDOM_NAMEDNODEMAPIMPL_H
#define JEZUK_SimpleDOM_NAMEDNODEMAPIMPL_H
2007-09-04 22:55:47 +00:00
#include <DOM/Simple/NodeImpl.hpp>
#include <DOM/NamedNodeMap.hpp>
2002-06-21 11:16:28 +00:00
#include <deque>
#include <algorithm>
#include <functional>
2007-09-05 11:47:13 +00:00
namespace Arabica
{
2002-06-21 11:16:28 +00:00
namespace SimpleDOM
{
template<class stringT, class string_adaptorT>
class nameIs
2002-06-21 11:16:28 +00:00
{
public:
explicit nameIs(const stringT& name) : name_(name) { }
2007-09-05 11:47:13 +00:00
bool operator()(const Arabica::SimpleDOM::NodeImpl<stringT, string_adaptorT>* node) const
2002-06-21 11:16:28 +00:00
{
return node->getNodeName() == name_;
2002-06-21 11:16:28 +00:00
} // operator()
private:
stringT name_;
}; // class nameIs
template<class stringT, class string_adaptorT>
class namespaceAndNameIs
2002-06-21 11:16:28 +00:00
{
public:
namespaceAndNameIs(const stringT& namespaceURI, const stringT& localName) :
namespaceURI_(namespaceURI),
localName_(localName) { }
bool operator()(const NodeImpl<stringT, string_adaptorT>* node) const
{
if(string_adaptorT::empty(namespaceURI_))
2002-06-21 11:16:28 +00:00
{
if((node->hasNamespaceURI() == false) &&
((node->getLocalName() == localName_) ||
(string_adaptorT::empty(node->getLocalName()) && node->getNodeName() == localName_)
2002-06-21 11:16:28 +00:00
)
)
return true;
} // if(namespaceURI_.empty())
if((node->getNamespaceURI() == namespaceURI_) && (node->getLocalName() == localName_))
return true;
return false;
} // operator()
private:
stringT namespaceURI_;
stringT localName_;
}; // class namespaceAndNameIs
template<class stringT, class string_adaptorT>
class NamedNodeMapImpl : public DOM::NamedNodeMap_impl<stringT, string_adaptorT>
2002-06-21 11:16:28 +00:00
{
public:
typedef DOM::Node_impl<stringT, string_adaptorT> DOMNode_implT;
typedef NodeImpl<stringT, string_adaptorT> NodeImplT;
typedef NamedNodeMapImpl<stringT, string_adaptorT> NamedNodeMapImplT;
typedef DocumentImpl<stringT, string_adaptorT> DocumentImplT;
NamedNodeMapImpl(DocumentImplT* ownerDoc) :
DOM::NamedNodeMap_impl<stringT, string_adaptorT>(),
nodes_(),
2002-06-21 11:16:28 +00:00
readOnly_(false),
ownerDoc_(ownerDoc)
{
} // NamedNodeMapImpl
virtual ~NamedNodeMapImpl()
{
2002-11-23 20:03:54 +00:00
for(typename NodeListT::iterator i = nodes_.begin(); i != nodes_.end(); ++i)
2002-06-21 11:16:28 +00:00
delete (*i);
} // ~NodeImpl
///////////////////////////////////////////////////////
// Ref counting
virtual void addRef()
{
if(ownerDoc_)
ownerDoc_->addRef();
} // addRef
virtual void releaseRef()
{
if(ownerDoc_)
ownerDoc_->releaseRef();
} // releaseRef
virtual void setOwnerDoc(DocumentImplT* ownerDoc)
2002-06-21 11:16:28 +00:00
{
ownerDoc_ = ownerDoc;
2002-11-23 20:03:54 +00:00
for(typename NodeListT::iterator i = nodes_.begin(); i != nodes_.end(); ++i)
2002-06-21 11:16:28 +00:00
(*i)->setOwnerDoc(ownerDoc);
} // setOwnerDoc
///////////////////////////////////////////////////
// DOM::NamedNodeMap methods
virtual DOMNode_implT* getNamedItem(const stringT& name) const
2002-06-21 11:16:28 +00:00
{
return getNode(const_cast<NamedNodeMapImplT*>(this)->findByName(name));
2002-06-21 11:16:28 +00:00
} // getNamedItem
virtual DOMNode_implT* setNamedItem(DOMNode_implT* arg)
2002-06-21 11:16:28 +00:00
{
throwIfReadOnly();
2010-12-26 22:20:33 +00:00
checkOwnerDocument(arg);
return setNode(findByName(arg->getNodeName()), dynamic_cast<NodeImplT*>(arg));
2002-06-21 11:16:28 +00:00
} // setNamedItem
virtual DOMNode_implT* removeNamedItem(const stringT& name)
2002-06-21 11:16:28 +00:00
{
throwIfReadOnly();
return removeNode(findByName(name));
} // removedNamedItem
virtual DOMNode_implT* item(unsigned int index) const
2002-06-21 11:16:28 +00:00
{
return do_item(index);
} // item
virtual unsigned int getLength() const
{
return static_cast<unsigned int>(nodes_.size());
2002-06-21 11:16:28 +00:00
} // getLength
virtual DOMNode_implT* getNamedItemNS(const stringT& namespaceURI, const stringT& localName) const
2002-06-21 11:16:28 +00:00
{
return getNode(const_cast<NamedNodeMapImplT*>(this)->findByNamespaceAndName(namespaceURI, localName));
2002-06-21 11:16:28 +00:00
} // getNamedItemNS
virtual DOMNode_implT* setNamedItemNS(DOMNode_implT* arg)
2002-06-21 11:16:28 +00:00
{
throwIfReadOnly();
2010-12-26 22:20:33 +00:00
checkOwnerDocument(arg);
return setNode(findByNamespaceAndName(arg->getNamespaceURI(), arg->getLocalName()), dynamic_cast<NodeImplT*>(arg));
2002-06-21 11:16:28 +00:00
} // setNamedItemNS
virtual DOMNode_implT* removeNamedItemNS(const stringT& namespaceURI, const stringT& localName)
2002-06-21 11:16:28 +00:00
{
throwIfReadOnly();
return removeNode(findByNamespaceAndName(namespaceURI, localName));
} // removeNamedItemNS
///////////////////////////////////////////////////////
// extensions
void setReadOnly(bool readOnly)
{
readOnly_ = readOnly;
} // setReadOnly
NodeImplT* do_item(unsigned int index) const
2002-06-21 11:16:28 +00:00
{
if(index >= nodes_.size())
return 0;
return nodes_[index];
} // item
protected:
2002-06-21 11:16:28 +00:00
void throwIfReadOnly() const
{
if(readOnly_)
throw DOM::DOMException(DOM::DOMException::NO_MODIFICATION_ALLOWED_ERR);
} // throwIfReadOnly
2010-12-26 22:20:33 +00:00
void checkOwnerDocument(const DOMNode_implT* arg)
{
if(!ownerDoc_)
return;
if(arg->getOwnerDocument() != ownerDoc_)
throw DOM::DOMException(DOM::DOMException::WRONG_DOCUMENT_ERR);
} // checkOwnerDocument
private:
typedef std::deque<NodeImplT*> NodeListT;
2002-06-21 11:16:28 +00:00
NodeImplT* getNode(typename NodeListT::const_iterator n) const
2002-06-21 11:16:28 +00:00
{
if(n == nodes_.end())
return 0;
return *n;
} // getNode
NodeImplT* setNode(typename NodeListT::iterator n, NodeImplT* arg)
2002-06-21 11:16:28 +00:00
{
if(ownerDoc_)
ownerDoc_->adopted(arg);
2002-06-21 11:16:28 +00:00
if(n == nodes_.end())
{
nodes_.push_back(arg);
return 0;
} // if(n == nodes_.end())
NodeImplT* removedNode = *n;
2002-06-21 11:16:28 +00:00
*n = arg;
ownerDoc_->orphaned(removedNode);
return removedNode;
} // setNode
NodeImplT* removeNode(typename NodeListT::iterator n)
2002-06-21 11:16:28 +00:00
{
if(n == nodes_.end())
throw DOM::DOMException(DOM::DOMException::NOT_FOUND_ERR);
NodeImplT* removedNode = *n;
2002-06-21 11:16:28 +00:00
nodes_.erase(n);
ownerDoc_->orphaned(removedNode);
return removedNode;
} // removeNode
2002-11-23 20:03:54 +00:00
typename NodeListT::iterator findByName(const stringT& name)
2002-06-21 11:16:28 +00:00
{
return std::find_if(nodes_.begin(), nodes_.end(), nameIs<stringT, string_adaptorT>(name));
} // findByName
2002-11-23 20:03:54 +00:00
typename NodeListT::iterator findByNamespaceAndName(const stringT& namespaceURI, const stringT& localName)
2002-06-21 11:16:28 +00:00
{
return std::find_if(nodes_.begin(), nodes_.end(), namespaceAndNameIs<stringT, string_adaptorT>(namespaceURI, localName));
} // findByName
NodeListT nodes_;
bool readOnly_;
protected:
DocumentImplT* ownerDoc_;
2002-06-21 11:16:28 +00:00
}; // class NamedNodeMapImpl
} // namespace SAX2DOM
2007-09-05 11:47:13 +00:00
} // namespace Arabica
2002-06-21 11:16:28 +00:00
#endif