1 /*
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 
8 #include "SkXMLWriter.h"
9 #include "SkStream.h"
10 
SkXMLWriter(bool doEscapeMarkup)11 SkXMLWriter::SkXMLWriter(bool doEscapeMarkup) : fDoEscapeMarkup(doEscapeMarkup)
12 {
13 }
14 
~SkXMLWriter()15 SkXMLWriter::~SkXMLWriter()
16 {
17     SkASSERT(fElems.count() == 0);
18 }
19 
flush()20 void SkXMLWriter::flush()
21 {
22     while (fElems.count())
23         this->endElement();
24 }
25 
addAttribute(const char name[],const char value[])26 void SkXMLWriter::addAttribute(const char name[], const char value[])
27 {
28     this->addAttributeLen(name, value, strlen(value));
29 }
30 
addS32Attribute(const char name[],int32_t value)31 void SkXMLWriter::addS32Attribute(const char name[], int32_t value)
32 {
33     SkString    tmp;
34     tmp.appendS32(value);
35     this->addAttribute(name, tmp.c_str());
36 }
37 
addHexAttribute(const char name[],uint32_t value,int minDigits)38 void SkXMLWriter::addHexAttribute(const char name[], uint32_t value, int minDigits)
39 {
40     SkString    tmp("0x");
41     tmp.appendHex(value, minDigits);
42     this->addAttribute(name, tmp.c_str());
43 }
44 
addScalarAttribute(const char name[],SkScalar value)45 void SkXMLWriter::addScalarAttribute(const char name[], SkScalar value)
46 {
47     SkString    tmp;
48     tmp.appendScalar(value);
49     this->addAttribute(name, tmp.c_str());
50 }
51 
addText(const char text[],size_t length)52 void SkXMLWriter::addText(const char text[], size_t length) {
53     if (fElems.isEmpty()) {
54         return;
55     }
56 
57     this->onAddText(text, length);
58 
59     fElems.top()->fHasText = true;
60 }
61 
doEnd(Elem * elem)62 void SkXMLWriter::doEnd(Elem* elem)
63 {
64     delete elem;
65 }
66 
doStart(const char name[],size_t length)67 bool SkXMLWriter::doStart(const char name[], size_t length)
68 {
69     int level = fElems.count();
70     bool firstChild = level > 0 && !fElems[level-1]->fHasChildren;
71     if (firstChild)
72         fElems[level-1]->fHasChildren = true;
73     Elem** elem = fElems.push();
74     *elem = new Elem(name, length);
75     return firstChild;
76 }
77 
getEnd()78 SkXMLWriter::Elem* SkXMLWriter::getEnd()
79 {
80     Elem* elem;
81     fElems.pop(&elem);
82     return elem;
83 }
84 
getHeader()85 const char* SkXMLWriter::getHeader()
86 {
87     static const char gHeader[] = "<?xml version=\"1.0\" encoding=\"utf-8\" ?>";
88     return gHeader;
89 }
90 
startElement(const char name[])91 void SkXMLWriter::startElement(const char name[])
92 {
93     this->startElementLen(name, strlen(name));
94 }
95 
escape_char(char c,char storage[2])96 static const char* escape_char(char c, char storage[2])
97 {
98     static const char* gEscapeChars[] = {
99         "<&lt;",
100         ">&gt;",
101         //"\"&quot;",
102         //"'&apos;",
103         "&&amp;"
104     };
105 
106     const char** array = gEscapeChars;
107     for (unsigned i = 0; i < SK_ARRAY_COUNT(gEscapeChars); i++)
108     {
109         if (array[i][0] == c)
110             return &array[i][1];
111     }
112     storage[0] = c;
113     storage[1] = 0;
114     return storage;
115 }
116 
escape_markup(char dst[],const char src[],size_t length)117 static size_t escape_markup(char dst[], const char src[], size_t length)
118 {
119     size_t      extra = 0;
120     const char* stop = src + length;
121 
122     while (src < stop)
123     {
124         char        orig[2];
125         const char* seq = escape_char(*src, orig);
126         size_t      seqSize = strlen(seq);
127 
128         if (dst)
129         {
130             memcpy(dst, seq, seqSize);
131             dst += seqSize;
132         }
133 
134         // now record the extra size needed
135         extra += seqSize - 1;   // minus one to subtract the original char
136 
137         // bump to the next src char
138         src += 1;
139     }
140     return extra;
141 }
142 
addAttributeLen(const char name[],const char value[],size_t length)143 void SkXMLWriter::addAttributeLen(const char name[], const char value[], size_t length)
144 {
145     SkString valueStr;
146 
147     if (fDoEscapeMarkup)
148     {
149         size_t   extra = escape_markup(nullptr, value, length);
150         if (extra)
151         {
152             valueStr.resize(length + extra);
153             (void)escape_markup(valueStr.writable_str(), value, length);
154             value = valueStr.c_str();
155             length += extra;
156         }
157     }
158     this->onAddAttributeLen(name, value, length);
159 }
160 
startElementLen(const char elem[],size_t length)161 void SkXMLWriter::startElementLen(const char elem[], size_t length)
162 {
163     this->onStartElementLen(elem, length);
164 }
165 
166 ////////////////////////////////////////////////////////////////////////////////////////
167 
write_dom(const SkDOM & dom,const SkDOM::Node * node,SkXMLWriter * w,bool skipRoot)168 static void write_dom(const SkDOM& dom, const SkDOM::Node* node, SkXMLWriter* w, bool skipRoot)
169 {
170     if (!skipRoot)
171     {
172         const char* elem = dom.getName(node);
173         if (dom.getType(node) == SkDOM::kText_Type) {
174             SkASSERT(dom.countChildren(node) == 0);
175             w->addText(elem, strlen(elem));
176             return;
177         }
178 
179         w->startElement(elem);
180 
181         SkDOM::AttrIter iter(dom, node);
182         const char* name;
183         const char* value;
184         while ((name = iter.next(&value)) != nullptr)
185             w->addAttribute(name, value);
186     }
187 
188     node = dom.getFirstChild(node, nullptr);
189     while (node)
190     {
191         write_dom(dom, node, w, false);
192         node = dom.getNextSibling(node, nullptr);
193     }
194 
195     if (!skipRoot)
196         w->endElement();
197 }
198 
writeDOM(const SkDOM & dom,const SkDOM::Node * node,bool skipRoot)199 void SkXMLWriter::writeDOM(const SkDOM& dom, const SkDOM::Node* node, bool skipRoot)
200 {
201     if (node)
202         write_dom(dom, node, this, skipRoot);
203 }
204 
writeHeader()205 void SkXMLWriter::writeHeader()
206 {
207 }
208 
209 // SkXMLStreamWriter
210 
tab(SkWStream & stream,int level)211 static void tab(SkWStream& stream, int level)
212 {
213     for (int i = 0; i < level; i++)
214         stream.writeText("\t");
215 }
216 
SkXMLStreamWriter(SkWStream * stream)217 SkXMLStreamWriter::SkXMLStreamWriter(SkWStream* stream) : fStream(*stream)
218 {
219 }
220 
~SkXMLStreamWriter()221 SkXMLStreamWriter::~SkXMLStreamWriter()
222 {
223     this->flush();
224 }
225 
onAddAttributeLen(const char name[],const char value[],size_t length)226 void SkXMLStreamWriter::onAddAttributeLen(const char name[], const char value[], size_t length)
227 {
228     SkASSERT(!fElems.top()->fHasChildren && !fElems.top()->fHasText);
229     fStream.writeText(" ");
230     fStream.writeText(name);
231     fStream.writeText("=\"");
232     fStream.write(value, length);
233     fStream.writeText("\"");
234 }
235 
onAddText(const char text[],size_t length)236 void SkXMLStreamWriter::onAddText(const char text[], size_t length) {
237     Elem* elem = fElems.top();
238 
239     if (!elem->fHasChildren && !elem->fHasText) {
240         fStream.writeText(">");
241         fStream.newline();
242     }
243 
244     tab(fStream, fElems.count() + 1);
245     fStream.write(text, length);
246     fStream.newline();
247 }
248 
onEndElement()249 void SkXMLStreamWriter::onEndElement()
250 {
251     Elem* elem = getEnd();
252     if (elem->fHasChildren || elem->fHasText)
253     {
254         tab(fStream, fElems.count());
255         fStream.writeText("</");
256         fStream.writeText(elem->fName.c_str());
257         fStream.writeText(">");
258     } else {
259         fStream.writeText("/>");
260     }
261     fStream.newline();
262     doEnd(elem);
263 }
264 
onStartElementLen(const char name[],size_t length)265 void SkXMLStreamWriter::onStartElementLen(const char name[], size_t length)
266 {
267     int level = fElems.count();
268     if (this->doStart(name, length))
269     {
270         // the first child, need to close with >
271         fStream.writeText(">");
272         fStream.newline();
273     }
274 
275     tab(fStream, level);
276     fStream.writeText("<");
277     fStream.write(name, length);
278 }
279 
writeHeader()280 void SkXMLStreamWriter::writeHeader()
281 {
282     const char* header = getHeader();
283     fStream.write(header, strlen(header));
284     fStream.newline();
285 }
286 
287 ////////////////////////////////////////////////////////////////////////////////////////////////
288 
289 #include "SkXMLParser.h"
290 
SkXMLParserWriter(SkXMLParser * parser)291 SkXMLParserWriter::SkXMLParserWriter(SkXMLParser* parser)
292     : SkXMLWriter(false), fParser(*parser)
293 {
294 }
295 
~SkXMLParserWriter()296 SkXMLParserWriter::~SkXMLParserWriter()
297 {
298     this->flush();
299 }
300 
onAddAttributeLen(const char name[],const char value[],size_t length)301 void SkXMLParserWriter::onAddAttributeLen(const char name[], const char value[], size_t length)
302 {
303     SkASSERT(fElems.count() == 0 || (!fElems.top()->fHasChildren && !fElems.top()->fHasText));
304     SkString str(value, length);
305     fParser.addAttribute(name, str.c_str());
306 }
307 
onAddText(const char text[],size_t length)308 void SkXMLParserWriter::onAddText(const char text[], size_t length) {
309     fParser.text(text, SkToInt(length));
310 }
311 
onEndElement()312 void SkXMLParserWriter::onEndElement()
313 {
314     Elem* elem = this->getEnd();
315     fParser.endElement(elem->fName.c_str());
316     this->doEnd(elem);
317 }
318 
onStartElementLen(const char name[],size_t length)319 void SkXMLParserWriter::onStartElementLen(const char name[], size_t length)
320 {
321     (void)this->doStart(name, length);
322     SkString str(name, length);
323     fParser.startElement(str.c_str());
324 }
325 
326 
327 ////////////////////////////////////////////////////////////////////////////////////////
328 ////////////////////////////////////////////////////////////////////////////////////////
329 
330 #ifdef SK_DEBUG
331 
UnitTest()332 void SkXMLStreamWriter::UnitTest()
333 {
334 #ifdef SK_SUPPORT_UNITTEST
335     SkDebugWStream  s;
336     SkXMLStreamWriter       w(&s);
337 
338     w.startElement("elem0");
339     w.addAttribute("hello", "world");
340     w.addS32Attribute("dec", 42);
341     w.addHexAttribute("hex", 0x42, 3);
342     w.addScalarAttribute("scalar", -4.2f);
343     w.startElement("elem1");
344         w.endElement();
345         w.startElement("elem1");
346         w.addAttribute("name", "value");
347         w.endElement();
348         w.startElement("elem1");
349             w.startElement("elem2");
350                 w.startElement("elem3");
351                 w.addAttribute("name", "value");
352                 w.endElement();
353             w.endElement();
354             w.startElement("elem2");
355             w.endElement();
356         w.endElement();
357     w.endElement();
358 #endif
359 }
360 
361 #endif
362