blob: 229b5c600a303141850906d78e3ccfb207e77bae [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 Malita094ccde2017-12-30 12:27:00 -050011#include "SkData.h"
Florin Malita49328072018-01-08 12:51:12 -050012#include "SkImage.h"
Florin Malita094ccde2017-12-30 12:27:00 -050013#include "SkMakeUnique.h"
Florin Malita49328072018-01-08 12:51:12 -050014#include "SkOSPath.h"
Florin Malita094ccde2017-12-30 12:27:00 -050015#include "SkPaint.h"
Florin Malita0e66fba2018-01-09 17:10:18 -050016#include "SkParse.h"
Florin Malita094ccde2017-12-30 12:27:00 -050017#include "SkPoint.h"
Florin Malita38ea40e2018-01-29 16:31:14 -050018#include "SkSGClipEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050019#include "SkSGColor.h"
20#include "SkSGDraw.h"
Florin Malita16d0ad02018-01-19 15:07:29 -050021#include "SkSGGeometryTransform.h"
Florin Malita6aaee592018-01-12 12:25:09 -050022#include "SkSGGradient.h"
Florin Malita094ccde2017-12-30 12:27:00 -050023#include "SkSGGroup.h"
Florin Malita49328072018-01-08 12:51:12 -050024#include "SkSGImage.h"
25#include "SkSGInvalidationController.h"
Florin Malita5f9102f2018-01-10 13:36:22 -050026#include "SkSGMaskEffect.h"
Florin Malitae6345d92018-01-03 23:37:54 -050027#include "SkSGMerge.h"
Florin Malitac0034172018-01-08 16:42:59 -050028#include "SkSGOpacityEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050029#include "SkSGPath.h"
Florin Malita2e1d7e22018-01-02 10:40:00 -050030#include "SkSGRect.h"
Florin Malita41dff6e2018-04-30 23:08:15 -040031#include "SkSGRoundEffect.h"
Florin Malita35efaa82018-01-22 12:57:06 -050032#include "SkSGScene.h"
Florin Malita094ccde2017-12-30 12:27:00 -050033#include "SkSGTransform.h"
Florin Malita51b8c892018-01-07 08:54:24 -050034#include "SkSGTrimEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050035#include "SkStream.h"
36#include "SkTArray.h"
Hal Canaryfdcfb8b2018-06-13 09:42:32 -040037#include "SkTHash.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -040038#include "SkTime.h"
39#include "SkTo.h"
40#include "SkottieAdapter.h"
41#include "SkottieAnimator.h"
42#include "SkottieJson.h"
43#include "SkottieValue.h"
Florin Malita094ccde2017-12-30 12:27:00 -050044
45#include <cmath>
Florin Malitae6345d92018-01-03 23:37:54 -050046#include <vector>
47
Florin Malita094ccde2017-12-30 12:27:00 -050048#include "stdlib.h"
49
Florin Malita54f65c42018-01-16 17:04:30 -050050namespace skottie {
Florin Malita094ccde2017-12-30 12:27:00 -050051
Florin Malitacf8ed522018-01-25 15:27:33 -050052#define LOG SkDebugf
53
Florin Malita094ccde2017-12-30 12:27:00 -050054namespace {
55
Florin Malita0cc01b72018-05-10 18:40:35 -040056struct AssetInfo {
57 json::ValueRef fAsset;
58 mutable bool fIsAttaching; // Used for cycle detection
59};
60
61using AssetMap = SkTHashMap<SkString, AssetInfo>;
Florin Malita094ccde2017-12-30 12:27:00 -050062
63struct AttachContext {
Florin Malitacca86f32018-01-29 10:49:49 -050064 const ResourceProvider& fResources;
65 const AssetMap& fAssets;
Florin Malita911ae402018-05-31 16:45:29 -040066 const float fDuration;
Florin Malitacca86f32018-01-29 10:49:49 -050067 sksg::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050068};
69
Florin Malitafa7e9a82018-05-04 15:10:54 -040070bool LogFail(const json::ValueRef& json, const char* msg) {
71 const auto dump = json.toString();
Florin Malita20880782018-05-09 11:35:00 -040072 LOG("!! %s: %s\n", msg, dump.c_str());
Florin Malita094ccde2017-12-30 12:27:00 -050073 return false;
74}
75
Florin Malitafa7e9a82018-05-04 15:10:54 -040076sk_sp<sksg::Matrix> AttachMatrix(const json::ValueRef& t, AttachContext* ctx,
Florin Malita18eafd92018-01-04 21:11:55 -050077 sk_sp<sksg::Matrix> parentMatrix) {
78 if (!t.isObject())
79 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -050080
Florin Malita18eafd92018-01-04 21:11:55 -050081 auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
Florin Malitaa6e30f72018-03-23 13:41:58 -040082 auto adapter = sk_make_sp<TransformAdapter>(matrix);
Florin Malitafc807c82018-01-25 22:35:09 -050083 auto anchor_attached = BindProperty<VectorValue>(t["a"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040084 [adapter](const VectorValue& a) {
85 adapter->setAnchorPoint(ValueTraits<VectorValue>::As<SkPoint>(a));
Florin Malita094ccde2017-12-30 12:27:00 -050086 });
Florin Malitafc807c82018-01-25 22:35:09 -050087 auto position_attached = BindProperty<VectorValue>(t["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040088 [adapter](const VectorValue& p) {
89 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malita094ccde2017-12-30 12:27:00 -050090 });
Florin Malitafc807c82018-01-25 22:35:09 -050091 auto scale_attached = BindProperty<VectorValue>(t["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040092 [adapter](const VectorValue& s) {
93 adapter->setScale(ValueTraits<VectorValue>::As<SkVector>(s));
Florin Malita094ccde2017-12-30 12:27:00 -050094 });
Florin Malita1eb98db2018-01-26 15:03:38 -050095
Florin Malitafa7e9a82018-05-04 15:10:54 -040096 auto jrotation = t["r"];
97 if (jrotation.isNull()) {
Florin Malita1eb98db2018-01-26 15:03:38 -050098 // 3d rotations have separate rx,ry,rz components. While we don't fully support them,
99 // we can still make use of rz.
Florin Malitafa7e9a82018-05-04 15:10:54 -0400100 jrotation = t["rz"];
Florin Malita1eb98db2018-01-26 15:03:38 -0500101 }
Florin Malitafa7e9a82018-05-04 15:10:54 -0400102 auto rotation_attached = BindProperty<ScalarValue>(jrotation, &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400103 [adapter](const ScalarValue& r) {
104 adapter->setRotation(r);
Florin Malita094ccde2017-12-30 12:27:00 -0500105 });
Florin Malitafc807c82018-01-25 22:35:09 -0500106 auto skew_attached = BindProperty<ScalarValue>(t["sk"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400107 [adapter](const ScalarValue& sk) {
108 adapter->setSkew(sk);
Florin Malita094ccde2017-12-30 12:27:00 -0500109 });
Florin Malitafc807c82018-01-25 22:35:09 -0500110 auto skewaxis_attached = BindProperty<ScalarValue>(t["sa"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400111 [adapter](const ScalarValue& sa) {
112 adapter->setSkewAxis(sa);
Florin Malita094ccde2017-12-30 12:27:00 -0500113 });
114
115 if (!anchor_attached &&
116 !position_attached &&
117 !scale_attached &&
118 !rotation_attached &&
119 !skew_attached &&
120 !skewaxis_attached) {
121 LogFail(t, "Could not parse transform");
Florin Malita18eafd92018-01-04 21:11:55 -0500122 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500123 }
124
Florin Malita18eafd92018-01-04 21:11:55 -0500125 return matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500126}
127
Florin Malitafa7e9a82018-05-04 15:10:54 -0400128sk_sp<sksg::RenderNode> AttachOpacity(const json::ValueRef& jtransform, AttachContext* ctx,
Florin Malitac0034172018-01-08 16:42:59 -0500129 sk_sp<sksg::RenderNode> childNode) {
130 if (!jtransform.isObject() || !childNode)
131 return childNode;
132
Florin Malita7ac2e3b2018-05-09 14:54:39 -0400133 static constexpr ScalarValue kNoopOpacity = 100;
Florin Malitac0034172018-01-08 16:42:59 -0500134 auto opacityNode = sksg::OpacityEffect::Make(childNode);
Florin Malita7ac2e3b2018-05-09 14:54:39 -0400135
136 if (!BindProperty<ScalarValue>(jtransform["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500137 [opacityNode](const ScalarValue& o) {
Florin Malitac0034172018-01-08 16:42:59 -0500138 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500139 opacityNode->setOpacity(o * 0.01f);
Florin Malita7ac2e3b2018-05-09 14:54:39 -0400140 }, &kNoopOpacity)) {
141 // We can ignore static full opacity.
142 return childNode;
143 }
Florin Malitac0034172018-01-08 16:42:59 -0500144
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400145 return std::move(opacityNode);
Florin Malitac0034172018-01-08 16:42:59 -0500146}
147
Florin Malitafa7e9a82018-05-04 15:10:54 -0400148sk_sp<sksg::RenderNode> AttachComposition(const json::ValueRef&, AttachContext* ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500149
Florin Malitafa7e9a82018-05-04 15:10:54 -0400150sk_sp<sksg::Path> AttachPath(const json::ValueRef& jpath, AttachContext* ctx) {
Florin Malita25366fa2018-01-23 13:37:59 -0500151 auto path_node = sksg::Path::Make();
Florin Malitafc807c82018-01-25 22:35:09 -0500152 return BindProperty<ShapeValue>(jpath, &ctx->fAnimators,
Florin Malitac353ee22018-04-30 21:49:41 -0400153 [path_node](const ShapeValue& p) {
154 path_node->setPath(ValueTraits<ShapeValue>::As<SkPath>(p));
155 })
Florin Malita25366fa2018-01-23 13:37:59 -0500156 ? path_node
157 : nullptr;
158}
159
Florin Malitafa7e9a82018-05-04 15:10:54 -0400160sk_sp<sksg::GeometryNode> AttachPathGeometry(const json::ValueRef& jpath, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500161 SkASSERT(jpath.isObject());
162
Florin Malita25366fa2018-01-23 13:37:59 -0500163 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500164}
165
Florin Malitafa7e9a82018-05-04 15:10:54 -0400166sk_sp<sksg::GeometryNode> AttachRRectGeometry(const json::ValueRef& jrect, AttachContext* ctx) {
Florin Malita2e1d7e22018-01-02 10:40:00 -0500167 SkASSERT(jrect.isObject());
168
169 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400170 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500171
Florin Malitafc807c82018-01-25 22:35:09 -0500172 auto p_attached = BindProperty<VectorValue>(jrect["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400173 [adapter](const VectorValue& p) {
174 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500175 });
Florin Malitafc807c82018-01-25 22:35:09 -0500176 auto s_attached = BindProperty<VectorValue>(jrect["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400177 [adapter](const VectorValue& s) {
178 adapter->setSize(ValueTraits<VectorValue>::As<SkSize>(s));
Florin Malitaf9590922018-01-09 11:56:09 -0500179 });
Florin Malitafc807c82018-01-25 22:35:09 -0500180 auto r_attached = BindProperty<ScalarValue>(jrect["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400181 [adapter](const ScalarValue& r) {
182 adapter->setRadius(SkSize::Make(r, r));
Florin Malitaf9590922018-01-09 11:56:09 -0500183 });
Florin Malita2e1d7e22018-01-02 10:40:00 -0500184
185 if (!p_attached && !s_attached && !r_attached) {
186 return nullptr;
187 }
188
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400189 return std::move(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500190}
191
Florin Malitafa7e9a82018-05-04 15:10:54 -0400192sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const json::ValueRef& jellipse, AttachContext* ctx) {
Florin Malitafbc13f12018-01-04 10:26:35 -0500193 SkASSERT(jellipse.isObject());
194
195 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400196 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500197
Florin Malitafc807c82018-01-25 22:35:09 -0500198 auto p_attached = BindProperty<VectorValue>(jellipse["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400199 [adapter](const VectorValue& p) {
200 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500201 });
Florin Malitafc807c82018-01-25 22:35:09 -0500202 auto s_attached = BindProperty<VectorValue>(jellipse["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400203 [adapter](const VectorValue& s) {
Florin Malitaf9590922018-01-09 11:56:09 -0500204 const auto sz = ValueTraits<VectorValue>::As<SkSize>(s);
Florin Malitaa6e30f72018-03-23 13:41:58 -0400205 adapter->setSize(sz);
206 adapter->setRadius(SkSize::Make(sz.width() / 2, sz.height() / 2));
Florin Malitaf9590922018-01-09 11:56:09 -0500207 });
Florin Malitafbc13f12018-01-04 10:26:35 -0500208
209 if (!p_attached && !s_attached) {
210 return nullptr;
211 }
212
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400213 return std::move(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500214}
215
Florin Malitafa7e9a82018-05-04 15:10:54 -0400216sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const json::ValueRef& jstar, AttachContext* ctx) {
Florin Malita02a32b02018-01-04 11:27:09 -0500217 SkASSERT(jstar.isObject());
218
Florin Malitaa6e30f72018-03-23 13:41:58 -0400219 static constexpr PolyStarAdapter::Type gTypes[] = {
220 PolyStarAdapter::Type::kStar, // "sy": 1
221 PolyStarAdapter::Type::kPoly, // "sy": 2
Florin Malita02a32b02018-01-04 11:27:09 -0500222 };
223
Florin Malitafa7e9a82018-05-04 15:10:54 -0400224 const auto type = jstar["sy"].toDefault<int>(0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500225 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
226 LogFail(jstar, "Unknown polystar type");
227 return nullptr;
228 }
229
230 auto path_node = sksg::Path::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400231 auto adapter = sk_make_sp<PolyStarAdapter>(path_node, gTypes[type]);
Florin Malita02a32b02018-01-04 11:27:09 -0500232
Florin Malitafc807c82018-01-25 22:35:09 -0500233 BindProperty<VectorValue>(jstar["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400234 [adapter](const VectorValue& p) {
235 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500236 });
Florin Malitafc807c82018-01-25 22:35:09 -0500237 BindProperty<ScalarValue>(jstar["pt"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400238 [adapter](const ScalarValue& pt) {
239 adapter->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500240 });
Florin Malitafc807c82018-01-25 22:35:09 -0500241 BindProperty<ScalarValue>(jstar["ir"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400242 [adapter](const ScalarValue& ir) {
243 adapter->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500244 });
Florin Malitafc807c82018-01-25 22:35:09 -0500245 BindProperty<ScalarValue>(jstar["or"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400246 [adapter](const ScalarValue& otr) {
247 adapter->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500248 });
Florin Malitafc807c82018-01-25 22:35:09 -0500249 BindProperty<ScalarValue>(jstar["is"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400250 [adapter](const ScalarValue& is) {
251 adapter->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500252 });
Florin Malitafc807c82018-01-25 22:35:09 -0500253 BindProperty<ScalarValue>(jstar["os"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400254 [adapter](const ScalarValue& os) {
255 adapter->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500256 });
Florin Malitafc807c82018-01-25 22:35:09 -0500257 BindProperty<ScalarValue>(jstar["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400258 [adapter](const ScalarValue& r) {
259 adapter->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500260 });
Florin Malita02a32b02018-01-04 11:27:09 -0500261
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400262 return std::move(path_node);
Florin Malita02a32b02018-01-04 11:27:09 -0500263}
264
Florin Malitafa7e9a82018-05-04 15:10:54 -0400265sk_sp<sksg::Color> AttachColor(const json::ValueRef& obj, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500266 SkASSERT(obj.isObject());
267
268 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malitafc807c82018-01-25 22:35:09 -0500269 auto color_attached = BindProperty<VectorValue>(obj["c"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500270 [color_node](const VectorValue& c) {
271 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500272 });
Florin Malita094ccde2017-12-30 12:27:00 -0500273
Florin Malita1586d852018-01-12 14:27:39 -0500274 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500275}
276
Florin Malitafa7e9a82018-05-04 15:10:54 -0400277sk_sp<sksg::Gradient> AttachGradient(const json::ValueRef& obj, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500278 SkASSERT(obj.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500279
Florin Malita20880782018-05-09 11:35:00 -0400280 const auto stops = obj["g"];
Florin Malita6aaee592018-01-12 12:25:09 -0500281 if (!stops.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500282 return nullptr;
283
Florin Malitafa7e9a82018-05-04 15:10:54 -0400284 const auto stopCount = stops["p"].toDefault<int>(-1);
Florin Malita6aaee592018-01-12 12:25:09 -0500285 if (stopCount < 0)
286 return nullptr;
287
288 sk_sp<sksg::Gradient> gradient_node;
Florin Malitaa6e30f72018-03-23 13:41:58 -0400289 sk_sp<GradientAdapter> adapter;
Florin Malita6aaee592018-01-12 12:25:09 -0500290
Florin Malitafa7e9a82018-05-04 15:10:54 -0400291 if (obj["t"].toDefault<int>(1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500292 auto linear_node = sksg::LinearGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400293 adapter = sk_make_sp<LinearGradientAdapter>(linear_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500294 gradient_node = std::move(linear_node);
295 } else {
296 auto radial_node = sksg::RadialGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400297 adapter = sk_make_sp<RadialGradientAdapter>(radial_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500298
299 // TODO: highlight, angle
300 gradient_node = std::move(radial_node);
301 }
302
Florin Malitafc807c82018-01-25 22:35:09 -0500303 BindProperty<VectorValue>(stops["k"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400304 [adapter](const VectorValue& stops) {
305 adapter->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500306 });
Florin Malitafc807c82018-01-25 22:35:09 -0500307 BindProperty<VectorValue>(obj["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400308 [adapter](const VectorValue& s) {
309 adapter->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500310 });
Florin Malitafc807c82018-01-25 22:35:09 -0500311 BindProperty<VectorValue>(obj["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400312 [adapter](const VectorValue& e) {
313 adapter->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500314 });
315
316 return gradient_node;
317}
318
Florin Malitafa7e9a82018-05-04 15:10:54 -0400319sk_sp<sksg::PaintNode> AttachPaint(const json::ValueRef& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500320 sk_sp<sksg::PaintNode> paint_node) {
321 if (paint_node) {
322 paint_node->setAntiAlias(true);
323
Florin Malitafc807c82018-01-25 22:35:09 -0500324 BindProperty<ScalarValue>(jpaint["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500325 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500326 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500327 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500328 });
Florin Malita6aaee592018-01-12 12:25:09 -0500329 }
330
331 return paint_node;
332}
333
Florin Malitafa7e9a82018-05-04 15:10:54 -0400334sk_sp<sksg::PaintNode> AttachStroke(const json::ValueRef& jstroke, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500335 sk_sp<sksg::PaintNode> stroke_node) {
336 SkASSERT(jstroke.isObject());
337
338 if (!stroke_node)
339 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500340
341 stroke_node->setStyle(SkPaint::kStroke_Style);
342
Florin Malitafc807c82018-01-25 22:35:09 -0500343 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500344 [stroke_node](const ScalarValue& w) {
345 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500346 });
Florin Malita094ccde2017-12-30 12:27:00 -0500347 if (!width_attached)
348 return nullptr;
349
Florin Malitafa7e9a82018-05-04 15:10:54 -0400350 stroke_node->setStrokeMiter(jstroke["ml"].toDefault(4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500351
352 static constexpr SkPaint::Join gJoins[] = {
353 SkPaint::kMiter_Join,
354 SkPaint::kRound_Join,
355 SkPaint::kBevel_Join,
356 };
Florin Malitafa7e9a82018-05-04 15:10:54 -0400357 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(jstroke["lj"].toDefault<int>(1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500358 0, SK_ARRAY_COUNT(gJoins) - 1)]);
359
360 static constexpr SkPaint::Cap gCaps[] = {
361 SkPaint::kButt_Cap,
362 SkPaint::kRound_Cap,
363 SkPaint::kSquare_Cap,
364 };
Florin Malitafa7e9a82018-05-04 15:10:54 -0400365 stroke_node->setStrokeCap(gCaps[SkTPin<int>(jstroke["lc"].toDefault<int>(1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500366 0, SK_ARRAY_COUNT(gCaps) - 1)]);
367
368 return stroke_node;
369}
370
Florin Malitafa7e9a82018-05-04 15:10:54 -0400371sk_sp<sksg::PaintNode> AttachColorFill(const json::ValueRef& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500372 SkASSERT(jfill.isObject());
373
374 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
375}
376
Florin Malitafa7e9a82018-05-04 15:10:54 -0400377sk_sp<sksg::PaintNode> AttachGradientFill(const json::ValueRef& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500378 SkASSERT(jfill.isObject());
379
380 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
381}
382
Florin Malitafa7e9a82018-05-04 15:10:54 -0400383sk_sp<sksg::PaintNode> AttachColorStroke(const json::ValueRef& jstroke, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500384 SkASSERT(jstroke.isObject());
385
386 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
387}
388
Florin Malitafa7e9a82018-05-04 15:10:54 -0400389sk_sp<sksg::PaintNode> AttachGradientStroke(const json::ValueRef& jstroke, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500390 SkASSERT(jstroke.isObject());
391
392 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
393}
394
Florin Malitae6345d92018-01-03 23:37:54 -0500395std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400396 const json::ValueRef& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malitae6345d92018-01-03 23:37:54 -0500397 std::vector<sk_sp<sksg::GeometryNode>> merged;
398
399 static constexpr sksg::Merge::Mode gModes[] = {
400 sksg::Merge::Mode::kMerge, // "mm": 1
401 sksg::Merge::Mode::kUnion, // "mm": 2
402 sksg::Merge::Mode::kDifference, // "mm": 3
403 sksg::Merge::Mode::kIntersect, // "mm": 4
404 sksg::Merge::Mode::kXOR , // "mm": 5
405 };
406
Florin Malitafa7e9a82018-05-04 15:10:54 -0400407 const auto mode = gModes[SkTPin<int>(jmerge["mm"].toDefault(1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500408 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malitae6345d92018-01-03 23:37:54 -0500409 merged.push_back(sksg::Merge::Make(std::move(geos), mode));
410
Florin Malitae6345d92018-01-03 23:37:54 -0500411 return merged;
412}
413
Florin Malita51b8c892018-01-07 08:54:24 -0500414std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400415 const json::ValueRef& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malita51b8c892018-01-07 08:54:24 -0500416
417 enum class Mode {
418 kMerged, // "m": 1
419 kSeparate, // "m": 2
420 } gModes[] = { Mode::kMerged, Mode::kSeparate };
421
Florin Malitafa7e9a82018-05-04 15:10:54 -0400422 const auto mode = gModes[SkTPin<int>(jtrim["m"].toDefault(1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500423 0, SK_ARRAY_COUNT(gModes) - 1)];
424
425 std::vector<sk_sp<sksg::GeometryNode>> inputs;
426 if (mode == Mode::kMerged) {
427 inputs.push_back(sksg::Merge::Make(std::move(geos), sksg::Merge::Mode::kMerge));
428 } else {
429 inputs = std::move(geos);
430 }
431
432 std::vector<sk_sp<sksg::GeometryNode>> trimmed;
433 trimmed.reserve(inputs.size());
434 for (const auto& i : inputs) {
Florin Malita69526b02018-03-22 12:20:02 -0400435 const auto trimEffect = sksg::TrimEffect::Make(i);
436 trimmed.push_back(trimEffect);
437
Florin Malitaa6e30f72018-03-23 13:41:58 -0400438 const auto adapter = sk_make_sp<TrimEffectAdapter>(std::move(trimEffect));
Florin Malitafc807c82018-01-25 22:35:09 -0500439 BindProperty<ScalarValue>(jtrim["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400440 [adapter](const ScalarValue& s) {
441 adapter->setStart(s);
Florin Malita51b8c892018-01-07 08:54:24 -0500442 });
Florin Malitafc807c82018-01-25 22:35:09 -0500443 BindProperty<ScalarValue>(jtrim["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400444 [adapter](const ScalarValue& e) {
445 adapter->setEnd(e);
Florin Malita51b8c892018-01-07 08:54:24 -0500446 });
Florin Malitafc807c82018-01-25 22:35:09 -0500447 BindProperty<ScalarValue>(jtrim["o"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400448 [adapter](const ScalarValue& o) {
449 adapter->setOffset(o);
Florin Malita51b8c892018-01-07 08:54:24 -0500450 });
451 }
452
453 return trimmed;
454}
455
Florin Malita41dff6e2018-04-30 23:08:15 -0400456std::vector<sk_sp<sksg::GeometryNode>> AttachRoundGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400457 const json::ValueRef& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malita41dff6e2018-04-30 23:08:15 -0400458
459 std::vector<sk_sp<sksg::GeometryNode>> rounded;
460 rounded.reserve(geos.size());
461
462 for (const auto& g : geos) {
463 const auto roundEffect = sksg::RoundEffect::Make(std::move(g));
464 rounded.push_back(roundEffect);
465
466 BindProperty<ScalarValue>(jtrim["r"], &ctx->fAnimators,
467 [roundEffect](const ScalarValue& r) {
468 roundEffect->setRadius(r);
469 });
470 }
471
472 return rounded;
473}
474
Florin Malitafa7e9a82018-05-04 15:10:54 -0400475using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const json::ValueRef&, AttachContext*);
Florin Malita094ccde2017-12-30 12:27:00 -0500476static constexpr GeometryAttacherT gGeometryAttachers[] = {
477 AttachPathGeometry,
Florin Malita2e1d7e22018-01-02 10:40:00 -0500478 AttachRRectGeometry,
Florin Malitafbc13f12018-01-04 10:26:35 -0500479 AttachEllipseGeometry,
Florin Malita02a32b02018-01-04 11:27:09 -0500480 AttachPolystarGeometry,
Florin Malita094ccde2017-12-30 12:27:00 -0500481};
482
Florin Malitafa7e9a82018-05-04 15:10:54 -0400483using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const json::ValueRef&, AttachContext*);
Florin Malita094ccde2017-12-30 12:27:00 -0500484static constexpr PaintAttacherT gPaintAttachers[] = {
Florin Malita6aaee592018-01-12 12:25:09 -0500485 AttachColorFill,
486 AttachColorStroke,
487 AttachGradientFill,
488 AttachGradientStroke,
Florin Malita094ccde2017-12-30 12:27:00 -0500489};
490
Florin Malitae6345d92018-01-03 23:37:54 -0500491using GeometryEffectAttacherT =
Florin Malitafa7e9a82018-05-04 15:10:54 -0400492 std::vector<sk_sp<sksg::GeometryNode>> (*)(const json::ValueRef&,
Florin Malitae6345d92018-01-03 23:37:54 -0500493 AttachContext*,
494 std::vector<sk_sp<sksg::GeometryNode>>&&);
495static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
496 AttachMergeGeometryEffect,
Florin Malita51b8c892018-01-07 08:54:24 -0500497 AttachTrimGeometryEffect,
Florin Malita41dff6e2018-04-30 23:08:15 -0400498 AttachRoundGeometryEffect,
Florin Malitae6345d92018-01-03 23:37:54 -0500499};
500
Florin Malita094ccde2017-12-30 12:27:00 -0500501enum class ShapeType {
502 kGeometry,
Florin Malitae6345d92018-01-03 23:37:54 -0500503 kGeometryEffect,
Florin Malita094ccde2017-12-30 12:27:00 -0500504 kPaint,
505 kGroup,
Florin Malitadacc02b2017-12-31 09:12:31 -0500506 kTransform,
Florin Malita094ccde2017-12-30 12:27:00 -0500507};
508
509struct ShapeInfo {
510 const char* fTypeString;
511 ShapeType fShapeType;
512 uint32_t fAttacherIndex; // index into respective attacher tables
513};
514
Florin Malitafa7e9a82018-05-04 15:10:54 -0400515const ShapeInfo* FindShapeInfo(const json::ValueRef& shape) {
Florin Malita094ccde2017-12-30 12:27:00 -0500516 static constexpr ShapeInfo gShapeInfo[] = {
Florin Malitafbc13f12018-01-04 10:26:35 -0500517 { "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500518 { "fl", ShapeType::kPaint , 0 }, // fill -> AttachColorFill
519 { "gf", ShapeType::kPaint , 2 }, // gfill -> AttachGradientFill
Florin Malita16d0ad02018-01-19 15:07:29 -0500520 { "gr", ShapeType::kGroup , 0 }, // group -> Inline handler
Florin Malita6aaee592018-01-12 12:25:09 -0500521 { "gs", ShapeType::kPaint , 3 }, // gstroke -> AttachGradientStroke
Florin Malitae6345d92018-01-03 23:37:54 -0500522 { "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
Florin Malita02a32b02018-01-04 11:27:09 -0500523 { "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
Florin Malita41dff6e2018-04-30 23:08:15 -0400524 { "rd", ShapeType::kGeometryEffect, 2 }, // round -> AttachRoundGeometryEffect
Florin Malitae6345d92018-01-03 23:37:54 -0500525 { "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
Florin Malita02a32b02018-01-04 11:27:09 -0500526 { "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500527 { "st", ShapeType::kPaint , 1 }, // stroke -> AttachColorStroke
Florin Malita51b8c892018-01-07 08:54:24 -0500528 { "tm", ShapeType::kGeometryEffect, 1 }, // trim -> AttachTrimGeometryEffect
Florin Malita16d0ad02018-01-19 15:07:29 -0500529 { "tr", ShapeType::kTransform , 0 }, // transform -> Inline handler
Florin Malita094ccde2017-12-30 12:27:00 -0500530 };
531
Florin Malita20880782018-05-09 11:35:00 -0400532 SkString type;
533 if (!shape["ty"].to(&type) || type.isEmpty())
Florin Malita094ccde2017-12-30 12:27:00 -0500534 return nullptr;
535
Florin Malitafa7e9a82018-05-04 15:10:54 -0400536 const auto* info = bsearch(type.c_str(),
Florin Malita094ccde2017-12-30 12:27:00 -0500537 gShapeInfo,
538 SK_ARRAY_COUNT(gShapeInfo),
539 sizeof(ShapeInfo),
540 [](const void* key, const void* info) {
541 return strcmp(static_cast<const char*>(key),
542 static_cast<const ShapeInfo*>(info)->fTypeString);
543 });
544
545 return static_cast<const ShapeInfo*>(info);
546}
547
Florin Malita16d0ad02018-01-19 15:07:29 -0500548struct GeometryEffectRec {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400549 const json::ValueRef fJson;
Florin Malita16d0ad02018-01-19 15:07:29 -0500550 GeometryEffectAttacherT fAttach;
551};
552
Florin Malitaca4439f2018-01-23 10:31:59 -0500553struct AttachShapeContext {
554 AttachShapeContext(AttachContext* ctx,
555 std::vector<sk_sp<sksg::GeometryNode>>* geos,
556 std::vector<GeometryEffectRec>* effects,
557 size_t committedAnimators)
558 : fCtx(ctx)
559 , fGeometryStack(geos)
560 , fGeometryEffectStack(effects)
561 , fCommittedAnimators(committedAnimators) {}
562
563 AttachContext* fCtx;
564 std::vector<sk_sp<sksg::GeometryNode>>* fGeometryStack;
565 std::vector<GeometryEffectRec>* fGeometryEffectStack;
566 size_t fCommittedAnimators;
567};
568
Florin Malitafa7e9a82018-05-04 15:10:54 -0400569sk_sp<sksg::RenderNode> AttachShape(const json::ValueRef& jshape, AttachShapeContext* shapeCtx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500570 if (!jshape.isArray())
Florin Malita094ccde2017-12-30 12:27:00 -0500571 return nullptr;
572
Florin Malitaca4439f2018-01-23 10:31:59 -0500573 SkDEBUGCODE(const auto initialGeometryEffects = shapeCtx->fGeometryEffectStack->size();)
Florin Malita094ccde2017-12-30 12:27:00 -0500574
Florin Malita16d0ad02018-01-19 15:07:29 -0500575 sk_sp<sksg::Group> shape_group = sksg::Group::Make();
576 sk_sp<sksg::RenderNode> shape_wrapper = shape_group;
577 sk_sp<sksg::Matrix> shape_matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500578
Florin Malita16d0ad02018-01-19 15:07:29 -0500579 struct ShapeRec {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400580 const json::ValueRef fJson;
Florin Malita16d0ad02018-01-19 15:07:29 -0500581 const ShapeInfo& fInfo;
582 };
583
584 // First pass (bottom->top):
585 //
586 // * pick up the group transform and opacity
587 // * push local geometry effects onto the stack
588 // * store recs for next pass
589 //
590 std::vector<ShapeRec> recs;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400591 for (size_t i = 0; i < jshape.size(); ++i) {
592 const auto s = jshape[jshape.size() - 1 - i];
Florin Malita094ccde2017-12-30 12:27:00 -0500593 const auto* info = FindShapeInfo(s);
594 if (!info) {
Florin Malita20880782018-05-09 11:35:00 -0400595 LogFail(s["ty"], "Unknown shape");
Florin Malita094ccde2017-12-30 12:27:00 -0500596 continue;
597 }
598
Florin Malita16d0ad02018-01-19 15:07:29 -0500599 recs.push_back({ s, *info });
600
Florin Malita094ccde2017-12-30 12:27:00 -0500601 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500602 case ShapeType::kTransform:
Florin Malitaca4439f2018-01-23 10:31:59 -0500603 if ((shape_matrix = AttachMatrix(s, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500604 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
605 }
Florin Malitaca4439f2018-01-23 10:31:59 -0500606 shape_wrapper = AttachOpacity(s, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500607 break;
608 case ShapeType::kGeometryEffect:
609 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500610 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita16d0ad02018-01-19 15:07:29 -0500611 { s, gGeometryEffectAttachers[info->fAttacherIndex] });
612 break;
613 default:
614 break;
615 }
616 }
617
618 // Second pass (top -> bottom, after 2x reverse):
619 //
620 // * track local geometry
621 // * emit local paints
622 //
623 std::vector<sk_sp<sksg::GeometryNode>> geos;
624 std::vector<sk_sp<sksg::RenderNode >> draws;
625 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
626 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500627 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500628 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500629 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
630 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500631 geos.push_back(std::move(geo));
632 }
633 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500634 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500635 // Apply the current effect and pop from the stack.
636 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500637 if (!geos.empty()) {
638 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500639 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500640 std::move(geos));
641 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500642
Florin Malitaca4439f2018-01-23 10:31:59 -0500643 SkASSERT(shapeCtx->fGeometryEffectStack->back().fJson == rec->fJson);
644 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500645 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500646 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500647 } break;
648 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500649 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
650 &geos,
651 shapeCtx->fGeometryEffectStack,
652 shapeCtx->fCommittedAnimators);
653 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500654 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500655 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
656 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500657 }
658 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500659 case ShapeType::kPaint: {
660 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500661 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500662 if (!paint || geos.empty())
663 break;
664
665 auto drawGeos = geos;
666
667 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500668 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
669 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
670 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500671 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500672
673 // If we still have multiple geos, reduce using 'merge'.
674 auto geo = drawGeos.size() > 1
675 ? sksg::Merge::Make(std::move(drawGeos), sksg::Merge::Mode::kMerge)
676 : drawGeos[0];
677
678 SkASSERT(geo);
679 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500680 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500681 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500682 default:
683 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500684 }
685 }
686
Florin Malita16d0ad02018-01-19 15:07:29 -0500687 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500688 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500689
690 // Push transformed local geometries to parent list, for subsequent paints.
691 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500692 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500693 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
694 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500695 }
696
Florin Malita16d0ad02018-01-19 15:07:29 -0500697 // Emit local draws reversed (bottom->top, per spec).
698 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
699 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500700 }
701
Florin Malita16d0ad02018-01-19 15:07:29 -0500702 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500703}
704
Florin Malita1022f742018-02-23 11:10:22 -0500705sk_sp<sksg::RenderNode> AttachNestedAnimation(const char* path, AttachContext* ctx) {
706 class SkottieSGAdapter final : public sksg::RenderNode {
707 public:
708 explicit SkottieSGAdapter(sk_sp<Animation> animation)
709 : fAnimation(std::move(animation)) {
710 SkASSERT(fAnimation);
711 }
712
713 protected:
714 SkRect onRevalidate(sksg::InvalidationController*, const SkMatrix&) override {
715 return SkRect::MakeSize(fAnimation->size());
716 }
717
718 void onRender(SkCanvas* canvas) const override {
719 fAnimation->render(canvas);
720 }
721
722 private:
723 const sk_sp<Animation> fAnimation;
724 };
725
726 class SkottieAnimatorAdapter final : public sksg::Animator {
727 public:
Florin Malita911ae402018-05-31 16:45:29 -0400728 SkottieAnimatorAdapter(sk_sp<Animation> animation, float time_scale)
Florin Malita1022f742018-02-23 11:10:22 -0500729 : fAnimation(std::move(animation))
Florin Malita911ae402018-05-31 16:45:29 -0400730 , fTimeScale(time_scale) {
Florin Malita1022f742018-02-23 11:10:22 -0500731 SkASSERT(fAnimation);
Florin Malita1022f742018-02-23 11:10:22 -0500732 }
733
734 protected:
735 void onTick(float t) {
Florin Malita911ae402018-05-31 16:45:29 -0400736 // TODO: we prolly need more sophisticated timeline mapping for nested animations.
737 fAnimation->seek(t * fTimeScale);
Florin Malita1022f742018-02-23 11:10:22 -0500738 }
739
740 private:
741 const sk_sp<Animation> fAnimation;
Florin Malita911ae402018-05-31 16:45:29 -0400742 const float fTimeScale;
Florin Malita1022f742018-02-23 11:10:22 -0500743 };
744
745 const auto resStream = ctx->fResources.openStream(path);
746 if (!resStream || !resStream->hasLength()) {
747 LOG("!! Could not open: %s\n", path);
748 return nullptr;
749 }
750
Florin Malitac83a0de2018-05-31 12:17:55 -0400751 auto animation = Animation::Make(resStream.get(), &ctx->fResources);
Florin Malita1022f742018-02-23 11:10:22 -0500752 if (!animation) {
753 LOG("!! Could not load nested animation: %s\n", path);
754 return nullptr;
755 }
756
Florin Malita911ae402018-05-31 16:45:29 -0400757
758 ctx->fAnimators.push_back(
759 skstd::make_unique<SkottieAnimatorAdapter>(animation,
760 animation->duration() / ctx->fDuration));
Florin Malita1022f742018-02-23 11:10:22 -0500761
762 return sk_make_sp<SkottieSGAdapter>(std::move(animation));
763}
764
Florin Malita0cc01b72018-05-10 18:40:35 -0400765sk_sp<sksg::RenderNode> AttachAssetRef(const json::ValueRef& jlayer, AttachContext* ctx,
766 sk_sp<sksg::RenderNode>(*attach_proc)(const json::ValueRef& comp, AttachContext* ctx)) {
767
768 const auto refId = jlayer["refId"].toDefault(SkString());
769 if (refId.isEmpty()) {
770 LOG("!! Layer missing refId\n");
771 return nullptr;
772 }
773
774 if (refId.startsWith("$")) {
775 return AttachNestedAnimation(refId.c_str() + 1, ctx);
776 }
777
778 const auto* asset_info = ctx->fAssets.find(refId);
779 if (!asset_info) {
780 LOG("!! Asset not found: '%s'\n", refId.c_str());
781 return nullptr;
782 }
783
784 if (asset_info->fIsAttaching) {
785 LOG("!! Asset cycle detected for: '%s'\n", refId.c_str());
786 return nullptr;
787 }
788
789 asset_info->fIsAttaching = true;
790 auto asset = attach_proc(asset_info->fAsset, ctx);
791 asset_info->fIsAttaching = false;
792
793 return asset;
794}
795
Florin Malitafa7e9a82018-05-04 15:10:54 -0400796sk_sp<sksg::RenderNode> AttachCompLayer(const json::ValueRef& jlayer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500797 float* time_bias, float* time_scale) {
798 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500799
Florin Malitafa7e9a82018-05-04 15:10:54 -0400800 const auto start_time = jlayer["st"].toDefault(0.0f),
801 stretch_time = jlayer["sr"].toDefault(1.0f);
Florin Malitaeb87d672018-01-29 15:28:24 -0500802
803 *time_bias = -start_time;
Mike Reed2d46a002018-05-10 10:36:58 -0400804 *time_scale = sk_ieee_float_divide(1, stretch_time);
Florin Malitaeb87d672018-01-29 15:28:24 -0500805 if (SkScalarIsNaN(*time_scale)) {
806 *time_scale = 1;
807 }
808
Florin Malita0cc01b72018-05-10 18:40:35 -0400809 return AttachAssetRef(jlayer, ctx, AttachComposition);
Florin Malita094ccde2017-12-30 12:27:00 -0500810}
811
Florin Malitafa7e9a82018-05-04 15:10:54 -0400812sk_sp<sksg::RenderNode> AttachSolidLayer(const json::ValueRef& jlayer, AttachContext*,
Florin Malitaeb87d672018-01-29 15:28:24 -0500813 float*, float*) {
Florin Malita0e66fba2018-01-09 17:10:18 -0500814 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500815
Florin Malitafa7e9a82018-05-04 15:10:54 -0400816 const auto size = SkSize::Make(jlayer["sw"].toDefault(0.0f),
817 jlayer["sh"].toDefault(0.0f));
818 const auto hex = jlayer["sc"].toDefault(SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500819 uint32_t c;
820 if (size.isEmpty() ||
821 !hex.startsWith("#") ||
822 !SkParse::FindHex(hex.c_str() + 1, &c)) {
823 LogFail(jlayer, "Could not parse solid layer");
824 return nullptr;
825 }
826
827 const SkColor color = 0xff000000 | c;
828
829 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
830 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500831}
832
Florin Malitafa7e9a82018-05-04 15:10:54 -0400833sk_sp<sksg::RenderNode> AttachImageAsset(const json::ValueRef& jimage, AttachContext* ctx) {
Florin Malita49328072018-01-08 12:51:12 -0500834 SkASSERT(jimage.isObject());
835
Florin Malitafa7e9a82018-05-04 15:10:54 -0400836 const auto name = jimage["p"].toDefault(SkString()),
837 path = jimage["u"].toDefault(SkString());
Florin Malita49328072018-01-08 12:51:12 -0500838 if (name.isEmpty())
839 return nullptr;
840
841 // TODO: plumb resource paths explicitly to ResourceProvider?
842 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
843 const auto resStream = ctx->fResources.openStream(resName.c_str());
844 if (!resStream || !resStream->hasLength()) {
845 LOG("!! Could not load image resource: %s\n", resName.c_str());
846 return nullptr;
847 }
848
849 // TODO: non-intrisic image sizing
850 return sksg::Image::Make(
851 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
852}
853
Florin Malita0cc01b72018-05-10 18:40:35 -0400854sk_sp<sksg::RenderNode> AttachImageLayer(const json::ValueRef& jlayer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500855 float*, float*) {
Florin Malita0cc01b72018-05-10 18:40:35 -0400856 SkASSERT(jlayer.isObject());
Florin Malitadd22cf962018-01-29 15:42:01 +0000857
Florin Malita0cc01b72018-05-10 18:40:35 -0400858 return AttachAssetRef(jlayer, ctx, AttachImageAsset);
Florin Malita094ccde2017-12-30 12:27:00 -0500859}
860
Florin Malitafa7e9a82018-05-04 15:10:54 -0400861sk_sp<sksg::RenderNode> AttachNullLayer(const json::ValueRef& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500862 SkASSERT(layer.isObject());
863
Florin Malita18eafd92018-01-04 21:11:55 -0500864 // Null layers are used solely to drive dependent transforms,
865 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500866 return nullptr;
867}
868
Florin Malitafa7e9a82018-05-04 15:10:54 -0400869sk_sp<sksg::RenderNode> AttachShapeLayer(const json::ValueRef& layer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500870 float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500871 SkASSERT(layer.isObject());
872
Florin Malita16d0ad02018-01-19 15:07:29 -0500873 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
874 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500875 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
876 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
877
878 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
879 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
880 // due to attached animators. To avoid this, we track committed animators and discard the
881 // orphans here.
882 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
883 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
884
885 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500886}
887
Florin Malitafa7e9a82018-05-04 15:10:54 -0400888sk_sp<sksg::RenderNode> AttachTextLayer(const json::ValueRef& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500889 SkASSERT(layer.isObject());
890
891 LOG("?? Text layer stub\n");
892 return nullptr;
893}
894
Florin Malita18eafd92018-01-04 21:11:55 -0500895struct AttachLayerContext {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400896 AttachLayerContext(const json::ValueRef& jlayers, AttachContext* ctx)
Florin Malita2919b612018-05-10 12:44:07 -0400897 : fLayerList(jlayers), fCtx(ctx) {
898 SkASSERT(fLayerList.isArray());
899 }
Florin Malita18eafd92018-01-04 21:11:55 -0500900
Florin Malita2919b612018-05-10 12:44:07 -0400901 const json::ValueRef fLayerList;
Florin Malita4a490682018-01-28 14:27:51 -0500902 AttachContext* fCtx;
903 SkTHashMap<int, sk_sp<sksg::Matrix>> fLayerMatrixMap;
904 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500905
Florin Malitafa7e9a82018-05-04 15:10:54 -0400906 sk_sp<sksg::Matrix> AttachLayerMatrix(const json::ValueRef& jlayer) {
Florin Malita18eafd92018-01-04 21:11:55 -0500907 SkASSERT(jlayer.isObject());
908
Florin Malitafa7e9a82018-05-04 15:10:54 -0400909 const auto layer_index = jlayer["ind"].toDefault<int>(-1);
Florin Malita4a490682018-01-28 14:27:51 -0500910 if (layer_index < 0)
911 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500912
Florin Malita4a490682018-01-28 14:27:51 -0500913 if (auto* m = fLayerMatrixMap.find(layer_index))
914 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500915
Florin Malita2919b612018-05-10 12:44:07 -0400916 return this->AttachLayerMatrixImpl(jlayer, layer_index);
917 }
918
919private:
920 sk_sp<sksg::Matrix> AttachParentLayerMatrix(const json::ValueRef& jlayer, int layer_index) {
921 SkASSERT(jlayer.isObject());
922
923 const auto parent_index = jlayer["parent"].toDefault<int>(-1);
924 if (parent_index < 0 || parent_index == layer_index)
925 return nullptr;
926
927 if (auto* m = fLayerMatrixMap.find(parent_index))
928 return *m;
929
930 for (const json::ValueRef l : fLayerList) {
931 if (l["ind"].toDefault<int>(-1) == parent_index) {
932 return this->AttachLayerMatrixImpl(l, parent_index);
933 }
934 }
935
936 return nullptr;
937 }
938
939 sk_sp<sksg::Matrix> AttachLayerMatrixImpl(const json::ValueRef& jlayer, int layer_index) {
940 SkASSERT(!fLayerMatrixMap.find(layer_index));
941
Florin Malita4a490682018-01-28 14:27:51 -0500942 // Add a stub entry to break recursion cycles.
943 fLayerMatrixMap.set(layer_index, nullptr);
Florin Malita18eafd92018-01-04 21:11:55 -0500944
Florin Malita2919b612018-05-10 12:44:07 -0400945 auto parent_matrix = this->AttachParentLayerMatrix(jlayer, layer_index);
Florin Malita18eafd92018-01-04 21:11:55 -0500946
Florin Malita2919b612018-05-10 12:44:07 -0400947 return *fLayerMatrixMap.set(layer_index, AttachMatrix(jlayer["ks"], fCtx, parent_matrix));
Florin Malita18eafd92018-01-04 21:11:55 -0500948 }
949};
950
Florin Malita25366fa2018-01-23 13:37:59 -0500951SkBlendMode MaskBlendMode(char mode) {
952 switch (mode) {
953 case 'a': return SkBlendMode::kSrcOver; // Additive
954 case 's': return SkBlendMode::kExclusion; // Subtract
955 case 'i': return SkBlendMode::kDstIn; // Intersect
956 case 'l': return SkBlendMode::kLighten; // Lighten
957 case 'd': return SkBlendMode::kDarken; // Darken
958 case 'f': return SkBlendMode::kDifference; // Difference
959 default: break;
960 }
961
962 return SkBlendMode::kSrcOver;
963}
964
Florin Malitafa7e9a82018-05-04 15:10:54 -0400965sk_sp<sksg::RenderNode> AttachMask(const json::ValueRef& jmask,
Florin Malita25366fa2018-01-23 13:37:59 -0500966 AttachContext* ctx,
967 sk_sp<sksg::RenderNode> childNode) {
968 if (!jmask.isArray())
969 return childNode;
970
Florin Malita0c51c212018-04-26 14:13:14 -0400971 struct MaskRecord {
972 sk_sp<sksg::Path> mask_path;
973 sk_sp<sksg::Color> mask_paint;
974 };
975
976 SkSTArray<4, MaskRecord, true> mask_stack;
977
978 bool opaque_mask = true;
Florin Malita25366fa2018-01-23 13:37:59 -0500979
Florin Malitafa7e9a82018-05-04 15:10:54 -0400980 for (const json::ValueRef m : jmask) {
Florin Malita25366fa2018-01-23 13:37:59 -0500981 if (!m.isObject())
982 continue;
983
Florin Malita25366fa2018-01-23 13:37:59 -0500984 auto mask_path = AttachPath(m["pt"], ctx);
985 if (!mask_path) {
986 LogFail(m, "Could not parse mask path");
987 continue;
988 }
989
Florin Malitafa7e9a82018-05-04 15:10:54 -0400990 mask_path->setFillType(m["inv"].toDefault(false)
Florin Malitacd05b192018-04-25 21:43:03 -0400991 ? SkPath::kInverseWinding_FillType
992 : SkPath::kWinding_FillType);
993
Florin Malitacf8ed522018-01-25 15:27:33 -0500994 SkString mode;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400995 if (!m["mode"].to(&mode) ||
Florin Malitacf8ed522018-01-25 15:27:33 -0500996 mode.size() != 1 ||
997 !strcmp(mode.c_str(), "n")) { // "None" masks have no effect.
Florin Malita25366fa2018-01-23 13:37:59 -0500998 continue;
Florin Malitacf8ed522018-01-25 15:27:33 -0500999 }
Florin Malita25366fa2018-01-23 13:37:59 -05001000
1001 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
Florin Malitaaa71c892018-01-30 09:27:01 -05001002 mask_paint->setAntiAlias(true);
Florin Malita25366fa2018-01-23 13:37:59 -05001003 mask_paint->setBlendMode(MaskBlendMode(mode.c_str()[0]));
Florin Malita0c51c212018-04-26 14:13:14 -04001004
1005 const auto animator_count = ctx->fAnimators.size();
Florin Malitafc807c82018-01-25 22:35:09 -05001006 BindProperty<ScalarValue>(m["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -05001007 [mask_paint](const ScalarValue& o) { mask_paint->setOpacity(o * 0.01f); });
Florin Malita25366fa2018-01-23 13:37:59 -05001008
Florin Malita0c51c212018-04-26 14:13:14 -04001009 opaque_mask &= (animator_count == ctx->fAnimators.size() && mask_paint->getOpacity() >= 1);
1010
1011 mask_stack.push_back({mask_path, mask_paint});
Florin Malita25366fa2018-01-23 13:37:59 -05001012 }
1013
Florin Malita0c51c212018-04-26 14:13:14 -04001014 if (mask_stack.empty())
1015 return childNode;
1016
1017 if (mask_stack.count() == 1 && opaque_mask) {
1018 // Single opaque mask => clip path.
1019 return sksg::ClipEffect::Make(std::move(childNode),
1020 std::move(mask_stack.front().mask_path),
1021 true);
1022 }
1023
1024 auto mask_group = sksg::Group::Make();
1025 for (const auto& rec : mask_stack) {
1026 mask_group->addChild(sksg::Draw::Make(std::move(rec.mask_path),
1027 std::move(rec.mask_paint)));
1028
1029 }
1030
1031 return sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
Florin Malita25366fa2018-01-23 13:37:59 -05001032}
1033
Florin Malitafa7e9a82018-05-04 15:10:54 -04001034sk_sp<sksg::RenderNode> AttachLayer(const json::ValueRef& jlayer, AttachLayerContext* layerCtx) {
Florin Malita18eafd92018-01-04 21:11:55 -05001035 if (!jlayer.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -05001036 return nullptr;
1037
Florin Malitafa7e9a82018-05-04 15:10:54 -04001038 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const json::ValueRef&, AttachContext*,
Florin Malitaeb87d672018-01-29 15:28:24 -05001039 float* time_bias, float* time_scale);
Florin Malita094ccde2017-12-30 12:27:00 -05001040 static constexpr LayerAttacher gLayerAttachers[] = {
1041 AttachCompLayer, // 'ty': 0
1042 AttachSolidLayer, // 'ty': 1
1043 AttachImageLayer, // 'ty': 2
1044 AttachNullLayer, // 'ty': 3
1045 AttachShapeLayer, // 'ty': 4
1046 AttachTextLayer, // 'ty': 5
1047 };
1048
Florin Malitafa7e9a82018-05-04 15:10:54 -04001049 int type = jlayer["ty"].toDefault<int>(-1);
Florin Malita094ccde2017-12-30 12:27:00 -05001050 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
1051 return nullptr;
1052 }
1053
Florin Malitacca86f32018-01-29 10:49:49 -05001054 sksg::AnimatorList layer_animators;
Florin Malita1022f742018-02-23 11:10:22 -05001055 AttachContext local_ctx = { layerCtx->fCtx->fResources,
1056 layerCtx->fCtx->fAssets,
Florin Malita911ae402018-05-31 16:45:29 -04001057 layerCtx->fCtx->fDuration,
Florin Malita1022f742018-02-23 11:10:22 -05001058 layer_animators};
Florin Malitacca86f32018-01-29 10:49:49 -05001059
Florin Malitaeb87d672018-01-29 15:28:24 -05001060 // Layer attachers may adjust these.
1061 float time_bias = 0,
1062 time_scale = 1;
1063
Florin Malita71cba8f2018-01-09 08:07:14 -05001064 // Layer content.
Florin Malitaeb87d672018-01-29 15:28:24 -05001065 auto layer = gLayerAttachers[type](jlayer, &local_ctx, &time_bias, &time_scale);
Florin Malita38ea40e2018-01-29 16:31:14 -05001066
1067 // Clip layers with explicit dimensions.
Florin Malitafa7e9a82018-05-04 15:10:54 -04001068 float w = 0, h = 0;
1069 if (jlayer["w"].to(&w) && jlayer["h"].to(&h)) {
Florin Malita38ea40e2018-01-29 16:31:14 -05001070 layer = sksg::ClipEffect::Make(std::move(layer),
1071 sksg::Rect::Make(SkRect::MakeWH(w, h)),
1072 true);
1073 }
1074
Florin Malita25366fa2018-01-23 13:37:59 -05001075 // Optional layer mask.
Florin Malitacca86f32018-01-29 10:49:49 -05001076 layer = AttachMask(jlayer["masksProperties"], &local_ctx, std::move(layer));
Florin Malita38ea40e2018-01-29 16:31:14 -05001077
Florin Malita25366fa2018-01-23 13:37:59 -05001078 // Optional layer transform.
Florin Malita71cba8f2018-01-09 08:07:14 -05001079 if (auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -05001080 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
1081 }
Florin Malita38ea40e2018-01-29 16:31:14 -05001082
Florin Malita71cba8f2018-01-09 08:07:14 -05001083 // Optional layer opacity.
Florin Malitacca86f32018-01-29 10:49:49 -05001084 layer = AttachOpacity(jlayer["ks"], &local_ctx, std::move(layer));
Florin Malita18eafd92018-01-04 21:11:55 -05001085
Florin Malitaeb87d672018-01-29 15:28:24 -05001086 class LayerController final : public sksg::GroupAnimator {
Florin Malita71cba8f2018-01-09 08:07:14 -05001087 public:
Florin Malitaeb87d672018-01-29 15:28:24 -05001088 LayerController(sksg::AnimatorList&& layer_animators,
1089 sk_sp<sksg::OpacityEffect> controlNode,
1090 float in, float out,
1091 float time_bias, float time_scale)
Florin Malitacca86f32018-01-29 10:49:49 -05001092 : INHERITED(std::move(layer_animators))
1093 , fControlNode(std::move(controlNode))
Florin Malita71cba8f2018-01-09 08:07:14 -05001094 , fIn(in)
Florin Malitaeb87d672018-01-29 15:28:24 -05001095 , fOut(out)
1096 , fTimeBias(time_bias)
1097 , fTimeScale(time_scale) {}
Florin Malita71cba8f2018-01-09 08:07:14 -05001098
Florin Malita35efaa82018-01-22 12:57:06 -05001099 void onTick(float t) override {
Florin Malitacca86f32018-01-29 10:49:49 -05001100 const auto active = (t >= fIn && t <= fOut);
1101
Florin Malita71cba8f2018-01-09 08:07:14 -05001102 // Keep the layer fully transparent except for its [in..out] lifespan.
1103 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
Florin Malitacca86f32018-01-29 10:49:49 -05001104 fControlNode->setOpacity(active ? 1 : 0);
1105
1106 // Dispatch ticks only while active.
1107 if (active)
Florin Malitaeb87d672018-01-29 15:28:24 -05001108 this->INHERITED::onTick((t + fTimeBias) * fTimeScale);
Florin Malita71cba8f2018-01-09 08:07:14 -05001109 }
1110
1111 private:
1112 const sk_sp<sksg::OpacityEffect> fControlNode;
1113 const float fIn,
Florin Malitaeb87d672018-01-29 15:28:24 -05001114 fOut,
1115 fTimeBias,
1116 fTimeScale;
Florin Malitacca86f32018-01-29 10:49:49 -05001117
1118 using INHERITED = sksg::GroupAnimator;
Florin Malita71cba8f2018-01-09 08:07:14 -05001119 };
1120
Florin Malitaeb87d672018-01-29 15:28:24 -05001121 auto controller_node = sksg::OpacityEffect::Make(std::move(layer));
Florin Malitafa7e9a82018-05-04 15:10:54 -04001122 const auto in = jlayer["ip"].toDefault(0.0f),
1123 out = jlayer["op"].toDefault(in);
Florin Malita71cba8f2018-01-09 08:07:14 -05001124
Florin Malitaeb87d672018-01-29 15:28:24 -05001125 if (!jlayer["tm"].isNull()) {
1126 LogFail(jlayer["tm"], "Unsupported time remapping");
1127 }
1128
1129 if (in >= out || !controller_node)
Florin Malita71cba8f2018-01-09 08:07:14 -05001130 return nullptr;
1131
Florin Malitacca86f32018-01-29 10:49:49 -05001132 layerCtx->fCtx->fAnimators.push_back(
Florin Malitaeb87d672018-01-29 15:28:24 -05001133 skstd::make_unique<LayerController>(std::move(layer_animators),
1134 controller_node,
1135 in,
1136 out,
1137 time_bias,
1138 time_scale));
Florin Malita71cba8f2018-01-09 08:07:14 -05001139
Florin Malitafa7e9a82018-05-04 15:10:54 -04001140 if (jlayer["td"].toDefault(false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -05001141 // This layer is a matte. We apply it as a mask to the next layer.
Florin Malitaeb87d672018-01-29 15:28:24 -05001142 layerCtx->fCurrentMatte = std::move(controller_node);
Florin Malita5f9102f2018-01-10 13:36:22 -05001143 return nullptr;
1144 }
1145
1146 if (layerCtx->fCurrentMatte) {
1147 // There is a pending matte. Apply and reset.
Florin Malitaa016be92018-03-05 14:01:41 -05001148 static constexpr sksg::MaskEffect::Mode gMaskModes[] = {
1149 sksg::MaskEffect::Mode::kNormal, // tt: 1
1150 sksg::MaskEffect::Mode::kInvert, // tt: 2
1151 };
Florin Malitafa7e9a82018-05-04 15:10:54 -04001152 const auto matteType = jlayer["tt"].toDefault<int>(1) - 1;
Florin Malitaa016be92018-03-05 14:01:41 -05001153
1154 if (matteType >= 0 && matteType < SkTo<int>(SK_ARRAY_COUNT(gMaskModes))) {
1155 return sksg::MaskEffect::Make(std::move(controller_node),
1156 std::move(layerCtx->fCurrentMatte),
1157 gMaskModes[matteType]);
1158 }
1159 layerCtx->fCurrentMatte.reset();
Florin Malita5f9102f2018-01-10 13:36:22 -05001160 }
1161
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001162 return std::move(controller_node);
Florin Malita094ccde2017-12-30 12:27:00 -05001163}
1164
Florin Malitafa7e9a82018-05-04 15:10:54 -04001165sk_sp<sksg::RenderNode> AttachComposition(const json::ValueRef& comp, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -05001166 if (!comp.isObject())
1167 return nullptr;
1168
Florin Malitafa7e9a82018-05-04 15:10:54 -04001169 const auto jlayers = comp["layers"];
Florin Malita18eafd92018-01-04 21:11:55 -05001170 if (!jlayers.isArray())
1171 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001172
Florin Malita18eafd92018-01-04 21:11:55 -05001173 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
1174 AttachLayerContext layerCtx(jlayers, ctx);
1175
Florin Malitafa7e9a82018-05-04 15:10:54 -04001176 for (const json::ValueRef l : jlayers) {
Florin Malita18eafd92018-01-04 21:11:55 -05001177 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001178 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001179 }
1180 }
1181
Florin Malita2a8275b2018-01-02 12:52:43 -05001182 if (layers.empty()) {
1183 return nullptr;
1184 }
1185
1186 // Layers are painted in bottom->top order.
1187 auto comp_group = sksg::Group::Make();
1188 for (int i = layers.count() - 1; i >= 0; --i) {
1189 comp_group->addChild(std::move(layers[i]));
1190 }
1191
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001192 return std::move(comp_group);
Florin Malita094ccde2017-12-30 12:27:00 -05001193}
1194
1195} // namespace
1196
Florin Malitac83a0de2018-05-31 12:17:55 -04001197sk_sp<Animation> Animation::Make(SkStream* stream, const ResourceProvider* provider, Stats* stats) {
Florin Malita6eb85a12018-04-30 10:32:18 -04001198 Stats stats_storage;
1199 if (!stats)
1200 stats = &stats_storage;
1201 memset(stats, 0, sizeof(struct Stats));
1202
Florin Malita094ccde2017-12-30 12:27:00 -05001203 if (!stream->hasLength()) {
1204 // TODO: handle explicit buffering?
1205 LOG("!! cannot parse streaming content\n");
1206 return nullptr;
1207 }
1208
Florin Malitafa7e9a82018-05-04 15:10:54 -04001209 stats->fJsonSize = stream->getLength();
Florin Malita6eb85a12018-04-30 10:32:18 -04001210 const auto t0 = SkTime::GetMSecs();
1211
Florin Malitafa7e9a82018-05-04 15:10:54 -04001212 const json::Document doc(stream);
1213 const auto json = doc.root();
1214 if (!json.isObject())
1215 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001216
Florin Malita6eb85a12018-04-30 10:32:18 -04001217 const auto t1 = SkTime::GetMSecs();
1218 stats->fJsonParseTimeMS = t1 - t0;
1219
Florin Malitafa7e9a82018-05-04 15:10:54 -04001220 const auto version = json["v"].toDefault(SkString());
1221 const auto size = SkSize::Make(json["w"].toDefault(0.0f),
1222 json["h"].toDefault(0.0f));
1223 const auto fps = json["fr"].toDefault(-1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001224
Florin Malita1022f742018-02-23 11:10:22 -05001225 if (size.isEmpty() || version.isEmpty() || fps <= 0) {
Florin Malita094ccde2017-12-30 12:27:00 -05001226 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1227 version.c_str(), size.width(), size.height(), fps);
1228 return nullptr;
1229 }
1230
Florin Malitac83a0de2018-05-31 12:17:55 -04001231 class NullResourceProvider final : public ResourceProvider {
1232 std::unique_ptr<SkStream> openStream(const char[]) const { return nullptr; }
1233 };
1234
Kevin Lubickf14bc982018-06-01 13:00:35 -04001235 NullResourceProvider null_provider;
Florin Malitac83a0de2018-05-31 12:17:55 -04001236 const auto anim = sk_sp<Animation>(new Animation(provider ? *provider : null_provider,
1237 std::move(version), size, fps, json, stats));
Florin Malita6eb85a12018-04-30 10:32:18 -04001238 const auto t2 = SkTime::GetMSecs();
1239 stats->fSceneParseTimeMS = t2 - t1;
1240 stats->fTotalLoadTimeMS = t2 - t0;
1241
1242 return anim;
Florin Malita094ccde2017-12-30 12:27:00 -05001243}
1244
Florin Malita6eb85a12018-04-30 10:32:18 -04001245sk_sp<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res,
1246 Stats* stats) {
Florin Malita49328072018-01-08 12:51:12 -05001247 class DirectoryResourceProvider final : public ResourceProvider {
1248 public:
1249 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1250
1251 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1252 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1253 return SkStream::MakeFromFile(resPath.c_str());
1254 }
1255
1256 private:
1257 const SkString fDir;
1258 };
1259
1260 const auto jsonStream = SkStream::MakeFromFile(path);
1261 if (!jsonStream)
1262 return nullptr;
1263
1264 std::unique_ptr<ResourceProvider> defaultProvider;
1265 if (!res) {
1266 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1267 }
1268
Florin Malitac83a0de2018-05-31 12:17:55 -04001269 return Make(jsonStream.get(), res ? res : defaultProvider.get(), stats);
Florin Malita49328072018-01-08 12:51:12 -05001270}
1271
1272Animation::Animation(const ResourceProvider& resources,
Florin Malitafa7e9a82018-05-04 15:10:54 -04001273 SkString version, const SkSize& size, SkScalar fps, const json::ValueRef& json,
Florin Malita6eb85a12018-04-30 10:32:18 -04001274 Stats* stats)
Florin Malita094ccde2017-12-30 12:27:00 -05001275 : fVersion(std::move(version))
1276 , fSize(size)
1277 , fFrameRate(fps)
Florin Malitafa7e9a82018-05-04 15:10:54 -04001278 , fInPoint(json["ip"].toDefault(0.0f))
1279 , fOutPoint(SkTMax(json["op"].toDefault(SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001280
1281 AssetMap assets;
Florin Malitafa7e9a82018-05-04 15:10:54 -04001282 for (const json::ValueRef asset : json["assets"]) {
1283 if (asset.isObject()) {
Florin Malita0cc01b72018-05-10 18:40:35 -04001284 assets.set(asset["id"].toDefault(SkString()), { asset, false });
Florin Malita094ccde2017-12-30 12:27:00 -05001285 }
Florin Malita094ccde2017-12-30 12:27:00 -05001286 }
1287
Florin Malitacca86f32018-01-29 10:49:49 -05001288 sksg::AnimatorList animators;
Florin Malita911ae402018-05-31 16:45:29 -04001289 AttachContext ctx = { resources, assets, this->duration(), animators };
Florin Malita35efaa82018-01-22 12:57:06 -05001290 auto root = AttachComposition(json, &ctx);
1291
Florin Malita6eb85a12018-04-30 10:32:18 -04001292 stats->fAnimatorCount = animators.size();
Florin Malita35efaa82018-01-22 12:57:06 -05001293
1294 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001295
Florin Malitadb385732018-01-09 12:19:32 -05001296 // In case the client calls render before the first tick.
Florin Malitaa33447d2018-05-29 13:46:54 -04001297 this->seek(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001298}
1299
1300Animation::~Animation() = default;
1301
Florin Malita35efaa82018-01-22 12:57:06 -05001302void Animation::setShowInval(bool show) {
1303 if (fScene) {
1304 fScene->setShowInval(show);
1305 }
1306}
1307
Mike Reed29859872018-01-08 08:25:27 -05001308void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001309 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001310 return;
1311
Mike Reed29859872018-01-08 08:25:27 -05001312 SkAutoCanvasRestore restore(canvas, true);
1313 const SkRect srcR = SkRect::MakeSize(this->size());
1314 if (dstR) {
1315 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1316 }
1317 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001318 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001319}
1320
Florin Malitaa33447d2018-05-29 13:46:54 -04001321void Animation::seek(SkScalar t) {
Florin Malita35efaa82018-01-22 12:57:06 -05001322 if (!fScene)
1323 return;
1324
Florin Malitaa33447d2018-05-29 13:46:54 -04001325 fScene->animate(fInPoint + SkTPin(t, 0.0f, 1.0f) * (fOutPoint - fInPoint));
Florin Malita094ccde2017-12-30 12:27:00 -05001326}
1327
Florin Malita54f65c42018-01-16 17:04:30 -05001328} // namespace skottie