/********************************************************************** Audacity: A Digital Audio Editor XMLWriter.cpp Leland Lucius *******************************************************************//** \class XMLWriter \brief Base class for XMLFileWriter and XMLStringWriter that provides the general functionality for creating XML in UTF8 encoding. *//****************************************************************//** \class XMLFileWriter \brief Wrapper to output XML data to files. *//****************************************************************//** \class XMLStringWriter \brief Wrapper to output XML data to strings. *//*******************************************************************/ #include "../Audacity.h" #include #include #include #include #include "../Internat.h" #include "XMLWriter.h" //table for xml encoding compatibility with expat decoding //see wxWidgets-2.8.12/src/expat/lib/xmltok_impl.h //and wxWidgets-2.8.12/src/expat/lib/asciitab.h static int charXMLCompatiblity[] = { /* 0x00 */ 0, 0, 0, 0, /* 0x04 */ 0, 0, 0, 0, /* 0x08 */ 0, 1, 1, 0, /* 0x0C */ 0, 1, 0, 0, /* 0x10 */ 0, 0, 0, 0, /* 0x14 */ 0, 0, 0, 0, /* 0x18 */ 0, 0, 0, 0, /* 0x1C */ 0, 0, 0, 0, }; /// /// XMLWriter base class /// XMLWriter::XMLWriter() { mDepth = 0; mInTag = false; mHasKids.Add(false); } XMLWriter::~XMLWriter() { } void XMLWriter::StartTag(const wxString &name) { int i; if (mInTag) { Write(wxT(">\n")); mInTag = false; } for (i = 0; i < mDepth; i++) { Write(wxT("\t")); } Write(wxString::Format(wxT("<%s"), name.c_str())); mTagstack.Insert(name, 0); mHasKids[0] = true; mHasKids.Insert(false, 0); mDepth++; mInTag = true; } void XMLWriter::EndTag(const wxString &name) { int i; if (mTagstack.GetCount() > 0) { if (mTagstack[0] == name) { if (mHasKids[1]) { // There will always be at least 2 at this point if (mInTag) { Write(wxT("/>\n")); } else { for (i = 0; i < mDepth - 1; i++) { Write(wxT("\t")); } Write(wxString::Format(wxT("\n"), name.c_str())); } } else { Write(wxT(">\n")); } mTagstack.RemoveAt(0); mHasKids.RemoveAt(0); } } mDepth--; mInTag = false; } void XMLWriter::WriteAttr(const wxString &name, const wxString &value) { Write(wxString::Format(wxT(" %s=\"%s\""), name.c_str(), XMLEsc(value).c_str())); } void XMLWriter::WriteAttr(const wxString &name, const wxChar *value) { WriteAttr(name, wxString(value)); } void XMLWriter::WriteAttr(const wxString &name, int value) { Write(wxString::Format(wxT(" %s=\"%d\""), name.c_str(), value)); } void XMLWriter::WriteAttr(const wxString &name, bool value) { Write(wxString::Format(wxT(" %s=\"%d\""), name.c_str(), value)); } void XMLWriter::WriteAttr(const wxString &name, long value) { Write(wxString::Format(wxT(" %s=\"%ld\""), name.c_str(), value)); } void XMLWriter::WriteAttr(const wxString &name, long long value) { Write(wxString::Format(wxT(" %s=\"%lld\""), name.c_str(), value)); } void XMLWriter::WriteAttr(const wxString &name, size_t value) { Write(wxString::Format(wxT(" %s=\"%lld\""), name.c_str(), (long long) value)); } void XMLWriter::WriteAttr(const wxString &name, float value, int digits) { Write(wxString::Format(wxT(" %s=\"%s\""), name.c_str(), Internat::ToString(value, digits).c_str())); } void XMLWriter::WriteAttr(const wxString &name, double value, int digits) { Write(wxString::Format(wxT(" %s=\"%s\""), name.c_str(), Internat::ToString(value, digits).c_str())); } void XMLWriter::WriteData(const wxString &value) { int i; for (i = 0; i < mDepth; i++) { Write(wxT("\t")); } Write(XMLEsc(value)); } void XMLWriter::WriteSubTree(const wxString &value) { if (mInTag) { Write(wxT(">\n")); mInTag = false; mHasKids[0] = true; } Write(value.c_str()); } // See http://www.w3.org/TR/REC-xml for reference wxString XMLWriter::XMLEsc(const wxString & s) { wxString result; int len = s.Length(); for(int i=0; i'): result += wxT(">"); break; default: if (!wxIsprint(c)) { //ignore several characters such ase eot (0x04) and stx (0x02) because it makes expat parser bail //see xmltok.c in expat checkCharRefNumber() to see how expat bails on these chars. //also see wxWidgets-2.8.12/src/expat/lib/asciitab.h to see which characters are nonxml compatible //post decode (we can still encode '&' and '<' with this table, but it prevents us from encoding eot) //everything is compatible past ascii 0x20, so we don't check higher than this. if(c> 0x1F || charXMLCompatiblity[c]!=0) result += wxString::Format(wxT("&#x%04x;"), c); } else { result += c; } break; } } return result; } /// /// XMLFileWriter class /// XMLFileWriter::XMLFileWriter() { } XMLFileWriter::~XMLFileWriter() { if (IsOpened()) { Close(); } } void XMLFileWriter::Open(const wxString &name, const wxString &mode) { if (!wxFFile::Open(name, mode)) throw new XMLFileWriterException(_("Error Opening File")); } void XMLFileWriter::Close() { while (mTagstack.GetCount()) { EndTag(mTagstack[0]); } CloseWithoutEndingTags(); } void XMLFileWriter::CloseWithoutEndingTags() { // Before closing, we first flush it, because if Flush() fails because of a // "disk full" condition, we can still at least try to close the file. if (!wxFFile::Flush()) { wxFFile::Close(); /* i18n-hint: 'flushing' means writing any remaining queued up changes * to disk that have not yet been written.*/ throw new XMLFileWriterException(_("Error Flushing File")); } // Note that this should never fail if flushing worked. if (!wxFFile::Close()) throw new XMLFileWriterException(_("Error Closing File")); } void XMLFileWriter::Write(const wxString &data) { if (!wxFFile::Write(data, wxConvUTF8)) { // When writing fails, we try to close the file before throwing the // exception, so it can at least be deleted. wxFFile::Close(); throw new XMLFileWriterException(_("Error Writing to File")); } } /// /// XMLStringWriter class /// XMLStringWriter::XMLStringWriter(size_t initialSize) { if (initialSize) { Alloc(initialSize); } } XMLStringWriter::~XMLStringWriter() { } void XMLStringWriter::Write(const wxString &data) { Append(data); }