Orb/Doxygen/src/xmlwriter.h
author szarinda <>
Thu, 21 Jan 2010 17:29:01 +0000
changeset 0 42188c7ea2d9
child 1 82f11024044a
permissions -rw-r--r--
Initial contribution of ORB delivering Feature bug 1460


#ifndef _XMLWRITER_H
#define _XMLWRITER_H

#include <qdir.h>
#include <qfile.h>
#include <qtextstream.h>
#include <qmap.h>
#include <qdict.h>
#include <qstack.h>

typedef QMap<QString, QString> AttributeMap;
typedef QMapConstIterator<QString, QString> AttributeMapIter;

static QString XML_INDENT("\t");
static QString XML_OUTPUT_ENDL("\n");

// TODO: If the output file fails to open then write to an internal string that can be
// accessed by the caller after they call close(). This might be useful for internal
// XML snippets.

/**
Class that handles writing to an XML file. It guarentees well-formedness,
handles the encoding of PCDATA and indents the output.

Individual elements are created as objects, they are written to the file on
creation and closed when the element is destroyed.
*/
class XmlStream
{
	friend class XmlElement;
public:
	/// Create an XML stream that writes to a file with the given encoding
	XmlStream(const QString &fileName,
				const QString &aEncoding="UTF-8",
				const QString &aStandalone="no",
				const QString &doctypeStr="");
	/// Returns true of the file tha tthis stream is writing to is currently open.
	bool isOpen() const { return mIsOpen; }
	/// Write PCDATA to the output file, entities will be translated
	void characters(const QString &aText);
	void characters(char c);
	/// Write a processing instruction to the output file.
	void processingInstruction(const QString &aText);
	/// Write a comment to the output file.
	void comment(const QString &aText);
	/// Output operator overloading, these equate to calling characters()
	XmlStream& operator<<(const QCString &s);
	XmlStream& operator<<(const char *s);
	XmlStream& operator<<(char s);
	/// Write a unicode character to the stream
	XmlStream& writeUnicode(const QCString &s);
	/// Close the XML file (if open), this will flush any unclosed elements
	void close();
	/// Get the file name
	QString getFileName() const { return mFile.name(); }
	/// Suspend and resume output
	void outputSuspend();
	void outputResume();
	/// Destructor, this will call close()
	~XmlStream();
private:
	QDict<char> unicodeCharTable;
	/// Write out the start of element and register that element name on the stack
	void startElement(const QString &aElemName, const AttributeMap &aAttrs);
	/// Write out the end of element and remove that element name on the stack
	void endElement(const QString &aElemName);
	void closeElementDeclIfOpen();
	void indent(unsigned int aInitVal=0);
	inline bool isLegalXmlChar(QChar c) const;
	QString encodeText(const QString &aStr) const;
	QString encodeChar(char c) const;
	bool mustEncodeChar(char c) const;
private:
	QFile mFile;
	QTextStream *mStreamP;
	QStack<QString> mElemStack;
	bool mInElement;
	bool mCanIndent;
	bool mIsOpen;
	bool mCanWrite;
private:
	// Private cctor and op=
	XmlStream (const XmlStream &rhs);
	XmlStream& operator=(const XmlStream &rhs);
};

/**
XmlElement

Usage example:
@code
// Start an XML file
XmlStream t(fileName);
if (!t.isOpen()) {
  err("Cannot open file %s for writing!\n", fileName.data());
  return;
}
// Create and persist a root element with no attributes
XmlElement elemRoot(t, QString("root"), AttributeMap());
// Note the use of {} that causes elem to go out of scope an thus
// its destructor will be called so closing the element
{
	// Create and persist a root element with attributes
	// First create the attributes
	AttributeMap attrs;
	attrs["attr_1"]	= "value_1";
	attrs["attr_2"]	= "value_2";
	// Create the element
	XmlElement elem(t, "Element", attrs);
	// Write some text
	t.characters("Some stuff");
	// Now elem is going out of scope
}
// Open and close and element
XmlElement(t, "OpenClose");
// Open and close and element with one attribute
XmlElement(t, "WhatsForDinner", "spam", "eggs");
// t goes out of scope and will close the file
@endcode

The result is:
@verbatim
<?xml version='1.0' encoding='UTF-8' standalone='no'?>
<root>
	<Element attr_1="value_1" attr_2="value_2">Some stuff</Element>
	<OpenClose/>
	<WhatsForDinner spam="eggs"/>
</root>
@endverbatim

*/
class XmlElement
{
public:
	/// Constructor with no attributes
	XmlElement(XmlStream &aStream, const QString &aElemName);
	/// Constructor with one attribute
	XmlElement(XmlStream &aStream, const QString &aElemName, const QString &aAttr, const QString &aAttrValue);
	/// Constructor with one attribute the values of which is a char
	XmlElement(XmlStream &aStream, const QString &aElemName, const QString &aAttr, char aAttrValue);
	/// Constructor with any number of attributes
	XmlElement(XmlStream &aStream, const QString &aElemName, AttributeMap &aAttrs);
	const QString& getElemName() const { return mElemName; }
	/// Destructor, this closes the element on the stream
	~XmlElement();
private:
	XmlStream& mStream;
	QString mElemName;
private:
	// Private cctor and op=
	XmlElement (const XmlElement &rhs);
	XmlElement& operator=(const XmlElement &rhs);
};

/*
This can be used in the case that elements need to persist over scope.

So given this kind of code:
void XmlDocVisitor::visitPre(DocHtmlDescTitle *)
{
  if (m_hide) return;
  m_t << "<varlistentry><term>";
}

void XmlDocVisitor::visitPost(DocHtmlDescTitle *) 
{
  if (m_hide) return;
  m_t << "</term></varlistentry>\n";
}

It can be replaced by this (assuming XmlDocVisitor has a member XmlElementStack m_es;):
void XmlDocVisitor::visitPre(DocHtmlDescTitle *)
{
  if (m_hide) return;
  m_es.push("varlistentry", AttributeMap());
  m_es.push("term", AttributeMap());
}

void XmlDocVisitor::visitPost(DocHtmlDescTitle *) 
{
  if (m_hide) return;
  m_es.pop();
  m_es.pop();
}

*/
class XmlElementStack
{
public:
	XmlElementStack(XmlStream &aStream);
	/// Push element with no attributes
	void push(const QString &aElemName);
	/// Constructor with one attribute
	void push(const QString &aElemName, const QString &aAttr, const QString &aAttrValue);
	/// Push element with any number of attributes
	void push(const QString &aElementName, AttributeMap &aAttrs);
	/// Method that will check the element name matches
	void pop(const QString &aElementName);
	/// Push and pop an element with no attributes
	void pushpop(const QString &aElemName);
	/// Push and pop an element with no attributes but with some content
	void pushpop(const QString &aElemName, const QString &aText);
	/// Add attribute to item currently on top of the stack
	void addAttribute(const QString &name, const QString &value);
	/// Return a reference to the top of the stack. The stack is not modified.
	const XmlElement& peek() const;
	/// Return true if the stack is empty
	bool isEmpty() const;
	void close();
	~XmlElementStack();
private:
	XmlStream& mStream;
	QStack<XmlElement> mElemStack;
private:
	// Private cctor and op=
	XmlElementStack (const XmlElementStack &rhs);
	XmlElementStack& operator=(const XmlElementStack &rhs);
	// Used by destructor
	void pop();
};

#endif // _XMLWRITER_H