arabica/SAX/filter/Writer.h

718 lines
21 KiB
C
Raw Normal View History

2003-09-11 12:26:53 +02:00
#ifndef ARABICA_SAX_WRITER_H
#define ARABICA_SAX_WRITER_H
2002-06-21 13:16:28 +02:00
2003-09-09 13:14:48 +02:00
#include <SAX/ArabicaConfig.h>
2002-06-21 13:16:28 +02:00
#include <SAX/helpers/XMLFilterImpl.h>
#include <SAX/ext/LexicalHandler.h>
#include <SAX/ext/DeclHandler.h>
2002-06-21 13:16:28 +02:00
#include <XML/UnicodeCharacters.h>
2003-04-04 18:58:05 +02:00
#include <SAX/helpers/PropertyNames.h>
2002-06-21 13:16:28 +02:00
#include <ostream>
2003-03-21 13:52:46 +01:00
#include <algorithm>
2003-04-03 16:01:44 +02:00
#include <typeinfo>
2002-06-21 13:16:28 +02:00
namespace SAX {
template<class string_type>
class basic_Writer : public basic_XMLFilterImpl<string_type>,
private basic_LexicalHandler<string_type>,
private basic_DeclHandler<string_type>
2002-06-21 13:16:28 +02:00
{
public:
typedef string_type stringT;
2003-03-21 13:52:46 +01:00
typedef basic_Writer<stringT> WriterT;
2002-06-21 13:16:28 +02:00
typedef typename string_type::value_type charT;
typedef typename string_type::traits_type traitsT;
typedef std::basic_ostream<charT, traitsT> ostreamT;
typedef basic_XMLReader<stringT> XMLReaderT;
typedef basic_XMLFilterImpl<stringT> XMLFilterT;
2003-09-10 17:15:55 +02:00
typedef typename basic_XMLFilterImpl<stringT>::AttributesT AttributesT;
2003-09-11 16:05:18 +02:00
typedef Arabica::Unicode<charT> UnicodeT;
private:
typedef basic_LexicalHandler<stringT> LexicalHandlerT;
typedef basic_DeclHandler<stringT> DeclHandlerT;
typedef typename XMLReaderT::InputSourceT InputSourceT;
typedef typename XMLReaderT::PropertyBase PropertyBase;
2002-06-21 13:16:28 +02:00
public:
2003-09-12 16:09:13 +02:00
basic_Writer(ostreamT& stream, unsigned int indent = 2) :
inCDATA_(false),
inDTD_(false),
internalSubset_(true),
2003-09-12 16:09:13 +02:00
indent_(indent),
depth_(0),
stream_(&stream),
2003-09-12 16:09:13 +02:00
lexicalHandler_(0),
declHandler_(0),
2003-09-12 16:09:13 +02:00
lastTag_(startTag)
2002-06-21 13:16:28 +02:00
{
} // basic_Writer
2003-09-12 16:09:13 +02:00
basic_Writer(ostreamT& stream, XMLReaderT& parent, unsigned int indent = 2) :
2002-06-21 13:16:28 +02:00
XMLFilterT(parent),
inCDATA_(false),
inDTD_(false),
internalSubset_(true),
2003-09-12 16:09:13 +02:00
indent_(indent),
depth_(0),
stream_(&stream),
2003-09-12 16:09:13 +02:00
lexicalHandler_(0),
declHandler_(0),
2003-09-12 16:09:13 +02:00
lastTag_(startTag)
2002-06-21 13:16:28 +02:00
{
} // basic_Writer
virtual void parse(InputSourceT& input);
protected:
// Parser
virtual std::auto_ptr<PropertyBase> doGetProperty(const stringT& name);
virtual void doSetProperty(const stringT& name, std::auto_ptr<PropertyBase> value);
// ContentHandler
2002-06-21 13:16:28 +02:00
virtual void startDocument();
virtual void endDocument();
virtual void startElement(const stringT& namespaceURI, const stringT& localName,
2003-09-10 17:15:55 +02:00
const stringT& qName, const AttributesT& atts);
2002-06-21 13:16:28 +02:00
virtual void endElement(const stringT& namespaceURI, const stringT& localName,
const stringT& qName);
virtual void characters(const stringT& ch);
2004-01-27 00:21:30 +01:00
virtual void ignorableWhitespace(const stringT& ch);
2002-06-21 13:16:28 +02:00
virtual void processingInstruction(const stringT& target, const stringT& data);
virtual void skippedEntity(const stringT& name);
// Lexical Handler
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);
// DTD Handler
virtual void notationDecl(const stringT& name, const stringT& publicId, const stringT& systemId);
virtual void unparsedEntityDecl(const stringT& name, const stringT& publicId, const stringT& systemId, const stringT& notationName);
// Decl Handler
virtual void elementDecl(const stringT& name, const stringT& model);
virtual void attributeDecl(const stringT& elementName, const stringT& attributeName,
const stringT& type, const stringT& valueDefault, const stringT& value);
virtual void internalEntityDecl(const stringT& name, const stringT& value);
virtual void externalEntityDecl(const stringT& name, const stringT& publicId, const stringT& systemId);
/////////////////
void startEntityDecl(const stringT& name);
void publicAndSystem(const stringT& publicId, const stringT& systemId);
2002-06-21 13:16:28 +02:00
void doIndent();
bool isDtd(const stringT& name);
private:
bool inCDATA_;
bool inDTD_;
bool internalSubset_;
2002-06-21 13:16:28 +02:00
int indent_;
int depth_;
ostreamT* stream_;
LexicalHandlerT* lexicalHandler_;
DeclHandlerT* declHandler_;
enum { startTag, endTag, docTag } lastTag_;
const SAX::PropertyNames<stringT> properties_;
2003-03-21 13:52:46 +01:00
2003-04-03 16:01:44 +02:00
template<typename char_type, typename traits_type>
2003-03-21 13:52:46 +01:00
class escaper
{
private:
2003-04-03 16:01:44 +02:00
typedef char_type charT;
typedef traits_type traitsT;
typedef std::basic_ostream<charT, traitsT> ostreamT;
2003-09-11 16:05:18 +02:00
typedef Arabica::Unicode<charT> UnicodeT;
2003-03-21 13:52:46 +01:00
public:
escaper(ostreamT* stream) : stream_(stream) { }
void operator()(charT ch)
{
if(ch == UnicodeT::LESS_THAN_SIGN)
{
2003-03-21 13:52:46 +01:00
*stream_ << UnicodeT::AMPERSAND
<< UnicodeT::LOWERCASE_L
<< UnicodeT::LOWERCASE_T
2003-03-21 13:52:46 +01:00
<< UnicodeT::SEMI_COLON;
return;
} // if(ch == UnicodeT::LESS_THAN_SIGN)
if(ch == UnicodeT::GREATER_THAN_SIGN)
{
2003-03-21 13:52:46 +01:00
*stream_ << UnicodeT::AMPERSAND
<< UnicodeT::LOWERCASE_G
<< UnicodeT::LOWERCASE_T
2003-03-21 13:52:46 +01:00
<< UnicodeT::SEMI_COLON;
return;
} // if(ch == UnicodeT::GREATER_THAN_SIGN)
if(ch == UnicodeT::AMPERSAND)
{
*stream_ << UnicodeT::AMPERSAND
<< UnicodeT::LOWERCASE_A
<< UnicodeT::LOWERCASE_M
<< UnicodeT::LOWERCASE_P
2003-03-21 13:52:46 +01:00
<< UnicodeT::SEMI_COLON;
return;
} // if(ch == case UnicodeT::AMPERSAND)
if(ch == UnicodeT::QUOTATION_MARK)
{
2003-03-21 13:52:46 +01:00
*stream_ << UnicodeT::AMPERSAND
<< UnicodeT::LOWERCASE_Q
<< UnicodeT::LOWERCASE_U
<< UnicodeT::LOWERCASE_O
<< UnicodeT::LOWERCASE_T
2003-03-21 13:52:46 +01:00
<< UnicodeT::SEMI_COLON;
return;
} // if(ch == UnicodeT::QUOTATION_MARK)
*stream_ << ch;
} // operator()
2003-03-21 13:52:46 +01:00
private:
ostreamT* stream_;
}; // escaper
2003-04-03 16:01:44 +02:00
2002-06-21 13:16:28 +02:00
}; // class basic_Writer
template<class string_type>
void basic_Writer<string_type>::startDocument()
{
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::QUESTION_MARK
<< UnicodeT::LOWERCASE_X
<< UnicodeT::LOWERCASE_M
<< UnicodeT::LOWERCASE_L
<< UnicodeT::SPACE
<< UnicodeT::LOWERCASE_V
<< UnicodeT::LOWERCASE_E
<< UnicodeT::LOWERCASE_R
<< UnicodeT::LOWERCASE_S
<< UnicodeT::LOWERCASE_I
<< UnicodeT::LOWERCASE_O
<< UnicodeT::LOWERCASE_N
<< UnicodeT::EQUALS_SIGN
<< UnicodeT::QUOTATION_MARK
<< UnicodeT::NUMBER_1
<< UnicodeT::FULL_STOP
<< UnicodeT::NUMBER_0
<< UnicodeT::QUOTATION_MARK
<< UnicodeT::QUESTION_MARK
<< UnicodeT::GREATER_THAN_SIGN
2003-09-12 16:09:13 +02:00
<< UnicodeT::LINE_FEED;
2002-06-21 13:16:28 +02:00
depth_ = 0;
inCDATA_ = false;
2002-06-21 13:16:28 +02:00
XMLFilterT::startDocument();
2003-09-12 16:09:13 +02:00
lastTag_ = docTag;
2002-06-21 13:16:28 +02:00
} // startDocument
template<class string_type>
void basic_Writer<string_type>::endDocument()
{
XMLFilterT::endDocument();
2003-09-12 16:09:13 +02:00
lastTag_ = endTag;
2002-06-21 13:16:28 +02:00
} // endDocument
template<class string_type>
void basic_Writer<string_type>::startElement(
const stringT& namespaceURI, const stringT& localName,
2003-09-10 17:15:55 +02:00
const stringT& qName, const AttributesT& atts)
2002-06-21 13:16:28 +02:00
{
if((lastTag_ == startTag) && (indent_ > 0))
2003-09-12 16:09:13 +02:00
*stream_ << UnicodeT::LINE_FEED;
2002-06-21 13:16:28 +02:00
doIndent();
*stream_ << UnicodeT::LESS_THAN_SIGN << (!qName.empty() ? qName : localName);
2002-06-21 13:16:28 +02:00
for(int i = 0; i < atts.getLength(); ++i)
2003-03-21 13:52:46 +01:00
{
2002-06-21 13:16:28 +02:00
*stream_ << UnicodeT::SPACE
<< atts.getQName(i)
<< UnicodeT::EQUALS_SIGN
<< UnicodeT::QUOTATION_MARK;
2003-03-21 13:52:46 +01:00
stringT value = atts.getValue(i);
std::for_each(value.begin(), value.end(), escaper<charT, traitsT>(stream_));
2003-03-21 13:52:46 +01:00
*stream_ << UnicodeT::QUOTATION_MARK;
}
2002-06-21 13:16:28 +02:00
*stream_ << UnicodeT::GREATER_THAN_SIGN;
depth_ += indent_;
2003-09-12 16:09:13 +02:00
lastTag_ = startTag;
2002-06-21 13:16:28 +02:00
XMLFilterT::startElement(namespaceURI, localName, qName, atts);
} // startElement
template<class string_type>
void basic_Writer<string_type>::endElement(
const stringT& namespaceURI, const stringT& localName,
const stringT& qName)
{
depth_ -= indent_;
2003-09-12 16:09:13 +02:00
if(lastTag_ == endTag)
doIndent();
2002-06-21 13:16:28 +02:00
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::SLASH
<< (!qName.empty() ? qName : localName)
<< UnicodeT::GREATER_THAN_SIGN;
if(indent_ != 0)
*stream_ << UnicodeT::LINE_FEED;
2003-09-12 16:09:13 +02:00
lastTag_ = endTag;
2002-06-21 13:16:28 +02:00
XMLFilterT::endElement(namespaceURI, localName, qName);
} // endElement
template<class string_type>
void basic_Writer<string_type>::characters(const stringT& ch)
{
2003-03-21 13:52:46 +01:00
if(!inCDATA_)
std::for_each(ch.begin(), ch.end(), escaper<charT, traitsT>(stream_));
2003-03-21 13:52:46 +01:00
else
*stream_ << ch;
2002-06-21 13:16:28 +02:00
XMLFilterT::characters(ch);
} // characters
2004-01-27 00:21:30 +01:00
template<class string_type>
void basic_Writer<string_type>::ignorableWhitespace(const stringT& ch)
{
*stream_ << ch;
XMLFilterT::ignorableWhitespace(ch);
} // ignorableWhitespace
2002-06-21 13:16:28 +02:00
template<class string_type>
void basic_Writer<string_type>::processingInstruction(const stringT& target, const stringT& data)
{
if((!inDTD_) || (inDTD_ && internalSubset_))
{
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::QUESTION_MARK
2004-01-27 00:21:30 +01:00
<< target;
if(data.length())
*stream_ << UnicodeT::SPACE << data;
2004-01-27 00:21:30 +01:00
*stream_ << UnicodeT::QUESTION_MARK
<< UnicodeT::GREATER_THAN_SIGN;
}
2002-06-21 13:16:28 +02:00
} // processingInstruction
template<class string_type>
void basic_Writer<string_type>::skippedEntity(const stringT& name)
{
if(!isDtd(name))
*stream_ << UnicodeT::AMPERSAND << name << UnicodeT::SEMI_COLON;
2002-06-21 13:16:28 +02:00
} // skippedEntity
template<class string_type>
void basic_Writer<string_type>::parse(InputSourceT& input)
{
2003-04-03 16:01:44 +02:00
try
{
XMLReaderT* parent = getParent();
if(parent)
parent->setProperty(properties_.lexicalHandler, static_cast<LexicalHandlerT&>(*this));
2003-04-03 16:01:44 +02:00
}
catch(...)
{ }
try
{
XMLReaderT* parent = getParent();
if(parent)
parent->setProperty(properties_.declHandler, static_cast<DeclHandlerT&>(*this));
}
catch(...)
{ }
XMLFilterT::parse(input);
} // parse
2002-06-21 13:16:28 +02:00
template<class string_type>
void basic_Writer<string_type>::doIndent()
{
for(int i = 0; i < depth_; ++i)
2003-09-12 16:09:13 +02:00
*stream_ << UnicodeT::SPACE;
2002-06-21 13:16:28 +02:00
} // doIndent
template<class string_type>
bool basic_Writer<string_type>::isDtd(const string_type& name)
{
return (name.length() == 5 &&
name[0] == UnicodeT::LEFT_SQUARE_BRACKET &&
name[1] == UnicodeT::LOWERCASE_D &&
name[2] == UnicodeT::LOWERCASE_T &&
name[3] == UnicodeT::LOWERCASE_D &&
2002-06-21 13:16:28 +02:00
name[4] == UnicodeT::RIGHT_SQUARE_BRACKET);
} // isDtd
2003-09-09 15:09:48 +02:00
#ifndef ARABICA_VS6_WORKAROUND
template<class string_type>
std::auto_ptr<typename basic_Writer<string_type>::PropertyBase> basic_Writer<string_type>::doGetProperty(const string_type& name)
2003-04-04 15:41:24 +02:00
#else
template<class string_type>
std::auto_ptr<basic_Writer<string_type>::PropertyBase> basic_Writer<string_type>::doGetProperty(const string_type& name)
#endif
{
if(name == properties_.lexicalHandler)
{
XMLReaderT::Property<LexicalHandlerT*>* prop =
new XMLReaderT::Property<LexicalHandlerT*>(lexicalHandler_);
return std::auto_ptr<PropertyBase>(prop);
}
if(name == properties_.declHandler)
{
XMLReaderT::Property<DeclHandlerT*>* prop =
new XMLReaderT::Property<DeclHandlerT*>(declHandler_);
return std::auto_ptr<PropertyBase>(prop);
}
return XMLFilterT::doGetProperty(name);
} // doGetProperty
2003-09-09 15:09:48 +02:00
#ifndef ARABICA_VS6_WORKAROUND
template<class string_type>
void basic_Writer<string_type>::doSetProperty(const string_type& name, std::auto_ptr<typename basic_Writer<string_type>::PropertyBase> value)
2003-04-04 15:41:24 +02:00
#else
template<class string_type>
void basic_Writer<string_type>::doSetProperty(const string_type& name, std::auto_ptr<basic_Writer<string_type>::PropertyBase> value)
#endif
{
if(name == properties_.lexicalHandler)
{
XMLReaderT::Property<LexicalHandlerT&>* prop =
dynamic_cast<XMLReaderT::Property<LexicalHandlerT&>*>(value.get());
if(!prop)
throw std::bad_cast();
lexicalHandler_ = &(prop->get());
}
if(name == properties_.declHandler)
{
XMLReaderT::Property<DeclHandlerT&>* prop =
dynamic_cast<XMLReaderT::Property<DeclHandlerT&>*>(value.get());
if(!prop)
throw std::bad_cast();
declHandler_ = &(prop->get());
}
2003-04-03 16:01:44 +02:00
XMLFilterT::doSetProperty(name, value);
} // doSetProperty
template<class string_type>
void basic_Writer<string_type>::startDTD(const stringT& name, const stringT& publicId, const stringT& systemId)
{
inDTD_ = true;
depth_ += indent_;
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::EXCLAMATION_MARK
<< UnicodeT::CAPITAL_D
<< UnicodeT::CAPITAL_O
<< UnicodeT::CAPITAL_C
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_Y
<< UnicodeT::CAPITAL_P
<< UnicodeT::CAPITAL_E
<< UnicodeT::SPACE
<< name;
publicAndSystem(publicId, systemId);
*stream_ << UnicodeT::SPACE
<< UnicodeT::LEFT_SQUARE_BRACKET
<< std::endl;
if(lexicalHandler_)
lexicalHandler_->startDTD(name, publicId, systemId);
} // startDTD
template<class string_type>
void basic_Writer<string_type>::endDTD()
{
*stream_ << UnicodeT::RIGHT_SQUARE_BRACKET
<< UnicodeT::GREATER_THAN_SIGN
<< std::endl;
inDTD_ = false;
depth_ -= indent_;
if(lexicalHandler_)
lexicalHandler_->endDTD();
} // endDTD
template<class string_type>
void basic_Writer<string_type>::startEntity(const stringT& name)
{
if(isDtd(name))
internalSubset_ = false;
if(lexicalHandler_)
lexicalHandler_->startEntity(name);
} // startEntity
template<class string_type>
void basic_Writer<string_type>::endEntity(const stringT& name)
{
if(isDtd(name))
internalSubset_ = false;
if(lexicalHandler_)
lexicalHandler_->endEntity(name);
} // endEntity
template<class string_type>
void basic_Writer<string_type>::startCDATA()
{
inCDATA_ = true;
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::EXCLAMATION_MARK
<< UnicodeT::LEFT_SQUARE_BRACKET
<< UnicodeT::CAPITAL_C
<< UnicodeT::CAPITAL_D
<< UnicodeT::CAPITAL_A
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_A
<< UnicodeT::LEFT_SQUARE_BRACKET;
if(lexicalHandler_)
lexicalHandler_->startCDATA();
} // startCDATA
template<class string_type>
void basic_Writer<string_type>::endCDATA()
{
*stream_ << UnicodeT::RIGHT_SQUARE_BRACKET
<< UnicodeT::RIGHT_SQUARE_BRACKET
<< UnicodeT::GREATER_THAN_SIGN;
inCDATA_ = false;
if(lexicalHandler_)
lexicalHandler_->endCDATA();
} // endCDATA
template<class string_type>
void basic_Writer<string_type>::comment(const stringT& text)
{
if((!inDTD_) || (inDTD_ && internalSubset_))
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::EXCLAMATION_MARK
<< UnicodeT::HYPHEN_MINUS
<< UnicodeT::HYPHEN_MINUS
<< text
<< UnicodeT::HYPHEN_MINUS
<< UnicodeT::HYPHEN_MINUS
<< UnicodeT::GREATER_THAN_SIGN;
if(lexicalHandler_)
lexicalHandler_->comment(text);
} // comment
2003-03-21 13:52:46 +01:00
template<class string_type>
void basic_Writer<string_type>::notationDecl(const stringT& name, const stringT& publicId, const stringT& systemId)
{
doIndent();
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::EXCLAMATION_MARK
<< UnicodeT::CAPITAL_N
<< UnicodeT::CAPITAL_O
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_A
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_I
<< UnicodeT::CAPITAL_O
<< UnicodeT::CAPITAL_N
<< UnicodeT::SPACE
<< name;
publicAndSystem(publicId, systemId);
*stream_ << UnicodeT::GREATER_THAN_SIGN
<< std::endl;
XMLFilterT::notationDecl(name, publicId, systemId);
} // notationDecl
template<class string_type>
void basic_Writer<string_type>::unparsedEntityDecl(const stringT& name, const stringT& publicId, const stringT& systemId, const stringT& notationName)
{
doIndent();
startEntityDecl(name);
publicAndSystem(publicId, systemId);
*stream_ << UnicodeT::SPACE
<< UnicodeT::CAPITAL_N
<< UnicodeT::CAPITAL_D
<< UnicodeT::CAPITAL_A
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_A
<< UnicodeT::SPACE
<< notationName
<< UnicodeT::GREATER_THAN_SIGN
<< std::endl;
XMLFilterT::unparsedEntityDecl(name, publicId, systemId, notationName);
} // unparsedEntityDecl
template<class string_type>
void basic_Writer<string_type>::elementDecl(const stringT& name, const stringT& model)
{
doIndent();
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::EXCLAMATION_MARK
<< UnicodeT::CAPITAL_E
<< UnicodeT::CAPITAL_L
<< UnicodeT::CAPITAL_E
<< UnicodeT::CAPITAL_M
<< UnicodeT::CAPITAL_E
<< UnicodeT::CAPITAL_N
<< UnicodeT::CAPITAL_T
<< UnicodeT::SPACE
<< name
<< UnicodeT::SPACE
<< model
<< UnicodeT::GREATER_THAN_SIGN
<< std::endl;
if(declHandler_)
declHandler_->elementDecl(name, model);
} // elementDecl
template<class string_type>
void basic_Writer<string_type>::attributeDecl(const stringT& elementName, const stringT& attributeName,
const stringT& type, const stringT& valueDefault, const stringT& value)
{
doIndent();
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::EXCLAMATION_MARK
<< UnicodeT::CAPITAL_A
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_L
<< UnicodeT::CAPITAL_I
<< UnicodeT::CAPITAL_S
<< UnicodeT::CAPITAL_T
<< UnicodeT::SPACE
<< elementName
<< UnicodeT::SPACE
<< attributeName
<< UnicodeT::SPACE
<< type;
if(!valueDefault.empty())
*stream_ << UnicodeT::SPACE
<< valueDefault;
if(!value.empty())
*stream_ << UnicodeT::SPACE
<< UnicodeT::QUOTATION_MARK
<< value
<< UnicodeT::QUOTATION_MARK;
*stream_ << UnicodeT::GREATER_THAN_SIGN
<< std::endl;
if(declHandler_)
declHandler_->attributeDecl(elementName, attributeName, type, valueDefault, value);
} // attributeDecl
template<class string_type>
void basic_Writer<string_type>::internalEntityDecl(const stringT& name, const stringT& value)
{
doIndent();
startEntityDecl(name);
*stream_ << UnicodeT::SPACE
<< UnicodeT::QUOTATION_MARK
<< value
<< UnicodeT::QUOTATION_MARK
<< UnicodeT::GREATER_THAN_SIGN
<< std::endl;
if(declHandler_)
declHandler_->internalEntityDecl(name, value);
} // internalEntityDecl
template<class string_type>
void basic_Writer<string_type>::externalEntityDecl(const stringT& name, const stringT& publicId, const stringT& systemId)
{
doIndent();
startEntityDecl(name);
publicAndSystem(publicId, systemId);
*stream_ << UnicodeT::GREATER_THAN_SIGN
<< std::endl;
if(declHandler_)
declHandler_->externalEntityDecl(name, publicId, systemId);
} // externalEntityDecl
template<class string_type>
void basic_Writer<string_type>::startEntityDecl(const stringT& name)
{
*stream_ << UnicodeT::LESS_THAN_SIGN
<< UnicodeT::EXCLAMATION_MARK
<< UnicodeT::CAPITAL_E
<< UnicodeT::CAPITAL_N
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_I
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_Y
<< UnicodeT::SPACE
<< name;
} // startEntityDecl
template<class string_type>
void basic_Writer<string_type>::publicAndSystem(const stringT& publicId, const stringT& systemId)
{
*stream_ << UnicodeT::SPACE;
if(!publicId.empty())
*stream_ << UnicodeT::CAPITAL_P
<< UnicodeT::CAPITAL_U
<< UnicodeT::CAPITAL_B
<< UnicodeT::CAPITAL_L
<< UnicodeT::CAPITAL_I
<< UnicodeT::CAPITAL_C
<< UnicodeT::SPACE
<< UnicodeT::QUOTATION_MARK
<< publicId
<< UnicodeT::QUOTATION_MARK;
if(!systemId.empty())
{
if(publicId.empty())
*stream_ << UnicodeT::CAPITAL_S
<< UnicodeT::CAPITAL_Y
<< UnicodeT::CAPITAL_S
<< UnicodeT::CAPITAL_T
<< UnicodeT::CAPITAL_E
<< UnicodeT::CAPITAL_M;
*stream_ << UnicodeT::SPACE
<< UnicodeT::QUOTATION_MARK
<< systemId
<< UnicodeT::QUOTATION_MARK;
} // if ...
} // publicAndSystem
2002-06-21 13:16:28 +02:00
typedef basic_Writer<std::string> Writer;
#ifndef ARABICA_NO_WCHAR_T
2002-06-21 13:16:28 +02:00
typedef basic_Writer<std::wstring> wWriter;
#endif
2002-06-21 13:16:28 +02:00
} // namespace SAX
#endif