blob: efd3d075d7ff72ae8eb7499d1a9766872df09346 [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 Malitafc807c82018-01-25 22:35:09 -050011#include "SkJSONCPP.h"
Florin Malita54f65c42018-01-16 17:04:30 -050012#include "SkottieAnimator.h"
Florin Malitacf8ed522018-01-25 15:27:33 -050013#include "SkottieParser.h"
Florin Malita54f65c42018-01-16 17:04:30 -050014#include "SkottieProperties.h"
Florin Malita094ccde2017-12-30 12:27:00 -050015#include "SkData.h"
Florin Malita49328072018-01-08 12:51:12 -050016#include "SkImage.h"
Florin Malita094ccde2017-12-30 12:27:00 -050017#include "SkMakeUnique.h"
Florin Malita49328072018-01-08 12:51:12 -050018#include "SkOSPath.h"
Florin Malita094ccde2017-12-30 12:27:00 -050019#include "SkPaint.h"
Florin Malita0e66fba2018-01-09 17:10:18 -050020#include "SkParse.h"
Florin Malita094ccde2017-12-30 12:27:00 -050021#include "SkPath.h"
22#include "SkPoint.h"
23#include "SkSGColor.h"
24#include "SkSGDraw.h"
Florin Malita16d0ad02018-01-19 15:07:29 -050025#include "SkSGGeometryTransform.h"
Florin Malita6aaee592018-01-12 12:25:09 -050026#include "SkSGGradient.h"
Florin Malita094ccde2017-12-30 12:27:00 -050027#include "SkSGGroup.h"
Florin Malita49328072018-01-08 12:51:12 -050028#include "SkSGImage.h"
29#include "SkSGInvalidationController.h"
Florin Malita5f9102f2018-01-10 13:36:22 -050030#include "SkSGMaskEffect.h"
Florin Malitae6345d92018-01-03 23:37:54 -050031#include "SkSGMerge.h"
Florin Malitac0034172018-01-08 16:42:59 -050032#include "SkSGOpacityEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050033#include "SkSGPath.h"
Florin Malita2e1d7e22018-01-02 10:40:00 -050034#include "SkSGRect.h"
Florin Malita35efaa82018-01-22 12:57:06 -050035#include "SkSGScene.h"
Florin Malita094ccde2017-12-30 12:27:00 -050036#include "SkSGTransform.h"
Florin Malita51b8c892018-01-07 08:54:24 -050037#include "SkSGTrimEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050038#include "SkStream.h"
39#include "SkTArray.h"
40#include "SkTHash.h"
41
42#include <cmath>
Florin Malita18eafd92018-01-04 21:11:55 -050043#include <unordered_map>
Florin Malitae6345d92018-01-03 23:37:54 -050044#include <vector>
45
Florin Malita094ccde2017-12-30 12:27:00 -050046#include "stdlib.h"
47
Florin Malita54f65c42018-01-16 17:04:30 -050048namespace skottie {
Florin Malita094ccde2017-12-30 12:27:00 -050049
Florin Malitacf8ed522018-01-25 15:27:33 -050050#define LOG SkDebugf
51
Florin Malita094ccde2017-12-30 12:27:00 -050052namespace {
53
54using AssetMap = SkTHashMap<SkString, const Json::Value*>;
55
56struct AttachContext {
Florin Malita35efaa82018-01-22 12:57:06 -050057 const ResourceProvider& fResources;
58 const AssetMap& fAssets;
59 sksg::Scene::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050060};
61
62bool LogFail(const Json::Value& json, const char* msg) {
63 const auto dump = json.toStyledString();
64 LOG("!! %s: %s", msg, dump.c_str());
65 return false;
66}
67
Florin Malita18eafd92018-01-04 21:11:55 -050068sk_sp<sksg::Matrix> AttachMatrix(const Json::Value& t, AttachContext* ctx,
69 sk_sp<sksg::Matrix> parentMatrix) {
70 if (!t.isObject())
71 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -050072
Florin Malita18eafd92018-01-04 21:11:55 -050073 auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
74 auto composite = sk_make_sp<CompositeTransform>(matrix);
Florin Malitafc807c82018-01-25 22:35:09 -050075 auto anchor_attached = BindProperty<VectorValue>(t["a"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -050076 [composite](const VectorValue& a) {
77 composite->setAnchorPoint(ValueTraits<VectorValue>::As<SkPoint>(a));
Florin Malita094ccde2017-12-30 12:27:00 -050078 });
Florin Malitafc807c82018-01-25 22:35:09 -050079 auto position_attached = BindProperty<VectorValue>(t["p"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -050080 [composite](const VectorValue& p) {
81 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malita094ccde2017-12-30 12:27:00 -050082 });
Florin Malitafc807c82018-01-25 22:35:09 -050083 auto scale_attached = BindProperty<VectorValue>(t["s"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -050084 [composite](const VectorValue& s) {
85 composite->setScale(ValueTraits<VectorValue>::As<SkVector>(s));
Florin Malita094ccde2017-12-30 12:27:00 -050086 });
Florin Malita1eb98db2018-01-26 15:03:38 -050087
88 auto* jrotation = &t["r"];
89 if (jrotation->isNull()) {
90 // 3d rotations have separate rx,ry,rz components. While we don't fully support them,
91 // we can still make use of rz.
92 jrotation = &t["rz"];
93 }
94 auto rotation_attached = BindProperty<ScalarValue>(*jrotation, &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -050095 [composite](const ScalarValue& r) {
96 composite->setRotation(r);
Florin Malita094ccde2017-12-30 12:27:00 -050097 });
Florin Malitafc807c82018-01-25 22:35:09 -050098 auto skew_attached = BindProperty<ScalarValue>(t["sk"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -050099 [composite](const ScalarValue& sk) {
100 composite->setSkew(sk);
Florin Malita094ccde2017-12-30 12:27:00 -0500101 });
Florin Malitafc807c82018-01-25 22:35:09 -0500102 auto skewaxis_attached = BindProperty<ScalarValue>(t["sa"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500103 [composite](const ScalarValue& sa) {
104 composite->setSkewAxis(sa);
Florin Malita094ccde2017-12-30 12:27:00 -0500105 });
106
107 if (!anchor_attached &&
108 !position_attached &&
109 !scale_attached &&
110 !rotation_attached &&
111 !skew_attached &&
112 !skewaxis_attached) {
113 LogFail(t, "Could not parse transform");
Florin Malita18eafd92018-01-04 21:11:55 -0500114 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500115 }
116
Florin Malita18eafd92018-01-04 21:11:55 -0500117 return matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500118}
119
Florin Malitac0034172018-01-08 16:42:59 -0500120sk_sp<sksg::RenderNode> AttachOpacity(const Json::Value& jtransform, AttachContext* ctx,
121 sk_sp<sksg::RenderNode> childNode) {
122 if (!jtransform.isObject() || !childNode)
123 return childNode;
124
125 // This is more peeky than other attachers, because we want to avoid redundant opacity
126 // nodes for the extremely common case of static opaciy == 100.
127 const auto& opacity = jtransform["o"];
128 if (opacity.isObject() &&
Florin Malitacf8ed522018-01-25 15:27:33 -0500129 !ParseDefault(opacity["a"], true) &&
130 ParseDefault(opacity["k"], -1) == 100) {
Florin Malitac0034172018-01-08 16:42:59 -0500131 // Ignoring static full opacity.
132 return childNode;
133 }
134
135 auto opacityNode = sksg::OpacityEffect::Make(childNode);
Florin Malitafc807c82018-01-25 22:35:09 -0500136 BindProperty<ScalarValue>(opacity, &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500137 [opacityNode](const ScalarValue& o) {
Florin Malitac0034172018-01-08 16:42:59 -0500138 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500139 opacityNode->setOpacity(o * 0.01f);
Florin Malitac0034172018-01-08 16:42:59 -0500140 });
141
142 return opacityNode;
143}
144
Florin Malita094ccde2017-12-30 12:27:00 -0500145sk_sp<sksg::RenderNode> AttachComposition(const Json::Value&, AttachContext* ctx);
146
Florin Malita25366fa2018-01-23 13:37:59 -0500147sk_sp<sksg::Path> AttachPath(const Json::Value& jpath, AttachContext* ctx) {
148 auto path_node = sksg::Path::Make();
Florin Malitafc807c82018-01-25 22:35:09 -0500149 return BindProperty<ShapeValue>(jpath, &ctx->fAnimators,
150 [path_node](const ShapeValue& p) { path_node->setPath(p); })
Florin Malita25366fa2018-01-23 13:37:59 -0500151 ? path_node
152 : nullptr;
153}
154
Florin Malita094ccde2017-12-30 12:27:00 -0500155sk_sp<sksg::GeometryNode> AttachPathGeometry(const Json::Value& jpath, AttachContext* ctx) {
156 SkASSERT(jpath.isObject());
157
Florin Malita25366fa2018-01-23 13:37:59 -0500158 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500159}
160
Florin Malita2e1d7e22018-01-02 10:40:00 -0500161sk_sp<sksg::GeometryNode> AttachRRectGeometry(const Json::Value& jrect, AttachContext* ctx) {
162 SkASSERT(jrect.isObject());
163
164 auto rect_node = sksg::RRect::Make();
165 auto composite = sk_make_sp<CompositeRRect>(rect_node);
166
Florin Malitafc807c82018-01-25 22:35:09 -0500167 auto p_attached = BindProperty<VectorValue>(jrect["p"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500168 [composite](const VectorValue& p) {
169 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500170 });
Florin Malitafc807c82018-01-25 22:35:09 -0500171 auto s_attached = BindProperty<VectorValue>(jrect["s"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500172 [composite](const VectorValue& s) {
173 composite->setSize(ValueTraits<VectorValue>::As<SkSize>(s));
Florin Malitaf9590922018-01-09 11:56:09 -0500174 });
Florin Malitafc807c82018-01-25 22:35:09 -0500175 auto r_attached = BindProperty<ScalarValue>(jrect["r"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500176 [composite](const ScalarValue& r) {
177 composite->setRadius(SkSize::Make(r, r));
Florin Malitaf9590922018-01-09 11:56:09 -0500178 });
Florin Malita2e1d7e22018-01-02 10:40:00 -0500179
180 if (!p_attached && !s_attached && !r_attached) {
181 return nullptr;
182 }
183
Florin Malitafbc13f12018-01-04 10:26:35 -0500184 LOG("** Attached (r)rect geometry\n");
185
186 return rect_node;
187}
188
189sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const Json::Value& jellipse, AttachContext* ctx) {
190 SkASSERT(jellipse.isObject());
191
192 auto rect_node = sksg::RRect::Make();
193 auto composite = sk_make_sp<CompositeRRect>(rect_node);
194
Florin Malitafc807c82018-01-25 22:35:09 -0500195 auto p_attached = BindProperty<VectorValue>(jellipse["p"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500196 [composite](const VectorValue& p) {
197 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500198 });
Florin Malitafc807c82018-01-25 22:35:09 -0500199 auto s_attached = BindProperty<VectorValue>(jellipse["s"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500200 [composite](const VectorValue& s) {
Florin Malitaf9590922018-01-09 11:56:09 -0500201 const auto sz = ValueTraits<VectorValue>::As<SkSize>(s);
Florin Malita2518a0a2018-01-24 18:29:00 -0500202 composite->setSize(sz);
203 composite->setRadius(SkSize::Make(sz.width() / 2, sz.height() / 2));
Florin Malitaf9590922018-01-09 11:56:09 -0500204 });
Florin Malitafbc13f12018-01-04 10:26:35 -0500205
206 if (!p_attached && !s_attached) {
207 return nullptr;
208 }
209
210 LOG("** Attached ellipse geometry\n");
211
Florin Malita2e1d7e22018-01-02 10:40:00 -0500212 return rect_node;
213}
214
Florin Malita02a32b02018-01-04 11:27:09 -0500215sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const Json::Value& jstar, AttachContext* ctx) {
216 SkASSERT(jstar.isObject());
217
218 static constexpr CompositePolyStar::Type gTypes[] = {
219 CompositePolyStar::Type::kStar, // "sy": 1
220 CompositePolyStar::Type::kPoly, // "sy": 2
221 };
222
Florin Malitacf8ed522018-01-25 15:27:33 -0500223 const auto type = ParseDefault(jstar["sy"], 0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500224 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
225 LogFail(jstar, "Unknown polystar type");
226 return nullptr;
227 }
228
229 auto path_node = sksg::Path::Make();
230 auto composite = sk_make_sp<CompositePolyStar>(path_node, gTypes[type]);
231
Florin Malitafc807c82018-01-25 22:35:09 -0500232 BindProperty<VectorValue>(jstar["p"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500233 [composite](const VectorValue& p) {
234 composite->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500235 });
Florin Malitafc807c82018-01-25 22:35:09 -0500236 BindProperty<ScalarValue>(jstar["pt"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500237 [composite](const ScalarValue& pt) {
238 composite->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500239 });
Florin Malitafc807c82018-01-25 22:35:09 -0500240 BindProperty<ScalarValue>(jstar["ir"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500241 [composite](const ScalarValue& ir) {
242 composite->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500243 });
Florin Malitafc807c82018-01-25 22:35:09 -0500244 BindProperty<ScalarValue>(jstar["or"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500245 [composite](const ScalarValue& otr) {
246 composite->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500247 });
Florin Malitafc807c82018-01-25 22:35:09 -0500248 BindProperty<ScalarValue>(jstar["is"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500249 [composite](const ScalarValue& is) {
250 composite->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500251 });
Florin Malitafc807c82018-01-25 22:35:09 -0500252 BindProperty<ScalarValue>(jstar["os"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500253 [composite](const ScalarValue& os) {
254 composite->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500255 });
Florin Malitafc807c82018-01-25 22:35:09 -0500256 BindProperty<ScalarValue>(jstar["r"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500257 [composite](const ScalarValue& r) {
258 composite->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500259 });
Florin Malita02a32b02018-01-04 11:27:09 -0500260
261 return path_node;
262}
263
Florin Malita6aaee592018-01-12 12:25:09 -0500264sk_sp<sksg::Color> AttachColor(const Json::Value& obj, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500265 SkASSERT(obj.isObject());
266
267 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malitafc807c82018-01-25 22:35:09 -0500268 auto color_attached = BindProperty<VectorValue>(obj["c"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500269 [color_node](const VectorValue& c) {
270 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500271 });
Florin Malita094ccde2017-12-30 12:27:00 -0500272
Florin Malita1586d852018-01-12 14:27:39 -0500273 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500274}
275
Florin Malita6aaee592018-01-12 12:25:09 -0500276sk_sp<sksg::Gradient> AttachGradient(const Json::Value& obj, AttachContext* ctx) {
277 SkASSERT(obj.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500278
Florin Malita6aaee592018-01-12 12:25:09 -0500279 const auto& stops = obj["g"];
280 if (!stops.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500281 return nullptr;
282
Florin Malitacf8ed522018-01-25 15:27:33 -0500283 const auto stopCount = ParseDefault(stops["p"], -1);
Florin Malita6aaee592018-01-12 12:25:09 -0500284 if (stopCount < 0)
285 return nullptr;
286
287 sk_sp<sksg::Gradient> gradient_node;
288 sk_sp<CompositeGradient> composite;
289
Florin Malitacf8ed522018-01-25 15:27:33 -0500290 if (ParseDefault(obj["t"], 1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500291 auto linear_node = sksg::LinearGradient::Make();
292 composite = sk_make_sp<CompositeLinearGradient>(linear_node, stopCount);
293 gradient_node = std::move(linear_node);
294 } else {
295 auto radial_node = sksg::RadialGradient::Make();
296 composite = sk_make_sp<CompositeRadialGradient>(radial_node, stopCount);
297
298 // TODO: highlight, angle
299 gradient_node = std::move(radial_node);
300 }
301
Florin Malitafc807c82018-01-25 22:35:09 -0500302 BindProperty<VectorValue>(stops["k"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500303 [composite](const VectorValue& stops) {
304 composite->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500305 });
Florin Malitafc807c82018-01-25 22:35:09 -0500306 BindProperty<VectorValue>(obj["s"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500307 [composite](const VectorValue& s) {
308 composite->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500309 });
Florin Malitafc807c82018-01-25 22:35:09 -0500310 BindProperty<VectorValue>(obj["e"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500311 [composite](const VectorValue& e) {
312 composite->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500313 });
314
315 return gradient_node;
316}
317
Florin Malita1586d852018-01-12 14:27:39 -0500318sk_sp<sksg::PaintNode> AttachPaint(const Json::Value& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500319 sk_sp<sksg::PaintNode> paint_node) {
320 if (paint_node) {
321 paint_node->setAntiAlias(true);
322
Florin Malitafc807c82018-01-25 22:35:09 -0500323 BindProperty<ScalarValue>(jpaint["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500324 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500325 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500326 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500327 });
Florin Malita6aaee592018-01-12 12:25:09 -0500328 }
329
330 return paint_node;
331}
332
333sk_sp<sksg::PaintNode> AttachStroke(const Json::Value& jstroke, AttachContext* ctx,
334 sk_sp<sksg::PaintNode> stroke_node) {
335 SkASSERT(jstroke.isObject());
336
337 if (!stroke_node)
338 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500339
340 stroke_node->setStyle(SkPaint::kStroke_Style);
341
Florin Malitafc807c82018-01-25 22:35:09 -0500342 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500343 [stroke_node](const ScalarValue& w) {
344 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500345 });
Florin Malita094ccde2017-12-30 12:27:00 -0500346 if (!width_attached)
347 return nullptr;
348
Florin Malitacf8ed522018-01-25 15:27:33 -0500349 stroke_node->setStrokeMiter(ParseDefault(jstroke["ml"], 4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500350
351 static constexpr SkPaint::Join gJoins[] = {
352 SkPaint::kMiter_Join,
353 SkPaint::kRound_Join,
354 SkPaint::kBevel_Join,
355 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500356 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(ParseDefault(jstroke["lj"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500357 0, SK_ARRAY_COUNT(gJoins) - 1)]);
358
359 static constexpr SkPaint::Cap gCaps[] = {
360 SkPaint::kButt_Cap,
361 SkPaint::kRound_Cap,
362 SkPaint::kSquare_Cap,
363 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500364 stroke_node->setStrokeCap(gCaps[SkTPin<int>(ParseDefault(jstroke["lc"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500365 0, SK_ARRAY_COUNT(gCaps) - 1)]);
366
367 return stroke_node;
368}
369
Florin Malita6aaee592018-01-12 12:25:09 -0500370sk_sp<sksg::PaintNode> AttachColorFill(const Json::Value& jfill, AttachContext* ctx) {
371 SkASSERT(jfill.isObject());
372
373 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
374}
375
376sk_sp<sksg::PaintNode> AttachGradientFill(const Json::Value& jfill, AttachContext* ctx) {
377 SkASSERT(jfill.isObject());
378
379 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
380}
381
382sk_sp<sksg::PaintNode> AttachColorStroke(const Json::Value& jstroke, AttachContext* ctx) {
383 SkASSERT(jstroke.isObject());
384
385 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
386}
387
388sk_sp<sksg::PaintNode> AttachGradientStroke(const Json::Value& jstroke, AttachContext* ctx) {
389 SkASSERT(jstroke.isObject());
390
391 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
392}
393
Florin Malitae6345d92018-01-03 23:37:54 -0500394std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
395 const Json::Value& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
396 std::vector<sk_sp<sksg::GeometryNode>> merged;
397
398 static constexpr sksg::Merge::Mode gModes[] = {
399 sksg::Merge::Mode::kMerge, // "mm": 1
400 sksg::Merge::Mode::kUnion, // "mm": 2
401 sksg::Merge::Mode::kDifference, // "mm": 3
402 sksg::Merge::Mode::kIntersect, // "mm": 4
403 sksg::Merge::Mode::kXOR , // "mm": 5
404 };
405
Florin Malitacf8ed522018-01-25 15:27:33 -0500406 const auto mode = gModes[SkTPin<int>(ParseDefault(jmerge["mm"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500407 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malitae6345d92018-01-03 23:37:54 -0500408 merged.push_back(sksg::Merge::Make(std::move(geos), mode));
409
410 LOG("** Attached merge path effect, mode: %d\n", mode);
411
412 return merged;
413}
414
Florin Malita51b8c892018-01-07 08:54:24 -0500415std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
416 const Json::Value& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
417
418 enum class Mode {
419 kMerged, // "m": 1
420 kSeparate, // "m": 2
421 } gModes[] = { Mode::kMerged, Mode::kSeparate };
422
Florin Malitacf8ed522018-01-25 15:27:33 -0500423 const auto mode = gModes[SkTPin<int>(ParseDefault(jtrim["m"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500424 0, SK_ARRAY_COUNT(gModes) - 1)];
425
426 std::vector<sk_sp<sksg::GeometryNode>> inputs;
427 if (mode == Mode::kMerged) {
428 inputs.push_back(sksg::Merge::Make(std::move(geos), sksg::Merge::Mode::kMerge));
429 } else {
430 inputs = std::move(geos);
431 }
432
433 std::vector<sk_sp<sksg::GeometryNode>> trimmed;
434 trimmed.reserve(inputs.size());
435 for (const auto& i : inputs) {
436 const auto trim = sksg::TrimEffect::Make(i);
437 trimmed.push_back(trim);
Florin Malitafc807c82018-01-25 22:35:09 -0500438 BindProperty<ScalarValue>(jtrim["s"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500439 [trim](const ScalarValue& s) {
440 trim->setStart(s * 0.01f);
Florin Malita51b8c892018-01-07 08:54:24 -0500441 });
Florin Malitafc807c82018-01-25 22:35:09 -0500442 BindProperty<ScalarValue>(jtrim["e"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500443 [trim](const ScalarValue& e) {
444 trim->setEnd(e * 0.01f);
Florin Malita51b8c892018-01-07 08:54:24 -0500445 });
Florin Malitafc807c82018-01-25 22:35:09 -0500446 BindProperty<ScalarValue>(jtrim["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500447 [trim](const ScalarValue& o) {
448 trim->setOffset(o / 360);
Florin Malita51b8c892018-01-07 08:54:24 -0500449 });
450 }
451
452 return trimmed;
453}
454
Florin Malita094ccde2017-12-30 12:27:00 -0500455using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const Json::Value&, AttachContext*);
456static constexpr GeometryAttacherT gGeometryAttachers[] = {
457 AttachPathGeometry,
Florin Malita2e1d7e22018-01-02 10:40:00 -0500458 AttachRRectGeometry,
Florin Malitafbc13f12018-01-04 10:26:35 -0500459 AttachEllipseGeometry,
Florin Malita02a32b02018-01-04 11:27:09 -0500460 AttachPolystarGeometry,
Florin Malita094ccde2017-12-30 12:27:00 -0500461};
462
463using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const Json::Value&, AttachContext*);
464static constexpr PaintAttacherT gPaintAttachers[] = {
Florin Malita6aaee592018-01-12 12:25:09 -0500465 AttachColorFill,
466 AttachColorStroke,
467 AttachGradientFill,
468 AttachGradientStroke,
Florin Malita094ccde2017-12-30 12:27:00 -0500469};
470
Florin Malitae6345d92018-01-03 23:37:54 -0500471using GeometryEffectAttacherT =
472 std::vector<sk_sp<sksg::GeometryNode>> (*)(const Json::Value&,
473 AttachContext*,
474 std::vector<sk_sp<sksg::GeometryNode>>&&);
475static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
476 AttachMergeGeometryEffect,
Florin Malita51b8c892018-01-07 08:54:24 -0500477 AttachTrimGeometryEffect,
Florin Malitae6345d92018-01-03 23:37:54 -0500478};
479
Florin Malita094ccde2017-12-30 12:27:00 -0500480enum class ShapeType {
481 kGeometry,
Florin Malitae6345d92018-01-03 23:37:54 -0500482 kGeometryEffect,
Florin Malita094ccde2017-12-30 12:27:00 -0500483 kPaint,
484 kGroup,
Florin Malitadacc02b2017-12-31 09:12:31 -0500485 kTransform,
Florin Malita094ccde2017-12-30 12:27:00 -0500486};
487
488struct ShapeInfo {
489 const char* fTypeString;
490 ShapeType fShapeType;
491 uint32_t fAttacherIndex; // index into respective attacher tables
492};
493
494const ShapeInfo* FindShapeInfo(const Json::Value& shape) {
495 static constexpr ShapeInfo gShapeInfo[] = {
Florin Malitafbc13f12018-01-04 10:26:35 -0500496 { "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500497 { "fl", ShapeType::kPaint , 0 }, // fill -> AttachColorFill
498 { "gf", ShapeType::kPaint , 2 }, // gfill -> AttachGradientFill
Florin Malita16d0ad02018-01-19 15:07:29 -0500499 { "gr", ShapeType::kGroup , 0 }, // group -> Inline handler
Florin Malita6aaee592018-01-12 12:25:09 -0500500 { "gs", ShapeType::kPaint , 3 }, // gstroke -> AttachGradientStroke
Florin Malitae6345d92018-01-03 23:37:54 -0500501 { "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
Florin Malita02a32b02018-01-04 11:27:09 -0500502 { "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
Florin Malitae6345d92018-01-03 23:37:54 -0500503 { "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
Florin Malita02a32b02018-01-04 11:27:09 -0500504 { "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500505 { "st", ShapeType::kPaint , 1 }, // stroke -> AttachColorStroke
Florin Malita51b8c892018-01-07 08:54:24 -0500506 { "tm", ShapeType::kGeometryEffect, 1 }, // trim -> AttachTrimGeometryEffect
Florin Malita16d0ad02018-01-19 15:07:29 -0500507 { "tr", ShapeType::kTransform , 0 }, // transform -> Inline handler
Florin Malita094ccde2017-12-30 12:27:00 -0500508 };
509
510 if (!shape.isObject())
511 return nullptr;
512
513 const auto& type = shape["ty"];
514 if (!type.isString())
515 return nullptr;
516
517 const auto* info = bsearch(type.asCString(),
518 gShapeInfo,
519 SK_ARRAY_COUNT(gShapeInfo),
520 sizeof(ShapeInfo),
521 [](const void* key, const void* info) {
522 return strcmp(static_cast<const char*>(key),
523 static_cast<const ShapeInfo*>(info)->fTypeString);
524 });
525
526 return static_cast<const ShapeInfo*>(info);
527}
528
Florin Malita16d0ad02018-01-19 15:07:29 -0500529struct GeometryEffectRec {
530 const Json::Value& fJson;
531 GeometryEffectAttacherT fAttach;
532};
533
Florin Malitaca4439f2018-01-23 10:31:59 -0500534struct AttachShapeContext {
535 AttachShapeContext(AttachContext* ctx,
536 std::vector<sk_sp<sksg::GeometryNode>>* geos,
537 std::vector<GeometryEffectRec>* effects,
538 size_t committedAnimators)
539 : fCtx(ctx)
540 , fGeometryStack(geos)
541 , fGeometryEffectStack(effects)
542 , fCommittedAnimators(committedAnimators) {}
543
544 AttachContext* fCtx;
545 std::vector<sk_sp<sksg::GeometryNode>>* fGeometryStack;
546 std::vector<GeometryEffectRec>* fGeometryEffectStack;
547 size_t fCommittedAnimators;
548};
549
550sk_sp<sksg::RenderNode> AttachShape(const Json::Value& jshape, AttachShapeContext* shapeCtx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500551 if (!jshape.isArray())
Florin Malita094ccde2017-12-30 12:27:00 -0500552 return nullptr;
553
Florin Malitaca4439f2018-01-23 10:31:59 -0500554 SkDEBUGCODE(const auto initialGeometryEffects = shapeCtx->fGeometryEffectStack->size();)
Florin Malita094ccde2017-12-30 12:27:00 -0500555
Florin Malita16d0ad02018-01-19 15:07:29 -0500556 sk_sp<sksg::Group> shape_group = sksg::Group::Make();
557 sk_sp<sksg::RenderNode> shape_wrapper = shape_group;
558 sk_sp<sksg::Matrix> shape_matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500559
Florin Malita16d0ad02018-01-19 15:07:29 -0500560 struct ShapeRec {
561 const Json::Value& fJson;
562 const ShapeInfo& fInfo;
563 };
564
565 // First pass (bottom->top):
566 //
567 // * pick up the group transform and opacity
568 // * push local geometry effects onto the stack
569 // * store recs for next pass
570 //
571 std::vector<ShapeRec> recs;
572 for (Json::ArrayIndex i = 0; i < jshape.size(); ++i) {
573 const auto& s = jshape[jshape.size() - 1 - i];
Florin Malita094ccde2017-12-30 12:27:00 -0500574 const auto* info = FindShapeInfo(s);
575 if (!info) {
576 LogFail(s.isObject() ? s["ty"] : s, "Unknown shape");
577 continue;
578 }
579
Florin Malita16d0ad02018-01-19 15:07:29 -0500580 recs.push_back({ s, *info });
581
Florin Malita094ccde2017-12-30 12:27:00 -0500582 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500583 case ShapeType::kTransform:
Florin Malitaca4439f2018-01-23 10:31:59 -0500584 if ((shape_matrix = AttachMatrix(s, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500585 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
586 }
Florin Malitaca4439f2018-01-23 10:31:59 -0500587 shape_wrapper = AttachOpacity(s, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500588 break;
589 case ShapeType::kGeometryEffect:
590 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500591 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita16d0ad02018-01-19 15:07:29 -0500592 { s, gGeometryEffectAttachers[info->fAttacherIndex] });
593 break;
594 default:
595 break;
596 }
597 }
598
599 // Second pass (top -> bottom, after 2x reverse):
600 //
601 // * track local geometry
602 // * emit local paints
603 //
604 std::vector<sk_sp<sksg::GeometryNode>> geos;
605 std::vector<sk_sp<sksg::RenderNode >> draws;
606 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
607 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500608 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500609 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500610 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
611 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500612 geos.push_back(std::move(geo));
613 }
614 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500615 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500616 // Apply the current effect and pop from the stack.
617 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500618 if (!geos.empty()) {
619 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500620 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500621 std::move(geos));
622 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500623
Florin Malitaca4439f2018-01-23 10:31:59 -0500624 SkASSERT(shapeCtx->fGeometryEffectStack->back().fJson == rec->fJson);
625 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500626 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500627 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500628 } break;
629 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500630 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
631 &geos,
632 shapeCtx->fGeometryEffectStack,
633 shapeCtx->fCommittedAnimators);
634 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500635 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500636 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
637 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500638 }
639 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500640 case ShapeType::kPaint: {
641 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500642 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500643 if (!paint || geos.empty())
644 break;
645
646 auto drawGeos = geos;
647
648 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500649 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
650 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
651 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500652 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500653
654 // If we still have multiple geos, reduce using 'merge'.
655 auto geo = drawGeos.size() > 1
656 ? sksg::Merge::Make(std::move(drawGeos), sksg::Merge::Mode::kMerge)
657 : drawGeos[0];
658
659 SkASSERT(geo);
660 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500661 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500662 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500663 default:
664 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500665 }
666 }
667
Florin Malita16d0ad02018-01-19 15:07:29 -0500668 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500669 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500670
671 // Push transformed local geometries to parent list, for subsequent paints.
672 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500673 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500674 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
675 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500676 }
677
Florin Malita16d0ad02018-01-19 15:07:29 -0500678 // Emit local draws reversed (bottom->top, per spec).
679 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
680 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500681 }
682
Florin Malita16d0ad02018-01-19 15:07:29 -0500683 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500684}
685
686sk_sp<sksg::RenderNode> AttachCompLayer(const Json::Value& layer, AttachContext* ctx) {
687 SkASSERT(layer.isObject());
688
Florin Malitacf8ed522018-01-25 15:27:33 -0500689 SkString refId;
690 if (!Parse(layer["refId"], &refId) || refId.isEmpty()) {
Florin Malita094ccde2017-12-30 12:27:00 -0500691 LOG("!! Comp layer missing refId\n");
692 return nullptr;
693 }
694
695 const auto* comp = ctx->fAssets.find(refId);
696 if (!comp) {
697 LOG("!! Pre-comp not found: '%s'\n", refId.c_str());
698 return nullptr;
699 }
700
701 // TODO: cycle detection
702 return AttachComposition(**comp, ctx);
703}
704
Florin Malita0e66fba2018-01-09 17:10:18 -0500705sk_sp<sksg::RenderNode> AttachSolidLayer(const Json::Value& jlayer, AttachContext*) {
706 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500707
Florin Malitacf8ed522018-01-25 15:27:33 -0500708 const auto size = SkSize::Make(ParseDefault(jlayer["sw"], 0.0f),
709 ParseDefault(jlayer["sh"], 0.0f));
710 const auto hex = ParseDefault(jlayer["sc"], SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500711 uint32_t c;
712 if (size.isEmpty() ||
713 !hex.startsWith("#") ||
714 !SkParse::FindHex(hex.c_str() + 1, &c)) {
715 LogFail(jlayer, "Could not parse solid layer");
716 return nullptr;
717 }
718
719 const SkColor color = 0xff000000 | c;
720
721 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
722 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500723}
724
Florin Malita49328072018-01-08 12:51:12 -0500725sk_sp<sksg::RenderNode> AttachImageAsset(const Json::Value& jimage, AttachContext* ctx) {
726 SkASSERT(jimage.isObject());
727
Florin Malitacf8ed522018-01-25 15:27:33 -0500728 const auto name = ParseDefault(jimage["p"], SkString()),
729 path = ParseDefault(jimage["u"], SkString());
Florin Malita49328072018-01-08 12:51:12 -0500730 if (name.isEmpty())
731 return nullptr;
732
733 // TODO: plumb resource paths explicitly to ResourceProvider?
734 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
735 const auto resStream = ctx->fResources.openStream(resName.c_str());
736 if (!resStream || !resStream->hasLength()) {
737 LOG("!! Could not load image resource: %s\n", resName.c_str());
738 return nullptr;
739 }
740
741 // TODO: non-intrisic image sizing
742 return sksg::Image::Make(
743 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
744}
745
746sk_sp<sksg::RenderNode> AttachImageLayer(const Json::Value& layer, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500747 SkASSERT(layer.isObject());
748
Florin Malitacf8ed522018-01-25 15:27:33 -0500749 SkString refId;
750 if (!Parse(layer["refId"], &refId) || refId.isEmpty()) {
Florin Malita49328072018-01-08 12:51:12 -0500751 LOG("!! Image layer missing refId\n");
752 return nullptr;
753 }
754
755 const auto* jimage = ctx->fAssets.find(refId);
756 if (!jimage) {
757 LOG("!! Image asset not found: '%s'\n", refId.c_str());
758 return nullptr;
759 }
760
761 return AttachImageAsset(**jimage, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500762}
763
764sk_sp<sksg::RenderNode> AttachNullLayer(const Json::Value& layer, AttachContext*) {
765 SkASSERT(layer.isObject());
766
Florin Malita18eafd92018-01-04 21:11:55 -0500767 // Null layers are used solely to drive dependent transforms,
768 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500769 return nullptr;
770}
771
772sk_sp<sksg::RenderNode> AttachShapeLayer(const Json::Value& layer, AttachContext* ctx) {
773 SkASSERT(layer.isObject());
774
Florin Malita16d0ad02018-01-19 15:07:29 -0500775 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
776 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500777 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
778 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
779
780 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
781 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
782 // due to attached animators. To avoid this, we track committed animators and discard the
783 // orphans here.
784 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
785 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
786
787 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500788}
789
790sk_sp<sksg::RenderNode> AttachTextLayer(const Json::Value& layer, AttachContext*) {
791 SkASSERT(layer.isObject());
792
793 LOG("?? Text layer stub\n");
794 return nullptr;
795}
796
Florin Malita18eafd92018-01-04 21:11:55 -0500797struct AttachLayerContext {
798 AttachLayerContext(const Json::Value& jlayers, AttachContext* ctx)
799 : fLayerList(jlayers), fCtx(ctx) {}
800
801 const Json::Value& fLayerList;
802 AttachContext* fCtx;
803 std::unordered_map<const Json::Value*, sk_sp<sksg::Matrix>> fLayerMatrixCache;
804 std::unordered_map<int, const Json::Value*> fLayerIndexCache;
Florin Malita5f9102f2018-01-10 13:36:22 -0500805 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500806
807 const Json::Value* findLayer(int index) {
808 SkASSERT(fLayerList.isArray());
809
810 if (index < 0) {
811 return nullptr;
812 }
813
814 const auto cached = fLayerIndexCache.find(index);
815 if (cached != fLayerIndexCache.end()) {
816 return cached->second;
817 }
818
819 for (const auto& l : fLayerList) {
820 if (!l.isObject()) {
821 continue;
822 }
823
Florin Malitacf8ed522018-01-25 15:27:33 -0500824 if (ParseDefault(l["ind"], -1) == index) {
Florin Malita18eafd92018-01-04 21:11:55 -0500825 fLayerIndexCache.insert(std::make_pair(index, &l));
826 return &l;
827 }
828 }
829
830 return nullptr;
831 }
832
833 sk_sp<sksg::Matrix> AttachLayerMatrix(const Json::Value& jlayer) {
834 SkASSERT(jlayer.isObject());
835
836 const auto cached = fLayerMatrixCache.find(&jlayer);
837 if (cached != fLayerMatrixCache.end()) {
838 return cached->second;
839 }
840
Florin Malitacf8ed522018-01-25 15:27:33 -0500841 const auto* parentLayer = this->findLayer(ParseDefault(jlayer["parent"], -1));
Florin Malita18eafd92018-01-04 21:11:55 -0500842
843 // TODO: cycle detection?
844 auto parentMatrix = (parentLayer && parentLayer != &jlayer)
845 ? this->AttachLayerMatrix(*parentLayer) : nullptr;
846
847 auto layerMatrix = AttachMatrix(jlayer["ks"], fCtx, std::move(parentMatrix));
848 fLayerMatrixCache.insert(std::make_pair(&jlayer, layerMatrix));
849
850 return layerMatrix;
851 }
852};
853
Florin Malita25366fa2018-01-23 13:37:59 -0500854SkBlendMode MaskBlendMode(char mode) {
855 switch (mode) {
856 case 'a': return SkBlendMode::kSrcOver; // Additive
857 case 's': return SkBlendMode::kExclusion; // Subtract
858 case 'i': return SkBlendMode::kDstIn; // Intersect
859 case 'l': return SkBlendMode::kLighten; // Lighten
860 case 'd': return SkBlendMode::kDarken; // Darken
861 case 'f': return SkBlendMode::kDifference; // Difference
862 default: break;
863 }
864
865 return SkBlendMode::kSrcOver;
866}
867
868sk_sp<sksg::RenderNode> AttachMask(const Json::Value& jmask,
869 AttachContext* ctx,
870 sk_sp<sksg::RenderNode> childNode) {
871 if (!jmask.isArray())
872 return childNode;
873
874 auto mask_group = sksg::Group::Make();
875
876 for (const auto& m : jmask) {
877 if (!m.isObject())
878 continue;
879
Florin Malitacf8ed522018-01-25 15:27:33 -0500880 const auto inverted = ParseDefault(m["inv"], false);
Florin Malita25366fa2018-01-23 13:37:59 -0500881 // TODO
882 if (inverted) {
883 LogFail(m, "Unsupported inverse mask");
884 continue;
885 }
886
887 auto mask_path = AttachPath(m["pt"], ctx);
888 if (!mask_path) {
889 LogFail(m, "Could not parse mask path");
890 continue;
891 }
892
Florin Malitacf8ed522018-01-25 15:27:33 -0500893 SkString mode;
894 if (!Parse(m["mode"], &mode) ||
895 mode.size() != 1 ||
896 !strcmp(mode.c_str(), "n")) { // "None" masks have no effect.
Florin Malita25366fa2018-01-23 13:37:59 -0500897 continue;
Florin Malitacf8ed522018-01-25 15:27:33 -0500898 }
Florin Malita25366fa2018-01-23 13:37:59 -0500899
900 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
901 mask_paint->setBlendMode(MaskBlendMode(mode.c_str()[0]));
Florin Malitafc807c82018-01-25 22:35:09 -0500902 BindProperty<ScalarValue>(m["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500903 [mask_paint](const ScalarValue& o) { mask_paint->setOpacity(o * 0.01f); });
Florin Malita25366fa2018-01-23 13:37:59 -0500904
905 mask_group->addChild(sksg::Draw::Make(std::move(mask_path), std::move(mask_paint)));
906 }
907
908 return mask_group->empty()
909 ? childNode
910 : sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
911}
912
Florin Malita18eafd92018-01-04 21:11:55 -0500913sk_sp<sksg::RenderNode> AttachLayer(const Json::Value& jlayer,
914 AttachLayerContext* layerCtx) {
915 if (!jlayer.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500916 return nullptr;
917
918 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const Json::Value&, AttachContext*);
919 static constexpr LayerAttacher gLayerAttachers[] = {
920 AttachCompLayer, // 'ty': 0
921 AttachSolidLayer, // 'ty': 1
922 AttachImageLayer, // 'ty': 2
923 AttachNullLayer, // 'ty': 3
924 AttachShapeLayer, // 'ty': 4
925 AttachTextLayer, // 'ty': 5
926 };
927
Florin Malitacf8ed522018-01-25 15:27:33 -0500928 int type = ParseDefault(jlayer["ty"], -1);
Florin Malita094ccde2017-12-30 12:27:00 -0500929 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
930 return nullptr;
931 }
932
Florin Malita71cba8f2018-01-09 08:07:14 -0500933 // Layer content.
934 auto layer = gLayerAttachers[type](jlayer, layerCtx->fCtx);
Florin Malita25366fa2018-01-23 13:37:59 -0500935 // Optional layer mask.
936 layer = AttachMask(jlayer["masksProperties"], layerCtx->fCtx, std::move(layer));
937 // Optional layer transform.
Florin Malita71cba8f2018-01-09 08:07:14 -0500938 if (auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -0500939 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
940 }
941 // Optional layer opacity.
942 layer = AttachOpacity(jlayer["ks"], layerCtx->fCtx, std::move(layer));
Florin Malita18eafd92018-01-04 21:11:55 -0500943
Florin Malita71cba8f2018-01-09 08:07:14 -0500944 // TODO: we should also disable related/inactive animators.
Florin Malita35efaa82018-01-22 12:57:06 -0500945 class Activator final : public sksg::Animator {
Florin Malita71cba8f2018-01-09 08:07:14 -0500946 public:
947 Activator(sk_sp<sksg::OpacityEffect> controlNode, float in, float out)
948 : fControlNode(std::move(controlNode))
949 , fIn(in)
950 , fOut(out) {}
951
Florin Malita35efaa82018-01-22 12:57:06 -0500952 void onTick(float t) override {
Florin Malita71cba8f2018-01-09 08:07:14 -0500953 // Keep the layer fully transparent except for its [in..out] lifespan.
954 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
955 fControlNode->setOpacity(t >= fIn && t <= fOut ? 1 : 0);
956 }
957
958 private:
959 const sk_sp<sksg::OpacityEffect> fControlNode;
960 const float fIn,
961 fOut;
962 };
963
964 auto layerControl = sksg::OpacityEffect::Make(std::move(layer));
Florin Malitacf8ed522018-01-25 15:27:33 -0500965 const auto in = ParseDefault(jlayer["ip"], 0.0f),
966 out = ParseDefault(jlayer["op"], in);
Florin Malita71cba8f2018-01-09 08:07:14 -0500967
968 if (in >= out || ! layerControl)
969 return nullptr;
970
971 layerCtx->fCtx->fAnimators.push_back(skstd::make_unique<Activator>(layerControl, in, out));
972
Florin Malitacf8ed522018-01-25 15:27:33 -0500973 if (ParseDefault(jlayer["td"], false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -0500974 // This layer is a matte. We apply it as a mask to the next layer.
975 layerCtx->fCurrentMatte = std::move(layerControl);
976 return nullptr;
977 }
978
979 if (layerCtx->fCurrentMatte) {
980 // There is a pending matte. Apply and reset.
981 return sksg::MaskEffect::Make(std::move(layerControl), std::move(layerCtx->fCurrentMatte));
982 }
983
Florin Malita71cba8f2018-01-09 08:07:14 -0500984 return layerControl;
Florin Malita094ccde2017-12-30 12:27:00 -0500985}
986
987sk_sp<sksg::RenderNode> AttachComposition(const Json::Value& comp, AttachContext* ctx) {
988 if (!comp.isObject())
989 return nullptr;
990
Florin Malita18eafd92018-01-04 21:11:55 -0500991 const auto& jlayers = comp["layers"];
992 if (!jlayers.isArray())
993 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500994
Florin Malita18eafd92018-01-04 21:11:55 -0500995 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
996 AttachLayerContext layerCtx(jlayers, ctx);
997
998 for (const auto& l : jlayers) {
999 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001000 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001001 }
1002 }
1003
Florin Malita2a8275b2018-01-02 12:52:43 -05001004 if (layers.empty()) {
1005 return nullptr;
1006 }
1007
1008 // Layers are painted in bottom->top order.
1009 auto comp_group = sksg::Group::Make();
1010 for (int i = layers.count() - 1; i >= 0; --i) {
1011 comp_group->addChild(std::move(layers[i]));
1012 }
1013
1014 LOG("** Attached composition '%s': %d layers.\n",
Florin Malitacf8ed522018-01-25 15:27:33 -05001015 ParseDefault(comp["id"], SkString()).c_str(), layers.count());
Florin Malita2a8275b2018-01-02 12:52:43 -05001016
Florin Malita094ccde2017-12-30 12:27:00 -05001017 return comp_group;
1018}
1019
1020} // namespace
1021
Florin Malita49328072018-01-08 12:51:12 -05001022std::unique_ptr<Animation> Animation::Make(SkStream* stream, const ResourceProvider& res) {
Florin Malita094ccde2017-12-30 12:27:00 -05001023 if (!stream->hasLength()) {
1024 // TODO: handle explicit buffering?
1025 LOG("!! cannot parse streaming content\n");
1026 return nullptr;
1027 }
1028
1029 Json::Value json;
1030 {
1031 auto data = SkData::MakeFromStream(stream, stream->getLength());
1032 if (!data) {
1033 LOG("!! could not read stream\n");
1034 return nullptr;
1035 }
1036
1037 Json::Reader reader;
1038
1039 auto dataStart = static_cast<const char*>(data->data());
1040 if (!reader.parse(dataStart, dataStart + data->size(), json, false) || !json.isObject()) {
1041 LOG("!! failed to parse json: %s\n", reader.getFormattedErrorMessages().c_str());
1042 return nullptr;
1043 }
1044 }
1045
Florin Malitacf8ed522018-01-25 15:27:33 -05001046 const auto version = ParseDefault(json["v"], SkString());
1047 const auto size = SkSize::Make(ParseDefault(json["w"], 0.0f),
1048 ParseDefault(json["h"], 0.0f));
1049 const auto fps = ParseDefault(json["fr"], -1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001050
1051 if (size.isEmpty() || version.isEmpty() || fps < 0) {
1052 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1053 version.c_str(), size.width(), size.height(), fps);
1054 return nullptr;
1055 }
1056
Florin Malita49328072018-01-08 12:51:12 -05001057 return std::unique_ptr<Animation>(new Animation(res, std::move(version), size, fps, json));
Florin Malita094ccde2017-12-30 12:27:00 -05001058}
1059
Florin Malita49328072018-01-08 12:51:12 -05001060std::unique_ptr<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res) {
1061 class DirectoryResourceProvider final : public ResourceProvider {
1062 public:
1063 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1064
1065 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1066 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1067 return SkStream::MakeFromFile(resPath.c_str());
1068 }
1069
1070 private:
1071 const SkString fDir;
1072 };
1073
1074 const auto jsonStream = SkStream::MakeFromFile(path);
1075 if (!jsonStream)
1076 return nullptr;
1077
1078 std::unique_ptr<ResourceProvider> defaultProvider;
1079 if (!res) {
1080 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1081 }
1082
1083 return Make(jsonStream.get(), res ? *res : *defaultProvider);
1084}
1085
1086Animation::Animation(const ResourceProvider& resources,
1087 SkString version, const SkSize& size, SkScalar fps, const Json::Value& json)
Florin Malita094ccde2017-12-30 12:27:00 -05001088 : fVersion(std::move(version))
1089 , fSize(size)
1090 , fFrameRate(fps)
Florin Malitacf8ed522018-01-25 15:27:33 -05001091 , fInPoint(ParseDefault(json["ip"], 0.0f))
1092 , fOutPoint(SkTMax(ParseDefault(json["op"], SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001093
1094 AssetMap assets;
1095 for (const auto& asset : json["assets"]) {
1096 if (!asset.isObject()) {
1097 continue;
1098 }
1099
Florin Malitacf8ed522018-01-25 15:27:33 -05001100 assets.set(ParseDefault(asset["id"], SkString()), &asset);
Florin Malita094ccde2017-12-30 12:27:00 -05001101 }
1102
Florin Malita35efaa82018-01-22 12:57:06 -05001103 sksg::Scene::AnimatorList animators;
1104 AttachContext ctx = { resources, assets, animators };
1105 auto root = AttachComposition(json, &ctx);
1106
1107 LOG("** Attached %d animators\n", animators.size());
1108
1109 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001110
Florin Malitadb385732018-01-09 12:19:32 -05001111 // In case the client calls render before the first tick.
1112 this->animationTick(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001113}
1114
1115Animation::~Animation() = default;
1116
Florin Malita35efaa82018-01-22 12:57:06 -05001117void Animation::setShowInval(bool show) {
1118 if (fScene) {
1119 fScene->setShowInval(show);
1120 }
1121}
1122
Mike Reed29859872018-01-08 08:25:27 -05001123void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001124 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001125 return;
1126
Mike Reed29859872018-01-08 08:25:27 -05001127 SkAutoCanvasRestore restore(canvas, true);
1128 const SkRect srcR = SkRect::MakeSize(this->size());
1129 if (dstR) {
1130 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1131 }
1132 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001133 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001134}
1135
1136void Animation::animationTick(SkMSec ms) {
Florin Malita35efaa82018-01-22 12:57:06 -05001137 if (!fScene)
1138 return;
1139
Florin Malita094ccde2017-12-30 12:27:00 -05001140 // 't' in the BM model really means 'frame #'
1141 auto t = static_cast<float>(ms) * fFrameRate / 1000;
1142
1143 t = fInPoint + std::fmod(t, fOutPoint - fInPoint);
1144
Florin Malita35efaa82018-01-22 12:57:06 -05001145 fScene->animate(t);
Florin Malita094ccde2017-12-30 12:27:00 -05001146}
1147
Florin Malita54f65c42018-01-16 17:04:30 -05001148} // namespace skottie