#ifndef SAXWriter_H #define SAXWriter_H #include #include #include #include #include namespace SAX { template class basic_Writer : public basic_XMLFilterImpl, private basic_LexicalHandler { public: typedef string_type stringT; typedef basic_Writer WriterT; typedef typename string_type::value_type charT; typedef typename string_type::traits_type traitsT; typedef std::basic_ostream ostreamT; typedef basic_XMLReader XMLReaderT; typedef basic_XMLFilterImpl XMLFilterT; typedef Unicode UnicodeT; private: typedef basic_LexicalHandler LexicalHandlerT; public: basic_Writer(ostreamT& stream) : lexicalHandler_(0), indent_(2), stream_(&stream), inCDATA_(false) { } // basic_Writer basic_Writer(ostreamT& stream, XMLReaderT& parent) : XMLFilterT(parent), lexicalHandler_(0), indent_(2), stream_(&stream), inCDATA_(false) { } // basic_Writer virtual void startDocument(); virtual void endDocument(); virtual void startElement(const stringT& namespaceURI, const stringT& localName, const stringT& qName, const typename XMLFilterT::AttributesT& atts); virtual void endElement(const stringT& namespaceURI, const stringT& localName, const stringT& qName); virtual void characters(const stringT& ch); virtual void processingInstruction(const stringT& target, const stringT& data); virtual void skippedEntity(const stringT& name); virtual void parse(InputSourceT& input); protected: virtual std::auto_ptr doGetProperty(const stringT& name); virtual void doSetProperty(const stringT& name, std::auto_ptr value); private: virtual void startDTD(const stringT& name, const stringT& publicId, const stringT& systemId); virtual void endDTD(); virtual void startEntity(const stringT& name); virtual void endEntity(const stringT& name); virtual void startCDATA(); virtual void endCDATA(); virtual void comment(const stringT& text); void doIndent(); bool isDtd(const stringT& name); bool inCDATA_; int indent_; int depth_; ostreamT* stream_; LexicalHandlerT* lexicalHandler_; const SAX::PropertyNames properties_; class escaper { private: typedef typename WriterT::ostreamT ostreamT; typedef typename WriterT::charT charT; typedef Unicode UnicodeT; public: escaper(ostreamT* stream) : stream_(stream) { } void operator()(charT ch) { switch(ch) { case UnicodeT::LESS_THAN_SIGN: *stream_ << UnicodeT::AMPERSAND << static_cast('l') << static_cast('t') << UnicodeT::SEMI_COLON; break; case UnicodeT::GREATER_THAN_SIGN: *stream_ << UnicodeT::AMPERSAND << static_cast('g') << static_cast('t') << UnicodeT::SEMI_COLON; break; case UnicodeT::AMPERSAND: *stream_ << UnicodeT::AMPERSAND << static_cast('a') << static_cast('m') << static_cast('p') << UnicodeT::SEMI_COLON; break; case UnicodeT::QUOTATION_MARK: *stream_ << UnicodeT::AMPERSAND << static_cast('q') << static_cast('u') << static_cast('o') << static_cast('t') << UnicodeT::SEMI_COLON; break; default: *stream_ << ch; } // switch } // operator() private: ostreamT* stream_; }; // escaper }; // class basic_Writer template void basic_Writer::startDocument() { *stream_ << "" << std::endl; depth_ = 0; inCDATA_ = false; XMLFilterT::startDocument(); } // startDocument template void basic_Writer::endDocument() { XMLFilterT::endDocument(); } // endDocument template void basic_Writer::startElement( const stringT& namespaceURI, const stringT& localName, const stringT& qName, const typename XMLFilterT::AttributesT& atts) { doIndent(); *stream_ << UnicodeT::LESS_THAN_SIGN << qName; for(int i = 0; i < atts.getLength(); ++i) { *stream_ << UnicodeT::SPACE << atts.getQName(i) << UnicodeT::EQUALS_SIGN << UnicodeT::QUOTATION_MARK; stringT value = atts.getValue(i); std::for_each(value.begin(), value.end(), escaper(stream_)); *stream_ << UnicodeT::QUOTATION_MARK; } *stream_ << UnicodeT::GREATER_THAN_SIGN; ++depth_; XMLFilterT::startElement(namespaceURI, localName, qName, atts); } // startElement template void basic_Writer::endElement( const stringT& namespaceURI, const stringT& localName, const stringT& qName) { --depth_; doIndent(); *stream_ << UnicodeT::LESS_THAN_SIGN << UnicodeT::SLASH << qName << UnicodeT::GREATER_THAN_SIGN << UnicodeT::LINE_FEED; XMLFilterT::endElement(namespaceURI, localName, qName); } // endElement template void basic_Writer::characters(const stringT& ch) { if(!inCDATA_) std::for_each(ch.begin(), ch.end(), escaper(stream_)); else *stream_ << ch; XMLFilterT::characters(ch); } // characters template void basic_Writer::processingInstruction(const stringT& target, const stringT& data) { std::cout << UnicodeT::LESS_THAN_SIGN << UnicodeT::QUESTION_MARK << target << UnicodeT::SPACE << data << UnicodeT::QUESTION_MARK << UnicodeT::GREATER_THAN_SIGN; } // processingInstruction template void basic_Writer::skippedEntity(const stringT& name) { if(!isDtd(name)) std::cout << UnicodeT::AMPERSAND << name << UnicodeT::SEMI_COLON; } // skippedEntity template void basic_Writer::parse(InputSourceT& input) { XMLReaderT* parent = getParent(); if(parent) parent->setProperty(properties_.lexicalHandler, static_cast(*this)); XMLFilterT::parse(input); } // parse template void basic_Writer::doIndent() { for(int i = 0; i < depth_; ++i) *stream_ << UnicodeT::SPACE << UnicodeT::SPACE; } // doIndent template bool basic_Writer::isDtd(const string_type& name) { return (name.length() == 5 && name[0] == UnicodeT::LEFT_SQUARE_BRACKET && name[1] == static_cast('d') && name[2] == static_cast('t') && name[3] == static_cast('d') && name[4] == UnicodeT::RIGHT_SQUARE_BRACKET); } // isDtd template std::auto_ptr::XMLReaderT::PropertyBase> basic_Writer::doGetProperty(const string_type& name) { if(name == properties_.lexicalHandler) { XMLReaderT::Property* prop = new XMLReaderT::Property(lexicalHandler_); return std::auto_ptr(prop); } return XMLFilterT::doGetProperty(name); } // doGetProperty template void basic_Writer::doSetProperty(const string_type& name, typename std::auto_ptr::XMLReaderT::PropertyBase> value) { if(name == properties_.lexicalHandler) { XMLReaderT::Property* prop = dynamic_cast*>(value.get()); if(!prop) throw std::bad_cast(); lexicalHandler_ = &(prop->get()); } return XMLFilterT::doSetProperty(name, value); } // doSetProperty template void basic_Writer::startDTD(const stringT& name, const stringT& publicId, const stringT& systemId) { if(lexicalHandler_) lexicalHandler_->startDTD(name, publicId, systemId); } // startDTD template void basic_Writer::endDTD() { if(lexicalHandler_) lexicalHandler_->endDTD(); } // endDTD template void basic_Writer::startEntity(const stringT& name) { if(lexicalHandler_) lexicalHandler_->startEntity(name); } // startEntity template void basic_Writer::endEntity(const stringT& name) { if(lexicalHandler_) lexicalHandler_->endEntity(name); } // endEntity template void basic_Writer::startCDATA() { inCDATA_ = true; std::cout << UnicodeT::LESS_THAN_SIGN << UnicodeT::EXCLAMATION_MARK << UnicodeT::LEFT_SQUARE_BRACKET << static_cast('C') << static_cast('D') << static_cast('A') << static_cast('T') << static_cast('A') << UnicodeT::LEFT_SQUARE_BRACKET; if(lexicalHandler_) lexicalHandler_->startCDATA(); } // startCDATA template void basic_Writer::endCDATA() { std::cout << UnicodeT::RIGHT_SQUARE_BRACKET << UnicodeT::RIGHT_SQUARE_BRACKET << UnicodeT::GREATER_THAN_SIGN; inCDATA_ = false; if(lexicalHandler_) lexicalHandler_->endCDATA(); } // endCDATA template void basic_Writer::comment(const stringT& text) { if(lexicalHandler_) lexicalHandler_->comment(text); } // comment typedef basic_Writer Writer; typedef basic_Writer wWriter; } // namespace SAX #endif