blob: bb98ad75f08b343741cb4d7dbd9c87fc656e3628 [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"
Florin Malita6eb85a12018-04-30 10:32:18 -040041#include "SkTime.h"
Florin Malita094ccde2017-12-30 12:27:00 -050042#include "SkTHash.h"
43
44#include <cmath>
Florin Malitae6345d92018-01-03 23:37:54 -050045#include <vector>
46
Florin Malita094ccde2017-12-30 12:27:00 -050047#include "stdlib.h"
48
Florin Malita54f65c42018-01-16 17:04:30 -050049namespace skottie {
Florin Malita094ccde2017-12-30 12:27:00 -050050
Florin Malitacf8ed522018-01-25 15:27:33 -050051#define LOG SkDebugf
52
Florin Malita094ccde2017-12-30 12:27:00 -050053namespace {
54
Florin Malitadd22cf962018-01-29 15:42:01 +000055using AssetMap = SkTHashMap<SkString, const Json::Value*>;
Florin Malita094ccde2017-12-30 12:27:00 -050056
57struct AttachContext {
Florin Malitacca86f32018-01-29 10:49:49 -050058 const ResourceProvider& fResources;
59 const AssetMap& fAssets;
Florin Malita1022f742018-02-23 11:10:22 -050060 const float fFrameRate;
Florin Malitacca86f32018-01-29 10:49:49 -050061 sksg::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050062};
63
64bool LogFail(const Json::Value& json, const char* msg) {
65 const auto dump = json.toStyledString();
66 LOG("!! %s: %s", msg, dump.c_str());
67 return false;
68}
69
Florin Malita18eafd92018-01-04 21:11:55 -050070sk_sp<sksg::Matrix> AttachMatrix(const Json::Value& t, AttachContext* ctx,
71 sk_sp<sksg::Matrix> parentMatrix) {
72 if (!t.isObject())
73 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -050074
Florin Malita18eafd92018-01-04 21:11:55 -050075 auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
Florin Malitaa6e30f72018-03-23 13:41:58 -040076 auto adapter = sk_make_sp<TransformAdapter>(matrix);
Florin Malitafc807c82018-01-25 22:35:09 -050077 auto anchor_attached = BindProperty<VectorValue>(t["a"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040078 [adapter](const VectorValue& a) {
79 adapter->setAnchorPoint(ValueTraits<VectorValue>::As<SkPoint>(a));
Florin Malita094ccde2017-12-30 12:27:00 -050080 });
Florin Malitafc807c82018-01-25 22:35:09 -050081 auto position_attached = BindProperty<VectorValue>(t["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040082 [adapter](const VectorValue& p) {
83 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malita094ccde2017-12-30 12:27:00 -050084 });
Florin Malitafc807c82018-01-25 22:35:09 -050085 auto scale_attached = BindProperty<VectorValue>(t["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040086 [adapter](const VectorValue& s) {
87 adapter->setScale(ValueTraits<VectorValue>::As<SkVector>(s));
Florin Malita094ccde2017-12-30 12:27:00 -050088 });
Florin Malita1eb98db2018-01-26 15:03:38 -050089
90 auto* jrotation = &t["r"];
91 if (jrotation->isNull()) {
92 // 3d rotations have separate rx,ry,rz components. While we don't fully support them,
93 // we can still make use of rz.
94 jrotation = &t["rz"];
95 }
96 auto rotation_attached = BindProperty<ScalarValue>(*jrotation, &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040097 [adapter](const ScalarValue& r) {
98 adapter->setRotation(r);
Florin Malita094ccde2017-12-30 12:27:00 -050099 });
Florin Malitafc807c82018-01-25 22:35:09 -0500100 auto skew_attached = BindProperty<ScalarValue>(t["sk"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400101 [adapter](const ScalarValue& sk) {
102 adapter->setSkew(sk);
Florin Malita094ccde2017-12-30 12:27:00 -0500103 });
Florin Malitafc807c82018-01-25 22:35:09 -0500104 auto skewaxis_attached = BindProperty<ScalarValue>(t["sa"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400105 [adapter](const ScalarValue& sa) {
106 adapter->setSkewAxis(sa);
Florin Malita094ccde2017-12-30 12:27:00 -0500107 });
108
109 if (!anchor_attached &&
110 !position_attached &&
111 !scale_attached &&
112 !rotation_attached &&
113 !skew_attached &&
114 !skewaxis_attached) {
115 LogFail(t, "Could not parse transform");
Florin Malita18eafd92018-01-04 21:11:55 -0500116 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500117 }
118
Florin Malita18eafd92018-01-04 21:11:55 -0500119 return matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500120}
121
Florin Malitac0034172018-01-08 16:42:59 -0500122sk_sp<sksg::RenderNode> AttachOpacity(const Json::Value& jtransform, AttachContext* ctx,
123 sk_sp<sksg::RenderNode> childNode) {
124 if (!jtransform.isObject() || !childNode)
125 return childNode;
126
127 // This is more peeky than other attachers, because we want to avoid redundant opacity
128 // nodes for the extremely common case of static opaciy == 100.
129 const auto& opacity = jtransform["o"];
130 if (opacity.isObject() &&
Florin Malitacf8ed522018-01-25 15:27:33 -0500131 !ParseDefault(opacity["a"], true) &&
132 ParseDefault(opacity["k"], -1) == 100) {
Florin Malitac0034172018-01-08 16:42:59 -0500133 // Ignoring static full opacity.
134 return childNode;
135 }
136
137 auto opacityNode = sksg::OpacityEffect::Make(childNode);
Florin Malitafc807c82018-01-25 22:35:09 -0500138 BindProperty<ScalarValue>(opacity, &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500139 [opacityNode](const ScalarValue& o) {
Florin Malitac0034172018-01-08 16:42:59 -0500140 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500141 opacityNode->setOpacity(o * 0.01f);
Florin Malitac0034172018-01-08 16:42:59 -0500142 });
143
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400144 return std::move(opacityNode);
Florin Malitac0034172018-01-08 16:42:59 -0500145}
146
Florin Malita094ccde2017-12-30 12:27:00 -0500147sk_sp<sksg::RenderNode> AttachComposition(const Json::Value&, AttachContext* ctx);
148
Florin Malita25366fa2018-01-23 13:37:59 -0500149sk_sp<sksg::Path> AttachPath(const Json::Value& jpath, AttachContext* ctx) {
150 auto path_node = sksg::Path::Make();
Florin Malitafc807c82018-01-25 22:35:09 -0500151 return BindProperty<ShapeValue>(jpath, &ctx->fAnimators,
152 [path_node](const ShapeValue& p) { path_node->setPath(p); })
Florin Malita25366fa2018-01-23 13:37:59 -0500153 ? path_node
154 : nullptr;
155}
156
Florin Malita094ccde2017-12-30 12:27:00 -0500157sk_sp<sksg::GeometryNode> AttachPathGeometry(const Json::Value& jpath, AttachContext* ctx) {
158 SkASSERT(jpath.isObject());
159
Florin Malita25366fa2018-01-23 13:37:59 -0500160 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500161}
162
Florin Malita2e1d7e22018-01-02 10:40:00 -0500163sk_sp<sksg::GeometryNode> AttachRRectGeometry(const Json::Value& jrect, AttachContext* ctx) {
164 SkASSERT(jrect.isObject());
165
166 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400167 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500168
Florin Malitafc807c82018-01-25 22:35:09 -0500169 auto p_attached = BindProperty<VectorValue>(jrect["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400170 [adapter](const VectorValue& p) {
171 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500172 });
Florin Malitafc807c82018-01-25 22:35:09 -0500173 auto s_attached = BindProperty<VectorValue>(jrect["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400174 [adapter](const VectorValue& s) {
175 adapter->setSize(ValueTraits<VectorValue>::As<SkSize>(s));
Florin Malitaf9590922018-01-09 11:56:09 -0500176 });
Florin Malitafc807c82018-01-25 22:35:09 -0500177 auto r_attached = BindProperty<ScalarValue>(jrect["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400178 [adapter](const ScalarValue& r) {
179 adapter->setRadius(SkSize::Make(r, r));
Florin Malitaf9590922018-01-09 11:56:09 -0500180 });
Florin Malita2e1d7e22018-01-02 10:40:00 -0500181
182 if (!p_attached && !s_attached && !r_attached) {
183 return nullptr;
184 }
185
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400186 return std::move(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500187}
188
189sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const Json::Value& jellipse, AttachContext* ctx) {
190 SkASSERT(jellipse.isObject());
191
192 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400193 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500194
Florin Malitafc807c82018-01-25 22:35:09 -0500195 auto p_attached = BindProperty<VectorValue>(jellipse["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400196 [adapter](const VectorValue& p) {
197 adapter->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 Malitaa6e30f72018-03-23 13:41:58 -0400200 [adapter](const VectorValue& s) {
Florin Malitaf9590922018-01-09 11:56:09 -0500201 const auto sz = ValueTraits<VectorValue>::As<SkSize>(s);
Florin Malitaa6e30f72018-03-23 13:41:58 -0400202 adapter->setSize(sz);
203 adapter->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
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400210 return std::move(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500211}
212
Florin Malita02a32b02018-01-04 11:27:09 -0500213sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const Json::Value& jstar, AttachContext* ctx) {
214 SkASSERT(jstar.isObject());
215
Florin Malitaa6e30f72018-03-23 13:41:58 -0400216 static constexpr PolyStarAdapter::Type gTypes[] = {
217 PolyStarAdapter::Type::kStar, // "sy": 1
218 PolyStarAdapter::Type::kPoly, // "sy": 2
Florin Malita02a32b02018-01-04 11:27:09 -0500219 };
220
Florin Malitacf8ed522018-01-25 15:27:33 -0500221 const auto type = ParseDefault(jstar["sy"], 0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500222 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
223 LogFail(jstar, "Unknown polystar type");
224 return nullptr;
225 }
226
227 auto path_node = sksg::Path::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400228 auto adapter = sk_make_sp<PolyStarAdapter>(path_node, gTypes[type]);
Florin Malita02a32b02018-01-04 11:27:09 -0500229
Florin Malitafc807c82018-01-25 22:35:09 -0500230 BindProperty<VectorValue>(jstar["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400231 [adapter](const VectorValue& p) {
232 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500233 });
Florin Malitafc807c82018-01-25 22:35:09 -0500234 BindProperty<ScalarValue>(jstar["pt"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400235 [adapter](const ScalarValue& pt) {
236 adapter->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500237 });
Florin Malitafc807c82018-01-25 22:35:09 -0500238 BindProperty<ScalarValue>(jstar["ir"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400239 [adapter](const ScalarValue& ir) {
240 adapter->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500241 });
Florin Malitafc807c82018-01-25 22:35:09 -0500242 BindProperty<ScalarValue>(jstar["or"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400243 [adapter](const ScalarValue& otr) {
244 adapter->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500245 });
Florin Malitafc807c82018-01-25 22:35:09 -0500246 BindProperty<ScalarValue>(jstar["is"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400247 [adapter](const ScalarValue& is) {
248 adapter->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500249 });
Florin Malitafc807c82018-01-25 22:35:09 -0500250 BindProperty<ScalarValue>(jstar["os"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400251 [adapter](const ScalarValue& os) {
252 adapter->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500253 });
Florin Malitafc807c82018-01-25 22:35:09 -0500254 BindProperty<ScalarValue>(jstar["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400255 [adapter](const ScalarValue& r) {
256 adapter->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500257 });
Florin Malita02a32b02018-01-04 11:27:09 -0500258
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400259 return std::move(path_node);
Florin Malita02a32b02018-01-04 11:27:09 -0500260}
261
Florin Malita6aaee592018-01-12 12:25:09 -0500262sk_sp<sksg::Color> AttachColor(const Json::Value& obj, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500263 SkASSERT(obj.isObject());
264
265 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malitafc807c82018-01-25 22:35:09 -0500266 auto color_attached = BindProperty<VectorValue>(obj["c"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500267 [color_node](const VectorValue& c) {
268 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500269 });
Florin Malita094ccde2017-12-30 12:27:00 -0500270
Florin Malita1586d852018-01-12 14:27:39 -0500271 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500272}
273
Florin Malita6aaee592018-01-12 12:25:09 -0500274sk_sp<sksg::Gradient> AttachGradient(const Json::Value& obj, AttachContext* ctx) {
275 SkASSERT(obj.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500276
Florin Malita6aaee592018-01-12 12:25:09 -0500277 const auto& stops = obj["g"];
278 if (!stops.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500279 return nullptr;
280
Florin Malitacf8ed522018-01-25 15:27:33 -0500281 const auto stopCount = ParseDefault(stops["p"], -1);
Florin Malita6aaee592018-01-12 12:25:09 -0500282 if (stopCount < 0)
283 return nullptr;
284
285 sk_sp<sksg::Gradient> gradient_node;
Florin Malitaa6e30f72018-03-23 13:41:58 -0400286 sk_sp<GradientAdapter> adapter;
Florin Malita6aaee592018-01-12 12:25:09 -0500287
Florin Malitacf8ed522018-01-25 15:27:33 -0500288 if (ParseDefault(obj["t"], 1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500289 auto linear_node = sksg::LinearGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400290 adapter = sk_make_sp<LinearGradientAdapter>(linear_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500291 gradient_node = std::move(linear_node);
292 } else {
293 auto radial_node = sksg::RadialGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400294 adapter = sk_make_sp<RadialGradientAdapter>(radial_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500295
296 // TODO: highlight, angle
297 gradient_node = std::move(radial_node);
298 }
299
Florin Malitafc807c82018-01-25 22:35:09 -0500300 BindProperty<VectorValue>(stops["k"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400301 [adapter](const VectorValue& stops) {
302 adapter->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500303 });
Florin Malitafc807c82018-01-25 22:35:09 -0500304 BindProperty<VectorValue>(obj["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400305 [adapter](const VectorValue& s) {
306 adapter->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500307 });
Florin Malitafc807c82018-01-25 22:35:09 -0500308 BindProperty<VectorValue>(obj["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400309 [adapter](const VectorValue& e) {
310 adapter->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500311 });
312
313 return gradient_node;
314}
315
Florin Malita1586d852018-01-12 14:27:39 -0500316sk_sp<sksg::PaintNode> AttachPaint(const Json::Value& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500317 sk_sp<sksg::PaintNode> paint_node) {
318 if (paint_node) {
319 paint_node->setAntiAlias(true);
320
Florin Malitafc807c82018-01-25 22:35:09 -0500321 BindProperty<ScalarValue>(jpaint["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500322 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500323 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500324 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500325 });
Florin Malita6aaee592018-01-12 12:25:09 -0500326 }
327
328 return paint_node;
329}
330
331sk_sp<sksg::PaintNode> AttachStroke(const Json::Value& jstroke, AttachContext* ctx,
332 sk_sp<sksg::PaintNode> stroke_node) {
333 SkASSERT(jstroke.isObject());
334
335 if (!stroke_node)
336 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500337
338 stroke_node->setStyle(SkPaint::kStroke_Style);
339
Florin Malitafc807c82018-01-25 22:35:09 -0500340 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500341 [stroke_node](const ScalarValue& w) {
342 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500343 });
Florin Malita094ccde2017-12-30 12:27:00 -0500344 if (!width_attached)
345 return nullptr;
346
Florin Malitacf8ed522018-01-25 15:27:33 -0500347 stroke_node->setStrokeMiter(ParseDefault(jstroke["ml"], 4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500348
349 static constexpr SkPaint::Join gJoins[] = {
350 SkPaint::kMiter_Join,
351 SkPaint::kRound_Join,
352 SkPaint::kBevel_Join,
353 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500354 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(ParseDefault(jstroke["lj"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500355 0, SK_ARRAY_COUNT(gJoins) - 1)]);
356
357 static constexpr SkPaint::Cap gCaps[] = {
358 SkPaint::kButt_Cap,
359 SkPaint::kRound_Cap,
360 SkPaint::kSquare_Cap,
361 };
Florin Malitacf8ed522018-01-25 15:27:33 -0500362 stroke_node->setStrokeCap(gCaps[SkTPin<int>(ParseDefault(jstroke["lc"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500363 0, SK_ARRAY_COUNT(gCaps) - 1)]);
364
365 return stroke_node;
366}
367
Florin Malita6aaee592018-01-12 12:25:09 -0500368sk_sp<sksg::PaintNode> AttachColorFill(const Json::Value& jfill, AttachContext* ctx) {
369 SkASSERT(jfill.isObject());
370
371 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
372}
373
374sk_sp<sksg::PaintNode> AttachGradientFill(const Json::Value& jfill, AttachContext* ctx) {
375 SkASSERT(jfill.isObject());
376
377 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
378}
379
380sk_sp<sksg::PaintNode> AttachColorStroke(const Json::Value& jstroke, AttachContext* ctx) {
381 SkASSERT(jstroke.isObject());
382
383 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
384}
385
386sk_sp<sksg::PaintNode> AttachGradientStroke(const Json::Value& jstroke, AttachContext* ctx) {
387 SkASSERT(jstroke.isObject());
388
389 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
390}
391
Florin Malitae6345d92018-01-03 23:37:54 -0500392std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
393 const Json::Value& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
394 std::vector<sk_sp<sksg::GeometryNode>> merged;
395
396 static constexpr sksg::Merge::Mode gModes[] = {
397 sksg::Merge::Mode::kMerge, // "mm": 1
398 sksg::Merge::Mode::kUnion, // "mm": 2
399 sksg::Merge::Mode::kDifference, // "mm": 3
400 sksg::Merge::Mode::kIntersect, // "mm": 4
401 sksg::Merge::Mode::kXOR , // "mm": 5
402 };
403
Florin Malitacf8ed522018-01-25 15:27:33 -0500404 const auto mode = gModes[SkTPin<int>(ParseDefault(jmerge["mm"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500405 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malitae6345d92018-01-03 23:37:54 -0500406 merged.push_back(sksg::Merge::Make(std::move(geos), mode));
407
Florin Malitae6345d92018-01-03 23:37:54 -0500408 return merged;
409}
410
Florin Malita51b8c892018-01-07 08:54:24 -0500411std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
412 const Json::Value& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
413
414 enum class Mode {
415 kMerged, // "m": 1
416 kSeparate, // "m": 2
417 } gModes[] = { Mode::kMerged, Mode::kSeparate };
418
Florin Malitacf8ed522018-01-25 15:27:33 -0500419 const auto mode = gModes[SkTPin<int>(ParseDefault(jtrim["m"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500420 0, SK_ARRAY_COUNT(gModes) - 1)];
421
422 std::vector<sk_sp<sksg::GeometryNode>> inputs;
423 if (mode == Mode::kMerged) {
424 inputs.push_back(sksg::Merge::Make(std::move(geos), sksg::Merge::Mode::kMerge));
425 } else {
426 inputs = std::move(geos);
427 }
428
429 std::vector<sk_sp<sksg::GeometryNode>> trimmed;
430 trimmed.reserve(inputs.size());
431 for (const auto& i : inputs) {
Florin Malita69526b02018-03-22 12:20:02 -0400432 const auto trimEffect = sksg::TrimEffect::Make(i);
433 trimmed.push_back(trimEffect);
434
Florin Malitaa6e30f72018-03-23 13:41:58 -0400435 const auto adapter = sk_make_sp<TrimEffectAdapter>(std::move(trimEffect));
Florin Malitafc807c82018-01-25 22:35:09 -0500436 BindProperty<ScalarValue>(jtrim["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400437 [adapter](const ScalarValue& s) {
438 adapter->setStart(s);
Florin Malita51b8c892018-01-07 08:54:24 -0500439 });
Florin Malitafc807c82018-01-25 22:35:09 -0500440 BindProperty<ScalarValue>(jtrim["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400441 [adapter](const ScalarValue& e) {
442 adapter->setEnd(e);
Florin Malita51b8c892018-01-07 08:54:24 -0500443 });
Florin Malitafc807c82018-01-25 22:35:09 -0500444 BindProperty<ScalarValue>(jtrim["o"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400445 [adapter](const ScalarValue& o) {
446 adapter->setOffset(o);
Florin Malita51b8c892018-01-07 08:54:24 -0500447 });
448 }
449
450 return trimmed;
451}
452
Florin Malita094ccde2017-12-30 12:27:00 -0500453using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const Json::Value&, AttachContext*);
454static constexpr GeometryAttacherT gGeometryAttachers[] = {
455 AttachPathGeometry,
Florin Malita2e1d7e22018-01-02 10:40:00 -0500456 AttachRRectGeometry,
Florin Malitafbc13f12018-01-04 10:26:35 -0500457 AttachEllipseGeometry,
Florin Malita02a32b02018-01-04 11:27:09 -0500458 AttachPolystarGeometry,
Florin Malita094ccde2017-12-30 12:27:00 -0500459};
460
461using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const Json::Value&, AttachContext*);
462static constexpr PaintAttacherT gPaintAttachers[] = {
Florin Malita6aaee592018-01-12 12:25:09 -0500463 AttachColorFill,
464 AttachColorStroke,
465 AttachGradientFill,
466 AttachGradientStroke,
Florin Malita094ccde2017-12-30 12:27:00 -0500467};
468
Florin Malitae6345d92018-01-03 23:37:54 -0500469using GeometryEffectAttacherT =
470 std::vector<sk_sp<sksg::GeometryNode>> (*)(const Json::Value&,
471 AttachContext*,
472 std::vector<sk_sp<sksg::GeometryNode>>&&);
473static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
474 AttachMergeGeometryEffect,
Florin Malita51b8c892018-01-07 08:54:24 -0500475 AttachTrimGeometryEffect,
Florin Malitae6345d92018-01-03 23:37:54 -0500476};
477
Florin Malita094ccde2017-12-30 12:27:00 -0500478enum class ShapeType {
479 kGeometry,
Florin Malitae6345d92018-01-03 23:37:54 -0500480 kGeometryEffect,
Florin Malita094ccde2017-12-30 12:27:00 -0500481 kPaint,
482 kGroup,
Florin Malitadacc02b2017-12-31 09:12:31 -0500483 kTransform,
Florin Malita094ccde2017-12-30 12:27:00 -0500484};
485
486struct ShapeInfo {
487 const char* fTypeString;
488 ShapeType fShapeType;
489 uint32_t fAttacherIndex; // index into respective attacher tables
490};
491
492const ShapeInfo* FindShapeInfo(const Json::Value& shape) {
493 static constexpr ShapeInfo gShapeInfo[] = {
Florin Malitafbc13f12018-01-04 10:26:35 -0500494 { "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500495 { "fl", ShapeType::kPaint , 0 }, // fill -> AttachColorFill
496 { "gf", ShapeType::kPaint , 2 }, // gfill -> AttachGradientFill
Florin Malita16d0ad02018-01-19 15:07:29 -0500497 { "gr", ShapeType::kGroup , 0 }, // group -> Inline handler
Florin Malita6aaee592018-01-12 12:25:09 -0500498 { "gs", ShapeType::kPaint , 3 }, // gstroke -> AttachGradientStroke
Florin Malitae6345d92018-01-03 23:37:54 -0500499 { "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
Florin Malita02a32b02018-01-04 11:27:09 -0500500 { "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
Florin Malitae6345d92018-01-03 23:37:54 -0500501 { "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
Florin Malita02a32b02018-01-04 11:27:09 -0500502 { "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500503 { "st", ShapeType::kPaint , 1 }, // stroke -> AttachColorStroke
Florin Malita51b8c892018-01-07 08:54:24 -0500504 { "tm", ShapeType::kGeometryEffect, 1 }, // trim -> AttachTrimGeometryEffect
Florin Malita16d0ad02018-01-19 15:07:29 -0500505 { "tr", ShapeType::kTransform , 0 }, // transform -> Inline handler
Florin Malita094ccde2017-12-30 12:27:00 -0500506 };
507
508 if (!shape.isObject())
509 return nullptr;
510
511 const auto& type = shape["ty"];
512 if (!type.isString())
513 return nullptr;
514
515 const auto* info = bsearch(type.asCString(),
516 gShapeInfo,
517 SK_ARRAY_COUNT(gShapeInfo),
518 sizeof(ShapeInfo),
519 [](const void* key, const void* info) {
520 return strcmp(static_cast<const char*>(key),
521 static_cast<const ShapeInfo*>(info)->fTypeString);
522 });
523
524 return static_cast<const ShapeInfo*>(info);
525}
526
Florin Malita16d0ad02018-01-19 15:07:29 -0500527struct GeometryEffectRec {
528 const Json::Value& fJson;
529 GeometryEffectAttacherT fAttach;
530};
531
Florin Malitaca4439f2018-01-23 10:31:59 -0500532struct AttachShapeContext {
533 AttachShapeContext(AttachContext* ctx,
534 std::vector<sk_sp<sksg::GeometryNode>>* geos,
535 std::vector<GeometryEffectRec>* effects,
536 size_t committedAnimators)
537 : fCtx(ctx)
538 , fGeometryStack(geos)
539 , fGeometryEffectStack(effects)
540 , fCommittedAnimators(committedAnimators) {}
541
542 AttachContext* fCtx;
543 std::vector<sk_sp<sksg::GeometryNode>>* fGeometryStack;
544 std::vector<GeometryEffectRec>* fGeometryEffectStack;
545 size_t fCommittedAnimators;
546};
547
548sk_sp<sksg::RenderNode> AttachShape(const Json::Value& jshape, AttachShapeContext* shapeCtx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500549 if (!jshape.isArray())
Florin Malita094ccde2017-12-30 12:27:00 -0500550 return nullptr;
551
Florin Malitaca4439f2018-01-23 10:31:59 -0500552 SkDEBUGCODE(const auto initialGeometryEffects = shapeCtx->fGeometryEffectStack->size();)
Florin Malita094ccde2017-12-30 12:27:00 -0500553
Florin Malita16d0ad02018-01-19 15:07:29 -0500554 sk_sp<sksg::Group> shape_group = sksg::Group::Make();
555 sk_sp<sksg::RenderNode> shape_wrapper = shape_group;
556 sk_sp<sksg::Matrix> shape_matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500557
Florin Malita16d0ad02018-01-19 15:07:29 -0500558 struct ShapeRec {
559 const Json::Value& fJson;
560 const ShapeInfo& fInfo;
561 };
562
563 // First pass (bottom->top):
564 //
565 // * pick up the group transform and opacity
566 // * push local geometry effects onto the stack
567 // * store recs for next pass
568 //
569 std::vector<ShapeRec> recs;
570 for (Json::ArrayIndex i = 0; i < jshape.size(); ++i) {
571 const auto& s = jshape[jshape.size() - 1 - i];
Florin Malita094ccde2017-12-30 12:27:00 -0500572 const auto* info = FindShapeInfo(s);
573 if (!info) {
574 LogFail(s.isObject() ? s["ty"] : s, "Unknown shape");
575 continue;
576 }
577
Florin Malita16d0ad02018-01-19 15:07:29 -0500578 recs.push_back({ s, *info });
579
Florin Malita094ccde2017-12-30 12:27:00 -0500580 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500581 case ShapeType::kTransform:
Florin Malitaca4439f2018-01-23 10:31:59 -0500582 if ((shape_matrix = AttachMatrix(s, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500583 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
584 }
Florin Malitaca4439f2018-01-23 10:31:59 -0500585 shape_wrapper = AttachOpacity(s, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500586 break;
587 case ShapeType::kGeometryEffect:
588 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500589 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita16d0ad02018-01-19 15:07:29 -0500590 { s, gGeometryEffectAttachers[info->fAttacherIndex] });
591 break;
592 default:
593 break;
594 }
595 }
596
597 // Second pass (top -> bottom, after 2x reverse):
598 //
599 // * track local geometry
600 // * emit local paints
601 //
602 std::vector<sk_sp<sksg::GeometryNode>> geos;
603 std::vector<sk_sp<sksg::RenderNode >> draws;
604 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
605 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500606 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500607 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500608 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
609 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500610 geos.push_back(std::move(geo));
611 }
612 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500613 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500614 // Apply the current effect and pop from the stack.
615 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500616 if (!geos.empty()) {
617 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500618 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500619 std::move(geos));
620 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500621
Florin Malitaca4439f2018-01-23 10:31:59 -0500622 SkASSERT(shapeCtx->fGeometryEffectStack->back().fJson == rec->fJson);
623 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500624 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500625 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500626 } break;
627 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500628 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
629 &geos,
630 shapeCtx->fGeometryEffectStack,
631 shapeCtx->fCommittedAnimators);
632 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500633 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500634 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
635 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500636 }
637 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500638 case ShapeType::kPaint: {
639 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500640 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500641 if (!paint || geos.empty())
642 break;
643
644 auto drawGeos = geos;
645
646 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500647 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
648 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
649 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500650 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500651
652 // If we still have multiple geos, reduce using 'merge'.
653 auto geo = drawGeos.size() > 1
654 ? sksg::Merge::Make(std::move(drawGeos), sksg::Merge::Mode::kMerge)
655 : drawGeos[0];
656
657 SkASSERT(geo);
658 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500659 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500660 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500661 default:
662 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500663 }
664 }
665
Florin Malita16d0ad02018-01-19 15:07:29 -0500666 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500667 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500668
669 // Push transformed local geometries to parent list, for subsequent paints.
670 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500671 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500672 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
673 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500674 }
675
Florin Malita16d0ad02018-01-19 15:07:29 -0500676 // Emit local draws reversed (bottom->top, per spec).
677 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
678 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500679 }
680
Florin Malita16d0ad02018-01-19 15:07:29 -0500681 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500682}
683
Florin Malita1022f742018-02-23 11:10:22 -0500684sk_sp<sksg::RenderNode> AttachNestedAnimation(const char* path, AttachContext* ctx) {
685 class SkottieSGAdapter final : public sksg::RenderNode {
686 public:
687 explicit SkottieSGAdapter(sk_sp<Animation> animation)
688 : fAnimation(std::move(animation)) {
689 SkASSERT(fAnimation);
690 }
691
692 protected:
693 SkRect onRevalidate(sksg::InvalidationController*, const SkMatrix&) override {
694 return SkRect::MakeSize(fAnimation->size());
695 }
696
697 void onRender(SkCanvas* canvas) const override {
698 fAnimation->render(canvas);
699 }
700
701 private:
702 const sk_sp<Animation> fAnimation;
703 };
704
705 class SkottieAnimatorAdapter final : public sksg::Animator {
706 public:
707 SkottieAnimatorAdapter(sk_sp<Animation> animation, float frameRate)
708 : fAnimation(std::move(animation))
709 , fFrameRate(frameRate) {
710 SkASSERT(fAnimation);
711 SkASSERT(fFrameRate > 0);
712 }
713
714 protected:
715 void onTick(float t) {
716 // map back from frame # to ms.
717 const auto t_ms = t * 1000 / fFrameRate;
718 fAnimation->animationTick(t_ms);
719 }
720
721 private:
722 const sk_sp<Animation> fAnimation;
723 const float fFrameRate;
724 };
725
726 const auto resStream = ctx->fResources.openStream(path);
727 if (!resStream || !resStream->hasLength()) {
728 LOG("!! Could not open: %s\n", path);
729 return nullptr;
730 }
731
732 auto animation = Animation::Make(resStream.get(), ctx->fResources);
733 if (!animation) {
734 LOG("!! Could not load nested animation: %s\n", path);
735 return nullptr;
736 }
737
738 ctx->fAnimators.push_back(skstd::make_unique<SkottieAnimatorAdapter>(animation,
739 ctx->fFrameRate));
740
741 return sk_make_sp<SkottieSGAdapter>(std::move(animation));
742}
743
Florin Malitaeb87d672018-01-29 15:28:24 -0500744sk_sp<sksg::RenderNode> AttachCompLayer(const Json::Value& jlayer, AttachContext* ctx,
745 float* time_bias, float* time_scale) {
746 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500747
Florin Malitacf8ed522018-01-25 15:27:33 -0500748 SkString refId;
Florin Malitaeb87d672018-01-29 15:28:24 -0500749 if (!Parse(jlayer["refId"], &refId) || refId.isEmpty()) {
Florin Malitadd22cf962018-01-29 15:42:01 +0000750 LOG("!! Comp layer missing refId\n");
Florin Malita094ccde2017-12-30 12:27:00 -0500751 return nullptr;
752 }
753
Florin Malitaeb87d672018-01-29 15:28:24 -0500754 const auto start_time = ParseDefault(jlayer["st"], 0.0f),
755 stretch_time = ParseDefault(jlayer["sr"], 1.0f);
756
757 *time_bias = -start_time;
758 *time_scale = 1 / stretch_time;
759 if (SkScalarIsNaN(*time_scale)) {
760 *time_scale = 1;
761 }
762
Florin Malita1022f742018-02-23 11:10:22 -0500763 if (refId.startsWith("$")) {
764 return AttachNestedAnimation(refId.c_str() + 1, ctx);
765 }
766
767 const auto* comp = ctx->fAssets.find(refId);
768 if (!comp) {
769 LOG("!! Pre-comp not found: '%s'\n", refId.c_str());
770 return nullptr;
771 }
772
Florin Malitadd22cf962018-01-29 15:42:01 +0000773 // TODO: cycle detection
774 return AttachComposition(**comp, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500775}
776
Florin Malitaeb87d672018-01-29 15:28:24 -0500777sk_sp<sksg::RenderNode> AttachSolidLayer(const Json::Value& jlayer, AttachContext*,
778 float*, float*) {
Florin Malita0e66fba2018-01-09 17:10:18 -0500779 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500780
Florin Malitacf8ed522018-01-25 15:27:33 -0500781 const auto size = SkSize::Make(ParseDefault(jlayer["sw"], 0.0f),
782 ParseDefault(jlayer["sh"], 0.0f));
783 const auto hex = ParseDefault(jlayer["sc"], SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500784 uint32_t c;
785 if (size.isEmpty() ||
786 !hex.startsWith("#") ||
787 !SkParse::FindHex(hex.c_str() + 1, &c)) {
788 LogFail(jlayer, "Could not parse solid layer");
789 return nullptr;
790 }
791
792 const SkColor color = 0xff000000 | c;
793
794 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
795 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500796}
797
Florin Malita49328072018-01-08 12:51:12 -0500798sk_sp<sksg::RenderNode> AttachImageAsset(const Json::Value& jimage, AttachContext* ctx) {
799 SkASSERT(jimage.isObject());
800
Florin Malitacf8ed522018-01-25 15:27:33 -0500801 const auto name = ParseDefault(jimage["p"], SkString()),
802 path = ParseDefault(jimage["u"], SkString());
Florin Malita49328072018-01-08 12:51:12 -0500803 if (name.isEmpty())
804 return nullptr;
805
806 // TODO: plumb resource paths explicitly to ResourceProvider?
807 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
808 const auto resStream = ctx->fResources.openStream(resName.c_str());
809 if (!resStream || !resStream->hasLength()) {
810 LOG("!! Could not load image resource: %s\n", resName.c_str());
811 return nullptr;
812 }
813
814 // TODO: non-intrisic image sizing
815 return sksg::Image::Make(
816 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
817}
818
Florin Malitaeb87d672018-01-29 15:28:24 -0500819sk_sp<sksg::RenderNode> AttachImageLayer(const Json::Value& layer, AttachContext* ctx,
820 float*, float*) {
Florin Malitadd22cf962018-01-29 15:42:01 +0000821 SkASSERT(layer.isObject());
822
823 SkString refId;
824 if (!Parse(layer["refId"], &refId) || refId.isEmpty()) {
825 LOG("!! Image layer missing refId\n");
826 return nullptr;
827 }
828
829 const auto* jimage = ctx->fAssets.find(refId);
830 if (!jimage) {
831 LOG("!! Image asset not found: '%s'\n", refId.c_str());
832 return nullptr;
833 }
834
835 return AttachImageAsset(**jimage, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500836}
837
Florin Malitaeb87d672018-01-29 15:28:24 -0500838sk_sp<sksg::RenderNode> AttachNullLayer(const Json::Value& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500839 SkASSERT(layer.isObject());
840
Florin Malita18eafd92018-01-04 21:11:55 -0500841 // Null layers are used solely to drive dependent transforms,
842 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500843 return nullptr;
844}
845
Florin Malitaeb87d672018-01-29 15:28:24 -0500846sk_sp<sksg::RenderNode> AttachShapeLayer(const Json::Value& layer, AttachContext* ctx,
847 float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500848 SkASSERT(layer.isObject());
849
Florin Malita16d0ad02018-01-19 15:07:29 -0500850 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
851 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500852 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
853 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
854
855 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
856 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
857 // due to attached animators. To avoid this, we track committed animators and discard the
858 // orphans here.
859 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
860 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
861
862 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500863}
864
Florin Malitaeb87d672018-01-29 15:28:24 -0500865sk_sp<sksg::RenderNode> AttachTextLayer(const Json::Value& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500866 SkASSERT(layer.isObject());
867
868 LOG("?? Text layer stub\n");
869 return nullptr;
870}
871
Florin Malita18eafd92018-01-04 21:11:55 -0500872struct AttachLayerContext {
873 AttachLayerContext(const Json::Value& jlayers, AttachContext* ctx)
874 : fLayerList(jlayers), fCtx(ctx) {}
875
Florin Malita4a490682018-01-28 14:27:51 -0500876 const Json::Value& fLayerList;
877 AttachContext* fCtx;
878 SkTHashMap<int, sk_sp<sksg::Matrix>> fLayerMatrixMap;
879 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500880
Florin Malita4a490682018-01-28 14:27:51 -0500881 sk_sp<sksg::Matrix> AttachParentLayerMatrix(const Json::Value& jlayer) {
882 SkASSERT(jlayer.isObject());
Florin Malita18eafd92018-01-04 21:11:55 -0500883 SkASSERT(fLayerList.isArray());
884
Florin Malita4a490682018-01-28 14:27:51 -0500885 const auto parent_index = ParseDefault(jlayer["parent"], -1);
886 if (parent_index < 0)
Florin Malita18eafd92018-01-04 21:11:55 -0500887 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500888
Florin Malita4a490682018-01-28 14:27:51 -0500889 if (auto* m = fLayerMatrixMap.find(parent_index))
890 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500891
892 for (const auto& l : fLayerList) {
893 if (!l.isObject()) {
894 continue;
895 }
896
Florin Malita4a490682018-01-28 14:27:51 -0500897 if (ParseDefault(l["ind"], -1) == parent_index) {
898 return this->AttachLayerMatrix(l);
Florin Malita18eafd92018-01-04 21:11:55 -0500899 }
900 }
901
902 return nullptr;
903 }
904
905 sk_sp<sksg::Matrix> AttachLayerMatrix(const Json::Value& jlayer) {
906 SkASSERT(jlayer.isObject());
907
Florin Malita4a490682018-01-28 14:27:51 -0500908 const auto layer_index = ParseDefault(jlayer["ind"], -1);
909 if (layer_index < 0)
910 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500911
Florin Malita4a490682018-01-28 14:27:51 -0500912 if (auto* m = fLayerMatrixMap.find(layer_index))
913 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500914
Florin Malita4a490682018-01-28 14:27:51 -0500915 // Add a stub entry to break recursion cycles.
916 fLayerMatrixMap.set(layer_index, nullptr);
Florin Malita18eafd92018-01-04 21:11:55 -0500917
Florin Malita4a490682018-01-28 14:27:51 -0500918 auto parent_matrix = this->AttachParentLayerMatrix(jlayer);
Florin Malita18eafd92018-01-04 21:11:55 -0500919
Florin Malita4a490682018-01-28 14:27:51 -0500920 return *fLayerMatrixMap.set(layer_index,
921 AttachMatrix(jlayer["ks"],
922 fCtx,
923 this->AttachParentLayerMatrix(jlayer)));
Florin Malita18eafd92018-01-04 21:11:55 -0500924 }
925};
926
Florin Malita25366fa2018-01-23 13:37:59 -0500927SkBlendMode MaskBlendMode(char mode) {
928 switch (mode) {
929 case 'a': return SkBlendMode::kSrcOver; // Additive
930 case 's': return SkBlendMode::kExclusion; // Subtract
931 case 'i': return SkBlendMode::kDstIn; // Intersect
932 case 'l': return SkBlendMode::kLighten; // Lighten
933 case 'd': return SkBlendMode::kDarken; // Darken
934 case 'f': return SkBlendMode::kDifference; // Difference
935 default: break;
936 }
937
938 return SkBlendMode::kSrcOver;
939}
940
941sk_sp<sksg::RenderNode> AttachMask(const Json::Value& jmask,
942 AttachContext* ctx,
943 sk_sp<sksg::RenderNode> childNode) {
944 if (!jmask.isArray())
945 return childNode;
946
Florin Malita0c51c212018-04-26 14:13:14 -0400947 struct MaskRecord {
948 sk_sp<sksg::Path> mask_path;
949 sk_sp<sksg::Color> mask_paint;
950 };
951
952 SkSTArray<4, MaskRecord, true> mask_stack;
953
954 bool opaque_mask = true;
Florin Malita25366fa2018-01-23 13:37:59 -0500955
956 for (const auto& m : jmask) {
957 if (!m.isObject())
958 continue;
959
Florin Malita25366fa2018-01-23 13:37:59 -0500960 auto mask_path = AttachPath(m["pt"], ctx);
961 if (!mask_path) {
962 LogFail(m, "Could not parse mask path");
963 continue;
964 }
965
Florin Malitacd05b192018-04-25 21:43:03 -0400966 mask_path->setFillType(ParseDefault(m["inv"], false)
967 ? SkPath::kInverseWinding_FillType
968 : SkPath::kWinding_FillType);
969
Florin Malitacf8ed522018-01-25 15:27:33 -0500970 SkString mode;
971 if (!Parse(m["mode"], &mode) ||
972 mode.size() != 1 ||
973 !strcmp(mode.c_str(), "n")) { // "None" masks have no effect.
Florin Malita25366fa2018-01-23 13:37:59 -0500974 continue;
Florin Malitacf8ed522018-01-25 15:27:33 -0500975 }
Florin Malita25366fa2018-01-23 13:37:59 -0500976
977 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
Florin Malitaaa71c892018-01-30 09:27:01 -0500978 mask_paint->setAntiAlias(true);
Florin Malita25366fa2018-01-23 13:37:59 -0500979 mask_paint->setBlendMode(MaskBlendMode(mode.c_str()[0]));
Florin Malita0c51c212018-04-26 14:13:14 -0400980
981 const auto animator_count = ctx->fAnimators.size();
Florin Malitafc807c82018-01-25 22:35:09 -0500982 BindProperty<ScalarValue>(m["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500983 [mask_paint](const ScalarValue& o) { mask_paint->setOpacity(o * 0.01f); });
Florin Malita25366fa2018-01-23 13:37:59 -0500984
Florin Malita0c51c212018-04-26 14:13:14 -0400985 opaque_mask &= (animator_count == ctx->fAnimators.size() && mask_paint->getOpacity() >= 1);
986
987 mask_stack.push_back({mask_path, mask_paint});
Florin Malita25366fa2018-01-23 13:37:59 -0500988 }
989
Florin Malita0c51c212018-04-26 14:13:14 -0400990 if (mask_stack.empty())
991 return childNode;
992
993 if (mask_stack.count() == 1 && opaque_mask) {
994 // Single opaque mask => clip path.
995 return sksg::ClipEffect::Make(std::move(childNode),
996 std::move(mask_stack.front().mask_path),
997 true);
998 }
999
1000 auto mask_group = sksg::Group::Make();
1001 for (const auto& rec : mask_stack) {
1002 mask_group->addChild(sksg::Draw::Make(std::move(rec.mask_path),
1003 std::move(rec.mask_paint)));
1004
1005 }
1006
1007 return sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
Florin Malita25366fa2018-01-23 13:37:59 -05001008}
1009
Florin Malita18eafd92018-01-04 21:11:55 -05001010sk_sp<sksg::RenderNode> AttachLayer(const Json::Value& jlayer,
1011 AttachLayerContext* layerCtx) {
1012 if (!jlayer.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -05001013 return nullptr;
1014
Florin Malitaeb87d672018-01-29 15:28:24 -05001015 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const Json::Value&, AttachContext*,
1016 float* time_bias, float* time_scale);
Florin Malita094ccde2017-12-30 12:27:00 -05001017 static constexpr LayerAttacher gLayerAttachers[] = {
1018 AttachCompLayer, // 'ty': 0
1019 AttachSolidLayer, // 'ty': 1
1020 AttachImageLayer, // 'ty': 2
1021 AttachNullLayer, // 'ty': 3
1022 AttachShapeLayer, // 'ty': 4
1023 AttachTextLayer, // 'ty': 5
1024 };
1025
Florin Malitacf8ed522018-01-25 15:27:33 -05001026 int type = ParseDefault(jlayer["ty"], -1);
Florin Malita094ccde2017-12-30 12:27:00 -05001027 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
1028 return nullptr;
1029 }
1030
Florin Malitacca86f32018-01-29 10:49:49 -05001031 sksg::AnimatorList layer_animators;
Florin Malita1022f742018-02-23 11:10:22 -05001032 AttachContext local_ctx = { layerCtx->fCtx->fResources,
1033 layerCtx->fCtx->fAssets,
1034 layerCtx->fCtx->fFrameRate,
1035 layer_animators};
Florin Malitacca86f32018-01-29 10:49:49 -05001036
Florin Malitaeb87d672018-01-29 15:28:24 -05001037 // Layer attachers may adjust these.
1038 float time_bias = 0,
1039 time_scale = 1;
1040
Florin Malita71cba8f2018-01-09 08:07:14 -05001041 // Layer content.
Florin Malitaeb87d672018-01-29 15:28:24 -05001042 auto layer = gLayerAttachers[type](jlayer, &local_ctx, &time_bias, &time_scale);
Florin Malita38ea40e2018-01-29 16:31:14 -05001043
1044 // Clip layers with explicit dimensions.
1045 float w, h;
1046 if (Parse(jlayer["w"], &w) && Parse(jlayer["h"], &h)) {
1047 layer = sksg::ClipEffect::Make(std::move(layer),
1048 sksg::Rect::Make(SkRect::MakeWH(w, h)),
1049 true);
1050 }
1051
Florin Malita25366fa2018-01-23 13:37:59 -05001052 // Optional layer mask.
Florin Malitacca86f32018-01-29 10:49:49 -05001053 layer = AttachMask(jlayer["masksProperties"], &local_ctx, std::move(layer));
Florin Malita38ea40e2018-01-29 16:31:14 -05001054
Florin Malita25366fa2018-01-23 13:37:59 -05001055 // Optional layer transform.
Florin Malita71cba8f2018-01-09 08:07:14 -05001056 if (auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -05001057 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
1058 }
Florin Malita38ea40e2018-01-29 16:31:14 -05001059
Florin Malita71cba8f2018-01-09 08:07:14 -05001060 // Optional layer opacity.
Florin Malitacca86f32018-01-29 10:49:49 -05001061 layer = AttachOpacity(jlayer["ks"], &local_ctx, std::move(layer));
Florin Malita18eafd92018-01-04 21:11:55 -05001062
Florin Malitaeb87d672018-01-29 15:28:24 -05001063 class LayerController final : public sksg::GroupAnimator {
Florin Malita71cba8f2018-01-09 08:07:14 -05001064 public:
Florin Malitaeb87d672018-01-29 15:28:24 -05001065 LayerController(sksg::AnimatorList&& layer_animators,
1066 sk_sp<sksg::OpacityEffect> controlNode,
1067 float in, float out,
1068 float time_bias, float time_scale)
Florin Malitacca86f32018-01-29 10:49:49 -05001069 : INHERITED(std::move(layer_animators))
1070 , fControlNode(std::move(controlNode))
Florin Malita71cba8f2018-01-09 08:07:14 -05001071 , fIn(in)
Florin Malitaeb87d672018-01-29 15:28:24 -05001072 , fOut(out)
1073 , fTimeBias(time_bias)
1074 , fTimeScale(time_scale) {}
Florin Malita71cba8f2018-01-09 08:07:14 -05001075
Florin Malita35efaa82018-01-22 12:57:06 -05001076 void onTick(float t) override {
Florin Malitacca86f32018-01-29 10:49:49 -05001077 const auto active = (t >= fIn && t <= fOut);
1078
Florin Malita71cba8f2018-01-09 08:07:14 -05001079 // Keep the layer fully transparent except for its [in..out] lifespan.
1080 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
Florin Malitacca86f32018-01-29 10:49:49 -05001081 fControlNode->setOpacity(active ? 1 : 0);
1082
1083 // Dispatch ticks only while active.
1084 if (active)
Florin Malitaeb87d672018-01-29 15:28:24 -05001085 this->INHERITED::onTick((t + fTimeBias) * fTimeScale);
Florin Malita71cba8f2018-01-09 08:07:14 -05001086 }
1087
1088 private:
1089 const sk_sp<sksg::OpacityEffect> fControlNode;
1090 const float fIn,
Florin Malitaeb87d672018-01-29 15:28:24 -05001091 fOut,
1092 fTimeBias,
1093 fTimeScale;
Florin Malitacca86f32018-01-29 10:49:49 -05001094
1095 using INHERITED = sksg::GroupAnimator;
Florin Malita71cba8f2018-01-09 08:07:14 -05001096 };
1097
Florin Malitaeb87d672018-01-29 15:28:24 -05001098 auto controller_node = sksg::OpacityEffect::Make(std::move(layer));
1099 const auto in = ParseDefault(jlayer["ip"], 0.0f),
1100 out = ParseDefault(jlayer["op"], in);
Florin Malita71cba8f2018-01-09 08:07:14 -05001101
Florin Malitaeb87d672018-01-29 15:28:24 -05001102 if (!jlayer["tm"].isNull()) {
1103 LogFail(jlayer["tm"], "Unsupported time remapping");
1104 }
1105
1106 if (in >= out || !controller_node)
Florin Malita71cba8f2018-01-09 08:07:14 -05001107 return nullptr;
1108
Florin Malitacca86f32018-01-29 10:49:49 -05001109 layerCtx->fCtx->fAnimators.push_back(
Florin Malitaeb87d672018-01-29 15:28:24 -05001110 skstd::make_unique<LayerController>(std::move(layer_animators),
1111 controller_node,
1112 in,
1113 out,
1114 time_bias,
1115 time_scale));
Florin Malita71cba8f2018-01-09 08:07:14 -05001116
Florin Malitacf8ed522018-01-25 15:27:33 -05001117 if (ParseDefault(jlayer["td"], false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -05001118 // This layer is a matte. We apply it as a mask to the next layer.
Florin Malitaeb87d672018-01-29 15:28:24 -05001119 layerCtx->fCurrentMatte = std::move(controller_node);
Florin Malita5f9102f2018-01-10 13:36:22 -05001120 return nullptr;
1121 }
1122
1123 if (layerCtx->fCurrentMatte) {
1124 // There is a pending matte. Apply and reset.
Florin Malitaa016be92018-03-05 14:01:41 -05001125 static constexpr sksg::MaskEffect::Mode gMaskModes[] = {
1126 sksg::MaskEffect::Mode::kNormal, // tt: 1
1127 sksg::MaskEffect::Mode::kInvert, // tt: 2
1128 };
1129 const auto matteType = ParseDefault(jlayer["tt"], 1) - 1;
1130
1131 if (matteType >= 0 && matteType < SkTo<int>(SK_ARRAY_COUNT(gMaskModes))) {
1132 return sksg::MaskEffect::Make(std::move(controller_node),
1133 std::move(layerCtx->fCurrentMatte),
1134 gMaskModes[matteType]);
1135 }
1136 layerCtx->fCurrentMatte.reset();
Florin Malita5f9102f2018-01-10 13:36:22 -05001137 }
1138
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001139 return std::move(controller_node);
Florin Malita094ccde2017-12-30 12:27:00 -05001140}
1141
1142sk_sp<sksg::RenderNode> AttachComposition(const Json::Value& comp, AttachContext* ctx) {
1143 if (!comp.isObject())
1144 return nullptr;
1145
Florin Malita18eafd92018-01-04 21:11:55 -05001146 const auto& jlayers = comp["layers"];
1147 if (!jlayers.isArray())
1148 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001149
Florin Malita18eafd92018-01-04 21:11:55 -05001150 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
1151 AttachLayerContext layerCtx(jlayers, ctx);
1152
1153 for (const auto& l : jlayers) {
1154 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001155 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001156 }
1157 }
1158
Florin Malita2a8275b2018-01-02 12:52:43 -05001159 if (layers.empty()) {
1160 return nullptr;
1161 }
1162
1163 // Layers are painted in bottom->top order.
1164 auto comp_group = sksg::Group::Make();
1165 for (int i = layers.count() - 1; i >= 0; --i) {
1166 comp_group->addChild(std::move(layers[i]));
1167 }
1168
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001169 return std::move(comp_group);
Florin Malita094ccde2017-12-30 12:27:00 -05001170}
1171
1172} // namespace
1173
Florin Malita6eb85a12018-04-30 10:32:18 -04001174sk_sp<Animation> Animation::Make(SkStream* stream, const ResourceProvider& res, Stats* stats) {
1175 Stats stats_storage;
1176 if (!stats)
1177 stats = &stats_storage;
1178 memset(stats, 0, sizeof(struct Stats));
1179
Florin Malita094ccde2017-12-30 12:27:00 -05001180 if (!stream->hasLength()) {
1181 // TODO: handle explicit buffering?
1182 LOG("!! cannot parse streaming content\n");
1183 return nullptr;
1184 }
1185
Florin Malita6eb85a12018-04-30 10:32:18 -04001186 const auto t0 = SkTime::GetMSecs();
1187
Florin Malita094ccde2017-12-30 12:27:00 -05001188 Json::Value json;
1189 {
1190 auto data = SkData::MakeFromStream(stream, stream->getLength());
1191 if (!data) {
1192 LOG("!! could not read stream\n");
1193 return nullptr;
1194 }
Florin Malita6eb85a12018-04-30 10:32:18 -04001195 stats->fJsonSize = data->size();
Florin Malita094ccde2017-12-30 12:27:00 -05001196
1197 Json::Reader reader;
1198
1199 auto dataStart = static_cast<const char*>(data->data());
1200 if (!reader.parse(dataStart, dataStart + data->size(), json, false) || !json.isObject()) {
1201 LOG("!! failed to parse json: %s\n", reader.getFormattedErrorMessages().c_str());
1202 return nullptr;
1203 }
1204 }
1205
Florin Malita6eb85a12018-04-30 10:32:18 -04001206 const auto t1 = SkTime::GetMSecs();
1207 stats->fJsonParseTimeMS = t1 - t0;
1208
Florin Malitacf8ed522018-01-25 15:27:33 -05001209 const auto version = ParseDefault(json["v"], SkString());
1210 const auto size = SkSize::Make(ParseDefault(json["w"], 0.0f),
1211 ParseDefault(json["h"], 0.0f));
1212 const auto fps = ParseDefault(json["fr"], -1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001213
Florin Malita1022f742018-02-23 11:10:22 -05001214 if (size.isEmpty() || version.isEmpty() || fps <= 0) {
Florin Malita094ccde2017-12-30 12:27:00 -05001215 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1216 version.c_str(), size.width(), size.height(), fps);
1217 return nullptr;
1218 }
1219
Florin Malita6eb85a12018-04-30 10:32:18 -04001220 const auto anim =
1221 sk_sp<Animation>(new Animation(res, std::move(version), size, fps, json, stats));
1222 const auto t2 = SkTime::GetMSecs();
1223 stats->fSceneParseTimeMS = t2 - t1;
1224 stats->fTotalLoadTimeMS = t2 - t0;
1225
1226 return anim;
Florin Malita094ccde2017-12-30 12:27:00 -05001227}
1228
Florin Malita6eb85a12018-04-30 10:32:18 -04001229sk_sp<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res,
1230 Stats* stats) {
Florin Malita49328072018-01-08 12:51:12 -05001231 class DirectoryResourceProvider final : public ResourceProvider {
1232 public:
1233 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1234
1235 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1236 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1237 return SkStream::MakeFromFile(resPath.c_str());
1238 }
1239
1240 private:
1241 const SkString fDir;
1242 };
1243
1244 const auto jsonStream = SkStream::MakeFromFile(path);
1245 if (!jsonStream)
1246 return nullptr;
1247
1248 std::unique_ptr<ResourceProvider> defaultProvider;
1249 if (!res) {
1250 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1251 }
1252
Florin Malita6eb85a12018-04-30 10:32:18 -04001253 return Make(jsonStream.get(), res ? *res : *defaultProvider, stats);
Florin Malita49328072018-01-08 12:51:12 -05001254}
1255
1256Animation::Animation(const ResourceProvider& resources,
Florin Malita6eb85a12018-04-30 10:32:18 -04001257 SkString version, const SkSize& size, SkScalar fps, const Json::Value& json,
1258 Stats* stats)
Florin Malita094ccde2017-12-30 12:27:00 -05001259 : fVersion(std::move(version))
1260 , fSize(size)
1261 , fFrameRate(fps)
Florin Malitacf8ed522018-01-25 15:27:33 -05001262 , fInPoint(ParseDefault(json["ip"], 0.0f))
1263 , fOutPoint(SkTMax(ParseDefault(json["op"], SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001264
1265 AssetMap assets;
1266 for (const auto& asset : json["assets"]) {
1267 if (!asset.isObject()) {
1268 continue;
1269 }
1270
Florin Malitadd22cf962018-01-29 15:42:01 +00001271 assets.set(ParseDefault(asset["id"], SkString()), &asset);
Florin Malita094ccde2017-12-30 12:27:00 -05001272 }
1273
Florin Malitacca86f32018-01-29 10:49:49 -05001274 sksg::AnimatorList animators;
Florin Malita1022f742018-02-23 11:10:22 -05001275 AttachContext ctx = { resources, assets, fFrameRate, animators };
Florin Malita35efaa82018-01-22 12:57:06 -05001276 auto root = AttachComposition(json, &ctx);
1277
Florin Malita6eb85a12018-04-30 10:32:18 -04001278 stats->fAnimatorCount = animators.size();
Florin Malita35efaa82018-01-22 12:57:06 -05001279
1280 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001281
Florin Malitadb385732018-01-09 12:19:32 -05001282 // In case the client calls render before the first tick.
1283 this->animationTick(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001284}
1285
1286Animation::~Animation() = default;
1287
Florin Malita35efaa82018-01-22 12:57:06 -05001288void Animation::setShowInval(bool show) {
1289 if (fScene) {
1290 fScene->setShowInval(show);
1291 }
1292}
1293
Mike Reed29859872018-01-08 08:25:27 -05001294void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001295 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001296 return;
1297
Mike Reed29859872018-01-08 08:25:27 -05001298 SkAutoCanvasRestore restore(canvas, true);
1299 const SkRect srcR = SkRect::MakeSize(this->size());
1300 if (dstR) {
1301 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1302 }
1303 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001304 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001305}
1306
1307void Animation::animationTick(SkMSec ms) {
Florin Malita35efaa82018-01-22 12:57:06 -05001308 if (!fScene)
1309 return;
1310
Florin Malita094ccde2017-12-30 12:27:00 -05001311 // 't' in the BM model really means 'frame #'
1312 auto t = static_cast<float>(ms) * fFrameRate / 1000;
1313
1314 t = fInPoint + std::fmod(t, fOutPoint - fInPoint);
1315
Florin Malita35efaa82018-01-22 12:57:06 -05001316 fScene->animate(t);
Florin Malita094ccde2017-12-30 12:27:00 -05001317}
1318
Florin Malita54f65c42018-01-16 17:04:30 -05001319} // namespace skottie