blob: 53f1c8803799f0f993b5b2ef25781f9c0e0d04dc [file] [log] [blame]
Florin Malita094ccde2017-12-30 12:27:00 -05001/*
2 * Copyright 2017 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
Florin Malita54f65c42018-01-16 17:04:30 -05008#include "Skottie.h"
Florin Malita094ccde2017-12-30 12:27:00 -05009
10#include "SkCanvas.h"
Florin Malita54f65c42018-01-16 17:04:30 -050011#include "SkottieAnimator.h"
Florin Malitacf8ed522018-01-25 15:27:33 -050012#include "SkottieParser.h"
Florin Malita54f65c42018-01-16 17:04:30 -050013#include "SkottieProperties.h"
Florin Malita094ccde2017-12-30 12:27:00 -050014#include "SkData.h"
Florin Malita49328072018-01-08 12:51:12 -050015#include "SkImage.h"
Florin Malita094ccde2017-12-30 12:27:00 -050016#include "SkMakeUnique.h"
Florin Malita49328072018-01-08 12:51:12 -050017#include "SkOSPath.h"
Florin Malita094ccde2017-12-30 12:27:00 -050018#include "SkPaint.h"
Florin Malita0e66fba2018-01-09 17:10:18 -050019#include "SkParse.h"
Florin Malita094ccde2017-12-30 12:27:00 -050020#include "SkPath.h"
21#include "SkPoint.h"
22#include "SkSGColor.h"
23#include "SkSGDraw.h"
Florin Malita16d0ad02018-01-19 15:07:29 -050024#include "SkSGGeometryTransform.h"
Florin Malita6aaee592018-01-12 12:25:09 -050025#include "SkSGGradient.h"
Florin Malita094ccde2017-12-30 12:27:00 -050026#include "SkSGGroup.h"
Florin Malita49328072018-01-08 12:51:12 -050027#include "SkSGImage.h"
28#include "SkSGInvalidationController.h"
Florin Malita5f9102f2018-01-10 13:36:22 -050029#include "SkSGMaskEffect.h"
Florin Malitae6345d92018-01-03 23:37:54 -050030#include "SkSGMerge.h"
Florin Malitac0034172018-01-08 16:42:59 -050031#include "SkSGOpacityEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050032#include "SkSGPath.h"
Florin Malita2e1d7e22018-01-02 10:40:00 -050033#include "SkSGRect.h"
Florin Malita35efaa82018-01-22 12:57:06 -050034#include "SkSGScene.h"
Florin Malita094ccde2017-12-30 12:27:00 -050035#include "SkSGTransform.h"
Florin Malita51b8c892018-01-07 08:54:24 -050036#include "SkSGTrimEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050037#include "SkStream.h"
38#include "SkTArray.h"
39#include "SkTHash.h"
40
41#include <cmath>
Florin Malita18eafd92018-01-04 21:11:55 -050042#include <unordered_map>
Florin Malitae6345d92018-01-03 23:37:54 -050043#include <vector>
44
Florin Malita094ccde2017-12-30 12:27:00 -050045#include "stdlib.h"
46
Florin Malita54f65c42018-01-16 17:04:30 -050047namespace skottie {
Florin Malita094ccde2017-12-30 12:27:00 -050048
Florin Malitacf8ed522018-01-25 15:27:33 -050049#define LOG SkDebugf
50
Florin Malita094ccde2017-12-30 12:27:00 -050051namespace {
52
53using AssetMap = SkTHashMap<SkString, const Json::Value*>;
54
55struct AttachContext {
Florin Malita35efaa82018-01-22 12:57:06 -050056 const ResourceProvider& fResources;
57 const AssetMap& fAssets;
58 sksg::Scene::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050059};
60
61bool LogFail(const Json::Value& json, const char* msg) {
62 const auto dump = json.toStyledString();
63 LOG("!! %s: %s", msg, dump.c_str());
64 return false;
65}
66
67// This is the workhorse for binding properties: depending on whether the property is animated,
68// it will either apply immediately or instantiate and attach a keyframe animator.
Florin Malita2518a0a2018-01-24 18:29:00 -050069template <typename T>
70bool BindProperty(const Json::Value& jprop, AttachContext* ctx,
71 typename Animator<T>::ApplyFuncT&& apply) {
Florin Malita094ccde2017-12-30 12:27:00 -050072 if (!jprop.isObject())
73 return false;
74
Florin Malita95448a92018-01-08 10:15:12 -050075 const auto& jpropA = jprop["a"];
76 const auto& jpropK = jprop["k"];
77
78 // Older Json versions don't have an "a" animation marker.
79 // For those, we attempt to parse both ways.
Florin Malitacf8ed522018-01-25 15:27:33 -050080 if (!ParseDefault(jpropA, false)) {
Florin Malita2518a0a2018-01-24 18:29:00 -050081 T val;
Florin Malitacf8ed522018-01-25 15:27:33 -050082 if (Parse<T>(jpropK, &val)) {
Florin Malita95448a92018-01-08 10:15:12 -050083 // Static property.
Florin Malita2518a0a2018-01-24 18:29:00 -050084 apply(val);
Florin Malita95448a92018-01-08 10:15:12 -050085 return true;
Florin Malita094ccde2017-12-30 12:27:00 -050086 }
87
Florin Malita95448a92018-01-08 10:15:12 -050088 if (!jpropA.isNull()) {
89 return LogFail(jprop, "Could not parse (explicit) static property");
Florin Malita094ccde2017-12-30 12:27:00 -050090 }
Florin Malita094ccde2017-12-30 12:27:00 -050091 }
92
Florin Malita95448a92018-01-08 10:15:12 -050093 // Keyframe property.
Florin Malita2518a0a2018-01-24 18:29:00 -050094 auto animator = Animator<T>::Make(jpropK, std::move(apply));
Florin Malita95448a92018-01-08 10:15:12 -050095
96 if (!animator) {
97 return LogFail(jprop, "Could not parse keyframed property");
98 }
99
100 ctx->fAnimators.push_back(std::move(animator));
101
Florin Malita094ccde2017-12-30 12:27:00 -0500102 return true;
103}
104
Florin Malita18eafd92018-01-04 21:11:55 -0500105sk_sp<sksg::Matrix> AttachMatrix(const Json::Value& t, AttachContext* ctx,
106 sk_sp<sksg::Matrix> parentMatrix) {
107 if (!t.isObject())
108 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500109
Florin Malita18eafd92018-01-04 21:11:55 -0500110 auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
111 auto composite = sk_make_sp<CompositeTransform>(matrix);
Florin Malita2518a0a2018-01-24 18:29:00 -0500112 auto anchor_attached = BindProperty<VectorValue>(t["a"], ctx,
113 [composite](const VectorValue& a) {
114 composite->setAnchorPoint(ValueTraits<VectorValue>::As<SkPoint>(a));
Florin Malita094ccde2017-12-30 12:27:00 -0500115 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500116 auto position_attached = BindProperty<VectorValue>(t["p"], ctx,
117 [composite](const VectorValue& p) {
118 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malita094ccde2017-12-30 12:27:00 -0500119 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500120 auto scale_attached = BindProperty<VectorValue>(t["s"], ctx,
121 [composite](const VectorValue& s) {
122 composite->setScale(ValueTraits<VectorValue>::As<SkVector>(s));
Florin Malita094ccde2017-12-30 12:27:00 -0500123 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500124 auto rotation_attached = BindProperty<ScalarValue>(t["r"], ctx,
125 [composite](const ScalarValue& r) {
126 composite->setRotation(r);
Florin Malita094ccde2017-12-30 12:27:00 -0500127 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500128 auto skew_attached = BindProperty<ScalarValue>(t["sk"], ctx,
129 [composite](const ScalarValue& sk) {
130 composite->setSkew(sk);
Florin Malita094ccde2017-12-30 12:27:00 -0500131 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500132 auto skewaxis_attached = BindProperty<ScalarValue>(t["sa"], ctx,
133 [composite](const ScalarValue& sa) {
134 composite->setSkewAxis(sa);
Florin Malita094ccde2017-12-30 12:27:00 -0500135 });
136
137 if (!anchor_attached &&
138 !position_attached &&
139 !scale_attached &&
140 !rotation_attached &&
141 !skew_attached &&
142 !skewaxis_attached) {
143 LogFail(t, "Could not parse transform");
Florin Malita18eafd92018-01-04 21:11:55 -0500144 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500145 }
146
Florin Malita18eafd92018-01-04 21:11:55 -0500147 return matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500148}
149
Florin Malitac0034172018-01-08 16:42:59 -0500150sk_sp<sksg::RenderNode> AttachOpacity(const Json::Value& jtransform, AttachContext* ctx,
151 sk_sp<sksg::RenderNode> childNode) {
152 if (!jtransform.isObject() || !childNode)
153 return childNode;
154
155 // This is more peeky than other attachers, because we want to avoid redundant opacity
156 // nodes for the extremely common case of static opaciy == 100.
157 const auto& opacity = jtransform["o"];
158 if (opacity.isObject() &&
Florin Malitacf8ed522018-01-25 15:27:33 -0500159 !ParseDefault(opacity["a"], true) &&
160 ParseDefault(opacity["k"], -1) == 100) {
Florin Malitac0034172018-01-08 16:42:59 -0500161 // Ignoring static full opacity.
162 return childNode;
163 }
164
165 auto opacityNode = sksg::OpacityEffect::Make(childNode);
Florin Malita2518a0a2018-01-24 18:29:00 -0500166 BindProperty<ScalarValue>(opacity, ctx,
167 [opacityNode](const ScalarValue& o) {
Florin Malitac0034172018-01-08 16:42:59 -0500168 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500169 opacityNode->setOpacity(o * 0.01f);
Florin Malitac0034172018-01-08 16:42:59 -0500170 });
171
172 return opacityNode;
173}
174
Florin Malita094ccde2017-12-30 12:27:00 -0500175sk_sp<sksg::RenderNode> AttachComposition(const Json::Value&, AttachContext* ctx);
176
Florin Malita25366fa2018-01-23 13:37:59 -0500177sk_sp<sksg::Path> AttachPath(const Json::Value& jpath, AttachContext* ctx) {
178 auto path_node = sksg::Path::Make();
Florin Malita2518a0a2018-01-24 18:29:00 -0500179 return BindProperty<ShapeValue>(jpath, ctx,
180 [path_node](const ShapeValue& p) { path_node->setPath(p); })
Florin Malita25366fa2018-01-23 13:37:59 -0500181 ? path_node
182 : nullptr;
183}
184
Florin Malita094ccde2017-12-30 12:27:00 -0500185sk_sp<sksg::GeometryNode> AttachPathGeometry(const Json::Value& jpath, AttachContext* ctx) {
186 SkASSERT(jpath.isObject());
187
Florin Malita25366fa2018-01-23 13:37:59 -0500188 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500189}
190
Florin Malita2e1d7e22018-01-02 10:40:00 -0500191sk_sp<sksg::GeometryNode> AttachRRectGeometry(const Json::Value& jrect, AttachContext* ctx) {
192 SkASSERT(jrect.isObject());
193
194 auto rect_node = sksg::RRect::Make();
195 auto composite = sk_make_sp<CompositeRRect>(rect_node);
196
Florin Malita2518a0a2018-01-24 18:29:00 -0500197 auto p_attached = BindProperty<VectorValue>(jrect["p"], ctx,
198 [composite](const VectorValue& p) {
199 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500200 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500201 auto s_attached = BindProperty<VectorValue>(jrect["s"], ctx,
202 [composite](const VectorValue& s) {
203 composite->setSize(ValueTraits<VectorValue>::As<SkSize>(s));
Florin Malitaf9590922018-01-09 11:56:09 -0500204 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500205 auto r_attached = BindProperty<ScalarValue>(jrect["r"], ctx,
206 [composite](const ScalarValue& r) {
207 composite->setRadius(SkSize::Make(r, r));
Florin Malitaf9590922018-01-09 11:56:09 -0500208 });
Florin Malita2e1d7e22018-01-02 10:40:00 -0500209
210 if (!p_attached && !s_attached && !r_attached) {
211 return nullptr;
212 }
213
Florin Malitafbc13f12018-01-04 10:26:35 -0500214 LOG("** Attached (r)rect geometry\n");
215
216 return rect_node;
217}
218
219sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const Json::Value& jellipse, AttachContext* ctx) {
220 SkASSERT(jellipse.isObject());
221
222 auto rect_node = sksg::RRect::Make();
223 auto composite = sk_make_sp<CompositeRRect>(rect_node);
224
Florin Malita2518a0a2018-01-24 18:29:00 -0500225 auto p_attached = BindProperty<VectorValue>(jellipse["p"], ctx,
226 [composite](const VectorValue& p) {
227 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500228 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500229 auto s_attached = BindProperty<VectorValue>(jellipse["s"], ctx,
230 [composite](const VectorValue& s) {
Florin Malitaf9590922018-01-09 11:56:09 -0500231 const auto sz = ValueTraits<VectorValue>::As<SkSize>(s);
Florin Malita2518a0a2018-01-24 18:29:00 -0500232 composite->setSize(sz);
233 composite->setRadius(SkSize::Make(sz.width() / 2, sz.height() / 2));
Florin Malitaf9590922018-01-09 11:56:09 -0500234 });
Florin Malitafbc13f12018-01-04 10:26:35 -0500235
236 if (!p_attached && !s_attached) {
237 return nullptr;
238 }
239
240 LOG("** Attached ellipse geometry\n");
241
Florin Malita2e1d7e22018-01-02 10:40:00 -0500242 return rect_node;
243}
244
Florin Malita02a32b02018-01-04 11:27:09 -0500245sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const Json::Value& jstar, AttachContext* ctx) {
246 SkASSERT(jstar.isObject());
247
248 static constexpr CompositePolyStar::Type gTypes[] = {
249 CompositePolyStar::Type::kStar, // "sy": 1
250 CompositePolyStar::Type::kPoly, // "sy": 2
251 };
252
Florin Malitacf8ed522018-01-25 15:27:33 -0500253 const auto type = ParseDefault(jstar["sy"], 0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500254 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
255 LogFail(jstar, "Unknown polystar type");
256 return nullptr;
257 }
258
259 auto path_node = sksg::Path::Make();
260 auto composite = sk_make_sp<CompositePolyStar>(path_node, gTypes[type]);
261
Florin Malita2518a0a2018-01-24 18:29:00 -0500262 BindProperty<VectorValue>(jstar["p"], ctx,
263 [composite](const VectorValue& p) {
264 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500265 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500266 BindProperty<ScalarValue>(jstar["pt"], ctx,
267 [composite](const ScalarValue& pt) {
268 composite->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500269 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500270 BindProperty<ScalarValue>(jstar["ir"], ctx,
271 [composite](const ScalarValue& ir) {
272 composite->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500273 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500274 BindProperty<ScalarValue>(jstar["or"], ctx,
275 [composite](const ScalarValue& otr) {
276 composite->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500277 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500278 BindProperty<ScalarValue>(jstar["is"], ctx,
279 [composite](const ScalarValue& is) {
280 composite->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500281 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500282 BindProperty<ScalarValue>(jstar["os"], ctx,
283 [composite](const ScalarValue& os) {
284 composite->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500285 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500286 BindProperty<ScalarValue>(jstar["r"], ctx,
287 [composite](const ScalarValue& r) {
288 composite->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500289 });
Florin Malita02a32b02018-01-04 11:27:09 -0500290
291 return path_node;
292}
293
Florin Malita6aaee592018-01-12 12:25:09 -0500294sk_sp<sksg::Color> AttachColor(const Json::Value& obj, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500295 SkASSERT(obj.isObject());
296
297 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malita2518a0a2018-01-24 18:29:00 -0500298 auto color_attached = BindProperty<VectorValue>(obj["c"], ctx,
299 [color_node](const VectorValue& c) {
300 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500301 });
Florin Malita094ccde2017-12-30 12:27:00 -0500302
Florin Malita1586d852018-01-12 14:27:39 -0500303 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500304}
305
Florin Malita6aaee592018-01-12 12:25:09 -0500306sk_sp<sksg::Gradient> AttachGradient(const Json::Value& obj, AttachContext* ctx) {
307 SkASSERT(obj.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500308
Florin Malita6aaee592018-01-12 12:25:09 -0500309 const auto& stops = obj["g"];
310 if (!stops.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500311 return nullptr;
312
Florin Malitacf8ed522018-01-25 15:27:33 -0500313 const auto stopCount = ParseDefault(stops["p"], -1);
Florin Malita6aaee592018-01-12 12:25:09 -0500314 if (stopCount < 0)
315 return nullptr;
316
317 sk_sp<sksg::Gradient> gradient_node;
318 sk_sp<CompositeGradient> composite;
319
Florin Malitacf8ed522018-01-25 15:27:33 -0500320 if (ParseDefault(obj["t"], 1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500321 auto linear_node = sksg::LinearGradient::Make();
322 composite = sk_make_sp<CompositeLinearGradient>(linear_node, stopCount);
323 gradient_node = std::move(linear_node);
324 } else {
325 auto radial_node = sksg::RadialGradient::Make();
326 composite = sk_make_sp<CompositeRadialGradient>(radial_node, stopCount);
327
328 // TODO: highlight, angle
329 gradient_node = std::move(radial_node);
330 }
331
Florin Malita2518a0a2018-01-24 18:29:00 -0500332 BindProperty<VectorValue>(stops["k"], ctx,
333 [composite](const VectorValue& stops) {
334 composite->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500335 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500336 BindProperty<VectorValue>(obj["s"], ctx,
337 [composite](const VectorValue& s) {
338 composite->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500339 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500340 BindProperty<VectorValue>(obj["e"], ctx,
341 [composite](const VectorValue& e) {
342 composite->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500343 });
344
345 return gradient_node;
346}
347
Florin Malita1586d852018-01-12 14:27:39 -0500348sk_sp<sksg::PaintNode> AttachPaint(const Json::Value& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500349 sk_sp<sksg::PaintNode> paint_node) {
350 if (paint_node) {
351 paint_node->setAntiAlias(true);
352
Florin Malita2518a0a2018-01-24 18:29:00 -0500353 BindProperty<ScalarValue>(jpaint["o"], ctx,
354 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500355 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500356 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500357 });
Florin Malita6aaee592018-01-12 12:25:09 -0500358 }
359
360 return paint_node;
361}
362
363sk_sp<sksg::PaintNode> AttachStroke(const Json::Value& jstroke, AttachContext* ctx,
364 sk_sp<sksg::PaintNode> stroke_node) {
365 SkASSERT(jstroke.isObject());
366
367 if (!stroke_node)
368 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500369
370 stroke_node->setStyle(SkPaint::kStroke_Style);
371
Florin Malita2518a0a2018-01-24 18:29:00 -0500372 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], ctx,
373 [stroke_node](const ScalarValue& w) {
374 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500375 });
Florin Malita094ccde2017-12-30 12:27:00 -0500376 if (!width_attached)
377 return nullptr;
378
Florin Malitacf8ed522018-01-25 15:27:33 -0500379 stroke_node->setStrokeMiter(ParseDefault(jstroke["ml"], 4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500380
381 static constexpr SkPaint::Join gJoins[] = {
382 SkPaint::kMiter_Join,
383 SkPaint::kRound_Join,
384 SkPaint::kBevel_Join,
385 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500386 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(ParseDefault(jstroke["lj"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500387 0, SK_ARRAY_COUNT(gJoins) - 1)]);
388
389 static constexpr SkPaint::Cap gCaps[] = {
390 SkPaint::kButt_Cap,
391 SkPaint::kRound_Cap,
392 SkPaint::kSquare_Cap,
393 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500394 stroke_node->setStrokeCap(gCaps[SkTPin<int>(ParseDefault(jstroke["lc"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500395 0, SK_ARRAY_COUNT(gCaps) - 1)]);
396
397 return stroke_node;
398}
399
Florin Malita6aaee592018-01-12 12:25:09 -0500400sk_sp<sksg::PaintNode> AttachColorFill(const Json::Value& jfill, AttachContext* ctx) {
401 SkASSERT(jfill.isObject());
402
403 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
404}
405
406sk_sp<sksg::PaintNode> AttachGradientFill(const Json::Value& jfill, AttachContext* ctx) {
407 SkASSERT(jfill.isObject());
408
409 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
410}
411
412sk_sp<sksg::PaintNode> AttachColorStroke(const Json::Value& jstroke, AttachContext* ctx) {
413 SkASSERT(jstroke.isObject());
414
415 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
416}
417
418sk_sp<sksg::PaintNode> AttachGradientStroke(const Json::Value& jstroke, AttachContext* ctx) {
419 SkASSERT(jstroke.isObject());
420
421 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
422}
423
Florin Malitae6345d92018-01-03 23:37:54 -0500424std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
425 const Json::Value& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
426 std::vector<sk_sp<sksg::GeometryNode>> merged;
427
428 static constexpr sksg::Merge::Mode gModes[] = {
429 sksg::Merge::Mode::kMerge, // "mm": 1
430 sksg::Merge::Mode::kUnion, // "mm": 2
431 sksg::Merge::Mode::kDifference, // "mm": 3
432 sksg::Merge::Mode::kIntersect, // "mm": 4
433 sksg::Merge::Mode::kXOR , // "mm": 5
434 };
435
Florin Malitacf8ed522018-01-25 15:27:33 -0500436 const auto mode = gModes[SkTPin<int>(ParseDefault(jmerge["mm"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500437 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malitae6345d92018-01-03 23:37:54 -0500438 merged.push_back(sksg::Merge::Make(std::move(geos), mode));
439
440 LOG("** Attached merge path effect, mode: %d\n", mode);
441
442 return merged;
443}
444
Florin Malita51b8c892018-01-07 08:54:24 -0500445std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
446 const Json::Value& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
447
448 enum class Mode {
449 kMerged, // "m": 1
450 kSeparate, // "m": 2
451 } gModes[] = { Mode::kMerged, Mode::kSeparate };
452
Florin Malitacf8ed522018-01-25 15:27:33 -0500453 const auto mode = gModes[SkTPin<int>(ParseDefault(jtrim["m"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500454 0, SK_ARRAY_COUNT(gModes) - 1)];
455
456 std::vector<sk_sp<sksg::GeometryNode>> inputs;
457 if (mode == Mode::kMerged) {
458 inputs.push_back(sksg::Merge::Make(std::move(geos), sksg::Merge::Mode::kMerge));
459 } else {
460 inputs = std::move(geos);
461 }
462
463 std::vector<sk_sp<sksg::GeometryNode>> trimmed;
464 trimmed.reserve(inputs.size());
465 for (const auto& i : inputs) {
466 const auto trim = sksg::TrimEffect::Make(i);
467 trimmed.push_back(trim);
Florin Malita2518a0a2018-01-24 18:29:00 -0500468 BindProperty<ScalarValue>(jtrim["s"], ctx,
469 [trim](const ScalarValue& s) {
470 trim->setStart(s * 0.01f);
Florin Malita51b8c892018-01-07 08:54:24 -0500471 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500472 BindProperty<ScalarValue>(jtrim["e"], ctx,
473 [trim](const ScalarValue& e) {
474 trim->setEnd(e * 0.01f);
Florin Malita51b8c892018-01-07 08:54:24 -0500475 });
Florin Malita2518a0a2018-01-24 18:29:00 -0500476 BindProperty<ScalarValue>(jtrim["o"], ctx,
477 [trim](const ScalarValue& o) {
478 trim->setOffset(o / 360);
Florin Malita51b8c892018-01-07 08:54:24 -0500479 });
480 }
481
482 return trimmed;
483}
484
Florin Malita094ccde2017-12-30 12:27:00 -0500485using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const Json::Value&, AttachContext*);
486static constexpr GeometryAttacherT gGeometryAttachers[] = {
487 AttachPathGeometry,
Florin Malita2e1d7e22018-01-02 10:40:00 -0500488 AttachRRectGeometry,
Florin Malitafbc13f12018-01-04 10:26:35 -0500489 AttachEllipseGeometry,
Florin Malita02a32b02018-01-04 11:27:09 -0500490 AttachPolystarGeometry,
Florin Malita094ccde2017-12-30 12:27:00 -0500491};
492
493using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const Json::Value&, AttachContext*);
494static constexpr PaintAttacherT gPaintAttachers[] = {
Florin Malita6aaee592018-01-12 12:25:09 -0500495 AttachColorFill,
496 AttachColorStroke,
497 AttachGradientFill,
498 AttachGradientStroke,
Florin Malita094ccde2017-12-30 12:27:00 -0500499};
500
Florin Malitae6345d92018-01-03 23:37:54 -0500501using GeometryEffectAttacherT =
502 std::vector<sk_sp<sksg::GeometryNode>> (*)(const Json::Value&,
503 AttachContext*,
504 std::vector<sk_sp<sksg::GeometryNode>>&&);
505static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
506 AttachMergeGeometryEffect,
Florin Malita51b8c892018-01-07 08:54:24 -0500507 AttachTrimGeometryEffect,
Florin Malitae6345d92018-01-03 23:37:54 -0500508};
509
Florin Malita094ccde2017-12-30 12:27:00 -0500510enum class ShapeType {
511 kGeometry,
Florin Malitae6345d92018-01-03 23:37:54 -0500512 kGeometryEffect,
Florin Malita094ccde2017-12-30 12:27:00 -0500513 kPaint,
514 kGroup,
Florin Malitadacc02b2017-12-31 09:12:31 -0500515 kTransform,
Florin Malita094ccde2017-12-30 12:27:00 -0500516};
517
518struct ShapeInfo {
519 const char* fTypeString;
520 ShapeType fShapeType;
521 uint32_t fAttacherIndex; // index into respective attacher tables
522};
523
524const ShapeInfo* FindShapeInfo(const Json::Value& shape) {
525 static constexpr ShapeInfo gShapeInfo[] = {
Florin Malitafbc13f12018-01-04 10:26:35 -0500526 { "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500527 { "fl", ShapeType::kPaint , 0 }, // fill -> AttachColorFill
528 { "gf", ShapeType::kPaint , 2 }, // gfill -> AttachGradientFill
Florin Malita16d0ad02018-01-19 15:07:29 -0500529 { "gr", ShapeType::kGroup , 0 }, // group -> Inline handler
Florin Malita6aaee592018-01-12 12:25:09 -0500530 { "gs", ShapeType::kPaint , 3 }, // gstroke -> AttachGradientStroke
Florin Malitae6345d92018-01-03 23:37:54 -0500531 { "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
Florin Malita02a32b02018-01-04 11:27:09 -0500532 { "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
Florin Malitae6345d92018-01-03 23:37:54 -0500533 { "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
Florin Malita02a32b02018-01-04 11:27:09 -0500534 { "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500535 { "st", ShapeType::kPaint , 1 }, // stroke -> AttachColorStroke
Florin Malita51b8c892018-01-07 08:54:24 -0500536 { "tm", ShapeType::kGeometryEffect, 1 }, // trim -> AttachTrimGeometryEffect
Florin Malita16d0ad02018-01-19 15:07:29 -0500537 { "tr", ShapeType::kTransform , 0 }, // transform -> Inline handler
Florin Malita094ccde2017-12-30 12:27:00 -0500538 };
539
540 if (!shape.isObject())
541 return nullptr;
542
543 const auto& type = shape["ty"];
544 if (!type.isString())
545 return nullptr;
546
547 const auto* info = bsearch(type.asCString(),
548 gShapeInfo,
549 SK_ARRAY_COUNT(gShapeInfo),
550 sizeof(ShapeInfo),
551 [](const void* key, const void* info) {
552 return strcmp(static_cast<const char*>(key),
553 static_cast<const ShapeInfo*>(info)->fTypeString);
554 });
555
556 return static_cast<const ShapeInfo*>(info);
557}
558
Florin Malita16d0ad02018-01-19 15:07:29 -0500559struct GeometryEffectRec {
560 const Json::Value& fJson;
561 GeometryEffectAttacherT fAttach;
562};
563
Florin Malitaca4439f2018-01-23 10:31:59 -0500564struct AttachShapeContext {
565 AttachShapeContext(AttachContext* ctx,
566 std::vector<sk_sp<sksg::GeometryNode>>* geos,
567 std::vector<GeometryEffectRec>* effects,
568 size_t committedAnimators)
569 : fCtx(ctx)
570 , fGeometryStack(geos)
571 , fGeometryEffectStack(effects)
572 , fCommittedAnimators(committedAnimators) {}
573
574 AttachContext* fCtx;
575 std::vector<sk_sp<sksg::GeometryNode>>* fGeometryStack;
576 std::vector<GeometryEffectRec>* fGeometryEffectStack;
577 size_t fCommittedAnimators;
578};
579
580sk_sp<sksg::RenderNode> AttachShape(const Json::Value& jshape, AttachShapeContext* shapeCtx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500581 if (!jshape.isArray())
Florin Malita094ccde2017-12-30 12:27:00 -0500582 return nullptr;
583
Florin Malitaca4439f2018-01-23 10:31:59 -0500584 SkDEBUGCODE(const auto initialGeometryEffects = shapeCtx->fGeometryEffectStack->size();)
Florin Malita094ccde2017-12-30 12:27:00 -0500585
Florin Malita16d0ad02018-01-19 15:07:29 -0500586 sk_sp<sksg::Group> shape_group = sksg::Group::Make();
587 sk_sp<sksg::RenderNode> shape_wrapper = shape_group;
588 sk_sp<sksg::Matrix> shape_matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500589
Florin Malita16d0ad02018-01-19 15:07:29 -0500590 struct ShapeRec {
591 const Json::Value& fJson;
592 const ShapeInfo& fInfo;
593 };
594
595 // First pass (bottom->top):
596 //
597 // * pick up the group transform and opacity
598 // * push local geometry effects onto the stack
599 // * store recs for next pass
600 //
601 std::vector<ShapeRec> recs;
602 for (Json::ArrayIndex i = 0; i < jshape.size(); ++i) {
603 const auto& s = jshape[jshape.size() - 1 - i];
Florin Malita094ccde2017-12-30 12:27:00 -0500604 const auto* info = FindShapeInfo(s);
605 if (!info) {
606 LogFail(s.isObject() ? s["ty"] : s, "Unknown shape");
607 continue;
608 }
609
Florin Malita16d0ad02018-01-19 15:07:29 -0500610 recs.push_back({ s, *info });
611
Florin Malita094ccde2017-12-30 12:27:00 -0500612 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500613 case ShapeType::kTransform:
Florin Malitaca4439f2018-01-23 10:31:59 -0500614 if ((shape_matrix = AttachMatrix(s, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500615 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
616 }
Florin Malitaca4439f2018-01-23 10:31:59 -0500617 shape_wrapper = AttachOpacity(s, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500618 break;
619 case ShapeType::kGeometryEffect:
620 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500621 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita16d0ad02018-01-19 15:07:29 -0500622 { s, gGeometryEffectAttachers[info->fAttacherIndex] });
623 break;
624 default:
625 break;
626 }
627 }
628
629 // Second pass (top -> bottom, after 2x reverse):
630 //
631 // * track local geometry
632 // * emit local paints
633 //
634 std::vector<sk_sp<sksg::GeometryNode>> geos;
635 std::vector<sk_sp<sksg::RenderNode >> draws;
636 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
637 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500638 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500639 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500640 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
641 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500642 geos.push_back(std::move(geo));
643 }
644 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500645 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500646 // Apply the current effect and pop from the stack.
647 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500648 if (!geos.empty()) {
649 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500650 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500651 std::move(geos));
652 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500653
Florin Malitaca4439f2018-01-23 10:31:59 -0500654 SkASSERT(shapeCtx->fGeometryEffectStack->back().fJson == rec->fJson);
655 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500656 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500657 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500658 } break;
659 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500660 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
661 &geos,
662 shapeCtx->fGeometryEffectStack,
663 shapeCtx->fCommittedAnimators);
664 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500665 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500666 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
667 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500668 }
669 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500670 case ShapeType::kPaint: {
671 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500672 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500673 if (!paint || geos.empty())
674 break;
675
676 auto drawGeos = geos;
677
678 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500679 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
680 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
681 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500682 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500683
684 // If we still have multiple geos, reduce using 'merge'.
685 auto geo = drawGeos.size() > 1
686 ? sksg::Merge::Make(std::move(drawGeos), sksg::Merge::Mode::kMerge)
687 : drawGeos[0];
688
689 SkASSERT(geo);
690 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500691 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500692 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500693 default:
694 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500695 }
696 }
697
Florin Malita16d0ad02018-01-19 15:07:29 -0500698 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500699 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500700
701 // Push transformed local geometries to parent list, for subsequent paints.
702 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500703 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500704 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
705 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500706 }
707
Florin Malita16d0ad02018-01-19 15:07:29 -0500708 // Emit local draws reversed (bottom->top, per spec).
709 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
710 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500711 }
712
Florin Malita16d0ad02018-01-19 15:07:29 -0500713 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500714}
715
716sk_sp<sksg::RenderNode> AttachCompLayer(const Json::Value& layer, AttachContext* ctx) {
717 SkASSERT(layer.isObject());
718
Florin Malitacf8ed522018-01-25 15:27:33 -0500719 SkString refId;
720 if (!Parse(layer["refId"], &refId) || refId.isEmpty()) {
Florin Malita094ccde2017-12-30 12:27:00 -0500721 LOG("!! Comp layer missing refId\n");
722 return nullptr;
723 }
724
725 const auto* comp = ctx->fAssets.find(refId);
726 if (!comp) {
727 LOG("!! Pre-comp not found: '%s'\n", refId.c_str());
728 return nullptr;
729 }
730
731 // TODO: cycle detection
732 return AttachComposition(**comp, ctx);
733}
734
Florin Malita0e66fba2018-01-09 17:10:18 -0500735sk_sp<sksg::RenderNode> AttachSolidLayer(const Json::Value& jlayer, AttachContext*) {
736 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500737
Florin Malitacf8ed522018-01-25 15:27:33 -0500738 const auto size = SkSize::Make(ParseDefault(jlayer["sw"], 0.0f),
739 ParseDefault(jlayer["sh"], 0.0f));
740 const auto hex = ParseDefault(jlayer["sc"], SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500741 uint32_t c;
742 if (size.isEmpty() ||
743 !hex.startsWith("#") ||
744 !SkParse::FindHex(hex.c_str() + 1, &c)) {
745 LogFail(jlayer, "Could not parse solid layer");
746 return nullptr;
747 }
748
749 const SkColor color = 0xff000000 | c;
750
751 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
752 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500753}
754
Florin Malita49328072018-01-08 12:51:12 -0500755sk_sp<sksg::RenderNode> AttachImageAsset(const Json::Value& jimage, AttachContext* ctx) {
756 SkASSERT(jimage.isObject());
757
Florin Malitacf8ed522018-01-25 15:27:33 -0500758 const auto name = ParseDefault(jimage["p"], SkString()),
759 path = ParseDefault(jimage["u"], SkString());
Florin Malita49328072018-01-08 12:51:12 -0500760 if (name.isEmpty())
761 return nullptr;
762
763 // TODO: plumb resource paths explicitly to ResourceProvider?
764 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
765 const auto resStream = ctx->fResources.openStream(resName.c_str());
766 if (!resStream || !resStream->hasLength()) {
767 LOG("!! Could not load image resource: %s\n", resName.c_str());
768 return nullptr;
769 }
770
771 // TODO: non-intrisic image sizing
772 return sksg::Image::Make(
773 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
774}
775
776sk_sp<sksg::RenderNode> AttachImageLayer(const Json::Value& layer, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500777 SkASSERT(layer.isObject());
778
Florin Malitacf8ed522018-01-25 15:27:33 -0500779 SkString refId;
780 if (!Parse(layer["refId"], &refId) || refId.isEmpty()) {
Florin Malita49328072018-01-08 12:51:12 -0500781 LOG("!! Image layer missing refId\n");
782 return nullptr;
783 }
784
785 const auto* jimage = ctx->fAssets.find(refId);
786 if (!jimage) {
787 LOG("!! Image asset not found: '%s'\n", refId.c_str());
788 return nullptr;
789 }
790
791 return AttachImageAsset(**jimage, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500792}
793
794sk_sp<sksg::RenderNode> AttachNullLayer(const Json::Value& layer, AttachContext*) {
795 SkASSERT(layer.isObject());
796
Florin Malita18eafd92018-01-04 21:11:55 -0500797 // Null layers are used solely to drive dependent transforms,
798 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500799 return nullptr;
800}
801
802sk_sp<sksg::RenderNode> AttachShapeLayer(const Json::Value& layer, AttachContext* ctx) {
803 SkASSERT(layer.isObject());
804
Florin Malita16d0ad02018-01-19 15:07:29 -0500805 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
806 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500807 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
808 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
809
810 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
811 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
812 // due to attached animators. To avoid this, we track committed animators and discard the
813 // orphans here.
814 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
815 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
816
817 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500818}
819
820sk_sp<sksg::RenderNode> AttachTextLayer(const Json::Value& layer, AttachContext*) {
821 SkASSERT(layer.isObject());
822
823 LOG("?? Text layer stub\n");
824 return nullptr;
825}
826
Florin Malita18eafd92018-01-04 21:11:55 -0500827struct AttachLayerContext {
828 AttachLayerContext(const Json::Value& jlayers, AttachContext* ctx)
829 : fLayerList(jlayers), fCtx(ctx) {}
830
831 const Json::Value& fLayerList;
832 AttachContext* fCtx;
833 std::unordered_map<const Json::Value*, sk_sp<sksg::Matrix>> fLayerMatrixCache;
834 std::unordered_map<int, const Json::Value*> fLayerIndexCache;
Florin Malita5f9102f2018-01-10 13:36:22 -0500835 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500836
837 const Json::Value* findLayer(int index) {
838 SkASSERT(fLayerList.isArray());
839
840 if (index < 0) {
841 return nullptr;
842 }
843
844 const auto cached = fLayerIndexCache.find(index);
845 if (cached != fLayerIndexCache.end()) {
846 return cached->second;
847 }
848
849 for (const auto& l : fLayerList) {
850 if (!l.isObject()) {
851 continue;
852 }
853
Florin Malitacf8ed522018-01-25 15:27:33 -0500854 if (ParseDefault(l["ind"], -1) == index) {
Florin Malita18eafd92018-01-04 21:11:55 -0500855 fLayerIndexCache.insert(std::make_pair(index, &l));
856 return &l;
857 }
858 }
859
860 return nullptr;
861 }
862
863 sk_sp<sksg::Matrix> AttachLayerMatrix(const Json::Value& jlayer) {
864 SkASSERT(jlayer.isObject());
865
866 const auto cached = fLayerMatrixCache.find(&jlayer);
867 if (cached != fLayerMatrixCache.end()) {
868 return cached->second;
869 }
870
Florin Malitacf8ed522018-01-25 15:27:33 -0500871 const auto* parentLayer = this->findLayer(ParseDefault(jlayer["parent"], -1));
Florin Malita18eafd92018-01-04 21:11:55 -0500872
873 // TODO: cycle detection?
874 auto parentMatrix = (parentLayer && parentLayer != &jlayer)
875 ? this->AttachLayerMatrix(*parentLayer) : nullptr;
876
877 auto layerMatrix = AttachMatrix(jlayer["ks"], fCtx, std::move(parentMatrix));
878 fLayerMatrixCache.insert(std::make_pair(&jlayer, layerMatrix));
879
880 return layerMatrix;
881 }
882};
883
Florin Malita25366fa2018-01-23 13:37:59 -0500884SkBlendMode MaskBlendMode(char mode) {
885 switch (mode) {
886 case 'a': return SkBlendMode::kSrcOver; // Additive
887 case 's': return SkBlendMode::kExclusion; // Subtract
888 case 'i': return SkBlendMode::kDstIn; // Intersect
889 case 'l': return SkBlendMode::kLighten; // Lighten
890 case 'd': return SkBlendMode::kDarken; // Darken
891 case 'f': return SkBlendMode::kDifference; // Difference
892 default: break;
893 }
894
895 return SkBlendMode::kSrcOver;
896}
897
898sk_sp<sksg::RenderNode> AttachMask(const Json::Value& jmask,
899 AttachContext* ctx,
900 sk_sp<sksg::RenderNode> childNode) {
901 if (!jmask.isArray())
902 return childNode;
903
904 auto mask_group = sksg::Group::Make();
905
906 for (const auto& m : jmask) {
907 if (!m.isObject())
908 continue;
909
Florin Malitacf8ed522018-01-25 15:27:33 -0500910 const auto inverted = ParseDefault(m["inv"], false);
Florin Malita25366fa2018-01-23 13:37:59 -0500911 // TODO
912 if (inverted) {
913 LogFail(m, "Unsupported inverse mask");
914 continue;
915 }
916
917 auto mask_path = AttachPath(m["pt"], ctx);
918 if (!mask_path) {
919 LogFail(m, "Could not parse mask path");
920 continue;
921 }
922
Florin Malitacf8ed522018-01-25 15:27:33 -0500923 SkString mode;
924 if (!Parse(m["mode"], &mode) ||
925 mode.size() != 1 ||
926 !strcmp(mode.c_str(), "n")) { // "None" masks have no effect.
Florin Malita25366fa2018-01-23 13:37:59 -0500927 continue;
Florin Malitacf8ed522018-01-25 15:27:33 -0500928 }
Florin Malita25366fa2018-01-23 13:37:59 -0500929
930 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
931 mask_paint->setBlendMode(MaskBlendMode(mode.c_str()[0]));
Florin Malita2518a0a2018-01-24 18:29:00 -0500932 BindProperty<ScalarValue>(m["o"], ctx,
933 [mask_paint](const ScalarValue& o) { mask_paint->setOpacity(o * 0.01f); });
Florin Malita25366fa2018-01-23 13:37:59 -0500934
935 mask_group->addChild(sksg::Draw::Make(std::move(mask_path), std::move(mask_paint)));
936 }
937
938 return mask_group->empty()
939 ? childNode
940 : sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
941}
942
Florin Malita18eafd92018-01-04 21:11:55 -0500943sk_sp<sksg::RenderNode> AttachLayer(const Json::Value& jlayer,
944 AttachLayerContext* layerCtx) {
945 if (!jlayer.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500946 return nullptr;
947
948 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const Json::Value&, AttachContext*);
949 static constexpr LayerAttacher gLayerAttachers[] = {
950 AttachCompLayer, // 'ty': 0
951 AttachSolidLayer, // 'ty': 1
952 AttachImageLayer, // 'ty': 2
953 AttachNullLayer, // 'ty': 3
954 AttachShapeLayer, // 'ty': 4
955 AttachTextLayer, // 'ty': 5
956 };
957
Florin Malitacf8ed522018-01-25 15:27:33 -0500958 int type = ParseDefault(jlayer["ty"], -1);
Florin Malita094ccde2017-12-30 12:27:00 -0500959 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
960 return nullptr;
961 }
962
Florin Malita71cba8f2018-01-09 08:07:14 -0500963 // Layer content.
964 auto layer = gLayerAttachers[type](jlayer, layerCtx->fCtx);
Florin Malita25366fa2018-01-23 13:37:59 -0500965 // Optional layer mask.
966 layer = AttachMask(jlayer["masksProperties"], layerCtx->fCtx, std::move(layer));
967 // Optional layer transform.
Florin Malita71cba8f2018-01-09 08:07:14 -0500968 if (auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -0500969 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
970 }
971 // Optional layer opacity.
972 layer = AttachOpacity(jlayer["ks"], layerCtx->fCtx, std::move(layer));
Florin Malita18eafd92018-01-04 21:11:55 -0500973
Florin Malita71cba8f2018-01-09 08:07:14 -0500974 // TODO: we should also disable related/inactive animators.
Florin Malita35efaa82018-01-22 12:57:06 -0500975 class Activator final : public sksg::Animator {
Florin Malita71cba8f2018-01-09 08:07:14 -0500976 public:
977 Activator(sk_sp<sksg::OpacityEffect> controlNode, float in, float out)
978 : fControlNode(std::move(controlNode))
979 , fIn(in)
980 , fOut(out) {}
981
Florin Malita35efaa82018-01-22 12:57:06 -0500982 void onTick(float t) override {
Florin Malita71cba8f2018-01-09 08:07:14 -0500983 // Keep the layer fully transparent except for its [in..out] lifespan.
984 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
985 fControlNode->setOpacity(t >= fIn && t <= fOut ? 1 : 0);
986 }
987
988 private:
989 const sk_sp<sksg::OpacityEffect> fControlNode;
990 const float fIn,
991 fOut;
992 };
993
994 auto layerControl = sksg::OpacityEffect::Make(std::move(layer));
Florin Malitacf8ed522018-01-25 15:27:33 -0500995 const auto in = ParseDefault(jlayer["ip"], 0.0f),
996 out = ParseDefault(jlayer["op"], in);
Florin Malita71cba8f2018-01-09 08:07:14 -0500997
998 if (in >= out || ! layerControl)
999 return nullptr;
1000
1001 layerCtx->fCtx->fAnimators.push_back(skstd::make_unique<Activator>(layerControl, in, out));
1002
Florin Malitacf8ed522018-01-25 15:27:33 -05001003 if (ParseDefault(jlayer["td"], false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -05001004 // This layer is a matte. We apply it as a mask to the next layer.
1005 layerCtx->fCurrentMatte = std::move(layerControl);
1006 return nullptr;
1007 }
1008
1009 if (layerCtx->fCurrentMatte) {
1010 // There is a pending matte. Apply and reset.
1011 return sksg::MaskEffect::Make(std::move(layerControl), std::move(layerCtx->fCurrentMatte));
1012 }
1013
Florin Malita71cba8f2018-01-09 08:07:14 -05001014 return layerControl;
Florin Malita094ccde2017-12-30 12:27:00 -05001015}
1016
1017sk_sp<sksg::RenderNode> AttachComposition(const Json::Value& comp, AttachContext* ctx) {
1018 if (!comp.isObject())
1019 return nullptr;
1020
Florin Malita18eafd92018-01-04 21:11:55 -05001021 const auto& jlayers = comp["layers"];
1022 if (!jlayers.isArray())
1023 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001024
Florin Malita18eafd92018-01-04 21:11:55 -05001025 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
1026 AttachLayerContext layerCtx(jlayers, ctx);
1027
1028 for (const auto& l : jlayers) {
1029 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001030 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001031 }
1032 }
1033
Florin Malita2a8275b2018-01-02 12:52:43 -05001034 if (layers.empty()) {
1035 return nullptr;
1036 }
1037
1038 // Layers are painted in bottom->top order.
1039 auto comp_group = sksg::Group::Make();
1040 for (int i = layers.count() - 1; i >= 0; --i) {
1041 comp_group->addChild(std::move(layers[i]));
1042 }
1043
1044 LOG("** Attached composition '%s': %d layers.\n",
Florin Malitacf8ed522018-01-25 15:27:33 -05001045 ParseDefault(comp["id"], SkString()).c_str(), layers.count());
Florin Malita2a8275b2018-01-02 12:52:43 -05001046
Florin Malita094ccde2017-12-30 12:27:00 -05001047 return comp_group;
1048}
1049
1050} // namespace
1051
Florin Malita49328072018-01-08 12:51:12 -05001052std::unique_ptr<Animation> Animation::Make(SkStream* stream, const ResourceProvider& res) {
Florin Malita094ccde2017-12-30 12:27:00 -05001053 if (!stream->hasLength()) {
1054 // TODO: handle explicit buffering?
1055 LOG("!! cannot parse streaming content\n");
1056 return nullptr;
1057 }
1058
1059 Json::Value json;
1060 {
1061 auto data = SkData::MakeFromStream(stream, stream->getLength());
1062 if (!data) {
1063 LOG("!! could not read stream\n");
1064 return nullptr;
1065 }
1066
1067 Json::Reader reader;
1068
1069 auto dataStart = static_cast<const char*>(data->data());
1070 if (!reader.parse(dataStart, dataStart + data->size(), json, false) || !json.isObject()) {
1071 LOG("!! failed to parse json: %s\n", reader.getFormattedErrorMessages().c_str());
1072 return nullptr;
1073 }
1074 }
1075
Florin Malitacf8ed522018-01-25 15:27:33 -05001076 const auto version = ParseDefault(json["v"], SkString());
1077 const auto size = SkSize::Make(ParseDefault(json["w"], 0.0f),
1078 ParseDefault(json["h"], 0.0f));
1079 const auto fps = ParseDefault(json["fr"], -1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001080
1081 if (size.isEmpty() || version.isEmpty() || fps < 0) {
1082 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1083 version.c_str(), size.width(), size.height(), fps);
1084 return nullptr;
1085 }
1086
Florin Malita49328072018-01-08 12:51:12 -05001087 return std::unique_ptr<Animation>(new Animation(res, std::move(version), size, fps, json));
Florin Malita094ccde2017-12-30 12:27:00 -05001088}
1089
Florin Malita49328072018-01-08 12:51:12 -05001090std::unique_ptr<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res) {
1091 class DirectoryResourceProvider final : public ResourceProvider {
1092 public:
1093 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1094
1095 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1096 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1097 return SkStream::MakeFromFile(resPath.c_str());
1098 }
1099
1100 private:
1101 const SkString fDir;
1102 };
1103
1104 const auto jsonStream = SkStream::MakeFromFile(path);
1105 if (!jsonStream)
1106 return nullptr;
1107
1108 std::unique_ptr<ResourceProvider> defaultProvider;
1109 if (!res) {
1110 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1111 }
1112
1113 return Make(jsonStream.get(), res ? *res : *defaultProvider);
1114}
1115
1116Animation::Animation(const ResourceProvider& resources,
1117 SkString version, const SkSize& size, SkScalar fps, const Json::Value& json)
Florin Malita094ccde2017-12-30 12:27:00 -05001118 : fVersion(std::move(version))
1119 , fSize(size)
1120 , fFrameRate(fps)
Florin Malitacf8ed522018-01-25 15:27:33 -05001121 , fInPoint(ParseDefault(json["ip"], 0.0f))
1122 , fOutPoint(SkTMax(ParseDefault(json["op"], SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001123
1124 AssetMap assets;
1125 for (const auto& asset : json["assets"]) {
1126 if (!asset.isObject()) {
1127 continue;
1128 }
1129
Florin Malitacf8ed522018-01-25 15:27:33 -05001130 assets.set(ParseDefault(asset["id"], SkString()), &asset);
Florin Malita094ccde2017-12-30 12:27:00 -05001131 }
1132
Florin Malita35efaa82018-01-22 12:57:06 -05001133 sksg::Scene::AnimatorList animators;
1134 AttachContext ctx = { resources, assets, animators };
1135 auto root = AttachComposition(json, &ctx);
1136
1137 LOG("** Attached %d animators\n", animators.size());
1138
1139 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001140
Florin Malitadb385732018-01-09 12:19:32 -05001141 // In case the client calls render before the first tick.
1142 this->animationTick(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001143}
1144
1145Animation::~Animation() = default;
1146
Florin Malita35efaa82018-01-22 12:57:06 -05001147void Animation::setShowInval(bool show) {
1148 if (fScene) {
1149 fScene->setShowInval(show);
1150 }
1151}
1152
Mike Reed29859872018-01-08 08:25:27 -05001153void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001154 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001155 return;
1156
Mike Reed29859872018-01-08 08:25:27 -05001157 SkAutoCanvasRestore restore(canvas, true);
1158 const SkRect srcR = SkRect::MakeSize(this->size());
1159 if (dstR) {
1160 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1161 }
1162 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001163 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001164}
1165
1166void Animation::animationTick(SkMSec ms) {
Florin Malita35efaa82018-01-22 12:57:06 -05001167 if (!fScene)
1168 return;
1169
Florin Malita094ccde2017-12-30 12:27:00 -05001170 // 't' in the BM model really means 'frame #'
1171 auto t = static_cast<float>(ms) * fFrameRate / 1000;
1172
1173 t = fInPoint + std::fmod(t, fOutPoint - fInPoint);
1174
Florin Malita35efaa82018-01-22 12:57:06 -05001175 fScene->animate(t);
Florin Malita094ccde2017-12-30 12:27:00 -05001176}
1177
Florin Malita54f65c42018-01-16 17:04:30 -05001178} // namespace skottie