// ********************************************************************** // // Copyright (c) 2001 // ZeroC, Inc. // Huntsville, AL, USA // // All Rights Reserved // // ********************************************************************** #include using namespace std; using namespace IceUtil; namespace IceUtil { NextLine nl; StartBlock sb; EndBlock eb; Separator sp; EndElement ee; StartEscapes startEscapes; EndEscapes endEscapes; } // ---------------------------------------------------------------------- // OutputBase // ---------------------------------------------------------------------- IceUtil::OutputBase::OutputBase() : _out(_fout), _pos(0), _indent(0), _indentSize(4), _useTab(true), _separator(true) { } IceUtil::OutputBase::OutputBase(ostream& os) : _out(os), _pos(0), _indent(0), _indentSize(4), _useTab(true), _separator(true) { } IceUtil::OutputBase::OutputBase(const char* s) : _out(_fout), _pos(0), _indent(0), _indentSize(4), _useTab(true), _separator(true) { open(s); } IceUtil::OutputBase::~OutputBase() { } void IceUtil::OutputBase::open(const char* s) { _fout.open(s); } void IceUtil::OutputBase::print(const char* s) { for(unsigned int i = 0; i < strlen(s); ++i) { if(s[i] == '\n') { _pos = 0; } else { ++_pos; } } _out << s; } void IceUtil::OutputBase::inc() { _indent += _indentSize; } void IceUtil::OutputBase::dec() { assert(_indent >= _indentSize); _indent -= _indentSize; } void IceUtil::OutputBase::useCurrentPosAsIndent() { _indentSave.push(_indent); _indent = _pos; } void IceUtil::OutputBase::zeroIndent() { _indentSave.push(_indent); _indent = 0; } void IceUtil::OutputBase::restoreIndent() { assert(!_indentSave.empty()); _indent = _indentSave.top(); _indentSave.pop(); } void IceUtil::OutputBase::setIndent(int indentSize) { _indentSize = indentSize; } void IceUtil::OutputBase::setUseTab(bool useTab) { _useTab = useTab; } void IceUtil::OutputBase::nl() { _out << '\n'; _pos = 0; _separator = true; int indent = _indent; if(_useTab) { while(indent >= 8) { indent -= 8; _out << '\t'; _pos += 8; } } else { while(indent >= _indentSize) { indent -= _indentSize; _out << " "; _pos += _indentSize; } } while(indent > 0) { --indent; _out << ' '; ++_pos; } _out.flush(); } void IceUtil::OutputBase::sp() { if(_separator) { _out << '\n'; } } bool IceUtil::OutputBase::operator!() const { return !_out; } streamsize IceUtil::OutputBase::width() const { return _out.width(); } streamsize IceUtil::OutputBase::width(streamsize newWidth) { return _out.width(newWidth); } ios_base::fmtflags IceUtil::OutputBase::flags() const { return _out.flags(); } ios_base::fmtflags IceUtil::OutputBase::flags(ios_base::fmtflags newFlags) { return _out.flags(newFlags); } ostream::char_type IceUtil::OutputBase::fill() const { return _out.fill(); } ostream::char_type IceUtil::OutputBase::fill(ostream::char_type newFill) { return _out.fill(newFill); } // ---------------------------------------------------------------------- // Output // ---------------------------------------------------------------------- IceUtil::Output::Output() : OutputBase(), _blockStart("{"), _blockEnd("}") { } IceUtil::Output::Output(ostream& os) : OutputBase(os), _blockStart("{"), _blockEnd("}") { } IceUtil::Output::Output(const char* s) : OutputBase(s), _blockStart("{"), _blockEnd("}") { } void IceUtil::Output::setBeginBlock(const char *bb) { _blockStart = bb; } void IceUtil::Output::setEndBlock(const char *eb) { _blockEnd = eb; } void IceUtil::Output::sb() { if(_blockStart.length()) { nl(); _out << _blockStart; } ++_pos; inc(); _separator = false; } void IceUtil::Output::eb() { dec(); if(_blockEnd.length()) { nl(); _out << _blockEnd; } --_pos; } Output& IceUtil::operator<<(Output& out, ios_base& (*val)(ios_base&)) { ostringstream s; s << val; out.print(s.str().c_str()); return out; } // ---------------------------------------------------------------------- // XMLOutput // ---------------------------------------------------------------------- IceUtil::XMLOutput::XMLOutput() : OutputBase(), _se(false), _text(false), _sgml(false), _escape(false) { } IceUtil::XMLOutput::XMLOutput(ostream& os) : OutputBase(os), _se(false), _text(false), _sgml(false), _escape(false) { } IceUtil::XMLOutput::XMLOutput(const char* s) : OutputBase(s), _se(false), _text(false), _sgml(false), _escape(false) { } void IceUtil::XMLOutput::setSGML(bool sgml) { _sgml = true; } void IceUtil::XMLOutput::print(const char* s) { if(_se) { _out << '>'; _se = false; } _text = true; if(_escape) { string escaped = escape(s); OutputBase::print(escaped.c_str()); } else { OutputBase::print(s); } } void IceUtil::XMLOutput::nl() { if(_se) { _se = false; _out << '>'; } OutputBase::nl(); } void IceUtil::XMLOutput::se(const string& element) { nl(); // // If we're not in SGML mode the output of the '>' character is // deferred until either the //end-element (in which case a /> is // emitted) or until something //is displayed. // if(_escape) { _out << '<' << escape(element); } else { _out << '<' << element; } _se = true; _text = false; string::size_type pos = element.find_first_of(" \t"); if(pos == string::npos) { _elementStack.push(element); } else { _elementStack.push(element.substr(0, pos)); } ++_pos; // TODO: ??? inc(); _separator = false; } void IceUtil::XMLOutput::ee() { string element = _elementStack.top(); _elementStack.pop(); dec(); if(_se) { // // SGML (docbook) doesn't support . // if(_sgml) { _out << ">'; } else { _out << "/>"; } } else { if(!_text) { nl(); } _out << "'; } --_pos; // TODO: ??? _se = false; _text = false; } void IceUtil::XMLOutput::attr(const string& name, const string& value) { // // Precondition: Attributes can only be attached to elements. // assert(_se); _out << " " << name << "=\"" << escape(value) << '"'; } void IceUtil::XMLOutput::startEscapes() { _escape = true; } void IceUtil::XMLOutput::endEscapes() { _escape = false; } string IceUtil::XMLOutput::currentElement() const { if(_elementStack.size() > 0) { return _elementStack.top(); } else { return string(); } } string IceUtil::XMLOutput::escape(const string& input) const { string v = input; // // Find out whether there is a reserved character to avoid // conversion if not necessary. // const string allReserved = "<>'\"&"; if(v.find_first_of(allReserved) != string::npos) { // // First convert all & to & // size_t pos = 0; while((pos = v.find_first_of('&', pos)) != string::npos) { v.insert(pos+1, "amp;"); pos += 4; } // // Next convert remaining reserved characters. // const string reserved = "<>'\""; pos = 0; while((pos = v.find_first_of(reserved, pos)) != string::npos) { string replace; switch(v[pos]) { case '>': replace = ">"; break; case '<': replace = "<"; break; case '\'': replace = "'"; break; case '"': replace = """; break; default: assert(false); } v.erase(pos, 1); v.insert(pos, replace); pos += replace.size(); } } return v; } XMLOutput& IceUtil::operator<<(XMLOutput& out, ios_base& (*val)(ios_base&)) { ostringstream s; s << val; out.print(s.str().c_str()); return out; } IceUtil::StartElement::StartElement(const string& name) : _name(name) { } const string& IceUtil::StartElement::getName() const { return _name; } IceUtil::Attribute::Attribute(const string& name, const string& value) : _name(name), _value(value) { } const string& IceUtil::Attribute::getName() const { return _name; } const string& IceUtil::Attribute::getValue() const { return _value; }