blob: a3b6d46b93eb473bdba8376340773f1fc4804c85 [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 Malitaa6e30f72018-03-23 13:41:58 -040012#include "SkottieAdapter.h"
Florin Malita54f65c42018-01-16 17:04:30 -050013#include "SkottieAnimator.h"
Florin Malitacf8ed522018-01-25 15:27:33 -050014#include "SkottieParser.h"
Florin Malitaa6e30f72018-03-23 13:41:58 -040015#include "SkottieValue.h"
Florin Malita094ccde2017-12-30 12:27:00 -050016#include "SkData.h"
Florin Malita49328072018-01-08 12:51:12 -050017#include "SkImage.h"
Florin Malita094ccde2017-12-30 12:27:00 -050018#include "SkMakeUnique.h"
Florin Malita49328072018-01-08 12:51:12 -050019#include "SkOSPath.h"
Florin Malita094ccde2017-12-30 12:27:00 -050020#include "SkPaint.h"
Florin Malita0e66fba2018-01-09 17:10:18 -050021#include "SkParse.h"
Florin Malita094ccde2017-12-30 12:27:00 -050022#include "SkPoint.h"
Florin Malita38ea40e2018-01-29 16:31:14 -050023#include "SkSGClipEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050024#include "SkSGColor.h"
25#include "SkSGDraw.h"
Florin Malita16d0ad02018-01-19 15:07:29 -050026#include "SkSGGeometryTransform.h"
Florin Malita6aaee592018-01-12 12:25:09 -050027#include "SkSGGradient.h"
Florin Malita094ccde2017-12-30 12:27:00 -050028#include "SkSGGroup.h"
Florin Malita49328072018-01-08 12:51:12 -050029#include "SkSGImage.h"
30#include "SkSGInvalidationController.h"
Florin Malita5f9102f2018-01-10 13:36:22 -050031#include "SkSGMaskEffect.h"
Florin Malitae6345d92018-01-03 23:37:54 -050032#include "SkSGMerge.h"
Florin Malitac0034172018-01-08 16:42:59 -050033#include "SkSGOpacityEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050034#include "SkSGPath.h"
Florin Malita2e1d7e22018-01-02 10:40:00 -050035#include "SkSGRect.h"
Florin Malita35efaa82018-01-22 12:57:06 -050036#include "SkSGScene.h"
Florin Malita094ccde2017-12-30 12:27:00 -050037#include "SkSGTransform.h"
Florin Malita51b8c892018-01-07 08:54:24 -050038#include "SkSGTrimEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050039#include "SkStream.h"
40#include "SkTArray.h"
41#include "SkTHash.h"
42
43#include <cmath>
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
Florin Malitadd22cf962018-01-29 15:42:01 +000054using AssetMap = SkTHashMap<SkString, const Json::Value*>;
Florin Malita094ccde2017-12-30 12:27:00 -050055
56struct AttachContext {
Florin Malitacca86f32018-01-29 10:49:49 -050057 const ResourceProvider& fResources;
58 const AssetMap& fAssets;
Florin Malita1022f742018-02-23 11:10:22 -050059 const float fFrameRate;
Florin Malitacca86f32018-01-29 10:49:49 -050060 sksg::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050061};
62
63bool LogFail(const Json::Value& json, const char* msg) {
64 const auto dump = json.toStyledString();
65 LOG("!! %s: %s", msg, dump.c_str());
66 return false;
67}
68
Florin Malita18eafd92018-01-04 21:11:55 -050069sk_sp<sksg::Matrix> AttachMatrix(const Json::Value& t, AttachContext* ctx,
70 sk_sp<sksg::Matrix> parentMatrix) {
71 if (!t.isObject())
72 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -050073
Florin Malita18eafd92018-01-04 21:11:55 -050074 auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
Florin Malitaa6e30f72018-03-23 13:41:58 -040075 auto adapter = sk_make_sp<TransformAdapter>(matrix);
Florin Malitafc807c82018-01-25 22:35:09 -050076 auto anchor_attached = BindProperty<VectorValue>(t["a"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040077 [adapter](const VectorValue& a) {
78 adapter->setAnchorPoint(ValueTraits<VectorValue>::As<SkPoint>(a));
Florin Malita094ccde2017-12-30 12:27:00 -050079 });
Florin Malitafc807c82018-01-25 22:35:09 -050080 auto position_attached = BindProperty<VectorValue>(t["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040081 [adapter](const VectorValue& p) {
82 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malita094ccde2017-12-30 12:27:00 -050083 });
Florin Malitafc807c82018-01-25 22:35:09 -050084 auto scale_attached = BindProperty<VectorValue>(t["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040085 [adapter](const VectorValue& s) {
86 adapter->setScale(ValueTraits<VectorValue>::As<SkVector>(s));
Florin Malita094ccde2017-12-30 12:27:00 -050087 });
Florin Malita1eb98db2018-01-26 15:03:38 -050088
89 auto* jrotation = &t["r"];
90 if (jrotation->isNull()) {
91 // 3d rotations have separate rx,ry,rz components. While we don't fully support them,
92 // we can still make use of rz.
93 jrotation = &t["rz"];
94 }
95 auto rotation_attached = BindProperty<ScalarValue>(*jrotation, &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040096 [adapter](const ScalarValue& r) {
97 adapter->setRotation(r);
Florin Malita094ccde2017-12-30 12:27:00 -050098 });
Florin Malitafc807c82018-01-25 22:35:09 -050099 auto skew_attached = BindProperty<ScalarValue>(t["sk"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400100 [adapter](const ScalarValue& sk) {
101 adapter->setSkew(sk);
Florin Malita094ccde2017-12-30 12:27:00 -0500102 });
Florin Malitafc807c82018-01-25 22:35:09 -0500103 auto skewaxis_attached = BindProperty<ScalarValue>(t["sa"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400104 [adapter](const ScalarValue& sa) {
105 adapter->setSkewAxis(sa);
Florin Malita094ccde2017-12-30 12:27:00 -0500106 });
107
108 if (!anchor_attached &&
109 !position_attached &&
110 !scale_attached &&
111 !rotation_attached &&
112 !skew_attached &&
113 !skewaxis_attached) {
114 LogFail(t, "Could not parse transform");
Florin Malita18eafd92018-01-04 21:11:55 -0500115 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500116 }
117
Florin Malita18eafd92018-01-04 21:11:55 -0500118 return matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500119}
120
Florin Malitac0034172018-01-08 16:42:59 -0500121sk_sp<sksg::RenderNode> AttachOpacity(const Json::Value& jtransform, AttachContext* ctx,
122 sk_sp<sksg::RenderNode> childNode) {
123 if (!jtransform.isObject() || !childNode)
124 return childNode;
125
126 // This is more peeky than other attachers, because we want to avoid redundant opacity
127 // nodes for the extremely common case of static opaciy == 100.
128 const auto& opacity = jtransform["o"];
129 if (opacity.isObject() &&
Florin Malitacf8ed522018-01-25 15:27:33 -0500130 !ParseDefault(opacity["a"], true) &&
131 ParseDefault(opacity["k"], -1) == 100) {
Florin Malitac0034172018-01-08 16:42:59 -0500132 // Ignoring static full opacity.
133 return childNode;
134 }
135
136 auto opacityNode = sksg::OpacityEffect::Make(childNode);
Florin Malitafc807c82018-01-25 22:35:09 -0500137 BindProperty<ScalarValue>(opacity, &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500138 [opacityNode](const ScalarValue& o) {
Florin Malitac0034172018-01-08 16:42:59 -0500139 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500140 opacityNode->setOpacity(o * 0.01f);
Florin Malitac0034172018-01-08 16:42:59 -0500141 });
142
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400143 return std::move(opacityNode);
Florin Malitac0034172018-01-08 16:42:59 -0500144}
145
Florin Malita094ccde2017-12-30 12:27:00 -0500146sk_sp<sksg::RenderNode> AttachComposition(const Json::Value&, AttachContext* ctx);
147
Florin Malita25366fa2018-01-23 13:37:59 -0500148sk_sp<sksg::Path> AttachPath(const Json::Value& jpath, AttachContext* ctx) {
149 auto path_node = sksg::Path::Make();
Florin Malitafc807c82018-01-25 22:35:09 -0500150 return BindProperty<ShapeValue>(jpath, &ctx->fAnimators,
151 [path_node](const ShapeValue& p) { path_node->setPath(p); })
Florin Malita25366fa2018-01-23 13:37:59 -0500152 ? path_node
153 : nullptr;
154}
155
Florin Malita094ccde2017-12-30 12:27:00 -0500156sk_sp<sksg::GeometryNode> AttachPathGeometry(const Json::Value& jpath, AttachContext* ctx) {
157 SkASSERT(jpath.isObject());
158
Florin Malita25366fa2018-01-23 13:37:59 -0500159 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500160}
161
Florin Malita2e1d7e22018-01-02 10:40:00 -0500162sk_sp<sksg::GeometryNode> AttachRRectGeometry(const Json::Value& jrect, AttachContext* ctx) {
163 SkASSERT(jrect.isObject());
164
165 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400166 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500167
Florin Malitafc807c82018-01-25 22:35:09 -0500168 auto p_attached = BindProperty<VectorValue>(jrect["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400169 [adapter](const VectorValue& p) {
170 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500171 });
Florin Malitafc807c82018-01-25 22:35:09 -0500172 auto s_attached = BindProperty<VectorValue>(jrect["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400173 [adapter](const VectorValue& s) {
174 adapter->setSize(ValueTraits<VectorValue>::As<SkSize>(s));
Florin Malitaf9590922018-01-09 11:56:09 -0500175 });
Florin Malitafc807c82018-01-25 22:35:09 -0500176 auto r_attached = BindProperty<ScalarValue>(jrect["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400177 [adapter](const ScalarValue& r) {
178 adapter->setRadius(SkSize::Make(r, r));
Florin Malitaf9590922018-01-09 11:56:09 -0500179 });
Florin Malita2e1d7e22018-01-02 10:40:00 -0500180
181 if (!p_attached && !s_attached && !r_attached) {
182 return nullptr;
183 }
184
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400185 return std::move(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500186}
187
188sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const Json::Value& jellipse, AttachContext* ctx) {
189 SkASSERT(jellipse.isObject());
190
191 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400192 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500193
Florin Malitafc807c82018-01-25 22:35:09 -0500194 auto p_attached = BindProperty<VectorValue>(jellipse["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400195 [adapter](const VectorValue& p) {
196 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500197 });
Florin Malitafc807c82018-01-25 22:35:09 -0500198 auto s_attached = BindProperty<VectorValue>(jellipse["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400199 [adapter](const VectorValue& s) {
Florin Malitaf9590922018-01-09 11:56:09 -0500200 const auto sz = ValueTraits<VectorValue>::As<SkSize>(s);
Florin Malitaa6e30f72018-03-23 13:41:58 -0400201 adapter->setSize(sz);
202 adapter->setRadius(SkSize::Make(sz.width() / 2, sz.height() / 2));
Florin Malitaf9590922018-01-09 11:56:09 -0500203 });
Florin Malitafbc13f12018-01-04 10:26:35 -0500204
205 if (!p_attached && !s_attached) {
206 return nullptr;
207 }
208
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400209 return std::move(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500210}
211
Florin Malita02a32b02018-01-04 11:27:09 -0500212sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const Json::Value& jstar, AttachContext* ctx) {
213 SkASSERT(jstar.isObject());
214
Florin Malitaa6e30f72018-03-23 13:41:58 -0400215 static constexpr PolyStarAdapter::Type gTypes[] = {
216 PolyStarAdapter::Type::kStar, // "sy": 1
217 PolyStarAdapter::Type::kPoly, // "sy": 2
Florin Malita02a32b02018-01-04 11:27:09 -0500218 };
219
Florin Malitacf8ed522018-01-25 15:27:33 -0500220 const auto type = ParseDefault(jstar["sy"], 0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500221 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
222 LogFail(jstar, "Unknown polystar type");
223 return nullptr;
224 }
225
226 auto path_node = sksg::Path::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400227 auto adapter = sk_make_sp<PolyStarAdapter>(path_node, gTypes[type]);
Florin Malita02a32b02018-01-04 11:27:09 -0500228
Florin Malitafc807c82018-01-25 22:35:09 -0500229 BindProperty<VectorValue>(jstar["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400230 [adapter](const VectorValue& p) {
231 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500232 });
Florin Malitafc807c82018-01-25 22:35:09 -0500233 BindProperty<ScalarValue>(jstar["pt"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400234 [adapter](const ScalarValue& pt) {
235 adapter->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500236 });
Florin Malitafc807c82018-01-25 22:35:09 -0500237 BindProperty<ScalarValue>(jstar["ir"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400238 [adapter](const ScalarValue& ir) {
239 adapter->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500240 });
Florin Malitafc807c82018-01-25 22:35:09 -0500241 BindProperty<ScalarValue>(jstar["or"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400242 [adapter](const ScalarValue& otr) {
243 adapter->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500244 });
Florin Malitafc807c82018-01-25 22:35:09 -0500245 BindProperty<ScalarValue>(jstar["is"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400246 [adapter](const ScalarValue& is) {
247 adapter->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500248 });
Florin Malitafc807c82018-01-25 22:35:09 -0500249 BindProperty<ScalarValue>(jstar["os"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400250 [adapter](const ScalarValue& os) {
251 adapter->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500252 });
Florin Malitafc807c82018-01-25 22:35:09 -0500253 BindProperty<ScalarValue>(jstar["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400254 [adapter](const ScalarValue& r) {
255 adapter->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500256 });
Florin Malita02a32b02018-01-04 11:27:09 -0500257
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400258 return std::move(path_node);
Florin Malita02a32b02018-01-04 11:27:09 -0500259}
260
Florin Malita6aaee592018-01-12 12:25:09 -0500261sk_sp<sksg::Color> AttachColor(const Json::Value& obj, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500262 SkASSERT(obj.isObject());
263
264 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malitafc807c82018-01-25 22:35:09 -0500265 auto color_attached = BindProperty<VectorValue>(obj["c"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500266 [color_node](const VectorValue& c) {
267 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500268 });
Florin Malita094ccde2017-12-30 12:27:00 -0500269
Florin Malita1586d852018-01-12 14:27:39 -0500270 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500271}
272
Florin Malita6aaee592018-01-12 12:25:09 -0500273sk_sp<sksg::Gradient> AttachGradient(const Json::Value& obj, AttachContext* ctx) {
274 SkASSERT(obj.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500275
Florin Malita6aaee592018-01-12 12:25:09 -0500276 const auto& stops = obj["g"];
277 if (!stops.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500278 return nullptr;
279
Florin Malitacf8ed522018-01-25 15:27:33 -0500280 const auto stopCount = ParseDefault(stops["p"], -1);
Florin Malita6aaee592018-01-12 12:25:09 -0500281 if (stopCount < 0)
282 return nullptr;
283
284 sk_sp<sksg::Gradient> gradient_node;
Florin Malitaa6e30f72018-03-23 13:41:58 -0400285 sk_sp<GradientAdapter> adapter;
Florin Malita6aaee592018-01-12 12:25:09 -0500286
Florin Malitacf8ed522018-01-25 15:27:33 -0500287 if (ParseDefault(obj["t"], 1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500288 auto linear_node = sksg::LinearGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400289 adapter = sk_make_sp<LinearGradientAdapter>(linear_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500290 gradient_node = std::move(linear_node);
291 } else {
292 auto radial_node = sksg::RadialGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400293 adapter = sk_make_sp<RadialGradientAdapter>(radial_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500294
295 // TODO: highlight, angle
296 gradient_node = std::move(radial_node);
297 }
298
Florin Malitafc807c82018-01-25 22:35:09 -0500299 BindProperty<VectorValue>(stops["k"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400300 [adapter](const VectorValue& stops) {
301 adapter->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500302 });
Florin Malitafc807c82018-01-25 22:35:09 -0500303 BindProperty<VectorValue>(obj["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400304 [adapter](const VectorValue& s) {
305 adapter->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500306 });
Florin Malitafc807c82018-01-25 22:35:09 -0500307 BindProperty<VectorValue>(obj["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400308 [adapter](const VectorValue& e) {
309 adapter->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500310 });
311
312 return gradient_node;
313}
314
Florin Malita1586d852018-01-12 14:27:39 -0500315sk_sp<sksg::PaintNode> AttachPaint(const Json::Value& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500316 sk_sp<sksg::PaintNode> paint_node) {
317 if (paint_node) {
318 paint_node->setAntiAlias(true);
319
Florin Malitafc807c82018-01-25 22:35:09 -0500320 BindProperty<ScalarValue>(jpaint["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500321 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500322 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500323 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500324 });
Florin Malita6aaee592018-01-12 12:25:09 -0500325 }
326
327 return paint_node;
328}
329
330sk_sp<sksg::PaintNode> AttachStroke(const Json::Value& jstroke, AttachContext* ctx,
331 sk_sp<sksg::PaintNode> stroke_node) {
332 SkASSERT(jstroke.isObject());
333
334 if (!stroke_node)
335 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500336
337 stroke_node->setStyle(SkPaint::kStroke_Style);
338
Florin Malitafc807c82018-01-25 22:35:09 -0500339 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500340 [stroke_node](const ScalarValue& w) {
341 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500342 });
Florin Malita094ccde2017-12-30 12:27:00 -0500343 if (!width_attached)
344 return nullptr;
345
Florin Malitacf8ed522018-01-25 15:27:33 -0500346 stroke_node->setStrokeMiter(ParseDefault(jstroke["ml"], 4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500347
348 static constexpr SkPaint::Join gJoins[] = {
349 SkPaint::kMiter_Join,
350 SkPaint::kRound_Join,
351 SkPaint::kBevel_Join,
352 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500353 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(ParseDefault(jstroke["lj"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500354 0, SK_ARRAY_COUNT(gJoins) - 1)]);
355
356 static constexpr SkPaint::Cap gCaps[] = {
357 SkPaint::kButt_Cap,
358 SkPaint::kRound_Cap,
359 SkPaint::kSquare_Cap,
360 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500361 stroke_node->setStrokeCap(gCaps[SkTPin<int>(ParseDefault(jstroke["lc"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500362 0, SK_ARRAY_COUNT(gCaps) - 1)]);
363
364 return stroke_node;
365}
366
Florin Malita6aaee592018-01-12 12:25:09 -0500367sk_sp<sksg::PaintNode> AttachColorFill(const Json::Value& jfill, AttachContext* ctx) {
368 SkASSERT(jfill.isObject());
369
370 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
371}
372
373sk_sp<sksg::PaintNode> AttachGradientFill(const Json::Value& jfill, AttachContext* ctx) {
374 SkASSERT(jfill.isObject());
375
376 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
377}
378
379sk_sp<sksg::PaintNode> AttachColorStroke(const Json::Value& jstroke, AttachContext* ctx) {
380 SkASSERT(jstroke.isObject());
381
382 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
383}
384
385sk_sp<sksg::PaintNode> AttachGradientStroke(const Json::Value& jstroke, AttachContext* ctx) {
386 SkASSERT(jstroke.isObject());
387
388 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
389}
390
Florin Malitae6345d92018-01-03 23:37:54 -0500391std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
392 const Json::Value& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
393 std::vector<sk_sp<sksg::GeometryNode>> merged;
394
395 static constexpr sksg::Merge::Mode gModes[] = {
396 sksg::Merge::Mode::kMerge, // "mm": 1
397 sksg::Merge::Mode::kUnion, // "mm": 2
398 sksg::Merge::Mode::kDifference, // "mm": 3
399 sksg::Merge::Mode::kIntersect, // "mm": 4
400 sksg::Merge::Mode::kXOR , // "mm": 5
401 };
402
Florin Malitacf8ed522018-01-25 15:27:33 -0500403 const auto mode = gModes[SkTPin<int>(ParseDefault(jmerge["mm"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500404 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malitae6345d92018-01-03 23:37:54 -0500405 merged.push_back(sksg::Merge::Make(std::move(geos), mode));
406
Florin Malitae6345d92018-01-03 23:37:54 -0500407 return merged;
408}
409
Florin Malita51b8c892018-01-07 08:54:24 -0500410std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
411 const Json::Value& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
412
413 enum class Mode {
414 kMerged, // "m": 1
415 kSeparate, // "m": 2
416 } gModes[] = { Mode::kMerged, Mode::kSeparate };
417
Florin Malitacf8ed522018-01-25 15:27:33 -0500418 const auto mode = gModes[SkTPin<int>(ParseDefault(jtrim["m"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500419 0, SK_ARRAY_COUNT(gModes) - 1)];
420
421 std::vector<sk_sp<sksg::GeometryNode>> inputs;
422 if (mode == Mode::kMerged) {
423 inputs.push_back(sksg::Merge::Make(std::move(geos), sksg::Merge::Mode::kMerge));
424 } else {
425 inputs = std::move(geos);
426 }
427
428 std::vector<sk_sp<sksg::GeometryNode>> trimmed;
429 trimmed.reserve(inputs.size());
430 for (const auto& i : inputs) {
Florin Malita69526b02018-03-22 12:20:02 -0400431 const auto trimEffect = sksg::TrimEffect::Make(i);
432 trimmed.push_back(trimEffect);
433
Florin Malitaa6e30f72018-03-23 13:41:58 -0400434 const auto adapter = sk_make_sp<TrimEffectAdapter>(std::move(trimEffect));
Florin Malitafc807c82018-01-25 22:35:09 -0500435 BindProperty<ScalarValue>(jtrim["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400436 [adapter](const ScalarValue& s) {
437 adapter->setStart(s);
Florin Malita51b8c892018-01-07 08:54:24 -0500438 });
Florin Malitafc807c82018-01-25 22:35:09 -0500439 BindProperty<ScalarValue>(jtrim["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400440 [adapter](const ScalarValue& e) {
441 adapter->setEnd(e);
Florin Malita51b8c892018-01-07 08:54:24 -0500442 });
Florin Malitafc807c82018-01-25 22:35:09 -0500443 BindProperty<ScalarValue>(jtrim["o"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400444 [adapter](const ScalarValue& o) {
445 adapter->setOffset(o);
Florin Malita51b8c892018-01-07 08:54:24 -0500446 });
447 }
448
449 return trimmed;
450}
451
Florin Malita094ccde2017-12-30 12:27:00 -0500452using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const Json::Value&, AttachContext*);
453static constexpr GeometryAttacherT gGeometryAttachers[] = {
454 AttachPathGeometry,
Florin Malita2e1d7e22018-01-02 10:40:00 -0500455 AttachRRectGeometry,
Florin Malitafbc13f12018-01-04 10:26:35 -0500456 AttachEllipseGeometry,
Florin Malita02a32b02018-01-04 11:27:09 -0500457 AttachPolystarGeometry,
Florin Malita094ccde2017-12-30 12:27:00 -0500458};
459
460using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const Json::Value&, AttachContext*);
461static constexpr PaintAttacherT gPaintAttachers[] = {
Florin Malita6aaee592018-01-12 12:25:09 -0500462 AttachColorFill,
463 AttachColorStroke,
464 AttachGradientFill,
465 AttachGradientStroke,
Florin Malita094ccde2017-12-30 12:27:00 -0500466};
467
Florin Malitae6345d92018-01-03 23:37:54 -0500468using GeometryEffectAttacherT =
469 std::vector<sk_sp<sksg::GeometryNode>> (*)(const Json::Value&,
470 AttachContext*,
471 std::vector<sk_sp<sksg::GeometryNode>>&&);
472static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
473 AttachMergeGeometryEffect,
Florin Malita51b8c892018-01-07 08:54:24 -0500474 AttachTrimGeometryEffect,
Florin Malitae6345d92018-01-03 23:37:54 -0500475};
476
Florin Malita094ccde2017-12-30 12:27:00 -0500477enum class ShapeType {
478 kGeometry,
Florin Malitae6345d92018-01-03 23:37:54 -0500479 kGeometryEffect,
Florin Malita094ccde2017-12-30 12:27:00 -0500480 kPaint,
481 kGroup,
Florin Malitadacc02b2017-12-31 09:12:31 -0500482 kTransform,
Florin Malita094ccde2017-12-30 12:27:00 -0500483};
484
485struct ShapeInfo {
486 const char* fTypeString;
487 ShapeType fShapeType;
488 uint32_t fAttacherIndex; // index into respective attacher tables
489};
490
491const ShapeInfo* FindShapeInfo(const Json::Value& shape) {
492 static constexpr ShapeInfo gShapeInfo[] = {
Florin Malitafbc13f12018-01-04 10:26:35 -0500493 { "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500494 { "fl", ShapeType::kPaint , 0 }, // fill -> AttachColorFill
495 { "gf", ShapeType::kPaint , 2 }, // gfill -> AttachGradientFill
Florin Malita16d0ad02018-01-19 15:07:29 -0500496 { "gr", ShapeType::kGroup , 0 }, // group -> Inline handler
Florin Malita6aaee592018-01-12 12:25:09 -0500497 { "gs", ShapeType::kPaint , 3 }, // gstroke -> AttachGradientStroke
Florin Malitae6345d92018-01-03 23:37:54 -0500498 { "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
Florin Malita02a32b02018-01-04 11:27:09 -0500499 { "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
Florin Malitae6345d92018-01-03 23:37:54 -0500500 { "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
Florin Malita02a32b02018-01-04 11:27:09 -0500501 { "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500502 { "st", ShapeType::kPaint , 1 }, // stroke -> AttachColorStroke
Florin Malita51b8c892018-01-07 08:54:24 -0500503 { "tm", ShapeType::kGeometryEffect, 1 }, // trim -> AttachTrimGeometryEffect
Florin Malita16d0ad02018-01-19 15:07:29 -0500504 { "tr", ShapeType::kTransform , 0 }, // transform -> Inline handler
Florin Malita094ccde2017-12-30 12:27:00 -0500505 };
506
507 if (!shape.isObject())
508 return nullptr;
509
510 const auto& type = shape["ty"];
511 if (!type.isString())
512 return nullptr;
513
514 const auto* info = bsearch(type.asCString(),
515 gShapeInfo,
516 SK_ARRAY_COUNT(gShapeInfo),
517 sizeof(ShapeInfo),
518 [](const void* key, const void* info) {
519 return strcmp(static_cast<const char*>(key),
520 static_cast<const ShapeInfo*>(info)->fTypeString);
521 });
522
523 return static_cast<const ShapeInfo*>(info);
524}
525
Florin Malita16d0ad02018-01-19 15:07:29 -0500526struct GeometryEffectRec {
527 const Json::Value& fJson;
528 GeometryEffectAttacherT fAttach;
529};
530
Florin Malitaca4439f2018-01-23 10:31:59 -0500531struct AttachShapeContext {
532 AttachShapeContext(AttachContext* ctx,
533 std::vector<sk_sp<sksg::GeometryNode>>* geos,
534 std::vector<GeometryEffectRec>* effects,
535 size_t committedAnimators)
536 : fCtx(ctx)
537 , fGeometryStack(geos)
538 , fGeometryEffectStack(effects)
539 , fCommittedAnimators(committedAnimators) {}
540
541 AttachContext* fCtx;
542 std::vector<sk_sp<sksg::GeometryNode>>* fGeometryStack;
543 std::vector<GeometryEffectRec>* fGeometryEffectStack;
544 size_t fCommittedAnimators;
545};
546
547sk_sp<sksg::RenderNode> AttachShape(const Json::Value& jshape, AttachShapeContext* shapeCtx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500548 if (!jshape.isArray())
Florin Malita094ccde2017-12-30 12:27:00 -0500549 return nullptr;
550
Florin Malitaca4439f2018-01-23 10:31:59 -0500551 SkDEBUGCODE(const auto initialGeometryEffects = shapeCtx->fGeometryEffectStack->size();)
Florin Malita094ccde2017-12-30 12:27:00 -0500552
Florin Malita16d0ad02018-01-19 15:07:29 -0500553 sk_sp<sksg::Group> shape_group = sksg::Group::Make();
554 sk_sp<sksg::RenderNode> shape_wrapper = shape_group;
555 sk_sp<sksg::Matrix> shape_matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500556
Florin Malita16d0ad02018-01-19 15:07:29 -0500557 struct ShapeRec {
558 const Json::Value& fJson;
559 const ShapeInfo& fInfo;
560 };
561
562 // First pass (bottom->top):
563 //
564 // * pick up the group transform and opacity
565 // * push local geometry effects onto the stack
566 // * store recs for next pass
567 //
568 std::vector<ShapeRec> recs;
569 for (Json::ArrayIndex i = 0; i < jshape.size(); ++i) {
570 const auto& s = jshape[jshape.size() - 1 - i];
Florin Malita094ccde2017-12-30 12:27:00 -0500571 const auto* info = FindShapeInfo(s);
572 if (!info) {
573 LogFail(s.isObject() ? s["ty"] : s, "Unknown shape");
574 continue;
575 }
576
Florin Malita16d0ad02018-01-19 15:07:29 -0500577 recs.push_back({ s, *info });
578
Florin Malita094ccde2017-12-30 12:27:00 -0500579 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500580 case ShapeType::kTransform:
Florin Malitaca4439f2018-01-23 10:31:59 -0500581 if ((shape_matrix = AttachMatrix(s, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500582 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
583 }
Florin Malitaca4439f2018-01-23 10:31:59 -0500584 shape_wrapper = AttachOpacity(s, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500585 break;
586 case ShapeType::kGeometryEffect:
587 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500588 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita16d0ad02018-01-19 15:07:29 -0500589 { s, gGeometryEffectAttachers[info->fAttacherIndex] });
590 break;
591 default:
592 break;
593 }
594 }
595
596 // Second pass (top -> bottom, after 2x reverse):
597 //
598 // * track local geometry
599 // * emit local paints
600 //
601 std::vector<sk_sp<sksg::GeometryNode>> geos;
602 std::vector<sk_sp<sksg::RenderNode >> draws;
603 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
604 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500605 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500606 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500607 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
608 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500609 geos.push_back(std::move(geo));
610 }
611 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500612 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500613 // Apply the current effect and pop from the stack.
614 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500615 if (!geos.empty()) {
616 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500617 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500618 std::move(geos));
619 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500620
Florin Malitaca4439f2018-01-23 10:31:59 -0500621 SkASSERT(shapeCtx->fGeometryEffectStack->back().fJson == rec->fJson);
622 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500623 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500624 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500625 } break;
626 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500627 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
628 &geos,
629 shapeCtx->fGeometryEffectStack,
630 shapeCtx->fCommittedAnimators);
631 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500632 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500633 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
634 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500635 }
636 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500637 case ShapeType::kPaint: {
638 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500639 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500640 if (!paint || geos.empty())
641 break;
642
643 auto drawGeos = geos;
644
645 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500646 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
647 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
648 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500649 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500650
651 // If we still have multiple geos, reduce using 'merge'.
652 auto geo = drawGeos.size() > 1
653 ? sksg::Merge::Make(std::move(drawGeos), sksg::Merge::Mode::kMerge)
654 : drawGeos[0];
655
656 SkASSERT(geo);
657 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500658 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500659 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500660 default:
661 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500662 }
663 }
664
Florin Malita16d0ad02018-01-19 15:07:29 -0500665 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500666 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500667
668 // Push transformed local geometries to parent list, for subsequent paints.
669 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500670 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500671 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
672 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500673 }
674
Florin Malita16d0ad02018-01-19 15:07:29 -0500675 // Emit local draws reversed (bottom->top, per spec).
676 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
677 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500678 }
679
Florin Malita16d0ad02018-01-19 15:07:29 -0500680 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500681}
682
Florin Malita1022f742018-02-23 11:10:22 -0500683sk_sp<sksg::RenderNode> AttachNestedAnimation(const char* path, AttachContext* ctx) {
684 class SkottieSGAdapter final : public sksg::RenderNode {
685 public:
686 explicit SkottieSGAdapter(sk_sp<Animation> animation)
687 : fAnimation(std::move(animation)) {
688 SkASSERT(fAnimation);
689 }
690
691 protected:
692 SkRect onRevalidate(sksg::InvalidationController*, const SkMatrix&) override {
693 return SkRect::MakeSize(fAnimation->size());
694 }
695
696 void onRender(SkCanvas* canvas) const override {
697 fAnimation->render(canvas);
698 }
699
700 private:
701 const sk_sp<Animation> fAnimation;
702 };
703
704 class SkottieAnimatorAdapter final : public sksg::Animator {
705 public:
706 SkottieAnimatorAdapter(sk_sp<Animation> animation, float frameRate)
707 : fAnimation(std::move(animation))
708 , fFrameRate(frameRate) {
709 SkASSERT(fAnimation);
710 SkASSERT(fFrameRate > 0);
711 }
712
713 protected:
714 void onTick(float t) {
715 // map back from frame # to ms.
716 const auto t_ms = t * 1000 / fFrameRate;
717 fAnimation->animationTick(t_ms);
718 }
719
720 private:
721 const sk_sp<Animation> fAnimation;
722 const float fFrameRate;
723 };
724
725 const auto resStream = ctx->fResources.openStream(path);
726 if (!resStream || !resStream->hasLength()) {
727 LOG("!! Could not open: %s\n", path);
728 return nullptr;
729 }
730
731 auto animation = Animation::Make(resStream.get(), ctx->fResources);
732 if (!animation) {
733 LOG("!! Could not load nested animation: %s\n", path);
734 return nullptr;
735 }
736
737 ctx->fAnimators.push_back(skstd::make_unique<SkottieAnimatorAdapter>(animation,
738 ctx->fFrameRate));
739
740 return sk_make_sp<SkottieSGAdapter>(std::move(animation));
741}
742
Florin Malitaeb87d672018-01-29 15:28:24 -0500743sk_sp<sksg::RenderNode> AttachCompLayer(const Json::Value& jlayer, AttachContext* ctx,
744 float* time_bias, float* time_scale) {
745 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500746
Florin Malitacf8ed522018-01-25 15:27:33 -0500747 SkString refId;
Florin Malitaeb87d672018-01-29 15:28:24 -0500748 if (!Parse(jlayer["refId"], &refId) || refId.isEmpty()) {
Florin Malitadd22cf962018-01-29 15:42:01 +0000749 LOG("!! Comp layer missing refId\n");
Florin Malita094ccde2017-12-30 12:27:00 -0500750 return nullptr;
751 }
752
Florin Malitaeb87d672018-01-29 15:28:24 -0500753 const auto start_time = ParseDefault(jlayer["st"], 0.0f),
754 stretch_time = ParseDefault(jlayer["sr"], 1.0f);
755
756 *time_bias = -start_time;
757 *time_scale = 1 / stretch_time;
758 if (SkScalarIsNaN(*time_scale)) {
759 *time_scale = 1;
760 }
761
Florin Malita1022f742018-02-23 11:10:22 -0500762 if (refId.startsWith("$")) {
763 return AttachNestedAnimation(refId.c_str() + 1, ctx);
764 }
765
766 const auto* comp = ctx->fAssets.find(refId);
767 if (!comp) {
768 LOG("!! Pre-comp not found: '%s'\n", refId.c_str());
769 return nullptr;
770 }
771
Florin Malitadd22cf962018-01-29 15:42:01 +0000772 // TODO: cycle detection
773 return AttachComposition(**comp, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500774}
775
Florin Malitaeb87d672018-01-29 15:28:24 -0500776sk_sp<sksg::RenderNode> AttachSolidLayer(const Json::Value& jlayer, AttachContext*,
777 float*, float*) {
Florin Malita0e66fba2018-01-09 17:10:18 -0500778 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500779
Florin Malitacf8ed522018-01-25 15:27:33 -0500780 const auto size = SkSize::Make(ParseDefault(jlayer["sw"], 0.0f),
781 ParseDefault(jlayer["sh"], 0.0f));
782 const auto hex = ParseDefault(jlayer["sc"], SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500783 uint32_t c;
784 if (size.isEmpty() ||
785 !hex.startsWith("#") ||
786 !SkParse::FindHex(hex.c_str() + 1, &c)) {
787 LogFail(jlayer, "Could not parse solid layer");
788 return nullptr;
789 }
790
791 const SkColor color = 0xff000000 | c;
792
793 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
794 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500795}
796
Florin Malita49328072018-01-08 12:51:12 -0500797sk_sp<sksg::RenderNode> AttachImageAsset(const Json::Value& jimage, AttachContext* ctx) {
798 SkASSERT(jimage.isObject());
799
Florin Malitacf8ed522018-01-25 15:27:33 -0500800 const auto name = ParseDefault(jimage["p"], SkString()),
801 path = ParseDefault(jimage["u"], SkString());
Florin Malita49328072018-01-08 12:51:12 -0500802 if (name.isEmpty())
803 return nullptr;
804
805 // TODO: plumb resource paths explicitly to ResourceProvider?
806 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
807 const auto resStream = ctx->fResources.openStream(resName.c_str());
808 if (!resStream || !resStream->hasLength()) {
809 LOG("!! Could not load image resource: %s\n", resName.c_str());
810 return nullptr;
811 }
812
813 // TODO: non-intrisic image sizing
814 return sksg::Image::Make(
815 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
816}
817
Florin Malitaeb87d672018-01-29 15:28:24 -0500818sk_sp<sksg::RenderNode> AttachImageLayer(const Json::Value& layer, AttachContext* ctx,
819 float*, float*) {
Florin Malitadd22cf962018-01-29 15:42:01 +0000820 SkASSERT(layer.isObject());
821
822 SkString refId;
823 if (!Parse(layer["refId"], &refId) || refId.isEmpty()) {
824 LOG("!! Image layer missing refId\n");
825 return nullptr;
826 }
827
828 const auto* jimage = ctx->fAssets.find(refId);
829 if (!jimage) {
830 LOG("!! Image asset not found: '%s'\n", refId.c_str());
831 return nullptr;
832 }
833
834 return AttachImageAsset(**jimage, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500835}
836
Florin Malitaeb87d672018-01-29 15:28:24 -0500837sk_sp<sksg::RenderNode> AttachNullLayer(const Json::Value& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500838 SkASSERT(layer.isObject());
839
Florin Malita18eafd92018-01-04 21:11:55 -0500840 // Null layers are used solely to drive dependent transforms,
841 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500842 return nullptr;
843}
844
Florin Malitaeb87d672018-01-29 15:28:24 -0500845sk_sp<sksg::RenderNode> AttachShapeLayer(const Json::Value& layer, AttachContext* ctx,
846 float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500847 SkASSERT(layer.isObject());
848
Florin Malita16d0ad02018-01-19 15:07:29 -0500849 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
850 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500851 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
852 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
853
854 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
855 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
856 // due to attached animators. To avoid this, we track committed animators and discard the
857 // orphans here.
858 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
859 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
860
861 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500862}
863
Florin Malitaeb87d672018-01-29 15:28:24 -0500864sk_sp<sksg::RenderNode> AttachTextLayer(const Json::Value& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500865 SkASSERT(layer.isObject());
866
867 LOG("?? Text layer stub\n");
868 return nullptr;
869}
870
Florin Malita18eafd92018-01-04 21:11:55 -0500871struct AttachLayerContext {
872 AttachLayerContext(const Json::Value& jlayers, AttachContext* ctx)
873 : fLayerList(jlayers), fCtx(ctx) {}
874
Florin Malita4a490682018-01-28 14:27:51 -0500875 const Json::Value& fLayerList;
876 AttachContext* fCtx;
877 SkTHashMap<int, sk_sp<sksg::Matrix>> fLayerMatrixMap;
878 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500879
Florin Malita4a490682018-01-28 14:27:51 -0500880 sk_sp<sksg::Matrix> AttachParentLayerMatrix(const Json::Value& jlayer) {
881 SkASSERT(jlayer.isObject());
Florin Malita18eafd92018-01-04 21:11:55 -0500882 SkASSERT(fLayerList.isArray());
883
Florin Malita4a490682018-01-28 14:27:51 -0500884 const auto parent_index = ParseDefault(jlayer["parent"], -1);
885 if (parent_index < 0)
Florin Malita18eafd92018-01-04 21:11:55 -0500886 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500887
Florin Malita4a490682018-01-28 14:27:51 -0500888 if (auto* m = fLayerMatrixMap.find(parent_index))
889 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500890
891 for (const auto& l : fLayerList) {
892 if (!l.isObject()) {
893 continue;
894 }
895
Florin Malita4a490682018-01-28 14:27:51 -0500896 if (ParseDefault(l["ind"], -1) == parent_index) {
897 return this->AttachLayerMatrix(l);
Florin Malita18eafd92018-01-04 21:11:55 -0500898 }
899 }
900
901 return nullptr;
902 }
903
904 sk_sp<sksg::Matrix> AttachLayerMatrix(const Json::Value& jlayer) {
905 SkASSERT(jlayer.isObject());
906
Florin Malita4a490682018-01-28 14:27:51 -0500907 const auto layer_index = ParseDefault(jlayer["ind"], -1);
908 if (layer_index < 0)
909 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500910
Florin Malita4a490682018-01-28 14:27:51 -0500911 if (auto* m = fLayerMatrixMap.find(layer_index))
912 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500913
Florin Malita4a490682018-01-28 14:27:51 -0500914 // Add a stub entry to break recursion cycles.
915 fLayerMatrixMap.set(layer_index, nullptr);
Florin Malita18eafd92018-01-04 21:11:55 -0500916
Florin Malita4a490682018-01-28 14:27:51 -0500917 auto parent_matrix = this->AttachParentLayerMatrix(jlayer);
Florin Malita18eafd92018-01-04 21:11:55 -0500918
Florin Malita4a490682018-01-28 14:27:51 -0500919 return *fLayerMatrixMap.set(layer_index,
920 AttachMatrix(jlayer["ks"],
921 fCtx,
922 this->AttachParentLayerMatrix(jlayer)));
Florin Malita18eafd92018-01-04 21:11:55 -0500923 }
924};
925
Florin Malita25366fa2018-01-23 13:37:59 -0500926SkBlendMode MaskBlendMode(char mode) {
927 switch (mode) {
928 case 'a': return SkBlendMode::kSrcOver; // Additive
929 case 's': return SkBlendMode::kExclusion; // Subtract
930 case 'i': return SkBlendMode::kDstIn; // Intersect
931 case 'l': return SkBlendMode::kLighten; // Lighten
932 case 'd': return SkBlendMode::kDarken; // Darken
933 case 'f': return SkBlendMode::kDifference; // Difference
934 default: break;
935 }
936
937 return SkBlendMode::kSrcOver;
938}
939
940sk_sp<sksg::RenderNode> AttachMask(const Json::Value& jmask,
941 AttachContext* ctx,
942 sk_sp<sksg::RenderNode> childNode) {
943 if (!jmask.isArray())
944 return childNode;
945
Florin Malita0c51c212018-04-26 14:13:14 -0400946 struct MaskRecord {
947 sk_sp<sksg::Path> mask_path;
948 sk_sp<sksg::Color> mask_paint;
949 };
950
951 SkSTArray<4, MaskRecord, true> mask_stack;
952
953 bool opaque_mask = true;
Florin Malita25366fa2018-01-23 13:37:59 -0500954
955 for (const auto& m : jmask) {
956 if (!m.isObject())
957 continue;
958
Florin Malita25366fa2018-01-23 13:37:59 -0500959 auto mask_path = AttachPath(m["pt"], ctx);
960 if (!mask_path) {
961 LogFail(m, "Could not parse mask path");
962 continue;
963 }
964
Florin Malitacd05b192018-04-25 21:43:03 -0400965 mask_path->setFillType(ParseDefault(m["inv"], false)
966 ? SkPath::kInverseWinding_FillType
967 : SkPath::kWinding_FillType);
968
Florin Malitacf8ed522018-01-25 15:27:33 -0500969 SkString mode;
970 if (!Parse(m["mode"], &mode) ||
971 mode.size() != 1 ||
972 !strcmp(mode.c_str(), "n")) { // "None" masks have no effect.
Florin Malita25366fa2018-01-23 13:37:59 -0500973 continue;
Florin Malitacf8ed522018-01-25 15:27:33 -0500974 }
Florin Malita25366fa2018-01-23 13:37:59 -0500975
976 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
Florin Malitaaa71c892018-01-30 09:27:01 -0500977 mask_paint->setAntiAlias(true);
Florin Malita25366fa2018-01-23 13:37:59 -0500978 mask_paint->setBlendMode(MaskBlendMode(mode.c_str()[0]));
Florin Malita0c51c212018-04-26 14:13:14 -0400979
980 const auto animator_count = ctx->fAnimators.size();
Florin Malitafc807c82018-01-25 22:35:09 -0500981 BindProperty<ScalarValue>(m["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500982 [mask_paint](const ScalarValue& o) { mask_paint->setOpacity(o * 0.01f); });
Florin Malita25366fa2018-01-23 13:37:59 -0500983
Florin Malita0c51c212018-04-26 14:13:14 -0400984 opaque_mask &= (animator_count == ctx->fAnimators.size() && mask_paint->getOpacity() >= 1);
985
986 mask_stack.push_back({mask_path, mask_paint});
Florin Malita25366fa2018-01-23 13:37:59 -0500987 }
988
Florin Malita0c51c212018-04-26 14:13:14 -0400989 if (mask_stack.empty())
990 return childNode;
991
992 if (mask_stack.count() == 1 && opaque_mask) {
993 // Single opaque mask => clip path.
994 return sksg::ClipEffect::Make(std::move(childNode),
995 std::move(mask_stack.front().mask_path),
996 true);
997 }
998
999 auto mask_group = sksg::Group::Make();
1000 for (const auto& rec : mask_stack) {
1001 mask_group->addChild(sksg::Draw::Make(std::move(rec.mask_path),
1002 std::move(rec.mask_paint)));
1003
1004 }
1005
1006 return sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
Florin Malita25366fa2018-01-23 13:37:59 -05001007}
1008
Florin Malita18eafd92018-01-04 21:11:55 -05001009sk_sp<sksg::RenderNode> AttachLayer(const Json::Value& jlayer,
1010 AttachLayerContext* layerCtx) {
1011 if (!jlayer.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -05001012 return nullptr;
1013
Florin Malitaeb87d672018-01-29 15:28:24 -05001014 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const Json::Value&, AttachContext*,
1015 float* time_bias, float* time_scale);
Florin Malita094ccde2017-12-30 12:27:00 -05001016 static constexpr LayerAttacher gLayerAttachers[] = {
1017 AttachCompLayer, // 'ty': 0
1018 AttachSolidLayer, // 'ty': 1
1019 AttachImageLayer, // 'ty': 2
1020 AttachNullLayer, // 'ty': 3
1021 AttachShapeLayer, // 'ty': 4
1022 AttachTextLayer, // 'ty': 5
1023 };
1024
Florin Malitacf8ed522018-01-25 15:27:33 -05001025 int type = ParseDefault(jlayer["ty"], -1);
Florin Malita094ccde2017-12-30 12:27:00 -05001026 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
1027 return nullptr;
1028 }
1029
Florin Malitacca86f32018-01-29 10:49:49 -05001030 sksg::AnimatorList layer_animators;
Florin Malita1022f742018-02-23 11:10:22 -05001031 AttachContext local_ctx = { layerCtx->fCtx->fResources,
1032 layerCtx->fCtx->fAssets,
1033 layerCtx->fCtx->fFrameRate,
1034 layer_animators};
Florin Malitacca86f32018-01-29 10:49:49 -05001035
Florin Malitaeb87d672018-01-29 15:28:24 -05001036 // Layer attachers may adjust these.
1037 float time_bias = 0,
1038 time_scale = 1;
1039
Florin Malita71cba8f2018-01-09 08:07:14 -05001040 // Layer content.
Florin Malitaeb87d672018-01-29 15:28:24 -05001041 auto layer = gLayerAttachers[type](jlayer, &local_ctx, &time_bias, &time_scale);
Florin Malita38ea40e2018-01-29 16:31:14 -05001042
1043 // Clip layers with explicit dimensions.
1044 float w, h;
1045 if (Parse(jlayer["w"], &w) && Parse(jlayer["h"], &h)) {
1046 layer = sksg::ClipEffect::Make(std::move(layer),
1047 sksg::Rect::Make(SkRect::MakeWH(w, h)),
1048 true);
1049 }
1050
Florin Malita25366fa2018-01-23 13:37:59 -05001051 // Optional layer mask.
Florin Malitacca86f32018-01-29 10:49:49 -05001052 layer = AttachMask(jlayer["masksProperties"], &local_ctx, std::move(layer));
Florin Malita38ea40e2018-01-29 16:31:14 -05001053
Florin Malita25366fa2018-01-23 13:37:59 -05001054 // Optional layer transform.
Florin Malita71cba8f2018-01-09 08:07:14 -05001055 if (auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -05001056 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
1057 }
Florin Malita38ea40e2018-01-29 16:31:14 -05001058
Florin Malita71cba8f2018-01-09 08:07:14 -05001059 // Optional layer opacity.
Florin Malitacca86f32018-01-29 10:49:49 -05001060 layer = AttachOpacity(jlayer["ks"], &local_ctx, std::move(layer));
Florin Malita18eafd92018-01-04 21:11:55 -05001061
Florin Malitaeb87d672018-01-29 15:28:24 -05001062 class LayerController final : public sksg::GroupAnimator {
Florin Malita71cba8f2018-01-09 08:07:14 -05001063 public:
Florin Malitaeb87d672018-01-29 15:28:24 -05001064 LayerController(sksg::AnimatorList&& layer_animators,
1065 sk_sp<sksg::OpacityEffect> controlNode,
1066 float in, float out,
1067 float time_bias, float time_scale)
Florin Malitacca86f32018-01-29 10:49:49 -05001068 : INHERITED(std::move(layer_animators))
1069 , fControlNode(std::move(controlNode))
Florin Malita71cba8f2018-01-09 08:07:14 -05001070 , fIn(in)
Florin Malitaeb87d672018-01-29 15:28:24 -05001071 , fOut(out)
1072 , fTimeBias(time_bias)
1073 , fTimeScale(time_scale) {}
Florin Malita71cba8f2018-01-09 08:07:14 -05001074
Florin Malita35efaa82018-01-22 12:57:06 -05001075 void onTick(float t) override {
Florin Malitacca86f32018-01-29 10:49:49 -05001076 const auto active = (t >= fIn && t <= fOut);
1077
Florin Malita71cba8f2018-01-09 08:07:14 -05001078 // Keep the layer fully transparent except for its [in..out] lifespan.
1079 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
Florin Malitacca86f32018-01-29 10:49:49 -05001080 fControlNode->setOpacity(active ? 1 : 0);
1081
1082 // Dispatch ticks only while active.
1083 if (active)
Florin Malitaeb87d672018-01-29 15:28:24 -05001084 this->INHERITED::onTick((t + fTimeBias) * fTimeScale);
Florin Malita71cba8f2018-01-09 08:07:14 -05001085 }
1086
1087 private:
1088 const sk_sp<sksg::OpacityEffect> fControlNode;
1089 const float fIn,
Florin Malitaeb87d672018-01-29 15:28:24 -05001090 fOut,
1091 fTimeBias,
1092 fTimeScale;
Florin Malitacca86f32018-01-29 10:49:49 -05001093
1094 using INHERITED = sksg::GroupAnimator;
Florin Malita71cba8f2018-01-09 08:07:14 -05001095 };
1096
Florin Malitaeb87d672018-01-29 15:28:24 -05001097 auto controller_node = sksg::OpacityEffect::Make(std::move(layer));
1098 const auto in = ParseDefault(jlayer["ip"], 0.0f),
1099 out = ParseDefault(jlayer["op"], in);
Florin Malita71cba8f2018-01-09 08:07:14 -05001100
Florin Malitaeb87d672018-01-29 15:28:24 -05001101 if (!jlayer["tm"].isNull()) {
1102 LogFail(jlayer["tm"], "Unsupported time remapping");
1103 }
1104
1105 if (in >= out || !controller_node)
Florin Malita71cba8f2018-01-09 08:07:14 -05001106 return nullptr;
1107
Florin Malitacca86f32018-01-29 10:49:49 -05001108 layerCtx->fCtx->fAnimators.push_back(
Florin Malitaeb87d672018-01-29 15:28:24 -05001109 skstd::make_unique<LayerController>(std::move(layer_animators),
1110 controller_node,
1111 in,
1112 out,
1113 time_bias,
1114 time_scale));
Florin Malita71cba8f2018-01-09 08:07:14 -05001115
Florin Malitacf8ed522018-01-25 15:27:33 -05001116 if (ParseDefault(jlayer["td"], false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -05001117 // This layer is a matte. We apply it as a mask to the next layer.
Florin Malitaeb87d672018-01-29 15:28:24 -05001118 layerCtx->fCurrentMatte = std::move(controller_node);
Florin Malita5f9102f2018-01-10 13:36:22 -05001119 return nullptr;
1120 }
1121
1122 if (layerCtx->fCurrentMatte) {
1123 // There is a pending matte. Apply and reset.
Florin Malitaa016be92018-03-05 14:01:41 -05001124 static constexpr sksg::MaskEffect::Mode gMaskModes[] = {
1125 sksg::MaskEffect::Mode::kNormal, // tt: 1
1126 sksg::MaskEffect::Mode::kInvert, // tt: 2
1127 };
1128 const auto matteType = ParseDefault(jlayer["tt"], 1) - 1;
1129
1130 if (matteType >= 0 && matteType < SkTo<int>(SK_ARRAY_COUNT(gMaskModes))) {
1131 return sksg::MaskEffect::Make(std::move(controller_node),
1132 std::move(layerCtx->fCurrentMatte),
1133 gMaskModes[matteType]);
1134 }
1135 layerCtx->fCurrentMatte.reset();
Florin Malita5f9102f2018-01-10 13:36:22 -05001136 }
1137
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001138 return std::move(controller_node);
Florin Malita094ccde2017-12-30 12:27:00 -05001139}
1140
1141sk_sp<sksg::RenderNode> AttachComposition(const Json::Value& comp, AttachContext* ctx) {
1142 if (!comp.isObject())
1143 return nullptr;
1144
Florin Malita18eafd92018-01-04 21:11:55 -05001145 const auto& jlayers = comp["layers"];
1146 if (!jlayers.isArray())
1147 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001148
Florin Malita18eafd92018-01-04 21:11:55 -05001149 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
1150 AttachLayerContext layerCtx(jlayers, ctx);
1151
1152 for (const auto& l : jlayers) {
1153 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001154 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001155 }
1156 }
1157
Florin Malita2a8275b2018-01-02 12:52:43 -05001158 if (layers.empty()) {
1159 return nullptr;
1160 }
1161
1162 // Layers are painted in bottom->top order.
1163 auto comp_group = sksg::Group::Make();
1164 for (int i = layers.count() - 1; i >= 0; --i) {
1165 comp_group->addChild(std::move(layers[i]));
1166 }
1167
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001168 return std::move(comp_group);
Florin Malita094ccde2017-12-30 12:27:00 -05001169}
1170
1171} // namespace
1172
Florin Malita1022f742018-02-23 11:10:22 -05001173sk_sp<Animation> Animation::Make(SkStream* stream, const ResourceProvider& res) {
Florin Malita094ccde2017-12-30 12:27:00 -05001174 if (!stream->hasLength()) {
1175 // TODO: handle explicit buffering?
1176 LOG("!! cannot parse streaming content\n");
1177 return nullptr;
1178 }
1179
1180 Json::Value json;
1181 {
1182 auto data = SkData::MakeFromStream(stream, stream->getLength());
1183 if (!data) {
1184 LOG("!! could not read stream\n");
1185 return nullptr;
1186 }
1187
1188 Json::Reader reader;
1189
1190 auto dataStart = static_cast<const char*>(data->data());
1191 if (!reader.parse(dataStart, dataStart + data->size(), json, false) || !json.isObject()) {
1192 LOG("!! failed to parse json: %s\n", reader.getFormattedErrorMessages().c_str());
1193 return nullptr;
1194 }
1195 }
1196
Florin Malitacf8ed522018-01-25 15:27:33 -05001197 const auto version = ParseDefault(json["v"], SkString());
1198 const auto size = SkSize::Make(ParseDefault(json["w"], 0.0f),
1199 ParseDefault(json["h"], 0.0f));
1200 const auto fps = ParseDefault(json["fr"], -1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001201
Florin Malita1022f742018-02-23 11:10:22 -05001202 if (size.isEmpty() || version.isEmpty() || fps <= 0) {
Florin Malita094ccde2017-12-30 12:27:00 -05001203 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1204 version.c_str(), size.width(), size.height(), fps);
1205 return nullptr;
1206 }
1207
Florin Malita1022f742018-02-23 11:10:22 -05001208 return sk_sp<Animation>(new Animation(res, std::move(version), size, fps, json));
Florin Malita094ccde2017-12-30 12:27:00 -05001209}
1210
Florin Malita1022f742018-02-23 11:10:22 -05001211sk_sp<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res) {
Florin Malita49328072018-01-08 12:51:12 -05001212 class DirectoryResourceProvider final : public ResourceProvider {
1213 public:
1214 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1215
1216 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1217 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1218 return SkStream::MakeFromFile(resPath.c_str());
1219 }
1220
1221 private:
1222 const SkString fDir;
1223 };
1224
1225 const auto jsonStream = SkStream::MakeFromFile(path);
1226 if (!jsonStream)
1227 return nullptr;
1228
1229 std::unique_ptr<ResourceProvider> defaultProvider;
1230 if (!res) {
1231 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1232 }
1233
1234 return Make(jsonStream.get(), res ? *res : *defaultProvider);
1235}
1236
1237Animation::Animation(const ResourceProvider& resources,
1238 SkString version, const SkSize& size, SkScalar fps, const Json::Value& json)
Florin Malita094ccde2017-12-30 12:27:00 -05001239 : fVersion(std::move(version))
1240 , fSize(size)
1241 , fFrameRate(fps)
Florin Malitacf8ed522018-01-25 15:27:33 -05001242 , fInPoint(ParseDefault(json["ip"], 0.0f))
1243 , fOutPoint(SkTMax(ParseDefault(json["op"], SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001244
1245 AssetMap assets;
1246 for (const auto& asset : json["assets"]) {
1247 if (!asset.isObject()) {
1248 continue;
1249 }
1250
Florin Malitadd22cf962018-01-29 15:42:01 +00001251 assets.set(ParseDefault(asset["id"], SkString()), &asset);
Florin Malita094ccde2017-12-30 12:27:00 -05001252 }
1253
Florin Malitacca86f32018-01-29 10:49:49 -05001254 sksg::AnimatorList animators;
Florin Malita1022f742018-02-23 11:10:22 -05001255 AttachContext ctx = { resources, assets, fFrameRate, animators };
Florin Malita35efaa82018-01-22 12:57:06 -05001256 auto root = AttachComposition(json, &ctx);
1257
1258 LOG("** Attached %d animators\n", animators.size());
1259
1260 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001261
Florin Malitadb385732018-01-09 12:19:32 -05001262 // In case the client calls render before the first tick.
1263 this->animationTick(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001264}
1265
1266Animation::~Animation() = default;
1267
Florin Malita35efaa82018-01-22 12:57:06 -05001268void Animation::setShowInval(bool show) {
1269 if (fScene) {
1270 fScene->setShowInval(show);
1271 }
1272}
1273
Mike Reed29859872018-01-08 08:25:27 -05001274void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001275 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001276 return;
1277
Mike Reed29859872018-01-08 08:25:27 -05001278 SkAutoCanvasRestore restore(canvas, true);
1279 const SkRect srcR = SkRect::MakeSize(this->size());
1280 if (dstR) {
1281 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1282 }
1283 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001284 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001285}
1286
1287void Animation::animationTick(SkMSec ms) {
Florin Malita35efaa82018-01-22 12:57:06 -05001288 if (!fScene)
1289 return;
1290
Florin Malita094ccde2017-12-30 12:27:00 -05001291 // 't' in the BM model really means 'frame #'
1292 auto t = static_cast<float>(ms) * fFrameRate / 1000;
1293
1294 t = fInPoint + std::fmod(t, fOutPoint - fInPoint);
1295
Florin Malita35efaa82018-01-22 12:57:06 -05001296 fScene->animate(t);
Florin Malita094ccde2017-12-30 12:27:00 -05001297}
1298
Florin Malita54f65c42018-01-16 17:04:30 -05001299} // namespace skottie