#ifndef ARABICA_XSLT_COPY_HPP #define ARABICA_XSLT_COPY_HPP #include "xslt_item.hpp" namespace Arabica { namespace XSLT { template class Copy_base : public ItemContainer { typedef StylesheetConstant SC; protected: Copy_base() { Arabica::XPath::XPath compiler; namespace_select_ = compiler.compile(SC::namespace_xpath); } // Copy_base virtual ~Copy_base() { } protected: void copy(const DOM::Node& node, ExecutionContext& context) const { switch(node.getNodeType()) { case DOM::Node_base::ATTRIBUTE_NODE: context.sink().add_attribute(node.getNamespaceURI(), node.getLocalName(), node.getNodeName(), node.getNodeValue()); break; case DOM::Node_base::COMMENT_NODE: context.sink().start_comment(); context.sink().characters(node.getNodeValue()); context.sink().end_comment(); break; case DOM::Node_base::DOCUMENT_NODE: case DOM::Node_base::DOCUMENT_FRAGMENT_NODE: process_content(node, context); break; case DOM::Node_base::ELEMENT_NODE: if(context.sink().start_element(node.getPrefix(), node.getLocalName(), node.getNamespaceURI())) { process_content(node, context); context.sink().end_element(node.getPrefix(), node.getLocalName(), node.getNamespaceURI()); } break; case DOM::Node_base::PROCESSING_INSTRUCTION_NODE: context.sink().start_processing_instruction(node.getNodeName()); context.sink().characters(node.getNodeValue()); context.sink().end_processing_instruction(); break; case DOM::Node_base::TEXT_NODE: case DOM::Node_base::CDATA_SECTION_NODE: context.sink().characters(node.getNodeValue()); break; } // switch } // execute virtual void process_content(const DOM::Node& node, ExecutionContext& context) const = 0; private: Arabica::XPath::XPathExpressionPtr namespace_select_; }; // class Copy_base template class Copy : public Copy_base { public: typedef stringT string_type; typedef adaptorT string_adaptor; Copy(const string_type& attributes_sets) : attributes_sets_(attributes_sets) { } // Copy virtual ~Copy() { } virtual void execute(const DOM::Node& node, ExecutionContext& context) const { this->copy(node, context); } // execute protected: virtual void process_content(const DOM::Node& node, ExecutionContext& context) const { this->execute_children(node, context); } // process_content private: string_type attributes_sets_; }; // class Copy template class CopyOf : public Copy_base { public: typedef stringT string_type; typedef adaptorT string_adaptor; CopyOf(const Arabica::XPath::XPathExpressionPtr& select) : select_(select) { } // CopyOf virtual ~CopyOf() { } virtual void execute(const DOM::Node& node, ExecutionContext& context) const { Arabica::XPath::XPathValue value = select_->evaluate(node, context.xpathContext()); if(value.type() != Arabica::XPath::NODE_SET) { context.sink().characters(value.asString()); return; } Arabica::XPath::NodeSet nodes = value.asNodeSet(); for(typename Arabica::XPath::NodeSet::const_iterator n = nodes.begin(), e = nodes.end(); n != e; ++n) this->copy(*n, context); } // execute protected: virtual void process_content(const DOM::Node& node, ExecutionContext& context) const { if(node.hasAttributes()) { const DOM::NamedNodeMap& attrs = node.getAttributes(); for(unsigned int a = 0; a < attrs.getLength(); ++a) this->copy(attrs.item(a), context); } // if ... for(DOM::Node n = node.getFirstChild(); n != 0; n = n.getNextSibling()) this->copy(n, context); } // process_content private: Arabica::XPath::XPathExpressionPtr select_; }; // class CopyOf } // namespace XSLT } // namespace Arabica #endif