#ifndef ARABICA_XSLT_ITEM_CONTAINER_HPP #define ARABICA_XSLT_ITEM_CONTAINER_HPP #include "xslt_create_handler.hpp" #include "../xslt_text.hpp" #include "xslt_value_validation.hpp" namespace Arabica { namespace XSLT { const ChildElement* AllowedChildren(); SAX::DefaultHandler* createInlineElementHandler(CompilationContext& context); template class ItemContainerHandler : public SAX::DefaultHandler { protected: ItemContainerHandler(CompilationContext& context) : context_(context), container_(0) { } // ItemContainerHandler ItemContainerHandler(CompilationContext& context, container_type* container) : context_(context), container_(container) { } // ItemContainerHandler virtual container_type* createContainer(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) = 0; CompilationContext& context() const { return context_; } container_type* container() const { return container_; } public: virtual void startElement(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) { if(container_ == 0) { container_ = createContainer(namespaceURI, localName, qName, atts); return; } // if(item_ == 0) if(createChild(namespaceURI, localName, qName, atts)) return; throw SAX::SAXException(qName + " <- Sorry, don't know about that yet :)"); } // startElement virtual void endElement(const std::string& /* namespaceURI */, const std::string& /* localName */, const std::string& /* qName */) { context_.parentContainer().add_item(container_); context_.pop(); } // endElement virtual void characters(const std::string& ch) { for(std::string::const_iterator i = ch.begin(), e = ch.end(); i != e; ++i) if(!Arabica::XML::is_space(*i)) { container_->add_item(new Text(ch)); return; } // if ... } // characters protected: virtual bool createChild(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) { if(namespaceURI == StylesheetConstant::NamespaceURI()) { for(const ChildElement* c = AllowedChildren(); c->name != 0; ++c) if(c->name == localName) { context_.push(container_, c->createHandler(context_), namespaceURI, localName, qName, atts); return true; } // if ... return false; } // if(namespaceURI ... context_.push(container_, createInlineElementHandler(context_), namespaceURI, localName, qName, atts); return true; } // createChild private: CompilationContext& context_; container_type* container_; }; // class ItemContainerHandler } // namespace XSLT } // namespace Arabica #include "xslt_apply_imports_handler.hpp" #include "xslt_apply_templates_handler.hpp" #include "xslt_attribute_handler.hpp" #include "xslt_call_template_handler.hpp" #include "xslt_choose_handler.hpp" #include "xslt_comment_handler.hpp" #include "xslt_copy_handler.hpp" #include "xslt_element_handler.hpp" #include "xslt_for_each_handler.hpp" #include "xslt_if_handler.hpp" #include "xslt_message_handler.hpp" #include "xslt_processing_instruction_handler.hpp" #include "xslt_text_handler.hpp" #include "xslt_value_of_handler.hpp" #include "xslt_variable_handler.hpp" #include "xslt_inline_element_handler.hpp" namespace Arabica { namespace XSLT { const ChildElement* AllowedChildren() { static const ChildElement allowedChildren[] = { { "apply-imports", CreateHandler }, { "apply-templates", CreateHandler }, { "attribute", CreateHandler }, { "call-template", CreateHandler }, { "choose", CreateHandler }, { "comment", CreateHandler }, { "copy", CreateHandler }, { "copy-of", CreateHandler }, { "element", CreateHandler }, { "fallback", CreateHandler}, { "for-each", CreateHandler }, { "if", CreateHandler }, { "message", CreateHandler}, { "number", CreateHandler}, { "processing-instruction", CreateHandler }, { "text", CreateHandler }, { "value-of", CreateHandler }, { "variable", CreateHandler > }, { 0, 0 } }; return allowedChildren; } // AllowedChildren SAX::DefaultHandler* createInlineElementHandler(CompilationContext& context) { return new InlineElementHandler(context); } // InlineElementHandler } // namespace XSLT } // namespace Arabica #endif