blob: 93c71e936a151bddc6f97d40dae17b237049f411 [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 Malitaa6e30f72018-03-23 13:41:58 -040011#include "SkottieAdapter.h"
Florin Malita54f65c42018-01-16 17:04:30 -050012#include "SkottieAnimator.h"
Florin Malitafa7e9a82018-05-04 15:10:54 -040013#include "SkottieJson.h"
Florin Malitaa6e30f72018-03-23 13:41:58 -040014#include "SkottieValue.h"
Florin Malita094ccde2017-12-30 12:27:00 -050015#include "SkData.h"
Florin Malita49328072018-01-08 12:51:12 -050016#include "SkImage.h"
Florin Malita094ccde2017-12-30 12:27:00 -050017#include "SkMakeUnique.h"
Florin Malita49328072018-01-08 12:51:12 -050018#include "SkOSPath.h"
Florin Malita094ccde2017-12-30 12:27:00 -050019#include "SkPaint.h"
Florin Malita0e66fba2018-01-09 17:10:18 -050020#include "SkParse.h"
Florin Malita094ccde2017-12-30 12:27:00 -050021#include "SkPoint.h"
Florin Malita38ea40e2018-01-29 16:31:14 -050022#include "SkSGClipEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050023#include "SkSGColor.h"
24#include "SkSGDraw.h"
Florin Malita16d0ad02018-01-19 15:07:29 -050025#include "SkSGGeometryTransform.h"
Florin Malita6aaee592018-01-12 12:25:09 -050026#include "SkSGGradient.h"
Florin Malita094ccde2017-12-30 12:27:00 -050027#include "SkSGGroup.h"
Florin Malita49328072018-01-08 12:51:12 -050028#include "SkSGImage.h"
29#include "SkSGInvalidationController.h"
Florin Malita5f9102f2018-01-10 13:36:22 -050030#include "SkSGMaskEffect.h"
Florin Malitae6345d92018-01-03 23:37:54 -050031#include "SkSGMerge.h"
Florin Malitac0034172018-01-08 16:42:59 -050032#include "SkSGOpacityEffect.h"
Florin Malita094ccde2017-12-30 12:27:00 -050033#include "SkSGPath.h"
Florin Malita2e1d7e22018-01-02 10:40:00 -050034#include "SkSGRect.h"
Florin Malita41dff6e2018-04-30 23:08:15 -040035#include "SkSGRoundEffect.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 Malita0cc01b72018-05-10 18:40:35 -040055struct AssetInfo {
56 json::ValueRef fAsset;
57 mutable bool fIsAttaching; // Used for cycle detection
58};
59
60using AssetMap = SkTHashMap<SkString, AssetInfo>;
Florin Malita094ccde2017-12-30 12:27:00 -050061
62struct AttachContext {
Florin Malitacca86f32018-01-29 10:49:49 -050063 const ResourceProvider& fResources;
64 const AssetMap& fAssets;
Florin Malita911ae402018-05-31 16:45:29 -040065 const float fDuration;
Florin Malitacca86f32018-01-29 10:49:49 -050066 sksg::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050067};
68
Florin Malitafa7e9a82018-05-04 15:10:54 -040069bool LogFail(const json::ValueRef& json, const char* msg) {
70 const auto dump = json.toString();
Florin Malita20880782018-05-09 11:35:00 -040071 LOG("!! %s: %s\n", msg, dump.c_str());
Florin Malita094ccde2017-12-30 12:27:00 -050072 return false;
73}
74
Florin Malitafa7e9a82018-05-04 15:10:54 -040075sk_sp<sksg::Matrix> AttachMatrix(const json::ValueRef& t, AttachContext* ctx,
Florin Malita18eafd92018-01-04 21:11:55 -050076 sk_sp<sksg::Matrix> parentMatrix) {
77 if (!t.isObject())
78 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -050079
Florin Malita18eafd92018-01-04 21:11:55 -050080 auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
Florin Malitaa6e30f72018-03-23 13:41:58 -040081 auto adapter = sk_make_sp<TransformAdapter>(matrix);
Florin Malitafc807c82018-01-25 22:35:09 -050082 auto anchor_attached = BindProperty<VectorValue>(t["a"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040083 [adapter](const VectorValue& a) {
84 adapter->setAnchorPoint(ValueTraits<VectorValue>::As<SkPoint>(a));
Florin Malita094ccde2017-12-30 12:27:00 -050085 });
Florin Malitafc807c82018-01-25 22:35:09 -050086 auto position_attached = BindProperty<VectorValue>(t["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040087 [adapter](const VectorValue& p) {
88 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malita094ccde2017-12-30 12:27:00 -050089 });
Florin Malitafc807c82018-01-25 22:35:09 -050090 auto scale_attached = BindProperty<VectorValue>(t["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040091 [adapter](const VectorValue& s) {
92 adapter->setScale(ValueTraits<VectorValue>::As<SkVector>(s));
Florin Malita094ccde2017-12-30 12:27:00 -050093 });
Florin Malita1eb98db2018-01-26 15:03:38 -050094
Florin Malitafa7e9a82018-05-04 15:10:54 -040095 auto jrotation = t["r"];
96 if (jrotation.isNull()) {
Florin Malita1eb98db2018-01-26 15:03:38 -050097 // 3d rotations have separate rx,ry,rz components. While we don't fully support them,
98 // we can still make use of rz.
Florin Malitafa7e9a82018-05-04 15:10:54 -040099 jrotation = t["rz"];
Florin Malita1eb98db2018-01-26 15:03:38 -0500100 }
Florin Malitafa7e9a82018-05-04 15:10:54 -0400101 auto rotation_attached = BindProperty<ScalarValue>(jrotation, &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400102 [adapter](const ScalarValue& r) {
103 adapter->setRotation(r);
Florin Malita094ccde2017-12-30 12:27:00 -0500104 });
Florin Malitafc807c82018-01-25 22:35:09 -0500105 auto skew_attached = BindProperty<ScalarValue>(t["sk"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400106 [adapter](const ScalarValue& sk) {
107 adapter->setSkew(sk);
Florin Malita094ccde2017-12-30 12:27:00 -0500108 });
Florin Malitafc807c82018-01-25 22:35:09 -0500109 auto skewaxis_attached = BindProperty<ScalarValue>(t["sa"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400110 [adapter](const ScalarValue& sa) {
111 adapter->setSkewAxis(sa);
Florin Malita094ccde2017-12-30 12:27:00 -0500112 });
113
114 if (!anchor_attached &&
115 !position_attached &&
116 !scale_attached &&
117 !rotation_attached &&
118 !skew_attached &&
119 !skewaxis_attached) {
120 LogFail(t, "Could not parse transform");
Florin Malita18eafd92018-01-04 21:11:55 -0500121 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500122 }
123
Florin Malita18eafd92018-01-04 21:11:55 -0500124 return matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500125}
126
Florin Malitafa7e9a82018-05-04 15:10:54 -0400127sk_sp<sksg::RenderNode> AttachOpacity(const json::ValueRef& jtransform, AttachContext* ctx,
Florin Malitac0034172018-01-08 16:42:59 -0500128 sk_sp<sksg::RenderNode> childNode) {
129 if (!jtransform.isObject() || !childNode)
130 return childNode;
131
Florin Malita7ac2e3b2018-05-09 14:54:39 -0400132 static constexpr ScalarValue kNoopOpacity = 100;
Florin Malitac0034172018-01-08 16:42:59 -0500133 auto opacityNode = sksg::OpacityEffect::Make(childNode);
Florin Malita7ac2e3b2018-05-09 14:54:39 -0400134
135 if (!BindProperty<ScalarValue>(jtransform["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500136 [opacityNode](const ScalarValue& o) {
Florin Malitac0034172018-01-08 16:42:59 -0500137 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500138 opacityNode->setOpacity(o * 0.01f);
Florin Malita7ac2e3b2018-05-09 14:54:39 -0400139 }, &kNoopOpacity)) {
140 // We can ignore static full opacity.
141 return childNode;
142 }
Florin Malitac0034172018-01-08 16:42:59 -0500143
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400144 return std::move(opacityNode);
Florin Malitac0034172018-01-08 16:42:59 -0500145}
146
Florin Malitafa7e9a82018-05-04 15:10:54 -0400147sk_sp<sksg::RenderNode> AttachComposition(const json::ValueRef&, AttachContext* ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500148
Florin Malitafa7e9a82018-05-04 15:10:54 -0400149sk_sp<sksg::Path> AttachPath(const json::ValueRef& jpath, AttachContext* ctx) {
Florin Malita25366fa2018-01-23 13:37:59 -0500150 auto path_node = sksg::Path::Make();
Florin Malitafc807c82018-01-25 22:35:09 -0500151 return BindProperty<ShapeValue>(jpath, &ctx->fAnimators,
Florin Malitac353ee22018-04-30 21:49:41 -0400152 [path_node](const ShapeValue& p) {
153 path_node->setPath(ValueTraits<ShapeValue>::As<SkPath>(p));
154 })
Florin Malita25366fa2018-01-23 13:37:59 -0500155 ? path_node
156 : nullptr;
157}
158
Florin Malitafa7e9a82018-05-04 15:10:54 -0400159sk_sp<sksg::GeometryNode> AttachPathGeometry(const json::ValueRef& jpath, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500160 SkASSERT(jpath.isObject());
161
Florin Malita25366fa2018-01-23 13:37:59 -0500162 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500163}
164
Florin Malitafa7e9a82018-05-04 15:10:54 -0400165sk_sp<sksg::GeometryNode> AttachRRectGeometry(const json::ValueRef& jrect, AttachContext* ctx) {
Florin Malita2e1d7e22018-01-02 10:40:00 -0500166 SkASSERT(jrect.isObject());
167
168 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400169 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500170
Florin Malitafc807c82018-01-25 22:35:09 -0500171 auto p_attached = BindProperty<VectorValue>(jrect["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400172 [adapter](const VectorValue& p) {
173 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500174 });
Florin Malitafc807c82018-01-25 22:35:09 -0500175 auto s_attached = BindProperty<VectorValue>(jrect["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400176 [adapter](const VectorValue& s) {
177 adapter->setSize(ValueTraits<VectorValue>::As<SkSize>(s));
Florin Malitaf9590922018-01-09 11:56:09 -0500178 });
Florin Malitafc807c82018-01-25 22:35:09 -0500179 auto r_attached = BindProperty<ScalarValue>(jrect["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400180 [adapter](const ScalarValue& r) {
181 adapter->setRadius(SkSize::Make(r, r));
Florin Malitaf9590922018-01-09 11:56:09 -0500182 });
Florin Malita2e1d7e22018-01-02 10:40:00 -0500183
184 if (!p_attached && !s_attached && !r_attached) {
185 return nullptr;
186 }
187
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400188 return std::move(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500189}
190
Florin Malitafa7e9a82018-05-04 15:10:54 -0400191sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const json::ValueRef& jellipse, AttachContext* ctx) {
Florin Malitafbc13f12018-01-04 10:26:35 -0500192 SkASSERT(jellipse.isObject());
193
194 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400195 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500196
Florin Malitafc807c82018-01-25 22:35:09 -0500197 auto p_attached = BindProperty<VectorValue>(jellipse["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400198 [adapter](const VectorValue& p) {
199 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500200 });
Florin Malitafc807c82018-01-25 22:35:09 -0500201 auto s_attached = BindProperty<VectorValue>(jellipse["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400202 [adapter](const VectorValue& s) {
Florin Malitaf9590922018-01-09 11:56:09 -0500203 const auto sz = ValueTraits<VectorValue>::As<SkSize>(s);
Florin Malitaa6e30f72018-03-23 13:41:58 -0400204 adapter->setSize(sz);
205 adapter->setRadius(SkSize::Make(sz.width() / 2, sz.height() / 2));
Florin Malitaf9590922018-01-09 11:56:09 -0500206 });
Florin Malitafbc13f12018-01-04 10:26:35 -0500207
208 if (!p_attached && !s_attached) {
209 return nullptr;
210 }
211
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400212 return std::move(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500213}
214
Florin Malitafa7e9a82018-05-04 15:10:54 -0400215sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const json::ValueRef& jstar, AttachContext* ctx) {
Florin Malita02a32b02018-01-04 11:27:09 -0500216 SkASSERT(jstar.isObject());
217
Florin Malitaa6e30f72018-03-23 13:41:58 -0400218 static constexpr PolyStarAdapter::Type gTypes[] = {
219 PolyStarAdapter::Type::kStar, // "sy": 1
220 PolyStarAdapter::Type::kPoly, // "sy": 2
Florin Malita02a32b02018-01-04 11:27:09 -0500221 };
222
Florin Malitafa7e9a82018-05-04 15:10:54 -0400223 const auto type = jstar["sy"].toDefault<int>(0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500224 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
225 LogFail(jstar, "Unknown polystar type");
226 return nullptr;
227 }
228
229 auto path_node = sksg::Path::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400230 auto adapter = sk_make_sp<PolyStarAdapter>(path_node, gTypes[type]);
Florin Malita02a32b02018-01-04 11:27:09 -0500231
Florin Malitafc807c82018-01-25 22:35:09 -0500232 BindProperty<VectorValue>(jstar["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400233 [adapter](const VectorValue& p) {
234 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500235 });
Florin Malitafc807c82018-01-25 22:35:09 -0500236 BindProperty<ScalarValue>(jstar["pt"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400237 [adapter](const ScalarValue& pt) {
238 adapter->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500239 });
Florin Malitafc807c82018-01-25 22:35:09 -0500240 BindProperty<ScalarValue>(jstar["ir"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400241 [adapter](const ScalarValue& ir) {
242 adapter->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500243 });
Florin Malitafc807c82018-01-25 22:35:09 -0500244 BindProperty<ScalarValue>(jstar["or"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400245 [adapter](const ScalarValue& otr) {
246 adapter->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500247 });
Florin Malitafc807c82018-01-25 22:35:09 -0500248 BindProperty<ScalarValue>(jstar["is"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400249 [adapter](const ScalarValue& is) {
250 adapter->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500251 });
Florin Malitafc807c82018-01-25 22:35:09 -0500252 BindProperty<ScalarValue>(jstar["os"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400253 [adapter](const ScalarValue& os) {
254 adapter->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500255 });
Florin Malitafc807c82018-01-25 22:35:09 -0500256 BindProperty<ScalarValue>(jstar["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400257 [adapter](const ScalarValue& r) {
258 adapter->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500259 });
Florin Malita02a32b02018-01-04 11:27:09 -0500260
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400261 return std::move(path_node);
Florin Malita02a32b02018-01-04 11:27:09 -0500262}
263
Florin Malitafa7e9a82018-05-04 15:10:54 -0400264sk_sp<sksg::Color> AttachColor(const json::ValueRef& obj, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500265 SkASSERT(obj.isObject());
266
267 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malitafc807c82018-01-25 22:35:09 -0500268 auto color_attached = BindProperty<VectorValue>(obj["c"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500269 [color_node](const VectorValue& c) {
270 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500271 });
Florin Malita094ccde2017-12-30 12:27:00 -0500272
Florin Malita1586d852018-01-12 14:27:39 -0500273 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500274}
275
Florin Malitafa7e9a82018-05-04 15:10:54 -0400276sk_sp<sksg::Gradient> AttachGradient(const json::ValueRef& obj, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500277 SkASSERT(obj.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500278
Florin Malita20880782018-05-09 11:35:00 -0400279 const auto stops = obj["g"];
Florin Malita6aaee592018-01-12 12:25:09 -0500280 if (!stops.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500281 return nullptr;
282
Florin Malitafa7e9a82018-05-04 15:10:54 -0400283 const auto stopCount = stops["p"].toDefault<int>(-1);
Florin Malita6aaee592018-01-12 12:25:09 -0500284 if (stopCount < 0)
285 return nullptr;
286
287 sk_sp<sksg::Gradient> gradient_node;
Florin Malitaa6e30f72018-03-23 13:41:58 -0400288 sk_sp<GradientAdapter> adapter;
Florin Malita6aaee592018-01-12 12:25:09 -0500289
Florin Malitafa7e9a82018-05-04 15:10:54 -0400290 if (obj["t"].toDefault<int>(1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500291 auto linear_node = sksg::LinearGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400292 adapter = sk_make_sp<LinearGradientAdapter>(linear_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500293 gradient_node = std::move(linear_node);
294 } else {
295 auto radial_node = sksg::RadialGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400296 adapter = sk_make_sp<RadialGradientAdapter>(radial_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500297
298 // TODO: highlight, angle
299 gradient_node = std::move(radial_node);
300 }
301
Florin Malitafc807c82018-01-25 22:35:09 -0500302 BindProperty<VectorValue>(stops["k"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400303 [adapter](const VectorValue& stops) {
304 adapter->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500305 });
Florin Malitafc807c82018-01-25 22:35:09 -0500306 BindProperty<VectorValue>(obj["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400307 [adapter](const VectorValue& s) {
308 adapter->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500309 });
Florin Malitafc807c82018-01-25 22:35:09 -0500310 BindProperty<VectorValue>(obj["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400311 [adapter](const VectorValue& e) {
312 adapter->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500313 });
314
315 return gradient_node;
316}
317
Florin Malitafa7e9a82018-05-04 15:10:54 -0400318sk_sp<sksg::PaintNode> AttachPaint(const json::ValueRef& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500319 sk_sp<sksg::PaintNode> paint_node) {
320 if (paint_node) {
321 paint_node->setAntiAlias(true);
322
Florin Malitafc807c82018-01-25 22:35:09 -0500323 BindProperty<ScalarValue>(jpaint["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500324 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500325 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500326 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500327 });
Florin Malita6aaee592018-01-12 12:25:09 -0500328 }
329
330 return paint_node;
331}
332
Florin Malitafa7e9a82018-05-04 15:10:54 -0400333sk_sp<sksg::PaintNode> AttachStroke(const json::ValueRef& jstroke, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500334 sk_sp<sksg::PaintNode> stroke_node) {
335 SkASSERT(jstroke.isObject());
336
337 if (!stroke_node)
338 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500339
340 stroke_node->setStyle(SkPaint::kStroke_Style);
341
Florin Malitafc807c82018-01-25 22:35:09 -0500342 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500343 [stroke_node](const ScalarValue& w) {
344 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500345 });
Florin Malita094ccde2017-12-30 12:27:00 -0500346 if (!width_attached)
347 return nullptr;
348
Florin Malitafa7e9a82018-05-04 15:10:54 -0400349 stroke_node->setStrokeMiter(jstroke["ml"].toDefault(4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500350
351 static constexpr SkPaint::Join gJoins[] = {
352 SkPaint::kMiter_Join,
353 SkPaint::kRound_Join,
354 SkPaint::kBevel_Join,
355 };
Florin Malitafa7e9a82018-05-04 15:10:54 -0400356 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(jstroke["lj"].toDefault<int>(1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500357 0, SK_ARRAY_COUNT(gJoins) - 1)]);
358
359 static constexpr SkPaint::Cap gCaps[] = {
360 SkPaint::kButt_Cap,
361 SkPaint::kRound_Cap,
362 SkPaint::kSquare_Cap,
363 };
Florin Malitafa7e9a82018-05-04 15:10:54 -0400364 stroke_node->setStrokeCap(gCaps[SkTPin<int>(jstroke["lc"].toDefault<int>(1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500365 0, SK_ARRAY_COUNT(gCaps) - 1)]);
366
367 return stroke_node;
368}
369
Florin Malitafa7e9a82018-05-04 15:10:54 -0400370sk_sp<sksg::PaintNode> AttachColorFill(const json::ValueRef& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500371 SkASSERT(jfill.isObject());
372
373 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
374}
375
Florin Malitafa7e9a82018-05-04 15:10:54 -0400376sk_sp<sksg::PaintNode> AttachGradientFill(const json::ValueRef& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500377 SkASSERT(jfill.isObject());
378
379 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
380}
381
Florin Malitafa7e9a82018-05-04 15:10:54 -0400382sk_sp<sksg::PaintNode> AttachColorStroke(const json::ValueRef& jstroke, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500383 SkASSERT(jstroke.isObject());
384
385 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
386}
387
Florin Malitafa7e9a82018-05-04 15:10:54 -0400388sk_sp<sksg::PaintNode> AttachGradientStroke(const json::ValueRef& jstroke, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500389 SkASSERT(jstroke.isObject());
390
391 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
392}
393
Florin Malitae6345d92018-01-03 23:37:54 -0500394std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400395 const json::ValueRef& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malitae6345d92018-01-03 23:37:54 -0500396 std::vector<sk_sp<sksg::GeometryNode>> merged;
397
398 static constexpr sksg::Merge::Mode gModes[] = {
399 sksg::Merge::Mode::kMerge, // "mm": 1
400 sksg::Merge::Mode::kUnion, // "mm": 2
401 sksg::Merge::Mode::kDifference, // "mm": 3
402 sksg::Merge::Mode::kIntersect, // "mm": 4
403 sksg::Merge::Mode::kXOR , // "mm": 5
404 };
405
Florin Malitafa7e9a82018-05-04 15:10:54 -0400406 const auto mode = gModes[SkTPin<int>(jmerge["mm"].toDefault(1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500407 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malitae6345d92018-01-03 23:37:54 -0500408 merged.push_back(sksg::Merge::Make(std::move(geos), mode));
409
Florin Malitae6345d92018-01-03 23:37:54 -0500410 return merged;
411}
412
Florin Malita51b8c892018-01-07 08:54:24 -0500413std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400414 const json::ValueRef& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malita51b8c892018-01-07 08:54:24 -0500415
416 enum class Mode {
417 kMerged, // "m": 1
418 kSeparate, // "m": 2
419 } gModes[] = { Mode::kMerged, Mode::kSeparate };
420
Florin Malitafa7e9a82018-05-04 15:10:54 -0400421 const auto mode = gModes[SkTPin<int>(jtrim["m"].toDefault(1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500422 0, SK_ARRAY_COUNT(gModes) - 1)];
423
424 std::vector<sk_sp<sksg::GeometryNode>> inputs;
425 if (mode == Mode::kMerged) {
426 inputs.push_back(sksg::Merge::Make(std::move(geos), sksg::Merge::Mode::kMerge));
427 } else {
428 inputs = std::move(geos);
429 }
430
431 std::vector<sk_sp<sksg::GeometryNode>> trimmed;
432 trimmed.reserve(inputs.size());
433 for (const auto& i : inputs) {
Florin Malita69526b02018-03-22 12:20:02 -0400434 const auto trimEffect = sksg::TrimEffect::Make(i);
435 trimmed.push_back(trimEffect);
436
Florin Malitaa6e30f72018-03-23 13:41:58 -0400437 const auto adapter = sk_make_sp<TrimEffectAdapter>(std::move(trimEffect));
Florin Malitafc807c82018-01-25 22:35:09 -0500438 BindProperty<ScalarValue>(jtrim["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400439 [adapter](const ScalarValue& s) {
440 adapter->setStart(s);
Florin Malita51b8c892018-01-07 08:54:24 -0500441 });
Florin Malitafc807c82018-01-25 22:35:09 -0500442 BindProperty<ScalarValue>(jtrim["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400443 [adapter](const ScalarValue& e) {
444 adapter->setEnd(e);
Florin Malita51b8c892018-01-07 08:54:24 -0500445 });
Florin Malitafc807c82018-01-25 22:35:09 -0500446 BindProperty<ScalarValue>(jtrim["o"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400447 [adapter](const ScalarValue& o) {
448 adapter->setOffset(o);
Florin Malita51b8c892018-01-07 08:54:24 -0500449 });
450 }
451
452 return trimmed;
453}
454
Florin Malita41dff6e2018-04-30 23:08:15 -0400455std::vector<sk_sp<sksg::GeometryNode>> AttachRoundGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400456 const json::ValueRef& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malita41dff6e2018-04-30 23:08:15 -0400457
458 std::vector<sk_sp<sksg::GeometryNode>> rounded;
459 rounded.reserve(geos.size());
460
461 for (const auto& g : geos) {
462 const auto roundEffect = sksg::RoundEffect::Make(std::move(g));
463 rounded.push_back(roundEffect);
464
465 BindProperty<ScalarValue>(jtrim["r"], &ctx->fAnimators,
466 [roundEffect](const ScalarValue& r) {
467 roundEffect->setRadius(r);
468 });
469 }
470
471 return rounded;
472}
473
Florin Malitafa7e9a82018-05-04 15:10:54 -0400474using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const json::ValueRef&, AttachContext*);
Florin Malita094ccde2017-12-30 12:27:00 -0500475static constexpr GeometryAttacherT gGeometryAttachers[] = {
476 AttachPathGeometry,
Florin Malita2e1d7e22018-01-02 10:40:00 -0500477 AttachRRectGeometry,
Florin Malitafbc13f12018-01-04 10:26:35 -0500478 AttachEllipseGeometry,
Florin Malita02a32b02018-01-04 11:27:09 -0500479 AttachPolystarGeometry,
Florin Malita094ccde2017-12-30 12:27:00 -0500480};
481
Florin Malitafa7e9a82018-05-04 15:10:54 -0400482using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const json::ValueRef&, AttachContext*);
Florin Malita094ccde2017-12-30 12:27:00 -0500483static constexpr PaintAttacherT gPaintAttachers[] = {
Florin Malita6aaee592018-01-12 12:25:09 -0500484 AttachColorFill,
485 AttachColorStroke,
486 AttachGradientFill,
487 AttachGradientStroke,
Florin Malita094ccde2017-12-30 12:27:00 -0500488};
489
Florin Malitae6345d92018-01-03 23:37:54 -0500490using GeometryEffectAttacherT =
Florin Malitafa7e9a82018-05-04 15:10:54 -0400491 std::vector<sk_sp<sksg::GeometryNode>> (*)(const json::ValueRef&,
Florin Malitae6345d92018-01-03 23:37:54 -0500492 AttachContext*,
493 std::vector<sk_sp<sksg::GeometryNode>>&&);
494static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
495 AttachMergeGeometryEffect,
Florin Malita51b8c892018-01-07 08:54:24 -0500496 AttachTrimGeometryEffect,
Florin Malita41dff6e2018-04-30 23:08:15 -0400497 AttachRoundGeometryEffect,
Florin Malitae6345d92018-01-03 23:37:54 -0500498};
499
Florin Malita094ccde2017-12-30 12:27:00 -0500500enum class ShapeType {
501 kGeometry,
Florin Malitae6345d92018-01-03 23:37:54 -0500502 kGeometryEffect,
Florin Malita094ccde2017-12-30 12:27:00 -0500503 kPaint,
504 kGroup,
Florin Malitadacc02b2017-12-31 09:12:31 -0500505 kTransform,
Florin Malita094ccde2017-12-30 12:27:00 -0500506};
507
508struct ShapeInfo {
509 const char* fTypeString;
510 ShapeType fShapeType;
511 uint32_t fAttacherIndex; // index into respective attacher tables
512};
513
Florin Malitafa7e9a82018-05-04 15:10:54 -0400514const ShapeInfo* FindShapeInfo(const json::ValueRef& shape) {
Florin Malita094ccde2017-12-30 12:27:00 -0500515 static constexpr ShapeInfo gShapeInfo[] = {
Florin Malitafbc13f12018-01-04 10:26:35 -0500516 { "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500517 { "fl", ShapeType::kPaint , 0 }, // fill -> AttachColorFill
518 { "gf", ShapeType::kPaint , 2 }, // gfill -> AttachGradientFill
Florin Malita16d0ad02018-01-19 15:07:29 -0500519 { "gr", ShapeType::kGroup , 0 }, // group -> Inline handler
Florin Malita6aaee592018-01-12 12:25:09 -0500520 { "gs", ShapeType::kPaint , 3 }, // gstroke -> AttachGradientStroke
Florin Malitae6345d92018-01-03 23:37:54 -0500521 { "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
Florin Malita02a32b02018-01-04 11:27:09 -0500522 { "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
Florin Malita41dff6e2018-04-30 23:08:15 -0400523 { "rd", ShapeType::kGeometryEffect, 2 }, // round -> AttachRoundGeometryEffect
Florin Malitae6345d92018-01-03 23:37:54 -0500524 { "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
Florin Malita02a32b02018-01-04 11:27:09 -0500525 { "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500526 { "st", ShapeType::kPaint , 1 }, // stroke -> AttachColorStroke
Florin Malita51b8c892018-01-07 08:54:24 -0500527 { "tm", ShapeType::kGeometryEffect, 1 }, // trim -> AttachTrimGeometryEffect
Florin Malita16d0ad02018-01-19 15:07:29 -0500528 { "tr", ShapeType::kTransform , 0 }, // transform -> Inline handler
Florin Malita094ccde2017-12-30 12:27:00 -0500529 };
530
Florin Malita20880782018-05-09 11:35:00 -0400531 SkString type;
532 if (!shape["ty"].to(&type) || type.isEmpty())
Florin Malita094ccde2017-12-30 12:27:00 -0500533 return nullptr;
534
Florin Malitafa7e9a82018-05-04 15:10:54 -0400535 const auto* info = bsearch(type.c_str(),
Florin Malita094ccde2017-12-30 12:27:00 -0500536 gShapeInfo,
537 SK_ARRAY_COUNT(gShapeInfo),
538 sizeof(ShapeInfo),
539 [](const void* key, const void* info) {
540 return strcmp(static_cast<const char*>(key),
541 static_cast<const ShapeInfo*>(info)->fTypeString);
542 });
543
544 return static_cast<const ShapeInfo*>(info);
545}
546
Florin Malita16d0ad02018-01-19 15:07:29 -0500547struct GeometryEffectRec {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400548 const json::ValueRef fJson;
Florin Malita16d0ad02018-01-19 15:07:29 -0500549 GeometryEffectAttacherT fAttach;
550};
551
Florin Malitaca4439f2018-01-23 10:31:59 -0500552struct AttachShapeContext {
553 AttachShapeContext(AttachContext* ctx,
554 std::vector<sk_sp<sksg::GeometryNode>>* geos,
555 std::vector<GeometryEffectRec>* effects,
556 size_t committedAnimators)
557 : fCtx(ctx)
558 , fGeometryStack(geos)
559 , fGeometryEffectStack(effects)
560 , fCommittedAnimators(committedAnimators) {}
561
562 AttachContext* fCtx;
563 std::vector<sk_sp<sksg::GeometryNode>>* fGeometryStack;
564 std::vector<GeometryEffectRec>* fGeometryEffectStack;
565 size_t fCommittedAnimators;
566};
567
Florin Malitafa7e9a82018-05-04 15:10:54 -0400568sk_sp<sksg::RenderNode> AttachShape(const json::ValueRef& jshape, AttachShapeContext* shapeCtx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500569 if (!jshape.isArray())
Florin Malita094ccde2017-12-30 12:27:00 -0500570 return nullptr;
571
Florin Malitaca4439f2018-01-23 10:31:59 -0500572 SkDEBUGCODE(const auto initialGeometryEffects = shapeCtx->fGeometryEffectStack->size();)
Florin Malita094ccde2017-12-30 12:27:00 -0500573
Florin Malita16d0ad02018-01-19 15:07:29 -0500574 sk_sp<sksg::Group> shape_group = sksg::Group::Make();
575 sk_sp<sksg::RenderNode> shape_wrapper = shape_group;
576 sk_sp<sksg::Matrix> shape_matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500577
Florin Malita16d0ad02018-01-19 15:07:29 -0500578 struct ShapeRec {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400579 const json::ValueRef fJson;
Florin Malita16d0ad02018-01-19 15:07:29 -0500580 const ShapeInfo& fInfo;
581 };
582
583 // First pass (bottom->top):
584 //
585 // * pick up the group transform and opacity
586 // * push local geometry effects onto the stack
587 // * store recs for next pass
588 //
589 std::vector<ShapeRec> recs;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400590 for (size_t i = 0; i < jshape.size(); ++i) {
591 const auto s = jshape[jshape.size() - 1 - i];
Florin Malita094ccde2017-12-30 12:27:00 -0500592 const auto* info = FindShapeInfo(s);
593 if (!info) {
Florin Malita20880782018-05-09 11:35:00 -0400594 LogFail(s["ty"], "Unknown shape");
Florin Malita094ccde2017-12-30 12:27:00 -0500595 continue;
596 }
597
Florin Malita16d0ad02018-01-19 15:07:29 -0500598 recs.push_back({ s, *info });
599
Florin Malita094ccde2017-12-30 12:27:00 -0500600 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500601 case ShapeType::kTransform:
Florin Malitaca4439f2018-01-23 10:31:59 -0500602 if ((shape_matrix = AttachMatrix(s, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500603 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
604 }
Florin Malitaca4439f2018-01-23 10:31:59 -0500605 shape_wrapper = AttachOpacity(s, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500606 break;
607 case ShapeType::kGeometryEffect:
608 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500609 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita16d0ad02018-01-19 15:07:29 -0500610 { s, gGeometryEffectAttachers[info->fAttacherIndex] });
611 break;
612 default:
613 break;
614 }
615 }
616
617 // Second pass (top -> bottom, after 2x reverse):
618 //
619 // * track local geometry
620 // * emit local paints
621 //
622 std::vector<sk_sp<sksg::GeometryNode>> geos;
623 std::vector<sk_sp<sksg::RenderNode >> draws;
624 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
625 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500626 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500627 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500628 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
629 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500630 geos.push_back(std::move(geo));
631 }
632 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500633 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500634 // Apply the current effect and pop from the stack.
635 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500636 if (!geos.empty()) {
637 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500638 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500639 std::move(geos));
640 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500641
Florin Malitaca4439f2018-01-23 10:31:59 -0500642 SkASSERT(shapeCtx->fGeometryEffectStack->back().fJson == rec->fJson);
643 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500644 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500645 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500646 } break;
647 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500648 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
649 &geos,
650 shapeCtx->fGeometryEffectStack,
651 shapeCtx->fCommittedAnimators);
652 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500653 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500654 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
655 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500656 }
657 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500658 case ShapeType::kPaint: {
659 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500660 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500661 if (!paint || geos.empty())
662 break;
663
664 auto drawGeos = geos;
665
666 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500667 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
668 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
669 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500670 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500671
672 // If we still have multiple geos, reduce using 'merge'.
673 auto geo = drawGeos.size() > 1
674 ? sksg::Merge::Make(std::move(drawGeos), sksg::Merge::Mode::kMerge)
675 : drawGeos[0];
676
677 SkASSERT(geo);
678 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500679 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500680 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500681 default:
682 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500683 }
684 }
685
Florin Malita16d0ad02018-01-19 15:07:29 -0500686 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500687 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500688
689 // Push transformed local geometries to parent list, for subsequent paints.
690 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500691 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500692 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
693 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500694 }
695
Florin Malita16d0ad02018-01-19 15:07:29 -0500696 // Emit local draws reversed (bottom->top, per spec).
697 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
698 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500699 }
700
Florin Malita16d0ad02018-01-19 15:07:29 -0500701 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500702}
703
Florin Malita1022f742018-02-23 11:10:22 -0500704sk_sp<sksg::RenderNode> AttachNestedAnimation(const char* path, AttachContext* ctx) {
705 class SkottieSGAdapter final : public sksg::RenderNode {
706 public:
707 explicit SkottieSGAdapter(sk_sp<Animation> animation)
708 : fAnimation(std::move(animation)) {
709 SkASSERT(fAnimation);
710 }
711
712 protected:
713 SkRect onRevalidate(sksg::InvalidationController*, const SkMatrix&) override {
714 return SkRect::MakeSize(fAnimation->size());
715 }
716
717 void onRender(SkCanvas* canvas) const override {
718 fAnimation->render(canvas);
719 }
720
721 private:
722 const sk_sp<Animation> fAnimation;
723 };
724
725 class SkottieAnimatorAdapter final : public sksg::Animator {
726 public:
Florin Malita911ae402018-05-31 16:45:29 -0400727 SkottieAnimatorAdapter(sk_sp<Animation> animation, float time_scale)
Florin Malita1022f742018-02-23 11:10:22 -0500728 : fAnimation(std::move(animation))
Florin Malita911ae402018-05-31 16:45:29 -0400729 , fTimeScale(time_scale) {
Florin Malita1022f742018-02-23 11:10:22 -0500730 SkASSERT(fAnimation);
Florin Malita1022f742018-02-23 11:10:22 -0500731 }
732
733 protected:
734 void onTick(float t) {
Florin Malita911ae402018-05-31 16:45:29 -0400735 // TODO: we prolly need more sophisticated timeline mapping for nested animations.
736 fAnimation->seek(t * fTimeScale);
Florin Malita1022f742018-02-23 11:10:22 -0500737 }
738
739 private:
740 const sk_sp<Animation> fAnimation;
Florin Malita911ae402018-05-31 16:45:29 -0400741 const float fTimeScale;
Florin Malita1022f742018-02-23 11:10:22 -0500742 };
743
744 const auto resStream = ctx->fResources.openStream(path);
745 if (!resStream || !resStream->hasLength()) {
746 LOG("!! Could not open: %s\n", path);
747 return nullptr;
748 }
749
Florin Malitac83a0de2018-05-31 12:17:55 -0400750 auto animation = Animation::Make(resStream.get(), &ctx->fResources);
Florin Malita1022f742018-02-23 11:10:22 -0500751 if (!animation) {
752 LOG("!! Could not load nested animation: %s\n", path);
753 return nullptr;
754 }
755
Florin Malita911ae402018-05-31 16:45:29 -0400756
757 ctx->fAnimators.push_back(
758 skstd::make_unique<SkottieAnimatorAdapter>(animation,
759 animation->duration() / ctx->fDuration));
Florin Malita1022f742018-02-23 11:10:22 -0500760
761 return sk_make_sp<SkottieSGAdapter>(std::move(animation));
762}
763
Florin Malita0cc01b72018-05-10 18:40:35 -0400764sk_sp<sksg::RenderNode> AttachAssetRef(const json::ValueRef& jlayer, AttachContext* ctx,
765 sk_sp<sksg::RenderNode>(*attach_proc)(const json::ValueRef& comp, AttachContext* ctx)) {
766
767 const auto refId = jlayer["refId"].toDefault(SkString());
768 if (refId.isEmpty()) {
769 LOG("!! Layer missing refId\n");
770 return nullptr;
771 }
772
773 if (refId.startsWith("$")) {
774 return AttachNestedAnimation(refId.c_str() + 1, ctx);
775 }
776
777 const auto* asset_info = ctx->fAssets.find(refId);
778 if (!asset_info) {
779 LOG("!! Asset not found: '%s'\n", refId.c_str());
780 return nullptr;
781 }
782
783 if (asset_info->fIsAttaching) {
784 LOG("!! Asset cycle detected for: '%s'\n", refId.c_str());
785 return nullptr;
786 }
787
788 asset_info->fIsAttaching = true;
789 auto asset = attach_proc(asset_info->fAsset, ctx);
790 asset_info->fIsAttaching = false;
791
792 return asset;
793}
794
Florin Malitafa7e9a82018-05-04 15:10:54 -0400795sk_sp<sksg::RenderNode> AttachCompLayer(const json::ValueRef& jlayer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500796 float* time_bias, float* time_scale) {
797 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500798
Florin Malitafa7e9a82018-05-04 15:10:54 -0400799 const auto start_time = jlayer["st"].toDefault(0.0f),
800 stretch_time = jlayer["sr"].toDefault(1.0f);
Florin Malitaeb87d672018-01-29 15:28:24 -0500801
802 *time_bias = -start_time;
Mike Reed2d46a002018-05-10 10:36:58 -0400803 *time_scale = sk_ieee_float_divide(1, stretch_time);
Florin Malitaeb87d672018-01-29 15:28:24 -0500804 if (SkScalarIsNaN(*time_scale)) {
805 *time_scale = 1;
806 }
807
Florin Malita0cc01b72018-05-10 18:40:35 -0400808 return AttachAssetRef(jlayer, ctx, AttachComposition);
Florin Malita094ccde2017-12-30 12:27:00 -0500809}
810
Florin Malitafa7e9a82018-05-04 15:10:54 -0400811sk_sp<sksg::RenderNode> AttachSolidLayer(const json::ValueRef& jlayer, AttachContext*,
Florin Malitaeb87d672018-01-29 15:28:24 -0500812 float*, float*) {
Florin Malita0e66fba2018-01-09 17:10:18 -0500813 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500814
Florin Malitafa7e9a82018-05-04 15:10:54 -0400815 const auto size = SkSize::Make(jlayer["sw"].toDefault(0.0f),
816 jlayer["sh"].toDefault(0.0f));
817 const auto hex = jlayer["sc"].toDefault(SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500818 uint32_t c;
819 if (size.isEmpty() ||
820 !hex.startsWith("#") ||
821 !SkParse::FindHex(hex.c_str() + 1, &c)) {
822 LogFail(jlayer, "Could not parse solid layer");
823 return nullptr;
824 }
825
826 const SkColor color = 0xff000000 | c;
827
828 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
829 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500830}
831
Florin Malitafa7e9a82018-05-04 15:10:54 -0400832sk_sp<sksg::RenderNode> AttachImageAsset(const json::ValueRef& jimage, AttachContext* ctx) {
Florin Malita49328072018-01-08 12:51:12 -0500833 SkASSERT(jimage.isObject());
834
Florin Malitafa7e9a82018-05-04 15:10:54 -0400835 const auto name = jimage["p"].toDefault(SkString()),
836 path = jimage["u"].toDefault(SkString());
Florin Malita49328072018-01-08 12:51:12 -0500837 if (name.isEmpty())
838 return nullptr;
839
840 // TODO: plumb resource paths explicitly to ResourceProvider?
841 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
842 const auto resStream = ctx->fResources.openStream(resName.c_str());
843 if (!resStream || !resStream->hasLength()) {
844 LOG("!! Could not load image resource: %s\n", resName.c_str());
845 return nullptr;
846 }
847
848 // TODO: non-intrisic image sizing
849 return sksg::Image::Make(
850 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
851}
852
Florin Malita0cc01b72018-05-10 18:40:35 -0400853sk_sp<sksg::RenderNode> AttachImageLayer(const json::ValueRef& jlayer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500854 float*, float*) {
Florin Malita0cc01b72018-05-10 18:40:35 -0400855 SkASSERT(jlayer.isObject());
Florin Malitadd22cf962018-01-29 15:42:01 +0000856
Florin Malita0cc01b72018-05-10 18:40:35 -0400857 return AttachAssetRef(jlayer, ctx, AttachImageAsset);
Florin Malita094ccde2017-12-30 12:27:00 -0500858}
859
Florin Malitafa7e9a82018-05-04 15:10:54 -0400860sk_sp<sksg::RenderNode> AttachNullLayer(const json::ValueRef& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500861 SkASSERT(layer.isObject());
862
Florin Malita18eafd92018-01-04 21:11:55 -0500863 // Null layers are used solely to drive dependent transforms,
864 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500865 return nullptr;
866}
867
Florin Malitafa7e9a82018-05-04 15:10:54 -0400868sk_sp<sksg::RenderNode> AttachShapeLayer(const json::ValueRef& layer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500869 float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500870 SkASSERT(layer.isObject());
871
Florin Malita16d0ad02018-01-19 15:07:29 -0500872 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
873 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500874 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
875 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
876
877 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
878 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
879 // due to attached animators. To avoid this, we track committed animators and discard the
880 // orphans here.
881 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
882 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
883
884 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500885}
886
Florin Malitafa7e9a82018-05-04 15:10:54 -0400887sk_sp<sksg::RenderNode> AttachTextLayer(const json::ValueRef& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500888 SkASSERT(layer.isObject());
889
890 LOG("?? Text layer stub\n");
891 return nullptr;
892}
893
Florin Malita18eafd92018-01-04 21:11:55 -0500894struct AttachLayerContext {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400895 AttachLayerContext(const json::ValueRef& jlayers, AttachContext* ctx)
Florin Malita2919b612018-05-10 12:44:07 -0400896 : fLayerList(jlayers), fCtx(ctx) {
897 SkASSERT(fLayerList.isArray());
898 }
Florin Malita18eafd92018-01-04 21:11:55 -0500899
Florin Malita2919b612018-05-10 12:44:07 -0400900 const json::ValueRef fLayerList;
Florin Malita4a490682018-01-28 14:27:51 -0500901 AttachContext* fCtx;
902 SkTHashMap<int, sk_sp<sksg::Matrix>> fLayerMatrixMap;
903 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500904
Florin Malitafa7e9a82018-05-04 15:10:54 -0400905 sk_sp<sksg::Matrix> AttachLayerMatrix(const json::ValueRef& jlayer) {
Florin Malita18eafd92018-01-04 21:11:55 -0500906 SkASSERT(jlayer.isObject());
907
Florin Malitafa7e9a82018-05-04 15:10:54 -0400908 const auto layer_index = jlayer["ind"].toDefault<int>(-1);
Florin Malita4a490682018-01-28 14:27:51 -0500909 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 Malita2919b612018-05-10 12:44:07 -0400915 return this->AttachLayerMatrixImpl(jlayer, layer_index);
916 }
917
918private:
919 sk_sp<sksg::Matrix> AttachParentLayerMatrix(const json::ValueRef& jlayer, int layer_index) {
920 SkASSERT(jlayer.isObject());
921
922 const auto parent_index = jlayer["parent"].toDefault<int>(-1);
923 if (parent_index < 0 || parent_index == layer_index)
924 return nullptr;
925
926 if (auto* m = fLayerMatrixMap.find(parent_index))
927 return *m;
928
929 for (const json::ValueRef l : fLayerList) {
930 if (l["ind"].toDefault<int>(-1) == parent_index) {
931 return this->AttachLayerMatrixImpl(l, parent_index);
932 }
933 }
934
935 return nullptr;
936 }
937
938 sk_sp<sksg::Matrix> AttachLayerMatrixImpl(const json::ValueRef& jlayer, int layer_index) {
939 SkASSERT(!fLayerMatrixMap.find(layer_index));
940
Florin Malita4a490682018-01-28 14:27:51 -0500941 // Add a stub entry to break recursion cycles.
942 fLayerMatrixMap.set(layer_index, nullptr);
Florin Malita18eafd92018-01-04 21:11:55 -0500943
Florin Malita2919b612018-05-10 12:44:07 -0400944 auto parent_matrix = this->AttachParentLayerMatrix(jlayer, layer_index);
Florin Malita18eafd92018-01-04 21:11:55 -0500945
Florin Malita2919b612018-05-10 12:44:07 -0400946 return *fLayerMatrixMap.set(layer_index, AttachMatrix(jlayer["ks"], fCtx, parent_matrix));
Florin Malita18eafd92018-01-04 21:11:55 -0500947 }
948};
949
Florin Malita25366fa2018-01-23 13:37:59 -0500950SkBlendMode MaskBlendMode(char mode) {
951 switch (mode) {
952 case 'a': return SkBlendMode::kSrcOver; // Additive
953 case 's': return SkBlendMode::kExclusion; // Subtract
954 case 'i': return SkBlendMode::kDstIn; // Intersect
955 case 'l': return SkBlendMode::kLighten; // Lighten
956 case 'd': return SkBlendMode::kDarken; // Darken
957 case 'f': return SkBlendMode::kDifference; // Difference
958 default: break;
959 }
960
961 return SkBlendMode::kSrcOver;
962}
963
Florin Malitafa7e9a82018-05-04 15:10:54 -0400964sk_sp<sksg::RenderNode> AttachMask(const json::ValueRef& jmask,
Florin Malita25366fa2018-01-23 13:37:59 -0500965 AttachContext* ctx,
966 sk_sp<sksg::RenderNode> childNode) {
967 if (!jmask.isArray())
968 return childNode;
969
Florin Malita0c51c212018-04-26 14:13:14 -0400970 struct MaskRecord {
971 sk_sp<sksg::Path> mask_path;
972 sk_sp<sksg::Color> mask_paint;
973 };
974
975 SkSTArray<4, MaskRecord, true> mask_stack;
976
977 bool opaque_mask = true;
Florin Malita25366fa2018-01-23 13:37:59 -0500978
Florin Malitafa7e9a82018-05-04 15:10:54 -0400979 for (const json::ValueRef m : jmask) {
Florin Malita25366fa2018-01-23 13:37:59 -0500980 if (!m.isObject())
981 continue;
982
Florin Malita25366fa2018-01-23 13:37:59 -0500983 auto mask_path = AttachPath(m["pt"], ctx);
984 if (!mask_path) {
985 LogFail(m, "Could not parse mask path");
986 continue;
987 }
988
Florin Malitafa7e9a82018-05-04 15:10:54 -0400989 mask_path->setFillType(m["inv"].toDefault(false)
Florin Malitacd05b192018-04-25 21:43:03 -0400990 ? SkPath::kInverseWinding_FillType
991 : SkPath::kWinding_FillType);
992
Florin Malitacf8ed522018-01-25 15:27:33 -0500993 SkString mode;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400994 if (!m["mode"].to(&mode) ||
Florin Malitacf8ed522018-01-25 15:27:33 -0500995 mode.size() != 1 ||
996 !strcmp(mode.c_str(), "n")) { // "None" masks have no effect.
Florin Malita25366fa2018-01-23 13:37:59 -0500997 continue;
Florin Malitacf8ed522018-01-25 15:27:33 -0500998 }
Florin Malita25366fa2018-01-23 13:37:59 -0500999
1000 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
Florin Malitaaa71c892018-01-30 09:27:01 -05001001 mask_paint->setAntiAlias(true);
Florin Malita25366fa2018-01-23 13:37:59 -05001002 mask_paint->setBlendMode(MaskBlendMode(mode.c_str()[0]));
Florin Malita0c51c212018-04-26 14:13:14 -04001003
1004 const auto animator_count = ctx->fAnimators.size();
Florin Malitafc807c82018-01-25 22:35:09 -05001005 BindProperty<ScalarValue>(m["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -05001006 [mask_paint](const ScalarValue& o) { mask_paint->setOpacity(o * 0.01f); });
Florin Malita25366fa2018-01-23 13:37:59 -05001007
Florin Malita0c51c212018-04-26 14:13:14 -04001008 opaque_mask &= (animator_count == ctx->fAnimators.size() && mask_paint->getOpacity() >= 1);
1009
1010 mask_stack.push_back({mask_path, mask_paint});
Florin Malita25366fa2018-01-23 13:37:59 -05001011 }
1012
Florin Malita0c51c212018-04-26 14:13:14 -04001013 if (mask_stack.empty())
1014 return childNode;
1015
1016 if (mask_stack.count() == 1 && opaque_mask) {
1017 // Single opaque mask => clip path.
1018 return sksg::ClipEffect::Make(std::move(childNode),
1019 std::move(mask_stack.front().mask_path),
1020 true);
1021 }
1022
1023 auto mask_group = sksg::Group::Make();
1024 for (const auto& rec : mask_stack) {
1025 mask_group->addChild(sksg::Draw::Make(std::move(rec.mask_path),
1026 std::move(rec.mask_paint)));
1027
1028 }
1029
1030 return sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
Florin Malita25366fa2018-01-23 13:37:59 -05001031}
1032
Florin Malitafa7e9a82018-05-04 15:10:54 -04001033sk_sp<sksg::RenderNode> AttachLayer(const json::ValueRef& jlayer, AttachLayerContext* layerCtx) {
Florin Malita18eafd92018-01-04 21:11:55 -05001034 if (!jlayer.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -05001035 return nullptr;
1036
Florin Malitafa7e9a82018-05-04 15:10:54 -04001037 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const json::ValueRef&, AttachContext*,
Florin Malitaeb87d672018-01-29 15:28:24 -05001038 float* time_bias, float* time_scale);
Florin Malita094ccde2017-12-30 12:27:00 -05001039 static constexpr LayerAttacher gLayerAttachers[] = {
1040 AttachCompLayer, // 'ty': 0
1041 AttachSolidLayer, // 'ty': 1
1042 AttachImageLayer, // 'ty': 2
1043 AttachNullLayer, // 'ty': 3
1044 AttachShapeLayer, // 'ty': 4
1045 AttachTextLayer, // 'ty': 5
1046 };
1047
Florin Malitafa7e9a82018-05-04 15:10:54 -04001048 int type = jlayer["ty"].toDefault<int>(-1);
Florin Malita094ccde2017-12-30 12:27:00 -05001049 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
1050 return nullptr;
1051 }
1052
Florin Malitacca86f32018-01-29 10:49:49 -05001053 sksg::AnimatorList layer_animators;
Florin Malita1022f742018-02-23 11:10:22 -05001054 AttachContext local_ctx = { layerCtx->fCtx->fResources,
1055 layerCtx->fCtx->fAssets,
Florin Malita911ae402018-05-31 16:45:29 -04001056 layerCtx->fCtx->fDuration,
Florin Malita1022f742018-02-23 11:10:22 -05001057 layer_animators};
Florin Malitacca86f32018-01-29 10:49:49 -05001058
Florin Malitaeb87d672018-01-29 15:28:24 -05001059 // Layer attachers may adjust these.
1060 float time_bias = 0,
1061 time_scale = 1;
1062
Florin Malita71cba8f2018-01-09 08:07:14 -05001063 // Layer content.
Florin Malitaeb87d672018-01-29 15:28:24 -05001064 auto layer = gLayerAttachers[type](jlayer, &local_ctx, &time_bias, &time_scale);
Florin Malita38ea40e2018-01-29 16:31:14 -05001065
1066 // Clip layers with explicit dimensions.
Florin Malitafa7e9a82018-05-04 15:10:54 -04001067 float w = 0, h = 0;
1068 if (jlayer["w"].to(&w) && jlayer["h"].to(&h)) {
Florin Malita38ea40e2018-01-29 16:31:14 -05001069 layer = sksg::ClipEffect::Make(std::move(layer),
1070 sksg::Rect::Make(SkRect::MakeWH(w, h)),
1071 true);
1072 }
1073
Florin Malita25366fa2018-01-23 13:37:59 -05001074 // Optional layer mask.
Florin Malitacca86f32018-01-29 10:49:49 -05001075 layer = AttachMask(jlayer["masksProperties"], &local_ctx, std::move(layer));
Florin Malita38ea40e2018-01-29 16:31:14 -05001076
Florin Malita25366fa2018-01-23 13:37:59 -05001077 // Optional layer transform.
Florin Malita71cba8f2018-01-09 08:07:14 -05001078 if (auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -05001079 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
1080 }
Florin Malita38ea40e2018-01-29 16:31:14 -05001081
Florin Malita71cba8f2018-01-09 08:07:14 -05001082 // Optional layer opacity.
Florin Malitacca86f32018-01-29 10:49:49 -05001083 layer = AttachOpacity(jlayer["ks"], &local_ctx, std::move(layer));
Florin Malita18eafd92018-01-04 21:11:55 -05001084
Florin Malitaeb87d672018-01-29 15:28:24 -05001085 class LayerController final : public sksg::GroupAnimator {
Florin Malita71cba8f2018-01-09 08:07:14 -05001086 public:
Florin Malitaeb87d672018-01-29 15:28:24 -05001087 LayerController(sksg::AnimatorList&& layer_animators,
1088 sk_sp<sksg::OpacityEffect> controlNode,
1089 float in, float out,
1090 float time_bias, float time_scale)
Florin Malitacca86f32018-01-29 10:49:49 -05001091 : INHERITED(std::move(layer_animators))
1092 , fControlNode(std::move(controlNode))
Florin Malita71cba8f2018-01-09 08:07:14 -05001093 , fIn(in)
Florin Malitaeb87d672018-01-29 15:28:24 -05001094 , fOut(out)
1095 , fTimeBias(time_bias)
1096 , fTimeScale(time_scale) {}
Florin Malita71cba8f2018-01-09 08:07:14 -05001097
Florin Malita35efaa82018-01-22 12:57:06 -05001098 void onTick(float t) override {
Florin Malitacca86f32018-01-29 10:49:49 -05001099 const auto active = (t >= fIn && t <= fOut);
1100
Florin Malita71cba8f2018-01-09 08:07:14 -05001101 // Keep the layer fully transparent except for its [in..out] lifespan.
1102 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
Florin Malitacca86f32018-01-29 10:49:49 -05001103 fControlNode->setOpacity(active ? 1 : 0);
1104
1105 // Dispatch ticks only while active.
1106 if (active)
Florin Malitaeb87d672018-01-29 15:28:24 -05001107 this->INHERITED::onTick((t + fTimeBias) * fTimeScale);
Florin Malita71cba8f2018-01-09 08:07:14 -05001108 }
1109
1110 private:
1111 const sk_sp<sksg::OpacityEffect> fControlNode;
1112 const float fIn,
Florin Malitaeb87d672018-01-29 15:28:24 -05001113 fOut,
1114 fTimeBias,
1115 fTimeScale;
Florin Malitacca86f32018-01-29 10:49:49 -05001116
1117 using INHERITED = sksg::GroupAnimator;
Florin Malita71cba8f2018-01-09 08:07:14 -05001118 };
1119
Florin Malitaeb87d672018-01-29 15:28:24 -05001120 auto controller_node = sksg::OpacityEffect::Make(std::move(layer));
Florin Malitafa7e9a82018-05-04 15:10:54 -04001121 const auto in = jlayer["ip"].toDefault(0.0f),
1122 out = jlayer["op"].toDefault(in);
Florin Malita71cba8f2018-01-09 08:07:14 -05001123
Florin Malitaeb87d672018-01-29 15:28:24 -05001124 if (!jlayer["tm"].isNull()) {
1125 LogFail(jlayer["tm"], "Unsupported time remapping");
1126 }
1127
1128 if (in >= out || !controller_node)
Florin Malita71cba8f2018-01-09 08:07:14 -05001129 return nullptr;
1130
Florin Malitacca86f32018-01-29 10:49:49 -05001131 layerCtx->fCtx->fAnimators.push_back(
Florin Malitaeb87d672018-01-29 15:28:24 -05001132 skstd::make_unique<LayerController>(std::move(layer_animators),
1133 controller_node,
1134 in,
1135 out,
1136 time_bias,
1137 time_scale));
Florin Malita71cba8f2018-01-09 08:07:14 -05001138
Florin Malitafa7e9a82018-05-04 15:10:54 -04001139 if (jlayer["td"].toDefault(false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -05001140 // This layer is a matte. We apply it as a mask to the next layer.
Florin Malitaeb87d672018-01-29 15:28:24 -05001141 layerCtx->fCurrentMatte = std::move(controller_node);
Florin Malita5f9102f2018-01-10 13:36:22 -05001142 return nullptr;
1143 }
1144
1145 if (layerCtx->fCurrentMatte) {
1146 // There is a pending matte. Apply and reset.
Florin Malitaa016be92018-03-05 14:01:41 -05001147 static constexpr sksg::MaskEffect::Mode gMaskModes[] = {
1148 sksg::MaskEffect::Mode::kNormal, // tt: 1
1149 sksg::MaskEffect::Mode::kInvert, // tt: 2
1150 };
Florin Malitafa7e9a82018-05-04 15:10:54 -04001151 const auto matteType = jlayer["tt"].toDefault<int>(1) - 1;
Florin Malitaa016be92018-03-05 14:01:41 -05001152
1153 if (matteType >= 0 && matteType < SkTo<int>(SK_ARRAY_COUNT(gMaskModes))) {
1154 return sksg::MaskEffect::Make(std::move(controller_node),
1155 std::move(layerCtx->fCurrentMatte),
1156 gMaskModes[matteType]);
1157 }
1158 layerCtx->fCurrentMatte.reset();
Florin Malita5f9102f2018-01-10 13:36:22 -05001159 }
1160
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001161 return std::move(controller_node);
Florin Malita094ccde2017-12-30 12:27:00 -05001162}
1163
Florin Malitafa7e9a82018-05-04 15:10:54 -04001164sk_sp<sksg::RenderNode> AttachComposition(const json::ValueRef& comp, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -05001165 if (!comp.isObject())
1166 return nullptr;
1167
Florin Malitafa7e9a82018-05-04 15:10:54 -04001168 const auto jlayers = comp["layers"];
Florin Malita18eafd92018-01-04 21:11:55 -05001169 if (!jlayers.isArray())
1170 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001171
Florin Malita18eafd92018-01-04 21:11:55 -05001172 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
1173 AttachLayerContext layerCtx(jlayers, ctx);
1174
Florin Malitafa7e9a82018-05-04 15:10:54 -04001175 for (const json::ValueRef l : jlayers) {
Florin Malita18eafd92018-01-04 21:11:55 -05001176 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001177 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001178 }
1179 }
1180
Florin Malita2a8275b2018-01-02 12:52:43 -05001181 if (layers.empty()) {
1182 return nullptr;
1183 }
1184
1185 // Layers are painted in bottom->top order.
1186 auto comp_group = sksg::Group::Make();
1187 for (int i = layers.count() - 1; i >= 0; --i) {
1188 comp_group->addChild(std::move(layers[i]));
1189 }
1190
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001191 return std::move(comp_group);
Florin Malita094ccde2017-12-30 12:27:00 -05001192}
1193
1194} // namespace
1195
Florin Malitac83a0de2018-05-31 12:17:55 -04001196sk_sp<Animation> Animation::Make(SkStream* stream, const ResourceProvider* provider, Stats* stats) {
Florin Malita6eb85a12018-04-30 10:32:18 -04001197 Stats stats_storage;
1198 if (!stats)
1199 stats = &stats_storage;
1200 memset(stats, 0, sizeof(struct Stats));
1201
Florin Malita094ccde2017-12-30 12:27:00 -05001202 if (!stream->hasLength()) {
1203 // TODO: handle explicit buffering?
1204 LOG("!! cannot parse streaming content\n");
1205 return nullptr;
1206 }
1207
Florin Malitafa7e9a82018-05-04 15:10:54 -04001208 stats->fJsonSize = stream->getLength();
Florin Malita6eb85a12018-04-30 10:32:18 -04001209 const auto t0 = SkTime::GetMSecs();
1210
Florin Malitafa7e9a82018-05-04 15:10:54 -04001211 const json::Document doc(stream);
1212 const auto json = doc.root();
1213 if (!json.isObject())
1214 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001215
Florin Malita6eb85a12018-04-30 10:32:18 -04001216 const auto t1 = SkTime::GetMSecs();
1217 stats->fJsonParseTimeMS = t1 - t0;
1218
Florin Malitafa7e9a82018-05-04 15:10:54 -04001219 const auto version = json["v"].toDefault(SkString());
1220 const auto size = SkSize::Make(json["w"].toDefault(0.0f),
1221 json["h"].toDefault(0.0f));
1222 const auto fps = json["fr"].toDefault(-1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001223
Florin Malita1022f742018-02-23 11:10:22 -05001224 if (size.isEmpty() || version.isEmpty() || fps <= 0) {
Florin Malita094ccde2017-12-30 12:27:00 -05001225 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1226 version.c_str(), size.width(), size.height(), fps);
1227 return nullptr;
1228 }
1229
Florin Malitac83a0de2018-05-31 12:17:55 -04001230 class NullResourceProvider final : public ResourceProvider {
1231 std::unique_ptr<SkStream> openStream(const char[]) const { return nullptr; }
1232 };
1233
Kevin Lubickf14bc982018-06-01 13:00:35 -04001234 NullResourceProvider null_provider;
Florin Malitac83a0de2018-05-31 12:17:55 -04001235 const auto anim = sk_sp<Animation>(new Animation(provider ? *provider : null_provider,
1236 std::move(version), size, fps, json, stats));
Florin Malita6eb85a12018-04-30 10:32:18 -04001237 const auto t2 = SkTime::GetMSecs();
1238 stats->fSceneParseTimeMS = t2 - t1;
1239 stats->fTotalLoadTimeMS = t2 - t0;
1240
1241 return anim;
Florin Malita094ccde2017-12-30 12:27:00 -05001242}
1243
Florin Malita6eb85a12018-04-30 10:32:18 -04001244sk_sp<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res,
1245 Stats* stats) {
Florin Malita49328072018-01-08 12:51:12 -05001246 class DirectoryResourceProvider final : public ResourceProvider {
1247 public:
1248 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1249
1250 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1251 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1252 return SkStream::MakeFromFile(resPath.c_str());
1253 }
1254
1255 private:
1256 const SkString fDir;
1257 };
1258
1259 const auto jsonStream = SkStream::MakeFromFile(path);
1260 if (!jsonStream)
1261 return nullptr;
1262
1263 std::unique_ptr<ResourceProvider> defaultProvider;
1264 if (!res) {
1265 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1266 }
1267
Florin Malitac83a0de2018-05-31 12:17:55 -04001268 return Make(jsonStream.get(), res ? res : defaultProvider.get(), stats);
Florin Malita49328072018-01-08 12:51:12 -05001269}
1270
1271Animation::Animation(const ResourceProvider& resources,
Florin Malitafa7e9a82018-05-04 15:10:54 -04001272 SkString version, const SkSize& size, SkScalar fps, const json::ValueRef& json,
Florin Malita6eb85a12018-04-30 10:32:18 -04001273 Stats* stats)
Florin Malita094ccde2017-12-30 12:27:00 -05001274 : fVersion(std::move(version))
1275 , fSize(size)
1276 , fFrameRate(fps)
Florin Malitafa7e9a82018-05-04 15:10:54 -04001277 , fInPoint(json["ip"].toDefault(0.0f))
1278 , fOutPoint(SkTMax(json["op"].toDefault(SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001279
1280 AssetMap assets;
Florin Malitafa7e9a82018-05-04 15:10:54 -04001281 for (const json::ValueRef asset : json["assets"]) {
1282 if (asset.isObject()) {
Florin Malita0cc01b72018-05-10 18:40:35 -04001283 assets.set(asset["id"].toDefault(SkString()), { asset, false });
Florin Malita094ccde2017-12-30 12:27:00 -05001284 }
Florin Malita094ccde2017-12-30 12:27:00 -05001285 }
1286
Florin Malitacca86f32018-01-29 10:49:49 -05001287 sksg::AnimatorList animators;
Florin Malita911ae402018-05-31 16:45:29 -04001288 AttachContext ctx = { resources, assets, this->duration(), animators };
Florin Malita35efaa82018-01-22 12:57:06 -05001289 auto root = AttachComposition(json, &ctx);
1290
Florin Malita6eb85a12018-04-30 10:32:18 -04001291 stats->fAnimatorCount = animators.size();
Florin Malita35efaa82018-01-22 12:57:06 -05001292
1293 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001294
Florin Malitadb385732018-01-09 12:19:32 -05001295 // In case the client calls render before the first tick.
Florin Malitaa33447d2018-05-29 13:46:54 -04001296 this->seek(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001297}
1298
1299Animation::~Animation() = default;
1300
Florin Malita35efaa82018-01-22 12:57:06 -05001301void Animation::setShowInval(bool show) {
1302 if (fScene) {
1303 fScene->setShowInval(show);
1304 }
1305}
1306
Mike Reed29859872018-01-08 08:25:27 -05001307void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001308 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001309 return;
1310
Mike Reed29859872018-01-08 08:25:27 -05001311 SkAutoCanvasRestore restore(canvas, true);
1312 const SkRect srcR = SkRect::MakeSize(this->size());
1313 if (dstR) {
1314 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1315 }
1316 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001317 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001318}
1319
Florin Malitaa33447d2018-05-29 13:46:54 -04001320void Animation::seek(SkScalar t) {
Florin Malita35efaa82018-01-22 12:57:06 -05001321 if (!fScene)
1322 return;
1323
Florin Malitaa33447d2018-05-29 13:46:54 -04001324 fScene->animate(fInPoint + SkTPin(t, 0.0f, 1.0f) * (fOutPoint - fInPoint));
Florin Malita094ccde2017-12-30 12:27:00 -05001325}
1326
Florin Malita54f65c42018-01-16 17:04:30 -05001327} // namespace skottie