arabica/include/DOM/Simple/NodeImpl.hpp

530 lines
16 KiB
C++
Raw Normal View History

2002-06-21 11:16:28 +00:00
#ifndef JEZUK_DOM_SimpleDOM_NODE_H
#define JEZUK_DOM_SimpleDOM_NODE_H
////////////////////////////
// C++ DOM definition
//
// $Id$
////////////////////////////
2005-06-08 21:45:07 +00:00
2007-09-04 22:55:47 +00:00
#include <DOM/Node.hpp>
#include <DOM/DOMException.hpp>
2002-06-21 11:16:28 +00:00
#include <deque>
#include <algorithm>
//#include <iostream>
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 DocumentImpl;
template<class stringT, class string_adaptorT>
class NodeImpl : virtual public DOM::Node_impl<stringT, string_adaptorT>
2002-06-21 11:16:28 +00:00
{
public:
typedef NodeImpl<stringT, string_adaptorT> NodeImplT;
typedef DocumentImpl<stringT, string_adaptorT> DocumentImplT;
typedef DOM::Node_impl<stringT, string_adaptorT> DOMNode_implT;
typedef DOM::Text_impl<stringT, string_adaptorT> DOMText_implT;
typedef DOM::Document_impl<stringT, string_adaptorT> DOMDocument_implT;
typedef DOM::NamedNodeMap_impl<stringT, string_adaptorT> DOMNamedNodeMap_implT;
typedef DOM::NodeList_impl<stringT, string_adaptorT> DOMNodeList_implT;
NodeImpl(DocumentImplT* ownerDoc) :
2002-06-21 11:16:28 +00:00
parentNode_(0),
ownerDoc_(ownerDoc),
prevSibling_(0),
nextSibling_(0),
readOnly_(false)
{
//std::cout << std::endl << "born " << this << std::endl;
} // NodeImpl
virtual ~NodeImpl()
{
//std::cout << std::endl << "die " << this << std::endl;
}
///////////////////////////////////////////////////////
// Ref counting
virtual void addRef()
{
if(ownerDoc_)
ownerDoc_->addRef();
} // addRef
virtual void releaseRef()
{
if(ownerDoc_)
ownerDoc_->releaseRef();
} // releaseRef
///////////////////////////////////////////////////////
// Node methods
2005-12-09 14:09:32 +00:00
virtual const stringT& getNodeName() const = 0;
2002-06-21 11:16:28 +00:00
virtual const stringT& getNodeValue() const { return ownerDoc_->empty_string(); }
virtual void setNodeValue(const stringT& /*nodeValue*/) { throwIfReadOnly(); }
2002-06-21 11:16:28 +00:00
virtual DOM::Node_base::Type getNodeType() const = 0;
virtual DOMNode_implT* getParentNode() const { return parentNode_; }
2002-06-21 11:16:28 +00:00
virtual DOMNodeList_implT* getChildNodes() const = 0;
2002-06-21 11:16:28 +00:00
virtual DOMNode_implT* getFirstChild() const = 0;
virtual DOMNode_implT* getLastChild() const = 0;
2002-06-21 11:16:28 +00:00
virtual DOMNode_implT* getPreviousSibling() const { return prevSibling_; }
virtual DOMNode_implT* getNextSibling() const { return nextSibling_; }
2002-06-21 11:16:28 +00:00
virtual DOMNamedNodeMap_implT* getAttributes() const { return 0; }
2002-06-21 11:16:28 +00:00
virtual DOMDocument_implT* getOwnerDocument() const { return ownerDoc_; }
2002-06-21 11:16:28 +00:00
virtual DOMNode_implT* insertBefore(DOMNode_implT* newChild, DOMNode_implT* refChild) = 0;
virtual DOMNode_implT* replaceChild(DOMNode_implT* newChild, DOMNode_implT* oldChild) = 0;
virtual DOMNode_implT* removeChild(DOMNode_implT* oldChild) = 0;
virtual DOMNode_implT* appendChild(DOMNode_implT* newChild) = 0;
virtual void purgeChild(DOMNode_implT* oldChild) = 0;
2002-06-21 11:16:28 +00:00
virtual bool hasChildNodes() const = 0;
virtual DOMNode_implT* cloneNode(bool deep) const = 0;
2002-06-21 11:16:28 +00:00
virtual void normalize()
{
DOMNode_implT*child = getFirstChild();
2002-06-21 11:16:28 +00:00
while(child != 0)
{
DOMNode_implT*next = child->getNextSibling();
2002-06-21 11:16:28 +00:00
2008-09-03 00:39:39 +01:00
if((child->getNodeType() == DOM::Node_base::TEXT_NODE) ||
(child->getNodeType() == DOM::Node_base::CDATA_SECTION_NODE))
2002-06-21 11:16:28 +00:00
{
DOMText_implT* textNode = dynamic_cast<DOMText_implT*>(child);
2008-09-03 00:39:39 +01:00
while((next != 0) &&
((next->getNodeType() == DOM::Node_base::TEXT_NODE) ||
(next->getNodeType() == DOM::Node_base::CDATA_SECTION_NODE)))
2002-06-21 11:16:28 +00:00
{
textNode->appendData(next->getNodeValue());
removeChild(next);
next = textNode->getNextSibling();
} // while
if(string_adaptorT::empty(textNode->getData()))
2002-06-21 11:16:28 +00:00
removeChild(textNode);
}
else
child->normalize();
child = next;
} // while
DOMNamedNodeMap_implT* attrs = getAttributes();
2002-06-21 11:16:28 +00:00
if(attrs)
for(unsigned int i = 0; i < attrs->getLength(); ++i)
attrs->item(i)->normalize();
} // normalize
virtual bool isSupported(const stringT& /*feature*/, const stringT& /*version*/) const
2002-06-21 11:16:28 +00:00
{
return false;
} // isSupported
virtual const stringT& getNamespaceURI() const { return ownerDoc_->empty_string(); }
virtual const stringT& getPrefix() const { return ownerDoc_->empty_string(); }
virtual void setPrefix(const stringT& /*prefix*/) { }
virtual const stringT& getLocalName() const { return ownerDoc_->empty_string(); }
2002-06-21 11:16:28 +00:00
// additional methods - since C++ std::string (and by implication
// stringT) don't differenciate between a null string and an empty string,
// but the DOM recommendation does, I have to introduce these three methods
// to disambiguate. If they return false, the corresponding attribute should be
// considered null. If they return true, the attribute has been set EVEN IF
// it has been set to the empty string
virtual bool hasNamespaceURI() const { return false; }
virtual bool hasPrefix() const { return false; }
virtual bool hasAttributes() const { return false; }
/////////////////////////////////////////////////////
// implementation specific method
void setParentNode(NodeImplT* parent)
2002-06-21 11:16:28 +00:00
{
if(!parent && parentNode_ && ownerDoc_)
ownerDoc_->orphaned(this);
if(!parentNode_ && parent && ownerDoc_)
ownerDoc_->adopted(this);
parentNode_ = parent;
} // setParentNode
NodeImplT* getFirst() { return dynamic_cast<NodeImplT*>(getFirstChild()); }
2002-06-21 11:16:28 +00:00
NodeImplT* getPrev() { return prevSibling_; }
2002-06-21 11:16:28 +00:00
void setPrev(NodeImplT* prevSibling)
2002-06-21 11:16:28 +00:00
{
prevSibling_ = prevSibling;
} // setPreviousSibling
NodeImplT* getNext() { return nextSibling_; }
2002-06-21 11:16:28 +00:00
void setNext(NodeImplT* nextSibling)
2002-06-21 11:16:28 +00:00
{
nextSibling_ = nextSibling;
} // setNextSibling
DocumentImplT* getOwnerDoc() const { return ownerDoc_; }
2002-06-21 11:16:28 +00:00
virtual void setOwnerDoc(DocumentImplT* ownerDoc)
2002-06-21 11:16:28 +00:00
{
ownerDoc_ = ownerDoc;
for(NodeImplT*child = getFirst(); child != 0; child = child->getNext())
2002-06-21 11:16:28 +00:00
child->setOwnerDoc(ownerDoc);
} // setOwnerDocument
bool getReadOnly() const { return readOnly_; }
virtual void setReadOnly(bool /*readOnly*/)
2002-06-21 11:16:28 +00:00
{
// Should be readOnly_ = readOnly; ?
2002-06-21 11:16:28 +00:00
readOnly_ = true;
for(NodeImplT*child = getFirst(); child != 0; child = child->getNext())
2002-06-21 11:16:28 +00:00
child->setReadOnly(readOnly_);
} // setReadOnly
void throwIfReadOnly() const
{
if(readOnly_)
throw DOM::DOMException(DOM::DOMException::NO_MODIFICATION_ALLOWED_ERR);
} // throwIfReadOnly
protected:
NodeImplT* parentNode_;
DocumentImplT* ownerDoc_;
NodeImplT* prevSibling_;
NodeImplT* nextSibling_;
2002-06-21 11:16:28 +00:00
bool readOnly_;
}; // class NodeImpl
template<class stringT, class string_adaptorT>
class ChildlessNodeImpl : public NodeImpl<stringT, string_adaptorT>
{
public:
typedef DocumentImpl<stringT, string_adaptorT> DocumentImplT;
typedef DOM::Node_impl<stringT, string_adaptorT> DOMNode_implT;
typedef DOM::NodeList_impl<stringT, string_adaptorT> DOMNodeList_implT;
ChildlessNodeImpl(DocumentImplT* ownerDoc) :
2002-06-21 11:16:28 +00:00
NodeImpl<stringT, string_adaptorT>(ownerDoc)
{
} // ChildlessNodeImpl
///////////////////////////////////////////////////////
// Node methods
virtual DOMNodeList_implT* getChildNodes() const { return 0; }
2002-06-21 11:16:28 +00:00
virtual DOMNode_implT* getFirstChild() const { return 0; }
virtual DOMNode_implT* getLastChild() const { return 0; }
2002-06-21 11:16:28 +00:00
virtual DOMNode_implT* insertBefore(DOMNode_implT* /*newChild*/, DOMNode_implT* /*refChild*/)
2002-06-21 11:16:28 +00:00
{
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
} // insertBefore
virtual DOMNode_implT* replaceChild(DOMNode_implT* /*newChild*/, DOMNode_implT* /*oldChild*/)
2002-06-21 11:16:28 +00:00
{
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
} // insertBefore
virtual DOMNode_implT* removeChild(DOMNode_implT* /*oldChild*/)
2002-06-21 11:16:28 +00:00
{
throw DOM::DOMException(DOM::DOMException::NOT_FOUND_ERR);
} // removeChild
virtual DOMNode_implT* appendChild(DOMNode_implT* /*newChild*/)
2002-06-21 11:16:28 +00:00
{
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
} // appendChild
virtual void purgeChild(DOMNode_implT* /*oldChild*/)
2005-11-25 22:39:16 +00:00
{
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
} // purgeChild
2002-06-21 11:16:28 +00:00
virtual bool hasChildNodes() const { return false; }
}; // class ChildlessNodeImpl
template<class stringT, class string_adaptorT>
class NodeImplWithChildren : public NodeImpl<stringT, string_adaptorT>,
public DOM::NodeList_impl<stringT, string_adaptorT>
2002-06-21 11:16:28 +00:00
{
public:
typedef NodeImpl<stringT, string_adaptorT> NodeImplT;
typedef DocumentImpl<stringT, string_adaptorT> DocumentImplT;
typedef DOM::Node_impl<stringT, string_adaptorT> DOMNode_implT;
typedef DOM::NodeList_impl<stringT, string_adaptorT> DOMNodeList_implT;
NodeImplWithChildren(DocumentImplT* ownerDoc) :
NodeImplT(ownerDoc)
2002-06-21 11:16:28 +00:00
{
} // NodeImplWithChildren
virtual ~NodeImplWithChildren()
{
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()
{
NodeImplT::addRef();
2002-06-21 11:16:28 +00:00
} // addRef
virtual void releaseRef()
{
NodeImplT::releaseRef();
2002-06-21 11:16:28 +00:00
} // releaseRef
///////////////////////////////////////////////////////
// Node methods
virtual DOMNodeList_implT* getChildNodes() const
2002-06-21 11:16:28 +00:00
{
return const_cast<DOMNodeList_implT*>(static_cast<const DOMNodeList_implT*>(this));
2002-06-21 11:16:28 +00:00
} // getChildNodes
virtual DOMNode_implT* getFirstChild() const
2002-06-21 11:16:28 +00:00
{
if(nodes_.size())
return nodes_[0];
return 0;
} // getFirstChild
virtual DOMNode_implT* getLastChild() const
2002-06-21 11:16:28 +00:00
{
if(nodes_.size())
return *nodes_.rbegin();
return 0;
} // getLastChild
virtual DOMNode_implT* insertBefore(DOMNode_implT*newChild, DOMNode_implT*refChild)
2002-06-21 11:16:28 +00:00
{
return do_insertBefore(dynamic_cast<NodeImplT*>(newChild),
dynamic_cast<NodeImplT*>(refChild));
2002-06-21 11:16:28 +00:00
} // insertBefore
virtual DOMNode_implT* replaceChild(DOMNode_implT*newChild, DOMNode_implT*oldChild)
2002-06-21 11:16:28 +00:00
{
return do_replaceChild(dynamic_cast<NodeImplT*>(newChild),
dynamic_cast<NodeImplT*>(oldChild));
2002-06-21 11:16:28 +00:00
} // replaceChild
virtual DOMNode_implT* removeChild(DOMNode_implT* oldChild)
2002-06-21 11:16:28 +00:00
{
return do_removeChild(dynamic_cast<NodeImplT*>(oldChild));
2002-06-21 11:16:28 +00:00
} // removeChild
virtual DOMNode_implT* appendChild(DOMNode_implT* newChild)
2002-06-21 11:16:28 +00:00
{
return do_appendChild(dynamic_cast<NodeImplT*>(newChild));
2002-06-21 11:16:28 +00:00
} // appendChild
virtual void purgeChild(DOMNode_implT* oldChild)
2005-11-25 22:39:16 +00:00
{
do_purgeChild(dynamic_cast<NodeImplT*>(oldChild));
2005-11-25 22:39:16 +00:00
} // purgeChild
2002-06-21 11:16:28 +00:00
virtual bool hasChildNodes() const
{
return nodes_.size() != 0;
} // hasChildNodes
///////////////////////////////////////////////////////
// NodeList methods
virtual DOMNode_implT* item(unsigned int index) const
2002-06-21 11:16:28 +00:00
{
if(index >= nodes_.size())
return 0;
return nodes_[index];
} // item
virtual unsigned int getLength() const
{
return static_cast<unsigned int>(nodes_.size());
2002-06-21 11:16:28 +00:00
} // getLength
/////////////////////////////////////////////////////////////
// implementation
protected:
NodeImplT* do_insertBefore(NodeImplT* newChild, NodeImplT* refChild)
2002-06-21 11:16:28 +00:00
{
NodeImplT::throwIfReadOnly();
2002-06-21 11:16:28 +00:00
if(newChild->getNodeType() == DOM::Node_base::DOCUMENT_FRAGMENT_NODE)
2002-06-21 11:16:28 +00:00
{
for(NodeImplT* child = newChild->getFirst(); child != 0; child = newChild->getFirst())
2002-06-21 11:16:28 +00:00
insertBefore(newChild->removeChild(child), refChild);
return newChild;
} // if ...
checkCanAdd(newChild);
removeIfRequired(newChild);
if(refChild)
{
nodes_.insert(findChild(refChild), newChild);
NodeImplT* prev = refChild->getPrev();
2002-06-21 11:16:28 +00:00
if(prev != 0)
prev->setNext(newChild);
newChild->setPrev(prev);
newChild->setNext(refChild);
refChild->setPrev(newChild);
}
else
{
if(!nodes_.empty())
{
(*nodes_.rbegin())->setNext(newChild);
newChild->setPrev(*nodes_.rbegin());
} //
nodes_.push_back(newChild);
}
newChild->setParentNode(this);
markChanged();
return newChild;
} // insertBefore
NodeImplT* do_replaceChild(NodeImplT* newChild, NodeImplT* oldChild)
2002-06-21 11:16:28 +00:00
{
NodeImplT::throwIfReadOnly();
2002-06-21 11:16:28 +00:00
if(newChild->getNodeType() == DOM::Node_base::DOCUMENT_FRAGMENT_NODE)
2002-06-21 11:16:28 +00:00
{
// not exception safe - but the it's not specified to be :(
DOMNode_implT* lc = newChild->getLastChild();
2002-06-21 11:16:28 +00:00
replaceChild(newChild->removeChild(lc), oldChild);
insertBefore(newChild, lc);
return newChild;
} // if ...
checkCanAdd(newChild);
typename std::deque<NodeImplT*>::iterator result = findChild(oldChild);
2002-06-21 11:16:28 +00:00
removeIfRequired(newChild);
*result = newChild;
newChild->setParentNode(this);
NodeImplT* prev = oldChild->getPrev();
NodeImplT* next = oldChild->getNext();
2002-06-21 11:16:28 +00:00
newChild->setPrev(prev);
newChild->setNext(next);
if(prev != 0)
prev->setNext(newChild);
if(next != 0)
next->setPrev(newChild);
oldChild->setParentNode(0);
oldChild->setPrev(0);
oldChild->setNext(0);
markChanged();
return oldChild;
} // replaceChild
NodeImplT* do_removeChild(NodeImplT* oldChild)
2002-06-21 11:16:28 +00:00
{
NodeImplT::throwIfReadOnly();
2002-06-21 11:16:28 +00:00
nodes_.erase(findChild(oldChild));
NodeImplT* prev = oldChild->getPrev();
NodeImplT* next = oldChild->getNext();
2002-06-21 11:16:28 +00:00
if(prev != 0)
prev->setNext(next);
if(next != 0)
next->setPrev(prev);
oldChild->setParentNode(0);
oldChild->setPrev(0);
oldChild->setNext(0);
markChanged();
return oldChild;
} // removeChild
NodeImplT* do_appendChild(NodeImplT* newChild)
2002-06-21 11:16:28 +00:00
{
return do_insertBefore(newChild, 0);
} // appendChild
void do_purgeChild(NodeImplT* oldChild)
2005-11-25 22:39:16 +00:00
{
oldChild = do_removeChild(oldChild);
NodeImplT::ownerDoc_->purge(oldChild);
2005-11-25 22:39:16 +00:00
} // do_purgeChild
2002-06-21 11:16:28 +00:00
private:
typedef std::deque<NodeImplT*> NodeListT;
2002-06-21 11:16:28 +00:00
typename NodeListT::iterator findChild(NodeImplT* refChild)
2002-06-21 11:16:28 +00:00
{
2002-11-23 20:03:54 +00:00
typename NodeListT::iterator result = std::find(nodes_.begin(), nodes_.end(), refChild);
2002-06-21 11:16:28 +00:00
if(result == nodes_.end())
throw DOM::DOMException(DOM::DOMException::NOT_FOUND_ERR);
return result;
} // findChild
void removeIfRequired(NodeImplT* newNode) const
2002-06-21 11:16:28 +00:00
{
if(newNode->getParentNode() != 0)
newNode->getParentNode()->removeChild(newNode);
} // removeIfRequired
void checkCanAdd(NodeImplT* child)
2002-06-21 11:16:28 +00:00
{
DocumentImplT* childDoc = child->getOwnerDoc();
2002-06-21 11:16:28 +00:00
if(childDoc == 0)
{
child->setOwnerDoc(NodeImplT::getOwnerDoc());
2002-06-21 11:16:28 +00:00
return;
} //
if(child->getNodeType() == DOM::Node_base::DOCUMENT_NODE)
2002-06-21 11:16:28 +00:00
{
if(childDoc != dynamic_cast<DocumentImplT*>(this))
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::WRONG_DOCUMENT_ERR);
return;
} // if(parent is a Document)
if(NodeImplT::getOwnerDocument() && childDoc != NodeImplT::getOwnerDocument())
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::WRONG_DOCUMENT_ERR);
checkChildType(child);
} // checkCanAdd
virtual void checkChildType(DOMNode_implT* child) = 0;
2002-06-21 11:16:28 +00:00
void markChanged()
{
if(NodeImplT::ownerDoc_)
NodeImplT::ownerDoc_->markChanged();
2002-06-21 11:16:28 +00:00
} // markChanged
NodeListT nodes_;
}; // class NodeImplWithChildren
} // namespace DOM
2007-09-05 11:47:13 +00:00
} // namespace Arabica
2002-06-21 11:16:28 +00:00
#endif // JEZUK_DOM_NODE_H
// end of file