1 /*
2 * Copyright 2016 Google Inc.
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 "SkCanvas.h"
9 #include "SkDOM.h"
10 #include "SkParsePath.h"
11 #include "SkString.h"
12 #include "SkSVGAttributeParser.h"
13 #include "SkSVGCircle.h"
14 #include "SkSVGClipPath.h"
15 #include "SkSVGDefs.h"
16 #include "SkSVGDOM.h"
17 #include "SkSVGEllipse.h"
18 #include "SkSVGG.h"
19 #include "SkSVGLine.h"
20 #include "SkSVGLinearGradient.h"
21 #include "SkSVGNode.h"
22 #include "SkSVGPath.h"
23 #include "SkSVGPoly.h"
24 #include "SkSVGRect.h"
25 #include "SkSVGRenderContext.h"
26 #include "SkSVGStop.h"
27 #include "SkSVGSVG.h"
28 #include "SkSVGTypes.h"
29 #include "SkSVGValue.h"
30 #include "SkTSearch.h"
31
32 namespace {
33
SetPaintAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)34 bool SetPaintAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
35 const char* stringValue) {
36 SkSVGPaint paint;
37 SkSVGAttributeParser parser(stringValue);
38 if (!parser.parsePaint(&paint)) {
39 return false;
40 }
41
42 node->setAttribute(attr, SkSVGPaintValue(paint));
43 return true;
44 }
45
SetColorAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)46 bool SetColorAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
47 const char* stringValue) {
48 SkSVGColorType color;
49 SkSVGAttributeParser parser(stringValue);
50 if (!parser.parseColor(&color)) {
51 return false;
52 }
53
54 node->setAttribute(attr, SkSVGColorValue(color));
55 return true;
56 }
57
SetIRIAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)58 bool SetIRIAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
59 const char* stringValue) {
60 SkSVGStringType iri;
61 SkSVGAttributeParser parser(stringValue);
62 if (!parser.parseIRI(&iri)) {
63 return false;
64 }
65
66 node->setAttribute(attr, SkSVGStringValue(iri));
67 return true;
68 }
69
SetClipPathAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)70 bool SetClipPathAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
71 const char* stringValue) {
72 SkSVGClip clip;
73 SkSVGAttributeParser parser(stringValue);
74 if (!parser.parseClipPath(&clip)) {
75 return false;
76 }
77
78 node->setAttribute(attr, SkSVGClipValue(clip));
79 return true;
80 }
81
82
SetPathDataAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)83 bool SetPathDataAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
84 const char* stringValue) {
85 SkPath path;
86 if (!SkParsePath::FromSVGString(stringValue, &path)) {
87 return false;
88 }
89
90 node->setAttribute(attr, SkSVGPathValue(path));
91 return true;
92 }
93
SetTransformAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)94 bool SetTransformAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
95 const char* stringValue) {
96 SkSVGTransformType transform;
97 SkSVGAttributeParser parser(stringValue);
98 if (!parser.parseTransform(&transform)) {
99 return false;
100 }
101
102 node->setAttribute(attr, SkSVGTransformValue(transform));
103 return true;
104 }
105
SetLengthAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)106 bool SetLengthAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
107 const char* stringValue) {
108 SkSVGLength length;
109 SkSVGAttributeParser parser(stringValue);
110 if (!parser.parseLength(&length)) {
111 return false;
112 }
113
114 node->setAttribute(attr, SkSVGLengthValue(length));
115 return true;
116 }
117
SetNumberAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)118 bool SetNumberAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
119 const char* stringValue) {
120 SkSVGNumberType number;
121 SkSVGAttributeParser parser(stringValue);
122 if (!parser.parseNumber(&number)) {
123 return false;
124 }
125
126 node->setAttribute(attr, SkSVGNumberValue(number));
127 return true;
128 }
129
SetViewBoxAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)130 bool SetViewBoxAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
131 const char* stringValue) {
132 SkSVGViewBoxType viewBox;
133 SkSVGAttributeParser parser(stringValue);
134 if (!parser.parseViewBox(&viewBox)) {
135 return false;
136 }
137
138 node->setAttribute(attr, SkSVGViewBoxValue(viewBox));
139 return true;
140 }
141
SetLineCapAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)142 bool SetLineCapAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
143 const char* stringValue) {
144 SkSVGLineCap lineCap;
145 SkSVGAttributeParser parser(stringValue);
146 if (!parser.parseLineCap(&lineCap)) {
147 return false;
148 }
149
150 node->setAttribute(attr, SkSVGLineCapValue(lineCap));
151 return true;
152 }
153
SetLineJoinAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)154 bool SetLineJoinAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
155 const char* stringValue) {
156 SkSVGLineJoin lineJoin;
157 SkSVGAttributeParser parser(stringValue);
158 if (!parser.parseLineJoin(&lineJoin)) {
159 return false;
160 }
161
162 node->setAttribute(attr, SkSVGLineJoinValue(lineJoin));
163 return true;
164 }
165
SetSpreadMethodAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)166 bool SetSpreadMethodAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
167 const char* stringValue) {
168 SkSVGSpreadMethod spread;
169 SkSVGAttributeParser parser(stringValue);
170 if (!parser.parseSpreadMethod(&spread)) {
171 return false;
172 }
173
174 node->setAttribute(attr, SkSVGSpreadMethodValue(spread));
175 return true;
176 }
177
SetPointsAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)178 bool SetPointsAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
179 const char* stringValue) {
180 SkSVGPointsType points;
181 SkSVGAttributeParser parser(stringValue);
182 if (!parser.parsePoints(&points)) {
183 return false;
184 }
185
186 node->setAttribute(attr, SkSVGPointsValue(points));
187 return true;
188 }
189
SetFillRuleAttribute(const sk_sp<SkSVGNode> & node,SkSVGAttribute attr,const char * stringValue)190 bool SetFillRuleAttribute(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr,
191 const char* stringValue) {
192 SkSVGFillRule fillRule;
193 SkSVGAttributeParser parser(stringValue);
194 if (!parser.parseFillRule(&fillRule)) {
195 return false;
196 }
197
198 node->setAttribute(attr, SkSVGFillRuleValue(fillRule));
199 return true;
200 }
201
TrimmedString(const char * first,const char * last)202 SkString TrimmedString(const char* first, const char* last) {
203 SkASSERT(first);
204 SkASSERT(last);
205 SkASSERT(first <= last);
206
207 while (first <= last && *first <= ' ') { first++; }
208 while (first <= last && *last <= ' ') { last--; }
209
210 SkASSERT(last - first + 1 >= 0);
211 return SkString(first, SkTo<size_t>(last - first + 1));
212 }
213
214 // Breaks a "foo: bar; baz: ..." string into key:value pairs.
215 class StyleIterator {
216 public:
StyleIterator(const char * str)217 StyleIterator(const char* str) : fPos(str) { }
218
next()219 std::tuple<SkString, SkString> next() {
220 SkString name, value;
221
222 if (fPos) {
223 const char* sep = this->nextSeparator();
224 SkASSERT(*sep == ';' || *sep == '\0');
225
226 const char* valueSep = strchr(fPos, ':');
227 if (valueSep && valueSep < sep) {
228 name = TrimmedString(fPos, valueSep - 1);
229 value = TrimmedString(valueSep + 1, sep - 1);
230 }
231
232 fPos = *sep ? sep + 1 : nullptr;
233 }
234
235 return std::make_tuple(name, value);
236 }
237
238 private:
nextSeparator() const239 const char* nextSeparator() const {
240 const char* sep = fPos;
241 while (*sep != ';' && *sep != '\0') {
242 sep++;
243 }
244 return sep;
245 }
246
247 const char* fPos;
248 };
249
250 void set_string_attribute(const sk_sp<SkSVGNode>& node, const char* name, const char* value);
251
SetStyleAttributes(const sk_sp<SkSVGNode> & node,SkSVGAttribute,const char * stringValue)252 bool SetStyleAttributes(const sk_sp<SkSVGNode>& node, SkSVGAttribute,
253 const char* stringValue) {
254
255 SkString name, value;
256 StyleIterator iter(stringValue);
257 for (;;) {
258 std::tie(name, value) = iter.next();
259 if (name.isEmpty()) {
260 break;
261 }
262 set_string_attribute(node, name.c_str(), value.c_str());
263 }
264
265 return true;
266 }
267
268 template<typename T>
269 struct SortedDictionaryEntry {
270 const char* fKey;
271 const T fValue;
272 };
273
274 struct AttrParseInfo {
275 SkSVGAttribute fAttr;
276 bool (*fSetter)(const sk_sp<SkSVGNode>& node, SkSVGAttribute attr, const char* stringValue);
277 };
278
279 SortedDictionaryEntry<AttrParseInfo> gAttributeParseInfo[] = {
280 { "clip-path" , { SkSVGAttribute::kClipPath , SetClipPathAttribute }},
281 { "cx" , { SkSVGAttribute::kCx , SetLengthAttribute }},
282 { "cy" , { SkSVGAttribute::kCy , SetLengthAttribute }},
283 { "d" , { SkSVGAttribute::kD , SetPathDataAttribute }},
284 { "fill" , { SkSVGAttribute::kFill , SetPaintAttribute }},
285 { "fill-opacity" , { SkSVGAttribute::kFillOpacity , SetNumberAttribute }},
286 { "fill-rule" , { SkSVGAttribute::kFillRule , SetFillRuleAttribute }},
287 { "gradientTransform", { SkSVGAttribute::kGradientTransform, SetTransformAttribute }},
288 { "height" , { SkSVGAttribute::kHeight , SetLengthAttribute }},
289 { "offset" , { SkSVGAttribute::kOffset , SetLengthAttribute }},
290 { "opacity" , { SkSVGAttribute::kOpacity , SetNumberAttribute }},
291 { "points" , { SkSVGAttribute::kPoints , SetPointsAttribute }},
292 { "r" , { SkSVGAttribute::kR , SetLengthAttribute }},
293 { "rx" , { SkSVGAttribute::kRx , SetLengthAttribute }},
294 { "ry" , { SkSVGAttribute::kRy , SetLengthAttribute }},
295 { "spreadMethod" , { SkSVGAttribute::kSpreadMethod , SetSpreadMethodAttribute }},
296 { "stop-color" , { SkSVGAttribute::kStopColor , SetColorAttribute }},
297 { "stop-opacity" , { SkSVGAttribute::kStopOpacity , SetNumberAttribute }},
298 { "stroke" , { SkSVGAttribute::kStroke , SetPaintAttribute }},
299 { "stroke-linecap" , { SkSVGAttribute::kStrokeLineCap , SetLineCapAttribute }},
300 { "stroke-linejoin" , { SkSVGAttribute::kStrokeLineJoin , SetLineJoinAttribute }},
301 { "stroke-opacity" , { SkSVGAttribute::kStrokeOpacity , SetNumberAttribute }},
302 { "stroke-width" , { SkSVGAttribute::kStrokeWidth , SetLengthAttribute }},
303 { "style" , { SkSVGAttribute::kUnknown , SetStyleAttributes }},
304 { "transform" , { SkSVGAttribute::kTransform , SetTransformAttribute }},
305 { "viewBox" , { SkSVGAttribute::kViewBox , SetViewBoxAttribute }},
306 { "width" , { SkSVGAttribute::kWidth , SetLengthAttribute }},
307 { "x" , { SkSVGAttribute::kX , SetLengthAttribute }},
308 { "x1" , { SkSVGAttribute::kX1 , SetLengthAttribute }},
309 { "x2" , { SkSVGAttribute::kX2 , SetLengthAttribute }},
310 { "xlink:href" , { SkSVGAttribute::kHref , SetIRIAttribute }},
311 { "y" , { SkSVGAttribute::kY , SetLengthAttribute }},
312 { "y1" , { SkSVGAttribute::kY1 , SetLengthAttribute }},
313 { "y2" , { SkSVGAttribute::kY2 , SetLengthAttribute }},
314 };
315
316 SortedDictionaryEntry<sk_sp<SkSVGNode>(*)()> gTagFactories[] = {
__anonfa75a9830202() 317 { "circle" , []() -> sk_sp<SkSVGNode> { return SkSVGCircle::Make(); }},
__anonfa75a9830302() 318 { "clipPath" , []() -> sk_sp<SkSVGNode> { return SkSVGClipPath::Make(); }},
__anonfa75a9830402() 319 { "defs" , []() -> sk_sp<SkSVGNode> { return SkSVGDefs::Make(); }},
__anonfa75a9830502() 320 { "ellipse" , []() -> sk_sp<SkSVGNode> { return SkSVGEllipse::Make(); }},
__anonfa75a9830602() 321 { "g" , []() -> sk_sp<SkSVGNode> { return SkSVGG::Make(); }},
__anonfa75a9830702() 322 { "line" , []() -> sk_sp<SkSVGNode> { return SkSVGLine::Make(); }},
__anonfa75a9830802() 323 { "linearGradient", []() -> sk_sp<SkSVGNode> { return SkSVGLinearGradient::Make(); }},
__anonfa75a9830902() 324 { "path" , []() -> sk_sp<SkSVGNode> { return SkSVGPath::Make(); }},
__anonfa75a9830a02() 325 { "polygon" , []() -> sk_sp<SkSVGNode> { return SkSVGPoly::MakePolygon(); }},
__anonfa75a9830b02() 326 { "polyline" , []() -> sk_sp<SkSVGNode> { return SkSVGPoly::MakePolyline(); }},
__anonfa75a9830c02() 327 { "rect" , []() -> sk_sp<SkSVGNode> { return SkSVGRect::Make(); }},
__anonfa75a9830d02() 328 { "stop" , []() -> sk_sp<SkSVGNode> { return SkSVGStop::Make(); }},
__anonfa75a9830e02() 329 { "svg" , []() -> sk_sp<SkSVGNode> { return SkSVGSVG::Make(); }},
330 };
331
332 struct ConstructionContext {
ConstructionContext__anonfa75a9830111::ConstructionContext333 ConstructionContext(SkSVGIDMapper* mapper) : fParent(nullptr), fIDMapper(mapper) {}
ConstructionContext__anonfa75a9830111::ConstructionContext334 ConstructionContext(const ConstructionContext& other, const sk_sp<SkSVGNode>& newParent)
335 : fParent(newParent.get()), fIDMapper(other.fIDMapper) {}
336
337 const SkSVGNode* fParent;
338 SkSVGIDMapper* fIDMapper;
339 };
340
set_string_attribute(const sk_sp<SkSVGNode> & node,const char * name,const char * value)341 void set_string_attribute(const sk_sp<SkSVGNode>& node, const char* name, const char* value) {
342 const int attrIndex = SkStrSearch(&gAttributeParseInfo[0].fKey,
343 SkTo<int>(SK_ARRAY_COUNT(gAttributeParseInfo)),
344 name, sizeof(gAttributeParseInfo[0]));
345 if (attrIndex < 0) {
346 #if defined(SK_VERBOSE_SVG_PARSING)
347 SkDebugf("unhandled attribute: %s\n", name);
348 #endif
349 return;
350 }
351
352 SkASSERT(SkTo<size_t>(attrIndex) < SK_ARRAY_COUNT(gAttributeParseInfo));
353 const auto& attrInfo = gAttributeParseInfo[attrIndex].fValue;
354 if (!attrInfo.fSetter(node, attrInfo.fAttr, value)) {
355 #if defined(SK_VERBOSE_SVG_PARSING)
356 SkDebugf("could not parse attribute: '%s=\"%s\"'\n", name, value);
357 #endif
358 }
359 }
360
parse_node_attributes(const SkDOM & xmlDom,const SkDOM::Node * xmlNode,const sk_sp<SkSVGNode> & svgNode,SkSVGIDMapper * mapper)361 void parse_node_attributes(const SkDOM& xmlDom, const SkDOM::Node* xmlNode,
362 const sk_sp<SkSVGNode>& svgNode, SkSVGIDMapper* mapper) {
363 const char* name, *value;
364 SkDOM::AttrIter attrIter(xmlDom, xmlNode);
365 while ((name = attrIter.next(&value))) {
366 // We're handling id attributes out of band for now.
367 if (!strcmp(name, "id")) {
368 mapper->set(SkString(value), svgNode);
369 continue;
370 }
371 set_string_attribute(svgNode, name, value);
372 }
373 }
374
construct_svg_node(const SkDOM & dom,const ConstructionContext & ctx,const SkDOM::Node * xmlNode)375 sk_sp<SkSVGNode> construct_svg_node(const SkDOM& dom, const ConstructionContext& ctx,
376 const SkDOM::Node* xmlNode) {
377 const char* elem = dom.getName(xmlNode);
378 const SkDOM::Type elemType = dom.getType(xmlNode);
379
380 if (elemType == SkDOM::kText_Type) {
381 SkASSERT(dom.countChildren(xmlNode) == 0);
382 // TODO: text handling
383 return nullptr;
384 }
385
386 SkASSERT(elemType == SkDOM::kElement_Type);
387
388 const int tagIndex = SkStrSearch(&gTagFactories[0].fKey,
389 SkTo<int>(SK_ARRAY_COUNT(gTagFactories)),
390 elem, sizeof(gTagFactories[0]));
391 if (tagIndex < 0) {
392 #if defined(SK_VERBOSE_SVG_PARSING)
393 SkDebugf("unhandled element: <%s>\n", elem);
394 #endif
395 return nullptr;
396 }
397
398 SkASSERT(SkTo<size_t>(tagIndex) < SK_ARRAY_COUNT(gTagFactories));
399 sk_sp<SkSVGNode> node = gTagFactories[tagIndex].fValue();
400 parse_node_attributes(dom, xmlNode, node, ctx.fIDMapper);
401
402 ConstructionContext localCtx(ctx, node);
403 for (auto* child = dom.getFirstChild(xmlNode, nullptr); child;
404 child = dom.getNextSibling(child)) {
405 sk_sp<SkSVGNode> childNode = construct_svg_node(dom, localCtx, child);
406 if (childNode) {
407 node->appendChild(std::move(childNode));
408 }
409 }
410
411 return node;
412 }
413
414 } // anonymous namespace
415
SkSVGDOM()416 SkSVGDOM::SkSVGDOM()
417 : fContainerSize(SkSize::Make(0, 0)) {
418 }
419
MakeFromDOM(const SkDOM & xmlDom)420 sk_sp<SkSVGDOM> SkSVGDOM::MakeFromDOM(const SkDOM& xmlDom) {
421 sk_sp<SkSVGDOM> dom = sk_make_sp<SkSVGDOM>();
422
423 ConstructionContext ctx(&dom->fIDMapper);
424 dom->fRoot = construct_svg_node(xmlDom, ctx, xmlDom.getRootNode());
425
426 // Reset the default container size to match the intrinsic SVG size.
427 dom->setContainerSize(dom->intrinsicSize());
428
429 return dom;
430 }
431
MakeFromStream(SkStream & svgStream)432 sk_sp<SkSVGDOM> SkSVGDOM::MakeFromStream(SkStream& svgStream) {
433 SkDOM xmlDom;
434 if (!xmlDom.build(svgStream)) {
435 return nullptr;
436 }
437
438 return MakeFromDOM(xmlDom);
439 }
440
render(SkCanvas * canvas) const441 void SkSVGDOM::render(SkCanvas* canvas) const {
442 if (fRoot) {
443 SkSVGRenderContext ctx(canvas,
444 fIDMapper,
445 SkSVGLengthContext(fContainerSize),
446 SkSVGPresentationContext());
447 fRoot->render(ctx);
448 }
449 }
450
intrinsicSize() const451 SkSize SkSVGDOM::intrinsicSize() const {
452 if (!fRoot || fRoot->tag() != SkSVGTag::kSvg) {
453 return SkSize::Make(0, 0);
454 }
455
456 // Intrinsic sizes are never relative, so the viewport size is irrelevant.
457 const SkSVGLengthContext lctx(SkSize::Make(0, 0));
458 return static_cast<const SkSVGSVG*>(fRoot.get())->intrinsicSize(lctx);
459 }
460
containerSize() const461 const SkSize& SkSVGDOM::containerSize() const {
462 return fContainerSize;
463 }
464
setContainerSize(const SkSize & containerSize)465 void SkSVGDOM::setContainerSize(const SkSize& containerSize) {
466 // TODO: inval
467 fContainerSize = containerSize;
468 }
469
setRoot(sk_sp<SkSVGNode> root)470 void SkSVGDOM::setRoot(sk_sp<SkSVGNode> root) {
471 fRoot = std::move(root);
472 }
473