arabica/include/XSLT/impl/xslt_copy.hpp
2007-07-19 17:01:42 +00:00

138 lines
4 KiB
C++
Executable file

#ifndef ARABICA_XSLT_COPY_HPP
#define ARABICA_XSLT_COPY_HPP
#include "xslt_item.hpp"
namespace Arabica
{
namespace XSLT
{
class Copy_base : public ItemContainer
{
protected:
Copy_base()
{
Arabica::XPath::XPath<std::string> compiler;
namespace_select_ = compiler.compile("namespace::node()");
} // Copy_base
virtual ~Copy_base() { }
protected:
void copy(const DOM::Node<std::string>& 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<std::string>& node, ExecutionContext& context) const = 0;
private:
Arabica::XPath::XPathExpressionPtr<std::string> namespace_select_;
}; // class Copy_base
class Copy : public Copy_base
{
public:
Copy(const std::string& attributes_sets) :
attributes_sets_(attributes_sets)
{
} // Copy
virtual ~Copy() { }
virtual void execute(const DOM::Node<std::string>& node, ExecutionContext& context) const
{
copy(node, context);
} // execute
protected:
virtual void process_content(const DOM::Node<std::string>& node, ExecutionContext& context) const
{
execute_children(node, context);
} // process_content
private:
std::string attributes_sets_;
}; // class Copy
class CopyOf : public Copy_base
{
public:
CopyOf(const Arabica::XPath::XPathExpressionPtr<std::string>& select) :
select_(select)
{
} // CopyOf
virtual ~CopyOf() { }
virtual void execute(const DOM::Node<std::string>& node, ExecutionContext& context) const
{
Arabica::XPath::XPathValuePtr<std::string> value = select_->evaluate(node, context.xpathContext());
if(value->type() != Arabica::XPath::NODE_SET)
{
context.sink().characters(value->asString());
return;
}
Arabica::XPath::NodeSet<std::string> nodes = value->asNodeSet();
for(Arabica::XPath::NodeSet<std::string>::const_iterator n = nodes.begin(), e = nodes.end(); n != e; ++n)
copy(*n, context);
} // execute
protected:
virtual void process_content(const DOM::Node<std::string>& node, ExecutionContext& context) const
{
if(node.hasAttributes())
{
const DOM::NamedNodeMap<std::string>& attrs = node.getAttributes();
for(unsigned int a = 0; a < attrs.getLength(); ++a)
copy(attrs.item(a), context);
} // if ...
for(DOM::Node<std::string> n = node.getFirstChild(); n != 0; n = n.getNextSibling())
copy(n, context);
} // process_content
private:
Arabica::XPath::XPathExpressionPtr<std::string> select_;
}; // class CopyOf
} // namespace XSLT
} // namespace Arabica
#endif