blob: ecaab8af22af49382607f102109da05c0c540f40 [file] [log] [blame]
Florin Malitab3418102020-10-15 18:10:29 -04001/*
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#ifndef SkSVGAttributeParser_DEFINED
9#define SkSVGAttributeParser_DEFINED
10
Florin Malita2d059fc2021-01-05 11:53:15 -050011#include <vector>
12
Florin Malitab3418102020-10-15 18:10:29 -040013#include "include/private/SkNoncopyable.h"
14#include "modules/svg/include/SkSVGTypes.h"
Tyler Denniston57154992020-11-04 16:08:30 -050015#include "src/core/SkTLazy.h"
Florin Malitab3418102020-10-15 18:10:29 -040016
17class SkSVGAttributeParser : public SkNoncopyable {
18public:
19 SkSVGAttributeParser(const char[]);
20
Tyler Dennistondada9602020-11-03 10:04:25 -050021 bool parseInteger(SkSVGIntegerType*);
Florin Malitab3418102020-10-15 18:10:29 -040022 bool parseViewBox(SkSVGViewBoxType*);
Florin Malitab3418102020-10-15 18:10:29 -040023 bool parsePoints(SkSVGPointsType*);
Florin Malita385e7442020-10-21 16:55:46 -040024 bool parsePreserveAspectRatio(SkSVGPreserveAspectRatio*);
Florin Malitab3418102020-10-15 18:10:29 -040025
Tyler Denniston57154992020-11-04 16:08:30 -050026 // TODO: Migrate all parse*() functions to this style (and delete the old version)
27 // so they can be used by parse<T>():
Tyler Denniston57154992020-11-04 16:08:30 -050028 bool parse(SkSVGIntegerType* v) { return parseInteger(v); }
29
30 template <typename T> using ParseResult = SkTLazy<T>;
31
Tyler Dennistona0a51462020-11-10 13:13:28 -050032 template <typename T> static ParseResult<T> parse(const char* value) {
33 ParseResult<T> result;
34 T parsedValue;
35 if (SkSVGAttributeParser(value).parse(&parsedValue)) {
36 result.set(std::move(parsedValue));
37 }
38 return result;
39 }
40
Tyler Denniston57154992020-11-04 16:08:30 -050041 template <typename T>
42 static ParseResult<T> parse(const char* expectedName,
43 const char* name,
Florin Malita401321d2020-11-05 11:27:24 -050044 const char* value) {
Florin Malita401321d2020-11-05 11:27:24 -050045 if (!strcmp(name, expectedName)) {
Tyler Dennistona0a51462020-11-10 13:13:28 -050046 return parse<T>(value);
Tyler Denniston57154992020-11-04 16:08:30 -050047 }
48
Tyler Dennistona0a51462020-11-10 13:13:28 -050049 return ParseResult<T>();
Tyler Denniston57154992020-11-04 16:08:30 -050050 }
51
Tyler Denniston79832e32020-11-18 09:34:07 -050052 template <typename PropertyT>
53 static ParseResult<PropertyT> parseProperty(const char* expectedName,
54 const char* name,
55 const char* value) {
56 if (strcmp(name, expectedName) != 0) {
57 return ParseResult<PropertyT>();
58 }
59
60 if (!strcmp(value, "inherit")) {
61 PropertyT result(SkSVGPropertyState::kInherit);
62 return ParseResult<PropertyT>(&result);
63 }
64
65 auto pr = parse<typename PropertyT::ValueT>(value);
66 if (pr.isValid()) {
67 PropertyT result(*pr);
68 return ParseResult<PropertyT>(&result);
69 }
70
71 return ParseResult<PropertyT>();
72 }
Tyler Dennistona0a51462020-11-10 13:13:28 -050073
Florin Malitab3418102020-10-15 18:10:29 -040074private:
75 // Stack-only
76 void* operator new(size_t) = delete;
77 void* operator new(size_t, void*) = delete;
78
Florin Malita401321d2020-11-05 11:27:24 -050079 template <typename T>
80 bool parse(T*);
81
Florin Malitab3418102020-10-15 18:10:29 -040082 template <typename F>
83 bool advanceWhile(F func);
84
85 bool parseWSToken();
Florin Malita401321d2020-11-05 11:27:24 -050086 bool parseEOSToken();
Florin Malitab3418102020-10-15 18:10:29 -040087 bool parseSepToken();
Florin Malita401321d2020-11-05 11:27:24 -050088 bool parseCommaWspToken();
89 bool parseExpectedStringToken(const char*);
Florin Malitab3418102020-10-15 18:10:29 -040090 bool parseScalarToken(SkScalar*);
Tyler Dennistondada9602020-11-03 10:04:25 -050091 bool parseInt32Token(int32_t*);
Florin Malitab3418102020-10-15 18:10:29 -040092 bool parseHexToken(uint32_t*);
93 bool parseLengthUnitToken(SkSVGLength::Unit*);
94 bool parseNamedColorToken(SkColor*);
95 bool parseHexColorToken(SkColor*);
96 bool parseColorComponentToken(int32_t*);
97 bool parseRGBColorToken(SkColor*);
98 bool parseFuncIRI(SkSVGStringType*);
99
100 // Transform helpers
101 bool parseMatrixToken(SkMatrix*);
102 bool parseTranslateToken(SkMatrix*);
103 bool parseScaleToken(SkMatrix*);
104 bool parseRotateToken(SkMatrix*);
105 bool parseSkewXToken(SkMatrix*);
106 bool parseSkewYToken(SkMatrix*);
107
108 // Parses a sequence of 'WS* <prefix> WS* (<nested>)', where the nested sequence
109 // is handled by the passed functor.
110 template <typename Func, typename T>
111 bool parseParenthesized(const char* prefix, Func, T* result);
112
Florin Malita2d059fc2021-01-05 11:53:15 -0500113 template <typename T>
114 bool parseList(std::vector<T>*);
115
Florin Malita39fe8c82020-10-20 10:43:03 -0400116 template <typename T, typename TArray>
Tyler Denniston70bb18d2020-11-06 12:07:53 -0500117 bool parseEnumMap(const TArray& arr, T* result) {
118 for (size_t i = 0; i < SK_ARRAY_COUNT(arr); ++i) {
119 if (this->parseExpectedStringToken(std::get<0>(arr[i]))) {
120 *result = std::get<1>(arr[i]);
121 return true;
122 }
123 }
124 return false;
125 }
Florin Malita39fe8c82020-10-20 10:43:03 -0400126
Florin Malitab3418102020-10-15 18:10:29 -0400127 // The current position in the input string.
128 const char* fCurPos;
129
130 using INHERITED = SkNoncopyable;
131};
132
133#endif // SkSVGAttributeParser_DEFINED