#ifndef JEZUK_SIMPLEDOM_ATTRIMPL_H #define JEZUK_SIMPLEDOM_ATTRIMPL_H #include #include #include #include namespace SimpleDOM { template class ElementImpl; template class AttrImpl : public DOM::Attr_impl, public NodeImplWithChildren { typedef NodeImplWithChildren NodeT; public: AttrImpl(DocumentImpl* ownerDoc, const stringT& name) : DOM::Attr_impl(), NodeImplWithChildren(ownerDoc), name_(ownerDoc->stringPool(name)), ownerElement_(0), specified_(true) { } // AttrImpl AttrImpl(DocumentImpl* ownerDoc, const stringT& name, const stringT& value) : DOM::Attr_impl(), NodeImplWithChildren(ownerDoc), name_(ownerDoc->stringPool(name)), ownerElement_(0), specified_(true) { setNodeValue(value); } // AttrImpl virtual ~AttrImpl() { } /////////////////////////////////////////////////// // DOM::Attribute methods stringT getName() const { return getNodeName(); } virtual bool getSpecified() const { return specified_; } // getSpecified stringT getValue() const { return getNodeValue(); } void setValue(const stringT& value) { setNodeValue(value); } virtual DOM::Element_impl* getOwnerElement() const { return ownerElement_; } // getOwnerElement ///////////////////////////////////////////////////// // DOM::Node methods virtual typename DOM::Node::Type getNodeType() const { return DOM::Node::ATTRIBUTE_NODE; } // getNodeType virtual DOM::Node_impl* getParentNode() const { return 0; } virtual DOM::Node_impl* getPreviousSibling() const { return 0; } virtual DOM::Node_impl* getNextSibling() const { return 0; } virtual DOM::Node_impl* cloneNode(bool deep) const { AttrImpl* a = dynamic_cast(NodeT::ownerDoc_->createAttribute(*name_)); cloneChildren(a); a->setSpecified(getSpecified()); return a; } // cloneNode virtual const stringT& getNodeName() const { return *name_; } // getNodeName virtual stringT getNodeValue() const { stringT value; for(DOM::Node_impl* c = NodeT::getFirstChild(); c != 0; c = c->getNextSibling()) string_adaptorT::append(value, c->getNodeValue()); return value; } // getNodeValue virtual void setNodeValue(const stringT& data) { NodeT::throwIfReadOnly(); // remove all children for(DOM::Node_impl* c = NodeT::getFirstChild(); c != 0; c = NodeT::getFirstChild()) NodeT::removeChild(c); // add a new text node NodeT::appendChild(new TextImpl(0, data)); specified_ = true; } // setNodeValue ///////////////////////////////////////////////////////////// // this implementation void setOwnerElement(ElementImpl* element) { ownerElement_ = element; if(NodeT::ownerDoc_) { NodeT::ownerDoc_->adopted(this); // don't have a parent but are owned DocumentTypeImpl* docType = dynamic_cast*>(NodeT::ownerDoc_->getDoctype()); if(!docType || docType->getElementIds()->empty()) return; std::vector* elemIds = docType->getElementIds(); if(std::find(elemIds->begin(), elemIds->end(), *name_) != elemIds->end()) NodeT::ownerDoc_->setElementId(this); } // if(ownerDoc_) } // setOwnerElement void setSpecified(bool specified) { specified_ = specified; } protected: void cloneChildren(AttrImpl* clone) const { for(DOM::Node_impl* c = NodeT::getFirstChild(); c != 0; c = c->getNextSibling()) clone->appendChild(c->cloneNode(true)); } // cloneChildren private: virtual void checkChildType(DOM::Node_impl* child) { typename DOM::Node::Type type = child->getNodeType(); if((type != DOM::Node::TEXT_NODE) && (type != DOM::Node::ENTITY_REFERENCE_NODE)) throw DOM::DOMException(DOM::DOMException::HIERARCHY_REQUEST_ERR); } // checkChildType protected: stringT const* name_; ElementImpl* ownerElement_; bool specified_; }; // class CDATAImpl } // namespace SimpleDOM #endif