1
0
Fork 0
mirror of https://github.com/jezhiggins/arabica synced 2025-01-17 18:12:04 +01:00
arabica/include/DOM/Simple/ElementImpl.hpp

213 lines
6.9 KiB
C++

#ifndef JEZUK_SimpleDOM_ELEMENTIMPL_H
#define JEZUK_SimpleDOM_ELEMENTIMPL_H
#include <DOM/Element.hpp>
#include <DOM/Simple/NodeImpl.hpp>
#include <DOM/Simple/ElementByTagImpl.hpp>
#include <DOM/Simple/AttrMap.hpp>
namespace Arabica
{
namespace SimpleDOM
{
template<class stringT, class string_adaptorT>
class ElementImpl : public DOM::Element_impl<stringT, string_adaptorT>,
public NodeImplWithChildren<stringT, string_adaptorT>
{
protected:
typedef NodeImplWithChildren<stringT, string_adaptorT> NodeT;
typedef DocumentImpl<stringT, string_adaptorT> DocumentImplT;
typedef AttrMap<stringT, string_adaptorT> AttrMapT;
typedef ElementByTagList<stringT, string_adaptorT> ElementByTagListT;
typedef DOM::Node_impl<stringT, string_adaptorT> DOMNode_implT;
typedef DOM::Attr_impl<stringT, string_adaptorT> DOMAttr_implT;
typedef DOM::NodeList_impl<stringT, string_adaptorT> DOMNodeList_implT;
typedef DOM::NamedNodeMap_impl<stringT, string_adaptorT> DOMNamedNodeMap_implT;
typedef DOM::Element_impl<stringT, string_adaptorT> DOMElement_implT;
public:
ElementImpl(DocumentImplT* ownerDoc, const stringT& tagName) :
DOMElement_implT(),
NodeT(ownerDoc),
attributes_(ownerDoc),
tagName_(ownerDoc->stringPool(tagName))
{
attributes_.setOwnerElement(this);
} // ElementImpl
virtual ~ElementImpl()
{
} // ~ElementImpl
/////////////////////////////////////////////////////
// DOM::Element functions
virtual const stringT& getTagName() const { return getNodeName(); }
virtual const stringT& getAttribute(const stringT& name) const
{
return attributes_.getAttribute(name);
} // getAttribute
virtual void setAttribute(const stringT& name, const stringT& value)
{
checkName(name);
attributes_.setAttribute(name, value);
} // setAttribute
virtual void removeAttribute(const stringT& name)
{
attributes_.removeAttribute(name);
} // removeAttribute
virtual DOMAttr_implT* getAttributeNode(const stringT& name) const
{
return attributes_.getAttributeNode(name);
} // getAttributeNode
virtual DOMAttr_implT* setAttributeNode(DOMAttr_implT* newAttr)
{
return attributes_.setAttributeNode(newAttr);
} // setAttributeNode
virtual DOMAttr_implT* removeAttributeNode(DOMAttr_implT* oldAttr)
{
return attributes_.removeAttributeNode(oldAttr);
} // removeAttributeNode
virtual DOMNodeList_implT* getElementsByTagName(const stringT& tagName) const
{
return new ElementByTagListT(NodeT::ownerDoc_,
const_cast<ElementImpl*>(this),
tagName);
} // getElementsByTagName
virtual stringT getAttributeNS(const stringT& namespaceURI, const stringT& localName) const
{
return attributes_.getAttributeNS(namespaceURI, localName);
} // getAttributeNS
virtual void setAttributeNS(const stringT& namespaceURI, const stringT& qualifiedName, const stringT& value)
{
checkName(qualifiedName);
attributes_.setAttributeNS(namespaceURI, qualifiedName, value);
} // setAttributeNS
virtual void removeAttributeNS(const stringT& namespaceURI, const stringT& localName)
{
attributes_.removeAttributeNS(namespaceURI, localName);
} // removeAttributeNS
virtual DOMAttr_implT* getAttributeNodeNS(const stringT& namespaceURI, const stringT& localName) const
{
return attributes_.getAttributeNodeNS(namespaceURI, localName);
} // getAttributeNodeNS
virtual DOMAttr_implT* setAttributeNodeNS(DOMAttr_implT* newAttr)
{
return attributes_.setAttributeNodeNS(newAttr);
} // setAttributeNodeNS
virtual DOMNodeList_implT* getElementsByTagNameNS(const stringT& namespaceURI, const stringT& localName) const
{
return new ElementByTagListT(NodeT::ownerDoc_,
const_cast<ElementImpl*>(this),
namespaceURI,
localName);
} // getElementsByTagNameNS
virtual bool hasAttribute(const stringT& name) const
{
return attributes_.hasAttribute(name);
} // hasAttribute
virtual bool hasAttributeNS(const stringT& namespaceURI, const stringT& localName) const
{
return attributes_.hasAttributeNS(namespaceURI, localName);
} // hasAttributeNS
///////////////////////////////////////////////////////
// DOM::Node methods
virtual DOM::Node_base::Type getNodeType() const
{
return DOM::Node_base::ELEMENT_NODE;
} // getNodeType
virtual const stringT& getNodeName() const
{
return *tagName_;
} // getNodeName
virtual DOMNamedNodeMap_implT* getAttributes() const
{
return const_cast<AttrMapT*>(&attributes_);
} // getAttributes
virtual DOMNode_implT* cloneNode(bool deep) const
{
ElementImpl* clone = dynamic_cast<ElementImpl*>(NodeT::ownerDoc_->createElement(*tagName_));
cloneChildren(clone, deep);
return clone;
} // cloneNode
virtual bool hasAttributes() const
{
return (attributes_.getLength() > 0);
} // hasAttributes
virtual void setOwnerDoc(DocumentImplT* ownerDoc)
{
attributes_.setOwnerDoc(ownerDoc);
NodeT::setOwnerDoc(ownerDoc);
} // setOwnerDoc
virtual void setReadOnly(bool readOnly)
{
attributes_.setReadOnly(readOnly);
NodeT::setReadOnly(readOnly);
} // setReadOnly
protected:
void cloneChildren(ElementImpl* clone, bool deep) const
{
for(unsigned int i = 0; i < attributes_.getLength(); ++i)
{
DOMAttr_implT* a = dynamic_cast<DOMAttr_implT*>(attributes_.item(i));
if(a->getSpecified())
{
DOMAttr_implT* newA = dynamic_cast<DOMAttr_implT*>(a->cloneNode(true));
if(string_adaptorT::empty(a->getLocalName()))
clone->setAttributeNode(newA);
else
clone->setAttributeNodeNS(newA);
} // if ...
} // for
if(deep)
for(DOMNode_implT* c = NodeT::getFirstChild(); c != 0; c = c->getNextSibling())
clone->appendChild(c->cloneNode(true));
} // cloneChildren
private:
virtual void checkChildType(DOMNode_implT* child)
{
typename DOM::Node_base::Type type = child->getNodeType();
if((type != DOM::Node_base::ELEMENT_NODE) &&
(type != DOM::Node_base::TEXT_NODE) &&
(type != DOM::Node_base::COMMENT_NODE) &&
(type != DOM::Node_base::PROCESSING_INSTRUCTION_NODE) &&
(type != DOM::Node_base::CDATA_SECTION_NODE) &&
(type != DOM::Node_base::ENTITY_REFERENCE_NODE))
throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR);
} // checkChildType
AttrMapT attributes_;
protected:
stringT const* tagName_;
}; // class ElementImpl
} // namespace SAX2DOM
} // namespace Arabica
#endif