blob: 3a2c3d40c9e658ad8c27618165f9f0d7f2a8771b [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001/*
2 * Copyright 2006 The Android Open Source Project
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
reed@android.com8a1c16f2008-12-17 15:59:43 +00008#include "SkXMLWriter.h"
9#include "SkStream.h"
10
11SkXMLWriter::SkXMLWriter(bool doEscapeMarkup) : fDoEscapeMarkup(doEscapeMarkup)
Mike Reedc3063e52017-01-07 16:16:02 -050012{}
reed@android.com8a1c16f2008-12-17 15:59:43 +000013
Mike Reedc3063e52017-01-07 16:16:02 -050014SkXMLWriter::~SkXMLWriter() {
reed@android.com8a1c16f2008-12-17 15:59:43 +000015 SkASSERT(fElems.count() == 0);
16}
17
Mike Reedc3063e52017-01-07 16:16:02 -050018void SkXMLWriter::flush() {
19 while (fElems.count()) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000020 this->endElement();
Mike Reedc3063e52017-01-07 16:16:02 -050021 }
reed@android.com8a1c16f2008-12-17 15:59:43 +000022}
23
Mike Reedc3063e52017-01-07 16:16:02 -050024void SkXMLWriter::addAttribute(const char name[], const char value[]) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000025 this->addAttributeLen(name, value, strlen(value));
26}
27
Mike Reedc3063e52017-01-07 16:16:02 -050028void SkXMLWriter::addS32Attribute(const char name[], int32_t value) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000029 SkString tmp;
30 tmp.appendS32(value);
31 this->addAttribute(name, tmp.c_str());
32}
33
Mike Reedc3063e52017-01-07 16:16:02 -050034void SkXMLWriter::addHexAttribute(const char name[], uint32_t value, int minDigits) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000035 SkString tmp("0x");
36 tmp.appendHex(value, minDigits);
37 this->addAttribute(name, tmp.c_str());
38}
39
Mike Reedc3063e52017-01-07 16:16:02 -050040void SkXMLWriter::addScalarAttribute(const char name[], SkScalar value) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000041 SkString tmp;
42 tmp.appendScalar(value);
43 this->addAttribute(name, tmp.c_str());
44}
45
reede73da402015-02-04 18:29:27 -080046void SkXMLWriter::addText(const char text[], size_t length) {
fmalitafe3f2602015-02-03 17:47:12 -080047 if (fElems.isEmpty()) {
48 return;
49 }
reede73da402015-02-04 18:29:27 -080050
51 this->onAddText(text, length);
52
fmalitafe3f2602015-02-03 17:47:12 -080053 fElems.top()->fHasText = true;
54}
55
Mike Reedc3063e52017-01-07 16:16:02 -050056void SkXMLWriter::doEnd(Elem* elem) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000057 delete elem;
58}
59
Mike Reedc3063e52017-01-07 16:16:02 -050060bool SkXMLWriter::doStart(const char name[], size_t length) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000061 int level = fElems.count();
62 bool firstChild = level > 0 && !fElems[level-1]->fHasChildren;
Mike Reedc3063e52017-01-07 16:16:02 -050063 if (firstChild) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000064 fElems[level-1]->fHasChildren = true;
Mike Reedc3063e52017-01-07 16:16:02 -050065 }
reed@android.com8a1c16f2008-12-17 15:59:43 +000066 Elem** elem = fElems.push();
fmalitafe3f2602015-02-03 17:47:12 -080067 *elem = new Elem(name, length);
reed@android.com8a1c16f2008-12-17 15:59:43 +000068 return firstChild;
69}
70
Mike Reedc3063e52017-01-07 16:16:02 -050071SkXMLWriter::Elem* SkXMLWriter::getEnd() {
reed@android.com8a1c16f2008-12-17 15:59:43 +000072 Elem* elem;
73 fElems.pop(&elem);
74 return elem;
75}
76
Mike Reedc3063e52017-01-07 16:16:02 -050077const char* SkXMLWriter::getHeader() {
reed@android.com8a1c16f2008-12-17 15:59:43 +000078 static const char gHeader[] = "<?xml version=\"1.0\" encoding=\"utf-8\" ?>";
79 return gHeader;
80}
81
Mike Reedc3063e52017-01-07 16:16:02 -050082void SkXMLWriter::startElement(const char name[]) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000083 this->startElementLen(name, strlen(name));
84}
85
Mike Reedc3063e52017-01-07 16:16:02 -050086static const char* escape_char(char c, char storage[2]) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000087 static const char* gEscapeChars[] = {
88 "<&lt;",
89 ">&gt;",
90 //"\"&quot;",
91 //"'&apos;",
92 "&&amp;"
93 };
94
95 const char** array = gEscapeChars;
Mike Reedc3063e52017-01-07 16:16:02 -050096 for (unsigned i = 0; i < SK_ARRAY_COUNT(gEscapeChars); i++) {
97 if (array[i][0] == c) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000098 return &array[i][1];
Mike Reedc3063e52017-01-07 16:16:02 -050099 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000100 }
101 storage[0] = c;
102 storage[1] = 0;
103 return storage;
104}
105
Mike Reedc3063e52017-01-07 16:16:02 -0500106static size_t escape_markup(char dst[], const char src[], size_t length) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000107 size_t extra = 0;
108 const char* stop = src + length;
109
Mike Reedc3063e52017-01-07 16:16:02 -0500110 while (src < stop) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000111 char orig[2];
112 const char* seq = escape_char(*src, orig);
113 size_t seqSize = strlen(seq);
114
Mike Reedc3063e52017-01-07 16:16:02 -0500115 if (dst) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000116 memcpy(dst, seq, seqSize);
117 dst += seqSize;
118 }
119
120 // now record the extra size needed
121 extra += seqSize - 1; // minus one to subtract the original char
122
123 // bump to the next src char
124 src += 1;
125 }
126 return extra;
127}
128
Mike Reedc3063e52017-01-07 16:16:02 -0500129void SkXMLWriter::addAttributeLen(const char name[], const char value[], size_t length) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000130 SkString valueStr;
131
Mike Reedc3063e52017-01-07 16:16:02 -0500132 if (fDoEscapeMarkup) {
halcanary96fcdcc2015-08-27 07:41:13 -0700133 size_t extra = escape_markup(nullptr, value, length);
Mike Reedc3063e52017-01-07 16:16:02 -0500134 if (extra) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000135 valueStr.resize(length + extra);
136 (void)escape_markup(valueStr.writable_str(), value, length);
137 value = valueStr.c_str();
138 length += extra;
139 }
140 }
141 this->onAddAttributeLen(name, value, length);
142}
143
Mike Reedc3063e52017-01-07 16:16:02 -0500144void SkXMLWriter::startElementLen(const char elem[], size_t length) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000145 this->onStartElementLen(elem, length);
146}
147
148////////////////////////////////////////////////////////////////////////////////////////
149
Mike Reedc3063e52017-01-07 16:16:02 -0500150static void write_dom(const SkDOM& dom, const SkDOM::Node* node, SkXMLWriter* w, bool skipRoot) {
151 if (!skipRoot) {
fmalita7a048692015-02-20 13:54:40 -0800152 const char* elem = dom.getName(node);
153 if (dom.getType(node) == SkDOM::kText_Type) {
154 SkASSERT(dom.countChildren(node) == 0);
155 w->addText(elem, strlen(elem));
156 return;
157 }
158
159 w->startElement(elem);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000160
161 SkDOM::AttrIter iter(dom, node);
162 const char* name;
163 const char* value;
Mike Reedc3063e52017-01-07 16:16:02 -0500164 while ((name = iter.next(&value)) != nullptr) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000165 w->addAttribute(name, value);
Mike Reedc3063e52017-01-07 16:16:02 -0500166 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000167 }
168
halcanary96fcdcc2015-08-27 07:41:13 -0700169 node = dom.getFirstChild(node, nullptr);
Mike Reedc3063e52017-01-07 16:16:02 -0500170 while (node) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000171 write_dom(dom, node, w, false);
halcanary96fcdcc2015-08-27 07:41:13 -0700172 node = dom.getNextSibling(node, nullptr);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000173 }
174
Mike Reedc3063e52017-01-07 16:16:02 -0500175 if (!skipRoot) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000176 w->endElement();
Mike Reedc3063e52017-01-07 16:16:02 -0500177 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000178}
179
Mike Reedc3063e52017-01-07 16:16:02 -0500180void SkXMLWriter::writeDOM(const SkDOM& dom, const SkDOM::Node* node, bool skipRoot) {
181 if (node) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000182 write_dom(dom, node, this, skipRoot);
Mike Reedc3063e52017-01-07 16:16:02 -0500183 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000184}
185
186void SkXMLWriter::writeHeader()
Mike Reedc3063e52017-01-07 16:16:02 -0500187{}
reed@android.com8a1c16f2008-12-17 15:59:43 +0000188
189// SkXMLStreamWriter
190
Mike Reedc3063e52017-01-07 16:16:02 -0500191static void tab(SkWStream& stream, int level) {
192 for (int i = 0; i < level; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000193 stream.writeText("\t");
Mike Reedc3063e52017-01-07 16:16:02 -0500194 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000195}
196
197SkXMLStreamWriter::SkXMLStreamWriter(SkWStream* stream) : fStream(*stream)
Mike Reedc3063e52017-01-07 16:16:02 -0500198{}
reed@android.com8a1c16f2008-12-17 15:59:43 +0000199
Mike Reedc3063e52017-01-07 16:16:02 -0500200SkXMLStreamWriter::~SkXMLStreamWriter() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000201 this->flush();
202}
203
Mike Reedc3063e52017-01-07 16:16:02 -0500204void SkXMLStreamWriter::onAddAttributeLen(const char name[], const char value[], size_t length) {
fmalitafe3f2602015-02-03 17:47:12 -0800205 SkASSERT(!fElems.top()->fHasChildren && !fElems.top()->fHasText);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000206 fStream.writeText(" ");
207 fStream.writeText(name);
208 fStream.writeText("=\"");
209 fStream.write(value, length);
210 fStream.writeText("\"");
211}
212
reede73da402015-02-04 18:29:27 -0800213void SkXMLStreamWriter::onAddText(const char text[], size_t length) {
fmalitafe3f2602015-02-03 17:47:12 -0800214 Elem* elem = fElems.top();
215
216 if (!elem->fHasChildren && !elem->fHasText) {
217 fStream.writeText(">");
218 fStream.newline();
219 }
220
221 tab(fStream, fElems.count() + 1);
reede73da402015-02-04 18:29:27 -0800222 fStream.write(text, length);
fmalitafe3f2602015-02-03 17:47:12 -0800223 fStream.newline();
224}
225
Mike Reedc3063e52017-01-07 16:16:02 -0500226void SkXMLStreamWriter::onEndElement() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000227 Elem* elem = getEnd();
Mike Reedc3063e52017-01-07 16:16:02 -0500228 if (elem->fHasChildren || elem->fHasText) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000229 tab(fStream, fElems.count());
230 fStream.writeText("</");
231 fStream.writeText(elem->fName.c_str());
232 fStream.writeText(">");
fmalitafe3f2602015-02-03 17:47:12 -0800233 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000234 fStream.writeText("/>");
fmalitafe3f2602015-02-03 17:47:12 -0800235 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000236 fStream.newline();
237 doEnd(elem);
238}
239
Mike Reedc3063e52017-01-07 16:16:02 -0500240void SkXMLStreamWriter::onStartElementLen(const char name[], size_t length) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000241 int level = fElems.count();
Mike Reedc3063e52017-01-07 16:16:02 -0500242 if (this->doStart(name, length)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000243 // the first child, need to close with >
244 fStream.writeText(">");
245 fStream.newline();
246 }
247
248 tab(fStream, level);
249 fStream.writeText("<");
250 fStream.write(name, length);
251}
252
Mike Reedc3063e52017-01-07 16:16:02 -0500253void SkXMLStreamWriter::writeHeader() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000254 const char* header = getHeader();
255 fStream.write(header, strlen(header));
256 fStream.newline();
257}
258
259////////////////////////////////////////////////////////////////////////////////////////////////
260
261#include "SkXMLParser.h"
262
263SkXMLParserWriter::SkXMLParserWriter(SkXMLParser* parser)
264 : SkXMLWriter(false), fParser(*parser)
265{
266}
267
Mike Reedc3063e52017-01-07 16:16:02 -0500268SkXMLParserWriter::~SkXMLParserWriter() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000269 this->flush();
270}
271
Mike Reedc3063e52017-01-07 16:16:02 -0500272void SkXMLParserWriter::onAddAttributeLen(const char name[], const char value[], size_t length) {
fmalitafe3f2602015-02-03 17:47:12 -0800273 SkASSERT(fElems.count() == 0 || (!fElems.top()->fHasChildren && !fElems.top()->fHasText));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000274 SkString str(value, length);
275 fParser.addAttribute(name, str.c_str());
276}
277
reede73da402015-02-04 18:29:27 -0800278void SkXMLParserWriter::onAddText(const char text[], size_t length) {
279 fParser.text(text, SkToInt(length));
fmalitafe3f2602015-02-03 17:47:12 -0800280}
281
Mike Reedc3063e52017-01-07 16:16:02 -0500282void SkXMLParserWriter::onEndElement() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000283 Elem* elem = this->getEnd();
284 fParser.endElement(elem->fName.c_str());
285 this->doEnd(elem);
286}
287
Mike Reedc3063e52017-01-07 16:16:02 -0500288void SkXMLParserWriter::onStartElementLen(const char name[], size_t length) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000289 (void)this->doStart(name, length);
290 SkString str(name, length);
291 fParser.startElement(str.c_str());
292}