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