Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 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 | #ifndef SkSVGAttributeParser_DEFINED |
| 9 | #define SkSVGAttributeParser_DEFINED |
| 10 | |
| 11 | #include "include/private/SkNoncopyable.h" |
| 12 | #include "modules/svg/include/SkSVGTypes.h" |
Tyler Denniston | 5715499 | 2020-11-04 16:08:30 -0500 | [diff] [blame] | 13 | #include "src/core/SkTLazy.h" |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 14 | |
| 15 | class SkSVGAttributeParser : public SkNoncopyable { |
| 16 | public: |
| 17 | SkSVGAttributeParser(const char[]); |
| 18 | |
| 19 | bool parseColor(SkSVGColorType*); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 20 | bool parseNumber(SkSVGNumberType*); |
Tyler Denniston | dada960 | 2020-11-03 10:04:25 -0500 | [diff] [blame] | 21 | bool parseInteger(SkSVGIntegerType*); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 22 | bool parseViewBox(SkSVGViewBoxType*); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 23 | bool parsePoints(SkSVGPointsType*); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 24 | bool parseStopColor(SkSVGStopColor*); |
Florin Malita | 385e744 | 2020-10-21 16:55:46 -0400 | [diff] [blame] | 25 | bool parsePreserveAspectRatio(SkSVGPreserveAspectRatio*); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 26 | |
Tyler Denniston | 5715499 | 2020-11-04 16:08:30 -0500 | [diff] [blame] | 27 | // TODO: Migrate all parse*() functions to this style (and delete the old version) |
| 28 | // so they can be used by parse<T>(): |
| 29 | bool parse(SkSVGNumberType* v) { return parseNumber(v); } |
| 30 | bool parse(SkSVGIntegerType* v) { return parseInteger(v); } |
| 31 | |
| 32 | template <typename T> using ParseResult = SkTLazy<T>; |
| 33 | |
Tyler Denniston | a0a5146 | 2020-11-10 13:13:28 -0500 | [diff] [blame] | 34 | template <typename T> static ParseResult<T> parse(const char* value) { |
| 35 | ParseResult<T> result; |
| 36 | T parsedValue; |
| 37 | if (SkSVGAttributeParser(value).parse(&parsedValue)) { |
| 38 | result.set(std::move(parsedValue)); |
| 39 | } |
| 40 | return result; |
| 41 | } |
| 42 | |
Tyler Denniston | 5715499 | 2020-11-04 16:08:30 -0500 | [diff] [blame] | 43 | template <typename T> |
| 44 | static ParseResult<T> parse(const char* expectedName, |
| 45 | const char* name, |
Florin Malita | 401321d | 2020-11-05 11:27:24 -0500 | [diff] [blame] | 46 | const char* value) { |
Florin Malita | 401321d | 2020-11-05 11:27:24 -0500 | [diff] [blame] | 47 | if (!strcmp(name, expectedName)) { |
Tyler Denniston | a0a5146 | 2020-11-10 13:13:28 -0500 | [diff] [blame] | 48 | return parse<T>(value); |
Tyler Denniston | 5715499 | 2020-11-04 16:08:30 -0500 | [diff] [blame] | 49 | } |
| 50 | |
Tyler Denniston | a0a5146 | 2020-11-10 13:13:28 -0500 | [diff] [blame] | 51 | return ParseResult<T>(); |
Tyler Denniston | 5715499 | 2020-11-04 16:08:30 -0500 | [diff] [blame] | 52 | } |
| 53 | |
Tyler Denniston | 79832e3 | 2020-11-18 09:34:07 -0500 | [diff] [blame^] | 54 | template <typename PropertyT> |
| 55 | static ParseResult<PropertyT> parseProperty(const char* expectedName, |
| 56 | const char* name, |
| 57 | const char* value) { |
| 58 | if (strcmp(name, expectedName) != 0) { |
| 59 | return ParseResult<PropertyT>(); |
| 60 | } |
| 61 | |
| 62 | if (!strcmp(value, "inherit")) { |
| 63 | PropertyT result(SkSVGPropertyState::kInherit); |
| 64 | return ParseResult<PropertyT>(&result); |
| 65 | } |
| 66 | |
| 67 | auto pr = parse<typename PropertyT::ValueT>(value); |
| 68 | if (pr.isValid()) { |
| 69 | PropertyT result(*pr); |
| 70 | return ParseResult<PropertyT>(&result); |
| 71 | } |
| 72 | |
| 73 | return ParseResult<PropertyT>(); |
| 74 | } |
Tyler Denniston | a0a5146 | 2020-11-10 13:13:28 -0500 | [diff] [blame] | 75 | |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 76 | private: |
| 77 | // Stack-only |
| 78 | void* operator new(size_t) = delete; |
| 79 | void* operator new(size_t, void*) = delete; |
| 80 | |
Florin Malita | 401321d | 2020-11-05 11:27:24 -0500 | [diff] [blame] | 81 | template <typename T> |
| 82 | bool parse(T*); |
| 83 | |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 84 | template <typename F> |
| 85 | bool advanceWhile(F func); |
| 86 | |
| 87 | bool parseWSToken(); |
Florin Malita | 401321d | 2020-11-05 11:27:24 -0500 | [diff] [blame] | 88 | bool parseEOSToken(); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 89 | bool parseSepToken(); |
Florin Malita | 401321d | 2020-11-05 11:27:24 -0500 | [diff] [blame] | 90 | bool parseCommaWspToken(); |
| 91 | bool parseExpectedStringToken(const char*); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 92 | bool parseScalarToken(SkScalar*); |
Tyler Denniston | dada960 | 2020-11-03 10:04:25 -0500 | [diff] [blame] | 93 | bool parseInt32Token(int32_t*); |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 94 | 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*); |
| 100 | bool parseFuncIRI(SkSVGStringType*); |
| 101 | |
| 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 Malita | 39fe8c8 | 2020-10-20 10:43:03 -0400 | [diff] [blame] | 115 | template <typename T, typename TArray> |
Tyler Denniston | 70bb18d | 2020-11-06 12:07:53 -0500 | [diff] [blame] | 116 | bool parseEnumMap(const TArray& arr, T* result) { |
| 117 | for (size_t i = 0; i < SK_ARRAY_COUNT(arr); ++i) { |
| 118 | if (this->parseExpectedStringToken(std::get<0>(arr[i]))) { |
| 119 | *result = std::get<1>(arr[i]); |
| 120 | return true; |
| 121 | } |
| 122 | } |
| 123 | return false; |
| 124 | } |
Florin Malita | 39fe8c8 | 2020-10-20 10:43:03 -0400 | [diff] [blame] | 125 | |
Florin Malita | b341810 | 2020-10-15 18:10:29 -0400 | [diff] [blame] | 126 | // The current position in the input string. |
| 127 | const char* fCurPos; |
| 128 | |
| 129 | using INHERITED = SkNoncopyable; |
| 130 | }; |
| 131 | |
| 132 | #endif // SkSVGAttributeParser_DEFINED |