#ifndef ARABICA_XSLT_PROCESSING_INSTRUCTION_HPP #define ARABICA_XSLT_PROCESSING_INSTRUCTION_HPP #include #include #include "xslt_item.hpp" namespace Arabica { namespace XSLT { template class ProcessingInstruction : public ItemContainer { public: typedef stringT string_type; typedef adaptorT string_adaptor; ProcessingInstruction(const Arabica::XPath::XPathExpressionPtr& name) : name_(name) { } // ProcessingInstruction virtual ~ProcessingInstruction() { } virtual void execute(const DOM::Node& node, ExecutionContext& context) const { string_type name = name_->evaluateAsString(node, context.xpathContext()); validate_name(name); context.sink().start_processing_instruction(name); execute_children(node, context); context.sink().end_processing_instruction(); } // execute private: void validate_name(const string_type& name) const { if(name.empty()) throw SAX::SAXException("xsl:processing-instruction : name attribute must evaluate to a valid name"); if(!Arabica::XML::is_ncname(name)) throw SAX::SAXException("xsl:processing-instruction : '" + string_adaptor::asStdString(name) + "' is not valid as the name"); if(name.length() != 3) return; typedef Arabica::text::Unicode UnicodeT; if((name[0] != UnicodeT::CAPITAL_X) && (name[0] != UnicodeT::LOWERCASE_X)) return; if((name[1] != UnicodeT::CAPITAL_M) && (name[1] != UnicodeT::LOWERCASE_M)) return; if((name[2] != UnicodeT::CAPITAL_L) && (name[2] != UnicodeT::LOWERCASE_L)) return; throw SAX::SAXException("xsl:processing-instruction name can not be 'xml'"); } // validate_name Arabica::XPath::XPathExpressionPtr name_; }; // class ProcessingInstruction } // namespace XSLT } // namespace Arabica #endif