blob: 15ed0c7ec71867bee749bf4cdcd982247fb97964 [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
Tyler Denniston209857c2021-01-27 14:22:19 -050085 bool matchStringToken(const char* token, const char** newPos = nullptr) const;
86
Florin Malitab3418102020-10-15 18:10:29 -040087 bool parseWSToken();
Florin Malita401321d2020-11-05 11:27:24 -050088 bool parseEOSToken();
Florin Malitab3418102020-10-15 18:10:29 -040089 bool parseSepToken();
Florin Malita401321d2020-11-05 11:27:24 -050090 bool parseCommaWspToken();
91 bool parseExpectedStringToken(const char*);
Florin Malitab3418102020-10-15 18:10:29 -040092 bool parseScalarToken(SkScalar*);
Tyler Dennistondada9602020-11-03 10:04:25 -050093 bool parseInt32Token(int32_t*);
Florin Malitab3418102020-10-15 18:10:29 -040094 bool parseHexToken(uint32_t*);
95 bool parseLengthUnitToken(SkSVGLength::Unit*);
96 bool parseNamedColorToken(SkColor*);
97 bool parseHexColorToken(SkColor*);
98 bool parseColorComponentToken(int32_t*);
99 bool parseRGBColorToken(SkColor*);
Tyler Dennistone71f5472021-01-27 13:30:59 -0500100 bool parseFuncIRI(SkSVGFuncIRI*);
Florin Malitab3418102020-10-15 18:10:29 -0400101
102 // Transform helpers
103 bool parseMatrixToken(SkMatrix*);
104 bool parseTranslateToken(SkMatrix*);
105 bool parseScaleToken(SkMatrix*);
106 bool parseRotateToken(SkMatrix*);
107 bool parseSkewXToken(SkMatrix*);
108 bool parseSkewYToken(SkMatrix*);
109
110 // Parses a sequence of 'WS* <prefix> WS* (<nested>)', where the nested sequence
111 // is handled by the passed functor.
112 template <typename Func, typename T>
113 bool parseParenthesized(const char* prefix, Func, T* result);
114
Florin Malita2d059fc2021-01-05 11:53:15 -0500115 template <typename T>
116 bool parseList(std::vector<T>*);
117
Florin Malita39fe8c82020-10-20 10:43:03 -0400118 template <typename T, typename TArray>
Tyler Denniston70bb18d2020-11-06 12:07:53 -0500119 bool parseEnumMap(const TArray& arr, T* result) {
120 for (size_t i = 0; i < SK_ARRAY_COUNT(arr); ++i) {
121 if (this->parseExpectedStringToken(std::get<0>(arr[i]))) {
122 *result = std::get<1>(arr[i]);
123 return true;
124 }
125 }
126 return false;
127 }
Florin Malita39fe8c82020-10-20 10:43:03 -0400128
Florin Malitab3418102020-10-15 18:10:29 -0400129 // The current position in the input string.
130 const char* fCurPos;
131
132 using INHERITED = SkNoncopyable;
133};
134
135#endif // SkSVGAttributeParser_DEFINED