mirror of
https://github.com/jezhiggins/arabica
synced 2025-01-15 15:40:56 +01:00
138 lines
3.9 KiB
C++
138 lines
3.9 KiB
C++
#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::XPathValue<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
|
|
|