arabica/include/DOM/Simple/DocumentImpl.hpp

507 lines
19 KiB
C++
Raw Normal View History

2002-06-21 11:16:28 +00:00
#ifndef JEZUK_SimpleDOM_DOCUMENTIMPL_H
#define JEZUK_SimpleDOM_DOCUMENTIMPL_H
2007-09-04 22:55:47 +00:00
#include <DOM/Document.hpp>
#include <DOM/Simple/ElementNSImpl.hpp>
#include <DOM/Simple/ProcessingInstructionImpl.hpp>
#include <DOM/Simple/DocumentFragmentImpl.hpp>
#include <DOM/Simple/CommentImpl.hpp>
#include <DOM/Simple/TextImpl.hpp>
#include <DOM/Simple/CDATASectionImpl.hpp>
#include <DOM/Simple/AttrNSImpl.hpp>
#include <DOM/Simple/EntityReferenceImpl.hpp>
#include <DOM/Simple/EntityImpl.hpp>
#include <DOM/Simple/NotationImpl.hpp>
#include <DOM/Simple/ElementByTagImpl.hpp>
#include <DOM/Simple/NodeImpl.hpp>
2002-06-21 11:16:28 +00:00
#include <set>
#include <list>
#include <algorithm>
2002-06-21 11:16:28 +00:00
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 valueIs : public std::unary_function<AttrImpl<stringT, string_adaptorT>*, bool>
{
public:
valueIs(const stringT& value) : value_(value) { }
bool operator()(const AttrImpl<stringT, string_adaptorT>* node) const
{
return (node->getNodeValue() == value_);
} // operator()
private:
stringT value_;
}; // class valueIs
template<class stringT, class string_adaptorT>
class DocumentImpl : public DOM::Document_impl<stringT, string_adaptorT>,
2002-06-21 11:16:28 +00:00
public NodeImplWithChildren<stringT, string_adaptorT>
{
typedef NodeImpl<stringT, string_adaptorT> NodeImplT;
typedef AttrImpl<stringT, string_adaptorT> AttrImplT;
typedef ElementImpl<stringT, string_adaptorT> ElementImplT;
typedef NodeImplWithChildren<stringT, string_adaptorT> NodeWithChildrenT;
2002-06-21 11:16:28 +00:00
public:
typedef DOM::Node_impl<stringT, string_adaptorT> DOMNode_implT;
typedef DOM::Attr_impl<stringT, string_adaptorT> DOMAttr_implT;
typedef DOM::Element_impl<stringT, string_adaptorT> DOMElement_implT;
typedef DOM::Document_impl<stringT, string_adaptorT> DOMDocument_implT;
typedef DOM::DocumentType_impl<stringT, string_adaptorT> DOMDocumentType_implT;
typedef DOM::DOMImplementation<stringT, string_adaptorT> DOMDOMImplementationT;
2002-06-21 11:16:28 +00:00
DocumentImpl() :
NodeWithChildrenT(0),
2002-06-21 11:16:28 +00:00
documentElement_(0),
documentType_(0),
domImplementation_(),
namespaceURI_(),
qualifiedName_(),
changesCount_(0),
refCount_(0),
empty_()
2002-06-21 11:16:28 +00:00
{
2004-10-12 20:49:30 +00:00
NodeImplT::setOwnerDoc(this);
2002-06-21 11:16:28 +00:00
} // DocumentBaseImpl
DocumentImpl(DOMDOMImplementationT domImpl) :
NodeWithChildrenT(0),
2002-06-21 11:16:28 +00:00
documentElement_(0),
documentType_(0),
domImplementation_(domImpl),
namespaceURI_(),
qualifiedName_(),
changesCount_(0),
refCount_(0)
{
2004-10-12 20:49:30 +00:00
NodeImplT::setOwnerDoc(this);
2002-06-21 11:16:28 +00:00
} // DocumentBaseImpl
DocumentImpl(const stringT& namespaceURI,
const stringT& qualifiedName,
DOMDocumentType_implT* docType,
DOMDOMImplementationT domImpl) :
NodeWithChildrenT(0),
2002-06-21 11:16:28 +00:00
documentElement_(0),
documentType_(0),
domImplementation_(domImpl),
namespaceURI_(namespaceURI),
qualifiedName_(qualifiedName),
changesCount_(0),
refCount_(0)
{
2004-10-12 20:49:30 +00:00
NodeImplT::setOwnerDoc(this);
2002-06-21 11:16:28 +00:00
if(docType)
{
if(docType->getOwnerDocument() != 0)
throw DOM::DOMException(DOM::DOMException::WRONG_DOCUMENT_ERR);
appendChild(docType);
} // if(docType)
} // DocumentBaseImpl
virtual ~DocumentImpl()
{
2002-11-23 20:03:54 +00:00
for(typename std::set<NodeImplT*>::iterator n = orphans_.begin(); n != orphans_.end(); ++n)
2002-06-21 11:16:28 +00:00
delete *n;
} // ~DocumentImpl
/////////////////////////////////////////////////////////////////////
// Ref counting
virtual void addRef()
{
++refCount_;
} // addRef
virtual void releaseRef()
{
if(--refCount_ == 0)
delete this;
} // releaseRef
/////////////////////////////////////////////////////////////////////
// DOM::Document functions
virtual DOMDocumentType_implT* getDoctype() const
2002-06-21 11:16:28 +00:00
{
return documentType_;
} // getDocType
virtual DOMDOMImplementationT getImplementation() const
2002-06-21 11:16:28 +00:00
{
return domImplementation_;
} // getImplementation
virtual DOMElement_implT* getDocumentElement() const
2002-06-21 11:16:28 +00:00
{
return documentElement_;
} // getDocumentElement
virtual DOMElement_implT* createElement(const stringT& tagName) const
2002-06-21 11:16:28 +00:00
{
ElementImplT* n =
new ElementImplT(const_cast<DocumentImpl*>(this), tagName);
2002-06-21 11:16:28 +00:00
orphaned(n);
return n;
} // createElement
virtual DOM::DocumentFragment_impl<stringT, string_adaptorT>* createDocumentFragment() const
2002-06-21 11:16:28 +00:00
{
DocumentFragmentImpl<stringT, string_adaptorT>* n = new DocumentFragmentImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this));
orphaned(n);
return n;
} // createDocumentFragment
virtual DOM::Text_impl<stringT, string_adaptorT>* createTextNode(const stringT& data) const
2002-06-21 11:16:28 +00:00
{
TextImpl<stringT, string_adaptorT>* n = new TextImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), data);
orphaned(n);
return n;
} // createTextNode
virtual DOM::Comment_impl<stringT, string_adaptorT>* createComment(const stringT& data) const
2002-06-21 11:16:28 +00:00
{
CommentImpl<stringT, string_adaptorT>* n = new CommentImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), data);
orphaned(n);
return n;
} // createComment
virtual DOM::CDATASection_impl<stringT, string_adaptorT>* createCDATASection(const stringT& data) const
2002-06-21 11:16:28 +00:00
{
CDATASectionImpl<stringT, string_adaptorT>* n = new CDATASectionImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), data);
orphaned(n);
return n;
} // createCDATASection
virtual DOM::ProcessingInstruction_impl<stringT, string_adaptorT>* createProcessingInstruction(const stringT& target, const stringT& data) const
2002-06-21 11:16:28 +00:00
{
ProcessingInstructionImpl<stringT, string_adaptorT>* n = new ProcessingInstructionImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), target, data);
orphaned(n);
return n;
} // createProcessingInstruction
virtual DOMAttr_implT* createAttribute(const stringT& name) const
2002-06-21 11:16:28 +00:00
{
AttrImpl<stringT, string_adaptorT>* n = new AttrImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), name);
orphaned(n);
return n;
} // createAttribute
virtual DOM::EntityReference_impl<stringT, string_adaptorT>* createEntityReference(const stringT& name) const
2002-06-21 11:16:28 +00:00
{
EntityReferenceImpl<stringT, string_adaptorT>* n = new EntityReferenceImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), name);
if((documentType_ != 0) && (documentType_->getEntities()->getNamedItem(name) != 0))
{
DOMNode_implT* entity = documentType_->getEntities()->getNamedItem(name);
for(DOMNode_implT* child = entity->getFirstChild(); child != 0; child = child->getNextSibling())
2002-06-21 11:16:28 +00:00
n->appendChild(importNode(child, true));
} // if ...
orphaned(n);
n->setReadOnly(true);
return n;
} // createEntityReference
virtual DOM::NodeList_impl<stringT, string_adaptorT>* getElementsByTagName(const stringT& tagname) const
2002-06-21 11:16:28 +00:00
{
return new ElementByTagList<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this),
const_cast<DocumentImpl*>(this),
tagname);
} // getElementsByTagName
virtual DOMNode_implT* importNode(DOMNode_implT* importedNode, bool deep) const
2002-06-21 11:16:28 +00:00
{
DOMNode_implT* newNode = 0;
2002-06-21 11:16:28 +00:00
switch(importedNode->getNodeType())
{
case DOM::Node_base::ATTRIBUTE_NODE:
if(string_adaptorT::empty(importedNode->getLocalName()))
2002-06-21 11:16:28 +00:00
newNode = createAttribute(importedNode->getNodeName());
else
newNode = createAttributeNS(importedNode->getNamespaceURI(), importedNode->getNodeName());
deep = true;
break;
case DOM::Node_base::DOCUMENT_FRAGMENT_NODE:
2002-06-21 11:16:28 +00:00
newNode = createDocumentFragment();
break;
case DOM::Node_base::DOCUMENT_NODE:
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::NOT_SUPPORTED_ERR);
case DOM::Node_base::DOCUMENT_TYPE_NODE:
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::NOT_SUPPORTED_ERR);
case DOM::Node_base::ELEMENT_NODE:
2002-06-21 11:16:28 +00:00
{
DOMElement_implT* elem;
if(string_adaptorT::empty(importedNode->getLocalName()))
2002-06-21 11:16:28 +00:00
elem = createElement(importedNode->getNodeName());
else
elem = createElementNS(importedNode->getNamespaceURI(), importedNode->getNodeName());
const DOM::NamedNodeMap_impl<stringT, string_adaptorT>* attrs = importedNode->getAttributes();
2002-06-21 11:16:28 +00:00
if(attrs)
for(unsigned int i = 0; i < attrs->getLength(); ++i)
{
DOMAttr_implT* a = dynamic_cast<DOMAttr_implT*>(attrs->item(i));
2002-06-21 11:16:28 +00:00
if(a->getSpecified())
{
DOMAttr_implT* newA = dynamic_cast<DOMAttr_implT*>(importNode(a, true));
if(string_adaptorT::empty(a->getLocalName()))
2002-06-21 11:16:28 +00:00
elem->setAttributeNode(newA);
else
elem->setAttributeNodeNS(newA);
} // if ...
} // for
newNode = elem;
}
break;
case DOM::Node_base::ENTITY_NODE:
2002-06-21 11:16:28 +00:00
{
DOM::Entity_impl<stringT, string_adaptorT>* entity = dynamic_cast<DOM::Entity_impl<stringT, string_adaptorT>*>(importedNode);
2002-06-21 11:16:28 +00:00
newNode = new EntityImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this),
entity->getNodeName(),
entity->getPublicId(),
entity->getSystemId(),
entity->getNotationName());
}
break;
case DOM::Node_base::ENTITY_REFERENCE_NODE:
2002-06-21 11:16:28 +00:00
newNode = createEntityReference(importedNode->getNodeName());
deep = false;
break;
case DOM::Node_base::NOTATION_NODE:
2002-06-21 11:16:28 +00:00
{
DOM::Notation_impl<stringT, string_adaptorT>* entity = dynamic_cast<DOM::Notation_impl<stringT, string_adaptorT>*>(importedNode);
2002-06-21 11:16:28 +00:00
newNode = new NotationImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this),
entity->getNodeName(),
entity->getPublicId(),
entity->getSystemId());
}
break;
case DOM::Node_base::PROCESSING_INSTRUCTION_NODE:
2002-06-21 11:16:28 +00:00
newNode = createProcessingInstruction(importedNode->getNodeName(), importedNode->getNodeValue());
break;
case DOM::Node_base::TEXT_NODE:
2002-06-21 11:16:28 +00:00
newNode = createTextNode(importedNode->getNodeValue());
break;
case DOM::Node_base::CDATA_SECTION_NODE:
2002-06-21 11:16:28 +00:00
newNode = createCDATASection(importedNode->getNodeValue());
break;
case DOM::Node_base::COMMENT_NODE:
2002-06-21 11:16:28 +00:00
newNode = createComment(importedNode->getNodeValue());
break;
2005-08-05 21:09:00 +00:00
default:
throw std::runtime_error("Bad node type value in importNode");
2002-06-21 11:16:28 +00:00
} // switch
if(deep)
{
for(DOMNode_implT* child = importedNode->getFirstChild(); child != 0; child = child->getNextSibling())
2002-06-21 11:16:28 +00:00
newNode->appendChild(importNode(child, true));
} // if(deep)
if(newNode->getNodeType() == DOM::Node_base::ENTITY_NODE)
2002-06-21 11:16:28 +00:00
dynamic_cast<NodeImplT*>(newNode)->setReadOnly(true);
orphaned(dynamic_cast<NodeImplT*>(newNode));
return newNode;
} // importNode
virtual DOMElement_implT* createElementNS(const stringT& namespaceURI, const stringT& qualifiedName) const
2002-06-21 11:16:28 +00:00
{
ElementNSImpl<stringT, string_adaptorT>* n =
new ElementNSImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), namespaceURI, !string_adaptorT::empty(namespaceURI), qualifiedName);
2002-06-21 11:16:28 +00:00
orphaned(n);
return n;
} // createElementNS
virtual DOMAttr_implT* createAttributeNS(const stringT& namespaceURI, const stringT& qualifiedName) const
2002-06-21 11:16:28 +00:00
{
AttrNSImpl<stringT, string_adaptorT>* n = new AttrNSImpl<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this), namespaceURI, !string_adaptorT::empty(namespaceURI), qualifiedName);
2002-06-21 11:16:28 +00:00
orphaned(n);
return n;
} // createAttrNS
virtual DOM::NodeList_impl<stringT, string_adaptorT>* getElementsByTagNameNS(const stringT& namespaceURI, const stringT& localName) const
2002-06-21 11:16:28 +00:00
{
return new ElementByTagList<stringT, string_adaptorT>(const_cast<DocumentImpl*>(this),
const_cast<DocumentImpl*>(this),
namespaceURI, localName);
} // getElementsByTagNameNS
virtual DOMElement_implT* getElementById(const stringT& elementId) const
2002-06-21 11:16:28 +00:00
{
2002-11-23 20:03:54 +00:00
typename std::set<AttrImplT*>::const_iterator i = std::find_if(idNodes_.begin(), idNodes_.end(), valueIs<stringT, string_adaptorT>(elementId));
2002-06-21 11:16:28 +00:00
if(i == idNodes_.end())
return 0;
return (*i)->getOwnerElement();
} // getElementById
////////////////////////////////////////////////////////
// DOM Node methods
virtual typename DOM::Node_base::Type getNodeType() const
2002-06-21 11:16:28 +00:00
{
return DOM::Node_base::DOCUMENT_NODE;
2002-06-21 11:16:28 +00:00
} // getNodeType
virtual DOMNode_implT* getParentNode() const { return 0; }
2002-06-21 11:16:28 +00:00
virtual DOMDocument_implT* getOwnerDocument() const { return 0; }
2002-06-21 11:16:28 +00:00
2005-12-09 14:09:32 +00:00
virtual const stringT& getNodeName() const
2002-06-21 11:16:28 +00:00
{
2005-12-09 14:09:32 +00:00
static const stringT doc = string_adaptorT::construct_from_utf8("#document");
return doc;
2002-06-21 11:16:28 +00:00
} // getNodeName
virtual DOMNode_implT* insertBefore(DOMNode_implT* newChild, DOMNode_implT* refChild)
2002-06-21 11:16:28 +00:00
{
if((newChild->getNodeType() == DOM::Node_base::ELEMENT_NODE) && (documentElement_ != 0))
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
if((newChild->getNodeType() == DOM::Node_base::DOCUMENT_TYPE_NODE) && (documentType_ != 0))
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
DOMNode_implT* result = NodeWithChildrenT::insertBefore(newChild, refChild);
2002-06-21 11:16:28 +00:00
if((newChild->getNodeType() == DOM::Node_base::ELEMENT_NODE) && (documentElement_ == 0))
documentElement_ = dynamic_cast<DOMElement_implT*>(newChild);
if((newChild->getNodeType() == DOM::Node_base::DOCUMENT_TYPE_NODE) && (documentType_ == 0))
documentType_ = dynamic_cast<DOMDocumentType_implT*>(newChild);
2002-06-21 11:16:28 +00:00
return result;
} // insertBefore
virtual DOMNode_implT* replaceChild(DOMNode_implT* newChild, DOMNode_implT* oldChild)
2002-06-21 11:16:28 +00:00
{
checkChildType(newChild);
if((newChild->getNodeType() == DOM::Node_base::DOCUMENT_TYPE_NODE) && (documentType_ == oldChild))
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
if((newChild->getNodeType() == DOM::Node_base::ELEMENT_NODE) &&
2002-06-21 11:16:28 +00:00
(documentElement_ != 0) &&
(documentElement_ != oldChild))
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
DOMNode_implT* result = NodeWithChildrenT::replaceChild(newChild, oldChild);
2002-06-21 11:16:28 +00:00
if((newChild->getNodeType() == DOM::Node_base::ELEMENT_NODE) &&
2002-06-21 11:16:28 +00:00
((documentElement_ == 0) || (documentElement_ == oldChild)))
documentElement_ = dynamic_cast<DOMElement_implT*>(newChild);
2002-06-21 11:16:28 +00:00
return result;
} // replaceChild
virtual DOMNode_implT* removeChild(DOMNode_implT* oldChild)
2002-06-21 11:16:28 +00:00
{
if((documentType_ == oldChild))
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
DOMNode_implT* result = NodeWithChildrenT::removeChild(oldChild);
2002-06-21 11:16:28 +00:00
if(documentElement_ == oldChild)
documentElement_ = static_cast<DOMElement_implT*>(0);
2002-06-21 11:16:28 +00:00
return result;
} // removeChild
virtual DOMNode_implT* appendChild(DOMNode_implT* newChild)
2002-06-21 11:16:28 +00:00
{
return insertBefore(newChild, 0);
} // appendChild
virtual DOMNode_implT* cloneNode(bool deep) const
2002-06-21 11:16:28 +00:00
{
DocumentImpl* clone = new DocumentImpl(namespaceURI_, qualifiedName_, 0, domImplementation_);
if(documentType_ != 0)
{
DocumentTypeImpl<stringT, string_adaptorT>* dt = dynamic_cast<DocumentTypeImpl<stringT, string_adaptorT>*>(documentType_->cloneNode(true));
dt->setOwnerDoc(clone);
clone->appendChild(dt);
}
if(deep)
for(DOMNode_implT* child = NodeWithChildrenT::getFirstChild(); child != 0; child = child->getNextSibling())
2002-06-21 11:16:28 +00:00
if((documentType_ != child) && (child != clone->getDocumentElement()))
clone->appendChild(clone->importNode(child, true));
return clone;
} // cloneNode
////////////////////////////
// extensions
void markChanged() { ++changesCount_; }
unsigned long changes() const { return changesCount_; }
void orphaned(NodeImplT* node) const
{
orphans_.insert(node);
} // orphaned
2005-11-09 21:13:22 +00:00
2005-11-25 22:39:16 +00:00
void purge(NodeImplT* node)
{
orphans_.erase(node);
delete node;
} // purge
2002-06-21 11:16:28 +00:00
void adopted(NodeImplT* node)
{
2002-11-23 20:03:54 +00:00
typename std::set<NodeImplT*>::iterator n = orphans_.find(node);
2002-06-21 11:16:28 +00:00
if(n != orphans_.end())
orphans_.erase(n);
} // adopted
void setElementId(AttrImplT* attr)
{
idNodes_.insert(attr);
} // setElementId
2002-06-21 11:16:28 +00:00
void removeElementId(AttrImplT* attr)
{
typename std::set<AttrImplT*>::iterator n = idNodes_.find(attr);
2002-06-21 11:16:28 +00:00
if(n != idNodes_.end())
idNodes_.erase(n);
} // removeElementId
2010-01-10 18:47:09 +00:00
stringT const* stringPool(const stringT& str) const
{
typename std::list<stringT>::const_iterator i = std::find(stringPool_.begin(), stringPool_.end(), str);
if(i != stringPool_.end())
return &(*i);
stringPool_.push_back(str);
return &(stringPool_.back());
} // stringPool
const stringT& empty_string() const { return empty_; }
2002-06-21 11:16:28 +00:00
private:
void checkChildType(DOMNode_implT* child)
2002-06-21 11:16:28 +00:00
{
typename DOM::Node_base::Type type = child->getNodeType();
if((type != DOM::Node_base::ELEMENT_NODE) &&
(type != DOM::Node_base::PROCESSING_INSTRUCTION_NODE) &&
(type != DOM::Node_base::COMMENT_NODE) &&
(type != DOM::Node_base::DOCUMENT_TYPE_NODE))
2002-06-21 11:16:28 +00:00
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
} // checkChildType
private:
DOMElement_implT* documentElement_;
DOMDocumentType_implT* documentType_;
DOMDOMImplementationT domImplementation_;
2002-06-21 11:16:28 +00:00
stringT namespaceURI_;
stringT qualifiedName_;
unsigned long changesCount_;
unsigned long refCount_;
mutable std::set<NodeImplT*> orphans_;
std::set<AttrImplT*> idNodes_;
mutable std::list<stringT> stringPool_;
const stringT empty_;
2002-06-21 11:16:28 +00:00
}; // class DocumentImpl
} // namespace SAX2DOM
2007-09-05 11:47:13 +00:00
} // namespace Arabica
2002-06-21 11:16:28 +00:00
#endif
// end of file