blob: 60009b9bc03934b6449c36ffc2d7e4ffe6edeed1 [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"
Florin Malita5dfd0692018-07-02 15:05:26 -040038#include "SkTLazy.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -040039#include "SkTime.h"
40#include "SkTo.h"
41#include "SkottieAdapter.h"
42#include "SkottieAnimator.h"
43#include "SkottieJson.h"
44#include "SkottieValue.h"
Florin Malita094ccde2017-12-30 12:27:00 -050045
46#include <cmath>
Florin Malitae6345d92018-01-03 23:37:54 -050047#include <vector>
48
Florin Malita094ccde2017-12-30 12:27:00 -050049#include "stdlib.h"
50
Florin Malita54f65c42018-01-16 17:04:30 -050051namespace skottie {
Florin Malita094ccde2017-12-30 12:27:00 -050052
Florin Malitacf8ed522018-01-25 15:27:33 -050053#define LOG SkDebugf
54
Florin Malita094ccde2017-12-30 12:27:00 -050055namespace {
56
Florin Malita0cc01b72018-05-10 18:40:35 -040057struct AssetInfo {
Florin Malita7d42c442018-06-14 16:16:01 -040058 const skjson::ObjectValue* fAsset;
59 mutable bool fIsAttaching; // Used for cycle detection
Florin Malita0cc01b72018-05-10 18:40:35 -040060};
61
62using AssetMap = SkTHashMap<SkString, AssetInfo>;
Florin Malita094ccde2017-12-30 12:27:00 -050063
64struct AttachContext {
Florin Malitacca86f32018-01-29 10:49:49 -050065 const ResourceProvider& fResources;
66 const AssetMap& fAssets;
Florin Malita5dfd0692018-07-02 15:05:26 -040067 const float fDuration,
68 fFrameRate;
Florin Malitacca86f32018-01-29 10:49:49 -050069 sksg::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050070};
71
Florin Malita7d42c442018-06-14 16:16:01 -040072bool LogFail(const skjson::Value& json, const char* msg) {
Florin Malitafa7e9a82018-05-04 15:10:54 -040073 const auto dump = json.toString();
Florin Malita20880782018-05-09 11:35:00 -040074 LOG("!! %s: %s\n", msg, dump.c_str());
Florin Malita094ccde2017-12-30 12:27:00 -050075 return false;
76}
77
Florin Malita7d42c442018-06-14 16:16:01 -040078sk_sp<sksg::Matrix> AttachMatrix(const skjson::ObjectValue& t, AttachContext* ctx,
79 sk_sp<sksg::Matrix> parentMatrix) {
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 Malita7d42c442018-06-14 16:16:01 -040095 const auto* jrotation = &t["r"];
96 if (jrotation->is<skjson::NullValue>()) {
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 Malita7d42c442018-06-14 16:16:01 -040099 jrotation = &t["rz"];
Florin Malita1eb98db2018-01-26 15:03:38 -0500100 }
Florin Malita7d42c442018-06-14 16:16:01 -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 Malita7d42c442018-06-14 16:16:01 -0400127sk_sp<sksg::RenderNode> AttachOpacity(const skjson::ObjectValue& jtransform, AttachContext* ctx,
Florin Malitac0034172018-01-08 16:42:59 -0500128 sk_sp<sksg::RenderNode> childNode) {
Florin Malita7d42c442018-06-14 16:16:01 -0400129 if (!childNode)
130 return nullptr;
Florin Malitac0034172018-01-08 16:42:59 -0500131
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 Malita7d42c442018-06-14 16:16:01 -0400147sk_sp<sksg::RenderNode> AttachComposition(const skjson::ObjectValue&, AttachContext* ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500148
Florin Malita7d42c442018-06-14 16:16:01 -0400149sk_sp<sksg::Path> AttachPath(const skjson::Value& 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) {
Florin Malita502c3ff2018-07-03 18:10:39 -0400153 // FillType is tracked in the SG node, not in keyframes -- make sure we preserve it.
154 auto path = ValueTraits<ShapeValue>::As<SkPath>(p);
155 path.setFillType(path_node->getFillType());
156 path_node->setPath(path);
Florin Malitac353ee22018-04-30 21:49:41 -0400157 })
Florin Malita25366fa2018-01-23 13:37:59 -0500158 ? path_node
159 : nullptr;
160}
161
Florin Malita7d42c442018-06-14 16:16:01 -0400162sk_sp<sksg::GeometryNode> AttachPathGeometry(const skjson::ObjectValue& jpath, AttachContext* ctx) {
Florin Malita25366fa2018-01-23 13:37:59 -0500163 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500164}
165
Florin Malita7d42c442018-06-14 16:16:01 -0400166sk_sp<sksg::GeometryNode> AttachRRectGeometry(const skjson::ObjectValue& jrect,
167 AttachContext* ctx) {
Florin Malita2e1d7e22018-01-02 10:40:00 -0500168 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 Malita7d42c442018-06-14 16:16:01 -0400191sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const skjson::ObjectValue& jellipse,
192 AttachContext* ctx) {
Florin Malitafbc13f12018-01-04 10:26:35 -0500193 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400194 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500195
Florin Malitafc807c82018-01-25 22:35:09 -0500196 auto p_attached = BindProperty<VectorValue>(jellipse["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400197 [adapter](const VectorValue& p) {
198 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500199 });
Florin Malitafc807c82018-01-25 22:35:09 -0500200 auto s_attached = BindProperty<VectorValue>(jellipse["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400201 [adapter](const VectorValue& s) {
Florin Malitaf9590922018-01-09 11:56:09 -0500202 const auto sz = ValueTraits<VectorValue>::As<SkSize>(s);
Florin Malitaa6e30f72018-03-23 13:41:58 -0400203 adapter->setSize(sz);
204 adapter->setRadius(SkSize::Make(sz.width() / 2, sz.height() / 2));
Florin Malitaf9590922018-01-09 11:56:09 -0500205 });
Florin Malitafbc13f12018-01-04 10:26:35 -0500206
207 if (!p_attached && !s_attached) {
208 return nullptr;
209 }
210
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400211 return std::move(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500212}
213
Florin Malita7d42c442018-06-14 16:16:01 -0400214sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const skjson::ObjectValue& jstar,
215 AttachContext* ctx) {
Florin Malitaa6e30f72018-03-23 13:41:58 -0400216 static constexpr PolyStarAdapter::Type gTypes[] = {
217 PolyStarAdapter::Type::kStar, // "sy": 1
218 PolyStarAdapter::Type::kPoly, // "sy": 2
Florin Malita02a32b02018-01-04 11:27:09 -0500219 };
220
Florin Malita7d42c442018-06-14 16:16:01 -0400221 const auto type = ParseDefault<int>(jstar["sy"], 0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500222 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
223 LogFail(jstar, "Unknown polystar type");
224 return nullptr;
225 }
226
227 auto path_node = sksg::Path::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400228 auto adapter = sk_make_sp<PolyStarAdapter>(path_node, gTypes[type]);
Florin Malita02a32b02018-01-04 11:27:09 -0500229
Florin Malitafc807c82018-01-25 22:35:09 -0500230 BindProperty<VectorValue>(jstar["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400231 [adapter](const VectorValue& p) {
232 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500233 });
Florin Malitafc807c82018-01-25 22:35:09 -0500234 BindProperty<ScalarValue>(jstar["pt"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400235 [adapter](const ScalarValue& pt) {
236 adapter->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500237 });
Florin Malitafc807c82018-01-25 22:35:09 -0500238 BindProperty<ScalarValue>(jstar["ir"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400239 [adapter](const ScalarValue& ir) {
240 adapter->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500241 });
Florin Malitafc807c82018-01-25 22:35:09 -0500242 BindProperty<ScalarValue>(jstar["or"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400243 [adapter](const ScalarValue& otr) {
244 adapter->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500245 });
Florin Malitafc807c82018-01-25 22:35:09 -0500246 BindProperty<ScalarValue>(jstar["is"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400247 [adapter](const ScalarValue& is) {
248 adapter->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500249 });
Florin Malitafc807c82018-01-25 22:35:09 -0500250 BindProperty<ScalarValue>(jstar["os"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400251 [adapter](const ScalarValue& os) {
252 adapter->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500253 });
Florin Malitafc807c82018-01-25 22:35:09 -0500254 BindProperty<ScalarValue>(jstar["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400255 [adapter](const ScalarValue& r) {
256 adapter->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500257 });
Florin Malita02a32b02018-01-04 11:27:09 -0500258
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400259 return std::move(path_node);
Florin Malita02a32b02018-01-04 11:27:09 -0500260}
261
Florin Malita7d42c442018-06-14 16:16:01 -0400262sk_sp<sksg::Color> AttachColor(const skjson::ObjectValue& jcolor, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500263 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malita7d42c442018-06-14 16:16:01 -0400264 auto color_attached = BindProperty<VectorValue>(jcolor["c"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500265 [color_node](const VectorValue& c) {
266 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500267 });
Florin Malita094ccde2017-12-30 12:27:00 -0500268
Florin Malita1586d852018-01-12 14:27:39 -0500269 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500270}
271
Florin Malita7d42c442018-06-14 16:16:01 -0400272sk_sp<sksg::Gradient> AttachGradient(const skjson::ObjectValue& jgrad, AttachContext* ctx) {
273 const skjson::ObjectValue* stops = jgrad["g"];
274 if (!stops)
Florin Malita094ccde2017-12-30 12:27:00 -0500275 return nullptr;
276
Florin Malita7d42c442018-06-14 16:16:01 -0400277 const auto stopCount = ParseDefault<int>((*stops)["p"], -1);
Florin Malita6aaee592018-01-12 12:25:09 -0500278 if (stopCount < 0)
279 return nullptr;
280
281 sk_sp<sksg::Gradient> gradient_node;
Florin Malitaa6e30f72018-03-23 13:41:58 -0400282 sk_sp<GradientAdapter> adapter;
Florin Malita6aaee592018-01-12 12:25:09 -0500283
Florin Malita7d42c442018-06-14 16:16:01 -0400284 if (ParseDefault<int>(jgrad["t"], 1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500285 auto linear_node = sksg::LinearGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400286 adapter = sk_make_sp<LinearGradientAdapter>(linear_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500287 gradient_node = std::move(linear_node);
288 } else {
289 auto radial_node = sksg::RadialGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400290 adapter = sk_make_sp<RadialGradientAdapter>(radial_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500291
292 // TODO: highlight, angle
293 gradient_node = std::move(radial_node);
294 }
295
Florin Malita7d42c442018-06-14 16:16:01 -0400296 BindProperty<VectorValue>((*stops)["k"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400297 [adapter](const VectorValue& stops) {
298 adapter->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500299 });
Florin Malita7d42c442018-06-14 16:16:01 -0400300 BindProperty<VectorValue>(jgrad["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400301 [adapter](const VectorValue& s) {
302 adapter->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500303 });
Florin Malita7d42c442018-06-14 16:16:01 -0400304 BindProperty<VectorValue>(jgrad["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400305 [adapter](const VectorValue& e) {
306 adapter->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500307 });
308
309 return gradient_node;
310}
311
Florin Malita7d42c442018-06-14 16:16:01 -0400312sk_sp<sksg::PaintNode> AttachPaint(const skjson::ObjectValue& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500313 sk_sp<sksg::PaintNode> paint_node) {
314 if (paint_node) {
315 paint_node->setAntiAlias(true);
316
Florin Malitafc807c82018-01-25 22:35:09 -0500317 BindProperty<ScalarValue>(jpaint["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500318 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500319 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500320 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500321 });
Florin Malita6aaee592018-01-12 12:25:09 -0500322 }
323
324 return paint_node;
325}
326
Florin Malita7d42c442018-06-14 16:16:01 -0400327sk_sp<sksg::PaintNode> AttachStroke(const skjson::ObjectValue& jstroke, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500328 sk_sp<sksg::PaintNode> stroke_node) {
Florin Malita6aaee592018-01-12 12:25:09 -0500329 if (!stroke_node)
330 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500331
332 stroke_node->setStyle(SkPaint::kStroke_Style);
333
Florin Malitafc807c82018-01-25 22:35:09 -0500334 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500335 [stroke_node](const ScalarValue& w) {
336 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500337 });
Florin Malita094ccde2017-12-30 12:27:00 -0500338 if (!width_attached)
339 return nullptr;
340
Florin Malita7d42c442018-06-14 16:16:01 -0400341 stroke_node->setStrokeMiter(ParseDefault<SkScalar>(jstroke["ml"], 4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500342
343 static constexpr SkPaint::Join gJoins[] = {
344 SkPaint::kMiter_Join,
345 SkPaint::kRound_Join,
346 SkPaint::kBevel_Join,
347 };
Florin Malita7d42c442018-06-14 16:16:01 -0400348 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(ParseDefault<int>(jstroke["lj"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500349 0, SK_ARRAY_COUNT(gJoins) - 1)]);
350
351 static constexpr SkPaint::Cap gCaps[] = {
352 SkPaint::kButt_Cap,
353 SkPaint::kRound_Cap,
354 SkPaint::kSquare_Cap,
355 };
Florin Malita7d42c442018-06-14 16:16:01 -0400356 stroke_node->setStrokeCap(gCaps[SkTPin<int>(ParseDefault(jstroke["lc"], 1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500357 0, SK_ARRAY_COUNT(gCaps) - 1)]);
358
359 return stroke_node;
360}
361
Florin Malita7d42c442018-06-14 16:16:01 -0400362sk_sp<sksg::PaintNode> AttachColorFill(const skjson::ObjectValue& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500363 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
364}
365
Florin Malita7d42c442018-06-14 16:16:01 -0400366sk_sp<sksg::PaintNode> AttachGradientFill(const skjson::ObjectValue& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500367 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
368}
369
Florin Malita7d42c442018-06-14 16:16:01 -0400370sk_sp<sksg::PaintNode> AttachColorStroke(const skjson::ObjectValue& jstroke, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500371 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
372}
373
Florin Malita7d42c442018-06-14 16:16:01 -0400374sk_sp<sksg::PaintNode> AttachGradientStroke(const skjson::ObjectValue& jstroke,
375 AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500376 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
377}
378
Florin Malita418e6582018-07-09 16:20:47 -0400379sk_sp<sksg::Merge> Merge(std::vector<sk_sp<sksg::GeometryNode>>&& geos, sksg::Merge::Mode mode) {
380 std::vector<sksg::Merge::Rec> merge_recs;
381 merge_recs.reserve(geos.size());
382
383 for (const auto& geo : geos) {
384 merge_recs.push_back(
385 {std::move(geo), merge_recs.empty() ? sksg::Merge::Mode::kMerge : mode});
386 }
387
388 return sksg::Merge::Make(std::move(merge_recs));
389}
390
Florin Malitae6345d92018-01-03 23:37:54 -0500391std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
Florin Malita7d42c442018-06-14 16:16:01 -0400392 const skjson::ObjectValue& jmerge, AttachContext*,
393 std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malitae6345d92018-01-03 23:37:54 -0500394 static constexpr sksg::Merge::Mode gModes[] = {
395 sksg::Merge::Mode::kMerge, // "mm": 1
396 sksg::Merge::Mode::kUnion, // "mm": 2
397 sksg::Merge::Mode::kDifference, // "mm": 3
398 sksg::Merge::Mode::kIntersect, // "mm": 4
399 sksg::Merge::Mode::kXOR , // "mm": 5
400 };
401
Florin Malita7d42c442018-06-14 16:16:01 -0400402 const auto mode = gModes[SkTPin<int>(ParseDefault<int>(jmerge["mm"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500403 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malita418e6582018-07-09 16:20:47 -0400404
405 std::vector<sk_sp<sksg::GeometryNode>> merged;
406 merged.push_back(Merge(std::move(geos), mode));
Florin Malitae6345d92018-01-03 23:37:54 -0500407
Florin Malitae6345d92018-01-03 23:37:54 -0500408 return merged;
409}
410
Florin Malita51b8c892018-01-07 08:54:24 -0500411std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
Florin Malita7d42c442018-06-14 16:16:01 -0400412 const skjson::ObjectValue& jtrim, AttachContext* ctx,
413 std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malita51b8c892018-01-07 08:54:24 -0500414
415 enum class Mode {
416 kMerged, // "m": 1
417 kSeparate, // "m": 2
418 } gModes[] = { Mode::kMerged, Mode::kSeparate };
419
Florin Malita7d42c442018-06-14 16:16:01 -0400420 const auto mode = gModes[SkTPin<int>(ParseDefault<int>(jtrim["m"], 1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500421 0, SK_ARRAY_COUNT(gModes) - 1)];
422
423 std::vector<sk_sp<sksg::GeometryNode>> inputs;
424 if (mode == Mode::kMerged) {
Florin Malita418e6582018-07-09 16:20:47 -0400425 inputs.push_back(Merge(std::move(geos), sksg::Merge::Mode::kMerge));
Florin Malita51b8c892018-01-07 08:54:24 -0500426 } else {
427 inputs = std::move(geos);
428 }
429
430 std::vector<sk_sp<sksg::GeometryNode>> trimmed;
431 trimmed.reserve(inputs.size());
432 for (const auto& i : inputs) {
Florin Malita69526b02018-03-22 12:20:02 -0400433 const auto trimEffect = sksg::TrimEffect::Make(i);
434 trimmed.push_back(trimEffect);
435
Florin Malitaa6e30f72018-03-23 13:41:58 -0400436 const auto adapter = sk_make_sp<TrimEffectAdapter>(std::move(trimEffect));
Florin Malitafc807c82018-01-25 22:35:09 -0500437 BindProperty<ScalarValue>(jtrim["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400438 [adapter](const ScalarValue& s) {
439 adapter->setStart(s);
Florin Malita51b8c892018-01-07 08:54:24 -0500440 });
Florin Malitafc807c82018-01-25 22:35:09 -0500441 BindProperty<ScalarValue>(jtrim["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400442 [adapter](const ScalarValue& e) {
443 adapter->setEnd(e);
Florin Malita51b8c892018-01-07 08:54:24 -0500444 });
Florin Malitafc807c82018-01-25 22:35:09 -0500445 BindProperty<ScalarValue>(jtrim["o"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400446 [adapter](const ScalarValue& o) {
447 adapter->setOffset(o);
Florin Malita51b8c892018-01-07 08:54:24 -0500448 });
449 }
450
451 return trimmed;
452}
453
Florin Malita41dff6e2018-04-30 23:08:15 -0400454std::vector<sk_sp<sksg::GeometryNode>> AttachRoundGeometryEffect(
Florin Malita7d42c442018-06-14 16:16:01 -0400455 const skjson::ObjectValue& jtrim, AttachContext* ctx,
456 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 Malita7d42c442018-06-14 16:16:01 -0400474using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const skjson::ObjectValue&, 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 Malita7d42c442018-06-14 16:16:01 -0400482using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const skjson::ObjectValue&, 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 Malita7d42c442018-06-14 16:16:01 -0400491 std::vector<sk_sp<sksg::GeometryNode>> (*)(const skjson::ObjectValue&,
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 Malita7d42c442018-06-14 16:16:01 -0400514const ShapeInfo* FindShapeInfo(const skjson::ObjectValue& jshape) {
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;
Florin Malita7d42c442018-06-14 16:16:01 -0400532 if (!Parse<SkString>(jshape["ty"], &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 Malita7d42c442018-06-14 16:16:01 -0400548 const skjson::ObjectValue& fJson;
549 GeometryEffectAttacherT fAttach;
Florin Malita16d0ad02018-01-19 15:07:29 -0500550};
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 Malita7d42c442018-06-14 16:16:01 -0400568sk_sp<sksg::RenderNode> AttachShape(const skjson::ArrayValue* jshape, AttachShapeContext* shapeCtx) {
569 if (!jshape)
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 Malita7d42c442018-06-14 16:16:01 -0400579 const skjson::ObjectValue& fJson;
580 const ShapeInfo& fInfo;
Florin Malita16d0ad02018-01-19 15:07:29 -0500581 };
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 Malita7d42c442018-06-14 16:16:01 -0400590 for (size_t i = 0; i < jshape->size(); ++i) {
591 const skjson::ObjectValue* shape = (*jshape)[jshape->size() - 1 - i];
592 if (!shape) continue;
593
594 const auto* info = FindShapeInfo(*shape);
Florin Malita094ccde2017-12-30 12:27:00 -0500595 if (!info) {
Florin Malita7d42c442018-06-14 16:16:01 -0400596 LogFail((*shape)["ty"], "Unknown shape");
Florin Malita094ccde2017-12-30 12:27:00 -0500597 continue;
598 }
599
Florin Malita7d42c442018-06-14 16:16:01 -0400600 recs.push_back({ *shape, *info });
Florin Malita16d0ad02018-01-19 15:07:29 -0500601
Florin Malita094ccde2017-12-30 12:27:00 -0500602 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500603 case ShapeType::kTransform:
Florin Malita7d42c442018-06-14 16:16:01 -0400604 if ((shape_matrix = AttachMatrix(*shape, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500605 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
606 }
Florin Malita7d42c442018-06-14 16:16:01 -0400607 shape_wrapper = AttachOpacity(*shape, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500608 break;
609 case ShapeType::kGeometryEffect:
610 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500611 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita7d42c442018-06-14 16:16:01 -0400612 { *shape, gGeometryEffectAttachers[info->fAttacherIndex] });
Florin Malita16d0ad02018-01-19 15:07:29 -0500613 break;
614 default:
615 break;
616 }
617 }
618
619 // Second pass (top -> bottom, after 2x reverse):
620 //
621 // * track local geometry
622 // * emit local paints
623 //
624 std::vector<sk_sp<sksg::GeometryNode>> geos;
625 std::vector<sk_sp<sksg::RenderNode >> draws;
626 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
627 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500628 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500629 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500630 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
631 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500632 geos.push_back(std::move(geo));
633 }
634 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500635 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500636 // Apply the current effect and pop from the stack.
637 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500638 if (!geos.empty()) {
639 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500640 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500641 std::move(geos));
642 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500643
Florin Malita7d42c442018-06-14 16:16:01 -0400644 SkASSERT(&shapeCtx->fGeometryEffectStack->back().fJson == &rec->fJson);
Florin Malitaca4439f2018-01-23 10:31:59 -0500645 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500646 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500647 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500648 } break;
649 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500650 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
651 &geos,
652 shapeCtx->fGeometryEffectStack,
653 shapeCtx->fCommittedAnimators);
654 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500655 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500656 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
657 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500658 }
659 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500660 case ShapeType::kPaint: {
661 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500662 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500663 if (!paint || geos.empty())
664 break;
665
666 auto drawGeos = geos;
667
668 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500669 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
670 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
671 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500672 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500673
674 // If we still have multiple geos, reduce using 'merge'.
675 auto geo = drawGeos.size() > 1
Florin Malita418e6582018-07-09 16:20:47 -0400676 ? Merge(std::move(drawGeos), sksg::Merge::Mode::kMerge)
Florin Malita16d0ad02018-01-19 15:07:29 -0500677 : drawGeos[0];
678
679 SkASSERT(geo);
680 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500681 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500682 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500683 default:
684 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500685 }
686 }
687
Florin Malita16d0ad02018-01-19 15:07:29 -0500688 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500689 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500690
691 // Push transformed local geometries to parent list, for subsequent paints.
692 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500693 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500694 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
695 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500696 }
697
Florin Malita16d0ad02018-01-19 15:07:29 -0500698 // Emit local draws reversed (bottom->top, per spec).
699 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
700 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500701 }
702
Florin Malita16d0ad02018-01-19 15:07:29 -0500703 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500704}
705
Florin Malita1022f742018-02-23 11:10:22 -0500706sk_sp<sksg::RenderNode> AttachNestedAnimation(const char* path, AttachContext* ctx) {
707 class SkottieSGAdapter final : public sksg::RenderNode {
708 public:
709 explicit SkottieSGAdapter(sk_sp<Animation> animation)
710 : fAnimation(std::move(animation)) {
711 SkASSERT(fAnimation);
712 }
713
714 protected:
715 SkRect onRevalidate(sksg::InvalidationController*, const SkMatrix&) override {
716 return SkRect::MakeSize(fAnimation->size());
717 }
718
719 void onRender(SkCanvas* canvas) const override {
720 fAnimation->render(canvas);
721 }
722
723 private:
724 const sk_sp<Animation> fAnimation;
725 };
726
727 class SkottieAnimatorAdapter final : public sksg::Animator {
728 public:
Florin Malita911ae402018-05-31 16:45:29 -0400729 SkottieAnimatorAdapter(sk_sp<Animation> animation, float time_scale)
Florin Malita1022f742018-02-23 11:10:22 -0500730 : fAnimation(std::move(animation))
Florin Malita911ae402018-05-31 16:45:29 -0400731 , fTimeScale(time_scale) {
Florin Malita1022f742018-02-23 11:10:22 -0500732 SkASSERT(fAnimation);
Florin Malita1022f742018-02-23 11:10:22 -0500733 }
734
735 protected:
736 void onTick(float t) {
Florin Malita911ae402018-05-31 16:45:29 -0400737 // TODO: we prolly need more sophisticated timeline mapping for nested animations.
738 fAnimation->seek(t * fTimeScale);
Florin Malita1022f742018-02-23 11:10:22 -0500739 }
740
741 private:
742 const sk_sp<Animation> fAnimation;
Florin Malita911ae402018-05-31 16:45:29 -0400743 const float fTimeScale;
Florin Malita1022f742018-02-23 11:10:22 -0500744 };
745
746 const auto resStream = ctx->fResources.openStream(path);
747 if (!resStream || !resStream->hasLength()) {
748 LOG("!! Could not open: %s\n", path);
749 return nullptr;
750 }
751
Florin Malitac83a0de2018-05-31 12:17:55 -0400752 auto animation = Animation::Make(resStream.get(), &ctx->fResources);
Florin Malita1022f742018-02-23 11:10:22 -0500753 if (!animation) {
754 LOG("!! Could not load nested animation: %s\n", path);
755 return nullptr;
756 }
757
Florin Malita911ae402018-05-31 16:45:29 -0400758
759 ctx->fAnimators.push_back(
760 skstd::make_unique<SkottieAnimatorAdapter>(animation,
761 animation->duration() / ctx->fDuration));
Florin Malita1022f742018-02-23 11:10:22 -0500762
763 return sk_make_sp<SkottieSGAdapter>(std::move(animation));
764}
765
Florin Malita7d42c442018-06-14 16:16:01 -0400766sk_sp<sksg::RenderNode> AttachAssetRef(const skjson::ObjectValue& jlayer, AttachContext* ctx,
767 sk_sp<sksg::RenderNode>(*attach_proc)(const skjson::ObjectValue& comp, AttachContext* ctx)) {
Florin Malita0cc01b72018-05-10 18:40:35 -0400768
Florin Malita7d42c442018-06-14 16:16:01 -0400769 const auto refId = ParseDefault<SkString>(jlayer["refId"], SkString());
Florin Malita0cc01b72018-05-10 18:40:35 -0400770 if (refId.isEmpty()) {
771 LOG("!! Layer missing refId\n");
772 return nullptr;
773 }
774
775 if (refId.startsWith("$")) {
776 return AttachNestedAnimation(refId.c_str() + 1, ctx);
777 }
778
779 const auto* asset_info = ctx->fAssets.find(refId);
780 if (!asset_info) {
781 LOG("!! Asset not found: '%s'\n", refId.c_str());
782 return nullptr;
783 }
784
785 if (asset_info->fIsAttaching) {
786 LOG("!! Asset cycle detected for: '%s'\n", refId.c_str());
787 return nullptr;
788 }
789
790 asset_info->fIsAttaching = true;
Florin Malita7d42c442018-06-14 16:16:01 -0400791 auto asset = attach_proc(*asset_info->fAsset, ctx);
Florin Malita0cc01b72018-05-10 18:40:35 -0400792 asset_info->fIsAttaching = false;
793
794 return asset;
795}
796
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400797sk_sp<sksg::RenderNode> AttachCompLayer(const skjson::ObjectValue& jlayer, AttachContext* ctx) {
Florin Malita5dfd0692018-07-02 15:05:26 -0400798 const skjson::ObjectValue* time_remap = jlayer["tm"];
Florin Malita7d42c442018-06-14 16:16:01 -0400799 const auto start_time = ParseDefault<float>(jlayer["st"], 0.0f),
800 stretch_time = ParseDefault<float>(jlayer["sr"], 1.0f);
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400801 const auto requires_time_mapping = !SkScalarNearlyEqual(start_time , 0) ||
Florin Malita5dfd0692018-07-02 15:05:26 -0400802 !SkScalarNearlyEqual(stretch_time, 1) ||
803 time_remap;
Florin Malitaeb87d672018-01-29 15:28:24 -0500804
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400805 sksg::AnimatorList local_animators;
806 AttachContext local_ctx = { ctx->fResources,
807 ctx->fAssets,
808 ctx->fDuration,
Florin Malita5dfd0692018-07-02 15:05:26 -0400809 ctx->fFrameRate,
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400810 requires_time_mapping ? local_animators : ctx->fAnimators };
811
812 auto comp_layer = AttachAssetRef(jlayer, &local_ctx, AttachComposition);
813
Florin Malita5dfd0692018-07-02 15:05:26 -0400814 // Applies a bias/scale/remap t-adjustment to child animators.
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400815 class CompTimeMapper final : public sksg::GroupAnimator {
816 public:
817 CompTimeMapper(sksg::AnimatorList&& layer_animators, float time_bias, float time_scale)
818 : INHERITED(std::move(layer_animators))
819 , fTimeBias(time_bias)
820 , fTimeScale(time_scale) {}
821
822 void onTick(float t) override {
Florin Malita5dfd0692018-07-02 15:05:26 -0400823 // When time remapping is active, |t| is driven externally.
824 if (fRemappedTime.isValid()) {
825 t = *fRemappedTime.get();
826 }
827
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400828 this->INHERITED::onTick((t + fTimeBias) * fTimeScale);
829 }
Florin Malita5dfd0692018-07-02 15:05:26 -0400830
831 void remapTime(float t) { fRemappedTime.set(t); }
832
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400833 private:
Florin Malita5dfd0692018-07-02 15:05:26 -0400834 const float fTimeBias,
835 fTimeScale;
836 SkTLazy<float> fRemappedTime;
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400837
838 using INHERITED = sksg::GroupAnimator;
839 };
840
841 if (requires_time_mapping) {
842 const auto t_bias = -start_time,
843 t_scale = sk_ieee_float_divide(1, stretch_time);
Florin Malita5dfd0692018-07-02 15:05:26 -0400844 auto time_mapper = skstd::make_unique<CompTimeMapper>(std::move(local_animators),
845 t_bias, t_scale);
846 if (time_remap) {
847 // The lambda below captures a raw pointer to the mapper object. That should be safe,
848 // because both the lambda and the mapper are scoped/owned by ctx->fAnimators.
849 auto* raw_mapper = time_mapper.get();
850 auto frame_rate = ctx->fFrameRate;
851 BindProperty<ScalarValue>(*time_remap, &ctx->fAnimators,
852 [raw_mapper, frame_rate](const ScalarValue& t) {
853 raw_mapper->remapTime(t * frame_rate);
854 });
855 }
856 ctx->fAnimators.push_back(std::move(time_mapper));
Florin Malitaeb87d672018-01-29 15:28:24 -0500857 }
858
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400859 return comp_layer;
Florin Malita094ccde2017-12-30 12:27:00 -0500860}
861
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400862sk_sp<sksg::RenderNode> AttachSolidLayer(const skjson::ObjectValue& jlayer, AttachContext*) {
Florin Malita7d42c442018-06-14 16:16:01 -0400863 const auto size = SkSize::Make(ParseDefault<float>(jlayer["sw"], 0.0f),
864 ParseDefault<float>(jlayer["sh"], 0.0f));
865 const auto hex = ParseDefault<SkString>(jlayer["sc"], SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500866 uint32_t c;
867 if (size.isEmpty() ||
868 !hex.startsWith("#") ||
869 !SkParse::FindHex(hex.c_str() + 1, &c)) {
870 LogFail(jlayer, "Could not parse solid layer");
871 return nullptr;
872 }
873
874 const SkColor color = 0xff000000 | c;
875
876 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
877 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500878}
879
Florin Malita7d42c442018-06-14 16:16:01 -0400880sk_sp<sksg::RenderNode> AttachImageAsset(const skjson::ObjectValue& jimage, AttachContext* ctx) {
881 const auto name = ParseDefault<SkString>(jimage["p"], SkString()),
882 path = ParseDefault<SkString>(jimage["u"], SkString());
Florin Malita49328072018-01-08 12:51:12 -0500883 if (name.isEmpty())
884 return nullptr;
885
886 // TODO: plumb resource paths explicitly to ResourceProvider?
887 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
888 const auto resStream = ctx->fResources.openStream(resName.c_str());
889 if (!resStream || !resStream->hasLength()) {
890 LOG("!! Could not load image resource: %s\n", resName.c_str());
891 return nullptr;
892 }
893
894 // TODO: non-intrisic image sizing
895 return sksg::Image::Make(
896 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
897}
898
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400899sk_sp<sksg::RenderNode> AttachImageLayer(const skjson::ObjectValue& jlayer, AttachContext* ctx) {
Florin Malita0cc01b72018-05-10 18:40:35 -0400900 return AttachAssetRef(jlayer, ctx, AttachImageAsset);
Florin Malita094ccde2017-12-30 12:27:00 -0500901}
902
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400903sk_sp<sksg::RenderNode> AttachNullLayer(const skjson::ObjectValue& layer, AttachContext*) {
Florin Malita18eafd92018-01-04 21:11:55 -0500904 // Null layers are used solely to drive dependent transforms,
905 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500906 return nullptr;
907}
908
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400909sk_sp<sksg::RenderNode> AttachShapeLayer(const skjson::ObjectValue& layer, AttachContext* ctx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500910 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
911 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500912 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
913 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
914
915 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
916 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
917 // due to attached animators. To avoid this, we track committed animators and discard the
918 // orphans here.
919 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
920 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
921
922 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500923}
924
Florin Malita8c5f9ef2018-06-20 11:19:53 -0400925sk_sp<sksg::RenderNode> AttachTextLayer(const skjson::ObjectValue& layer, AttachContext*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500926 LOG("?? Text layer stub\n");
927 return nullptr;
928}
929
Florin Malita18eafd92018-01-04 21:11:55 -0500930struct AttachLayerContext {
Florin Malita7d42c442018-06-14 16:16:01 -0400931 AttachLayerContext(const skjson::ArrayValue& jlayers, AttachContext* ctx)
932 : fLayerList(jlayers), fCtx(ctx) {}
Florin Malita18eafd92018-01-04 21:11:55 -0500933
Florin Malita7d42c442018-06-14 16:16:01 -0400934 const skjson::ArrayValue& fLayerList;
Florin Malita4a490682018-01-28 14:27:51 -0500935 AttachContext* fCtx;
936 SkTHashMap<int, sk_sp<sksg::Matrix>> fLayerMatrixMap;
937 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500938
Florin Malita7d42c442018-06-14 16:16:01 -0400939 sk_sp<sksg::Matrix> AttachLayerMatrix(const skjson::ObjectValue& jlayer) {
940 const auto layer_index = ParseDefault<int>(jlayer["ind"], -1);
Florin Malita4a490682018-01-28 14:27:51 -0500941 if (layer_index < 0)
942 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500943
Florin Malita4a490682018-01-28 14:27:51 -0500944 if (auto* m = fLayerMatrixMap.find(layer_index))
945 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500946
Florin Malita2919b612018-05-10 12:44:07 -0400947 return this->AttachLayerMatrixImpl(jlayer, layer_index);
948 }
949
950private:
Florin Malita7d42c442018-06-14 16:16:01 -0400951 sk_sp<sksg::Matrix> AttachParentLayerMatrix(const skjson::ObjectValue& jlayer,
952 int layer_index) {
953 const auto parent_index = ParseDefault<int>(jlayer["parent"], -1);
Florin Malita2919b612018-05-10 12:44:07 -0400954 if (parent_index < 0 || parent_index == layer_index)
955 return nullptr;
956
957 if (auto* m = fLayerMatrixMap.find(parent_index))
958 return *m;
959
Florin Malita7d42c442018-06-14 16:16:01 -0400960 for (const skjson::ObjectValue* l : fLayerList) {
961 if (!l) continue;
962
963 if (ParseDefault<int>((*l)["ind"], -1) == parent_index) {
964 return this->AttachLayerMatrixImpl(*l, parent_index);
Florin Malita2919b612018-05-10 12:44:07 -0400965 }
966 }
967
968 return nullptr;
969 }
970
Florin Malita7d42c442018-06-14 16:16:01 -0400971 sk_sp<sksg::Matrix> AttachLayerMatrixImpl(const skjson::ObjectValue& jlayer, int layer_index) {
Florin Malita2919b612018-05-10 12:44:07 -0400972 SkASSERT(!fLayerMatrixMap.find(layer_index));
973
Florin Malita4a490682018-01-28 14:27:51 -0500974 // Add a stub entry to break recursion cycles.
975 fLayerMatrixMap.set(layer_index, nullptr);
Florin Malita18eafd92018-01-04 21:11:55 -0500976
Florin Malita2919b612018-05-10 12:44:07 -0400977 auto parent_matrix = this->AttachParentLayerMatrix(jlayer, layer_index);
Florin Malita18eafd92018-01-04 21:11:55 -0500978
Florin Malita7d42c442018-06-14 16:16:01 -0400979 if (const skjson::ObjectValue* jtransform = jlayer["ks"]) {
980 return *fLayerMatrixMap.set(layer_index, AttachMatrix(*jtransform, fCtx,
981 std::move(parent_matrix)));
982
983 }
984 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500985 }
986};
987
Florin Malita8deab3a2018-06-29 13:05:03 -0400988struct MaskInfo {
Florin Malita418e6582018-07-09 16:20:47 -0400989 SkBlendMode fBlendMode; // used when masking with layers/blending
990 sksg::Merge::Mode fMergeMode; // used when clipping
991 bool fInvertGeometry;
Florin Malita8deab3a2018-06-29 13:05:03 -0400992};
993
994const MaskInfo* GetMaskInfo(char mode) {
Florin Malita418e6582018-07-09 16:20:47 -0400995 static constexpr MaskInfo k_add_info =
996 { SkBlendMode::kSrcOver , sksg::Merge::Mode::kUnion , false };
997 static constexpr MaskInfo k_int_info =
998 { SkBlendMode::kSrcIn , sksg::Merge::Mode::kIntersect , false };
Florin Malita8deab3a2018-06-29 13:05:03 -0400999 // AE 'subtract' is the same as 'intersect' + inverted geometry
1000 // (draws the opacity-adjusted paint *outside* the shape).
Florin Malita418e6582018-07-09 16:20:47 -04001001 static constexpr MaskInfo k_sub_info =
1002 { SkBlendMode::kSrcIn , sksg::Merge::Mode::kIntersect , true };
1003 static constexpr MaskInfo k_dif_info =
1004 { SkBlendMode::kDifference, sksg::Merge::Mode::kDifference, false };
Florin Malita8deab3a2018-06-29 13:05:03 -04001005
Florin Malita25366fa2018-01-23 13:37:59 -05001006 switch (mode) {
Florin Malita8deab3a2018-06-29 13:05:03 -04001007 case 'a': return &k_add_info;
1008 case 'f': return &k_dif_info;
1009 case 'i': return &k_int_info;
1010 case 's': return &k_sub_info;
Florin Malita25366fa2018-01-23 13:37:59 -05001011 default: break;
1012 }
1013
Florin Malita8deab3a2018-06-29 13:05:03 -04001014 return nullptr;
Florin Malita25366fa2018-01-23 13:37:59 -05001015}
1016
Florin Malita7d42c442018-06-14 16:16:01 -04001017sk_sp<sksg::RenderNode> AttachMask(const skjson::ArrayValue* jmask,
Florin Malita25366fa2018-01-23 13:37:59 -05001018 AttachContext* ctx,
1019 sk_sp<sksg::RenderNode> childNode) {
Florin Malita7d42c442018-06-14 16:16:01 -04001020 if (!jmask) return childNode;
Florin Malita25366fa2018-01-23 13:37:59 -05001021
Florin Malita0c51c212018-04-26 14:13:14 -04001022 struct MaskRecord {
Florin Malita418e6582018-07-09 16:20:47 -04001023 sk_sp<sksg::Path> mask_path; // for clipping and masking
1024 sk_sp<sksg::Color> mask_paint; // for masking
1025 sksg::Merge::Mode merge_mode; // for clipping
Florin Malita0c51c212018-04-26 14:13:14 -04001026 };
1027
1028 SkSTArray<4, MaskRecord, true> mask_stack;
1029
Florin Malita53619352018-06-21 12:28:14 -04001030 const SkScalar full_opacity = 100;
1031 bool has_opacity = false;
Florin Malita25366fa2018-01-23 13:37:59 -05001032
Florin Malita7d42c442018-06-14 16:16:01 -04001033 for (const skjson::ObjectValue* m : *jmask) {
1034 if (!m) continue;
Florin Malita25366fa2018-01-23 13:37:59 -05001035
Florin Malita8deab3a2018-06-29 13:05:03 -04001036 SkString mode;
1037 if (!Parse<SkString>((*m)["mode"], &mode) || mode.size() != 1) {
1038 LogFail((*m)["mode"], "Invalid mask mode");
1039 continue;
1040 }
1041
1042 if (mode[0] == 'n') {
1043 // "None" masks have no effect.
1044 continue;
1045 }
1046
1047 const auto* mask_info = GetMaskInfo(mode[0]);
1048 if (!mask_info) {
1049 LOG("?? Unsupported mask mode: '%c'\n", mode[0]);
1050 continue;
1051 }
1052
Florin Malita7d42c442018-06-14 16:16:01 -04001053 auto mask_path = AttachPath((*m)["pt"], ctx);
Florin Malita25366fa2018-01-23 13:37:59 -05001054 if (!mask_path) {
Florin Malita7d42c442018-06-14 16:16:01 -04001055 LogFail(*m, "Could not parse mask path");
Florin Malita25366fa2018-01-23 13:37:59 -05001056 continue;
1057 }
1058
Florin Malita8deab3a2018-06-29 13:05:03 -04001059 // "inv" is cumulative with mask info fInvertGeometry
1060 const auto inverted =
1061 (mask_info->fInvertGeometry != ParseDefault<bool>((*m)["inv"], false));
1062 mask_path->setFillType(inverted ? SkPath::kInverseWinding_FillType
1063 : SkPath::kWinding_FillType);
Florin Malita25366fa2018-01-23 13:37:59 -05001064
1065 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
Florin Malitaaa71c892018-01-30 09:27:01 -05001066 mask_paint->setAntiAlias(true);
Florin Malita8deab3a2018-06-29 13:05:03 -04001067 // First mask in the stack initializes the mask buffer.
1068 mask_paint->setBlendMode(mask_stack.empty() ? SkBlendMode::kSrc
1069 : mask_info->fBlendMode);
Florin Malita0c51c212018-04-26 14:13:14 -04001070
Florin Malita53619352018-06-21 12:28:14 -04001071 has_opacity |= BindProperty<ScalarValue>((*m)["o"], &ctx->fAnimators,
1072 [mask_paint](const ScalarValue& o) {
1073 mask_paint->setOpacity(o * 0.01f);
1074 }, &full_opacity);
Florin Malita0c51c212018-04-26 14:13:14 -04001075
Florin Malita418e6582018-07-09 16:20:47 -04001076 mask_stack.push_back({mask_path, mask_paint, mask_info->fMergeMode});
Florin Malita25366fa2018-01-23 13:37:59 -05001077 }
1078
Florin Malita0c51c212018-04-26 14:13:14 -04001079 if (mask_stack.empty())
1080 return childNode;
1081
Florin Malita418e6582018-07-09 16:20:47 -04001082 // If the masks are fully opaque, we can clip.
1083 if (!has_opacity) {
1084 sk_sp<sksg::GeometryNode> clip_node;
1085
1086 if (mask_stack.count() == 1) {
1087 // Single path -> just clip.
1088 clip_node = std::move(mask_stack.front().mask_path);
1089 } else {
1090 // Multiple clip paths -> merge.
1091 std::vector<sksg::Merge::Rec> merge_recs;
1092 merge_recs.reserve(SkToSizeT(mask_stack.count()));
1093
1094 for (const auto& mask : mask_stack) {
1095 const auto mode = merge_recs.empty() ? sksg::Merge::Mode::kMerge : mask.merge_mode;
1096 merge_recs.push_back({std::move(mask.mask_path), mode});
1097 }
1098 clip_node = sksg::Merge::Make(std::move(merge_recs));
1099 }
1100
1101 return sksg::ClipEffect::Make(std::move(childNode), std::move(clip_node), true);
Florin Malita0c51c212018-04-26 14:13:14 -04001102 }
1103
1104 auto mask_group = sksg::Group::Make();
1105 for (const auto& rec : mask_stack) {
1106 mask_group->addChild(sksg::Draw::Make(std::move(rec.mask_path),
1107 std::move(rec.mask_paint)));
1108
1109 }
1110
1111 return sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
Florin Malita25366fa2018-01-23 13:37:59 -05001112}
1113
Florin Malita7d42c442018-06-14 16:16:01 -04001114sk_sp<sksg::RenderNode> AttachLayer(const skjson::ObjectValue* jlayer,
1115 AttachLayerContext* layerCtx) {
1116 if (!jlayer) return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001117
Florin Malita8c5f9ef2018-06-20 11:19:53 -04001118 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const skjson::ObjectValue&, AttachContext*);
Florin Malita094ccde2017-12-30 12:27:00 -05001119 static constexpr LayerAttacher gLayerAttachers[] = {
1120 AttachCompLayer, // 'ty': 0
1121 AttachSolidLayer, // 'ty': 1
1122 AttachImageLayer, // 'ty': 2
1123 AttachNullLayer, // 'ty': 3
1124 AttachShapeLayer, // 'ty': 4
1125 AttachTextLayer, // 'ty': 5
1126 };
1127
Florin Malita7d42c442018-06-14 16:16:01 -04001128 int type = ParseDefault<int>((*jlayer)["ty"], -1);
Florin Malita094ccde2017-12-30 12:27:00 -05001129 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
1130 return nullptr;
1131 }
1132
Florin Malitacca86f32018-01-29 10:49:49 -05001133 sksg::AnimatorList layer_animators;
Florin Malita1022f742018-02-23 11:10:22 -05001134 AttachContext local_ctx = { layerCtx->fCtx->fResources,
1135 layerCtx->fCtx->fAssets,
Florin Malita911ae402018-05-31 16:45:29 -04001136 layerCtx->fCtx->fDuration,
Florin Malita5dfd0692018-07-02 15:05:26 -04001137 layerCtx->fCtx->fFrameRate,
Florin Malita1022f742018-02-23 11:10:22 -05001138 layer_animators};
Florin Malitacca86f32018-01-29 10:49:49 -05001139
Florin Malita71cba8f2018-01-09 08:07:14 -05001140 // Layer content.
Florin Malita8c5f9ef2018-06-20 11:19:53 -04001141 auto layer = gLayerAttachers[type](*jlayer, &local_ctx);
Florin Malita38ea40e2018-01-29 16:31:14 -05001142
1143 // Clip layers with explicit dimensions.
Florin Malitafa7e9a82018-05-04 15:10:54 -04001144 float w = 0, h = 0;
Florin Malita7d42c442018-06-14 16:16:01 -04001145 if (Parse<float>((*jlayer)["w"], &w) && Parse<float>((*jlayer)["h"], &h)) {
Florin Malita38ea40e2018-01-29 16:31:14 -05001146 layer = sksg::ClipEffect::Make(std::move(layer),
1147 sksg::Rect::Make(SkRect::MakeWH(w, h)),
1148 true);
1149 }
1150
Florin Malita25366fa2018-01-23 13:37:59 -05001151 // Optional layer mask.
Florin Malita7d42c442018-06-14 16:16:01 -04001152 layer = AttachMask((*jlayer)["masksProperties"], &local_ctx, std::move(layer));
Florin Malita38ea40e2018-01-29 16:31:14 -05001153
Florin Malita25366fa2018-01-23 13:37:59 -05001154 // Optional layer transform.
Florin Malita7d42c442018-06-14 16:16:01 -04001155 if (auto layerMatrix = layerCtx->AttachLayerMatrix(*jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -05001156 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
1157 }
Florin Malita38ea40e2018-01-29 16:31:14 -05001158
Florin Malita71cba8f2018-01-09 08:07:14 -05001159 // Optional layer opacity.
Florin Malita7d42c442018-06-14 16:16:01 -04001160 // TODO: de-dupe this "ks" lookup with matrix above.
1161 if (const skjson::ObjectValue* jtransform = (*jlayer)["ks"]) {
1162 layer = AttachOpacity(*jtransform, &local_ctx, std::move(layer));
1163 }
Florin Malita18eafd92018-01-04 21:11:55 -05001164
Florin Malitaeb87d672018-01-29 15:28:24 -05001165 class LayerController final : public sksg::GroupAnimator {
Florin Malita71cba8f2018-01-09 08:07:14 -05001166 public:
Florin Malitaeb87d672018-01-29 15:28:24 -05001167 LayerController(sksg::AnimatorList&& layer_animators,
1168 sk_sp<sksg::OpacityEffect> controlNode,
Florin Malita8c5f9ef2018-06-20 11:19:53 -04001169 float in, float out)
Florin Malitacca86f32018-01-29 10:49:49 -05001170 : INHERITED(std::move(layer_animators))
1171 , fControlNode(std::move(controlNode))
Florin Malita71cba8f2018-01-09 08:07:14 -05001172 , fIn(in)
Florin Malita8c5f9ef2018-06-20 11:19:53 -04001173 , fOut(out) {}
Florin Malita71cba8f2018-01-09 08:07:14 -05001174
Florin Malita35efaa82018-01-22 12:57:06 -05001175 void onTick(float t) override {
Florin Malitacca86f32018-01-29 10:49:49 -05001176 const auto active = (t >= fIn && t <= fOut);
1177
Florin Malita71cba8f2018-01-09 08:07:14 -05001178 // Keep the layer fully transparent except for its [in..out] lifespan.
1179 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
Florin Malitacca86f32018-01-29 10:49:49 -05001180 fControlNode->setOpacity(active ? 1 : 0);
1181
1182 // Dispatch ticks only while active.
Florin Malita8c5f9ef2018-06-20 11:19:53 -04001183 if (active) this->INHERITED::onTick(t);
Florin Malita71cba8f2018-01-09 08:07:14 -05001184 }
1185
1186 private:
1187 const sk_sp<sksg::OpacityEffect> fControlNode;
1188 const float fIn,
Florin Malita8c5f9ef2018-06-20 11:19:53 -04001189 fOut;
Florin Malitacca86f32018-01-29 10:49:49 -05001190
1191 using INHERITED = sksg::GroupAnimator;
Florin Malita71cba8f2018-01-09 08:07:14 -05001192 };
1193
Florin Malitaeb87d672018-01-29 15:28:24 -05001194 auto controller_node = sksg::OpacityEffect::Make(std::move(layer));
Florin Malita7d42c442018-06-14 16:16:01 -04001195 const auto in = ParseDefault<float>((*jlayer)["ip"], 0.0f),
1196 out = ParseDefault<float>((*jlayer)["op"], in);
Florin Malita71cba8f2018-01-09 08:07:14 -05001197
Florin Malitaeb87d672018-01-29 15:28:24 -05001198 if (in >= out || !controller_node)
Florin Malita71cba8f2018-01-09 08:07:14 -05001199 return nullptr;
1200
Florin Malitacca86f32018-01-29 10:49:49 -05001201 layerCtx->fCtx->fAnimators.push_back(
Florin Malita8c5f9ef2018-06-20 11:19:53 -04001202 skstd::make_unique<LayerController>(std::move(layer_animators), controller_node, in, out));
Florin Malita71cba8f2018-01-09 08:07:14 -05001203
Florin Malita7d42c442018-06-14 16:16:01 -04001204 if (ParseDefault<bool>((*jlayer)["td"], false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -05001205 // This layer is a matte. We apply it as a mask to the next layer.
Florin Malitaeb87d672018-01-29 15:28:24 -05001206 layerCtx->fCurrentMatte = std::move(controller_node);
Florin Malita5f9102f2018-01-10 13:36:22 -05001207 return nullptr;
1208 }
1209
1210 if (layerCtx->fCurrentMatte) {
1211 // There is a pending matte. Apply and reset.
Florin Malitaa016be92018-03-05 14:01:41 -05001212 static constexpr sksg::MaskEffect::Mode gMaskModes[] = {
1213 sksg::MaskEffect::Mode::kNormal, // tt: 1
1214 sksg::MaskEffect::Mode::kInvert, // tt: 2
1215 };
Florin Malita7d42c442018-06-14 16:16:01 -04001216 const auto matteType = ParseDefault<int>((*jlayer)["tt"], 1) - 1;
Florin Malitaa016be92018-03-05 14:01:41 -05001217
1218 if (matteType >= 0 && matteType < SkTo<int>(SK_ARRAY_COUNT(gMaskModes))) {
1219 return sksg::MaskEffect::Make(std::move(controller_node),
1220 std::move(layerCtx->fCurrentMatte),
1221 gMaskModes[matteType]);
1222 }
1223 layerCtx->fCurrentMatte.reset();
Florin Malita5f9102f2018-01-10 13:36:22 -05001224 }
1225
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001226 return std::move(controller_node);
Florin Malita094ccde2017-12-30 12:27:00 -05001227}
1228
Florin Malita7d42c442018-06-14 16:16:01 -04001229sk_sp<sksg::RenderNode> AttachComposition(const skjson::ObjectValue& comp, AttachContext* ctx) {
1230 const skjson::ArrayValue* jlayers = comp["layers"];
1231 if (!jlayers) return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001232
Florin Malita18eafd92018-01-04 21:11:55 -05001233 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
Florin Malita7d42c442018-06-14 16:16:01 -04001234 AttachLayerContext layerCtx(*jlayers, ctx);
Florin Malita18eafd92018-01-04 21:11:55 -05001235
Florin Malita7d42c442018-06-14 16:16:01 -04001236 for (const auto& l : *jlayers) {
Florin Malita18eafd92018-01-04 21:11:55 -05001237 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001238 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001239 }
1240 }
1241
Florin Malita2a8275b2018-01-02 12:52:43 -05001242 if (layers.empty()) {
1243 return nullptr;
1244 }
1245
1246 // Layers are painted in bottom->top order.
1247 auto comp_group = sksg::Group::Make();
1248 for (int i = layers.count() - 1; i >= 0; --i) {
1249 comp_group->addChild(std::move(layers[i]));
1250 }
1251
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001252 return std::move(comp_group);
Florin Malita094ccde2017-12-30 12:27:00 -05001253}
1254
1255} // namespace
1256
Florin Malitac83a0de2018-05-31 12:17:55 -04001257sk_sp<Animation> Animation::Make(SkStream* stream, const ResourceProvider* provider, Stats* stats) {
Florin Malita6eb85a12018-04-30 10:32:18 -04001258 Stats stats_storage;
1259 if (!stats)
1260 stats = &stats_storage;
1261 memset(stats, 0, sizeof(struct Stats));
1262
Florin Malita094ccde2017-12-30 12:27:00 -05001263 if (!stream->hasLength()) {
1264 // TODO: handle explicit buffering?
1265 LOG("!! cannot parse streaming content\n");
1266 return nullptr;
1267 }
1268
Florin Malitafa7e9a82018-05-04 15:10:54 -04001269 stats->fJsonSize = stream->getLength();
Florin Malita6eb85a12018-04-30 10:32:18 -04001270 const auto t0 = SkTime::GetMSecs();
1271
Florin Malita7d42c442018-06-14 16:16:01 -04001272 auto data = SkData::MakeFromStream(stream, stream->getLength());
1273 if (!data) {
1274 SkDebugf("!! Failed to read the input stream.\n");
Florin Malitafa7e9a82018-05-04 15:10:54 -04001275 return nullptr;
Florin Malita7d42c442018-06-14 16:16:01 -04001276 }
1277
1278 const skjson::DOM dom(static_cast<const char*>(data->data()), data->size());
1279 if (!dom.root().is<skjson::ObjectValue>()) {
1280 // TODO: more error info.
1281 SkDebugf("!! Failed to parse JSON input.\n");
1282 return nullptr;
1283 }
1284 const auto& json = dom.root().as<skjson::ObjectValue>();
Florin Malita094ccde2017-12-30 12:27:00 -05001285
Florin Malita6eb85a12018-04-30 10:32:18 -04001286 const auto t1 = SkTime::GetMSecs();
1287 stats->fJsonParseTimeMS = t1 - t0;
1288
Florin Malita7d42c442018-06-14 16:16:01 -04001289 const auto version = ParseDefault<SkString>(json["v"], SkString());
1290 const auto size = SkSize::Make(ParseDefault<float>(json["w"], 0.0f),
1291 ParseDefault<float>(json["h"], 0.0f));
1292 const auto fps = ParseDefault<float>(json["fr"], -1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001293
Florin Malita1022f742018-02-23 11:10:22 -05001294 if (size.isEmpty() || version.isEmpty() || fps <= 0) {
Florin Malita094ccde2017-12-30 12:27:00 -05001295 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1296 version.c_str(), size.width(), size.height(), fps);
1297 return nullptr;
1298 }
1299
Florin Malitac83a0de2018-05-31 12:17:55 -04001300 class NullResourceProvider final : public ResourceProvider {
1301 std::unique_ptr<SkStream> openStream(const char[]) const { return nullptr; }
1302 };
1303
Kevin Lubickf14bc982018-06-01 13:00:35 -04001304 NullResourceProvider null_provider;
Florin Malitac83a0de2018-05-31 12:17:55 -04001305 const auto anim = sk_sp<Animation>(new Animation(provider ? *provider : null_provider,
1306 std::move(version), size, fps, json, stats));
Florin Malita6eb85a12018-04-30 10:32:18 -04001307 const auto t2 = SkTime::GetMSecs();
1308 stats->fSceneParseTimeMS = t2 - t1;
1309 stats->fTotalLoadTimeMS = t2 - t0;
1310
1311 return anim;
Florin Malita094ccde2017-12-30 12:27:00 -05001312}
1313
Florin Malita6eb85a12018-04-30 10:32:18 -04001314sk_sp<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res,
1315 Stats* stats) {
Florin Malita49328072018-01-08 12:51:12 -05001316 class DirectoryResourceProvider final : public ResourceProvider {
1317 public:
1318 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1319
1320 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1321 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1322 return SkStream::MakeFromFile(resPath.c_str());
1323 }
1324
1325 private:
1326 const SkString fDir;
1327 };
1328
1329 const auto jsonStream = SkStream::MakeFromFile(path);
1330 if (!jsonStream)
1331 return nullptr;
1332
1333 std::unique_ptr<ResourceProvider> defaultProvider;
1334 if (!res) {
1335 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1336 }
1337
Florin Malitac83a0de2018-05-31 12:17:55 -04001338 return Make(jsonStream.get(), res ? res : defaultProvider.get(), stats);
Florin Malita49328072018-01-08 12:51:12 -05001339}
1340
1341Animation::Animation(const ResourceProvider& resources,
Florin Malita7d42c442018-06-14 16:16:01 -04001342 SkString version, const SkSize& size, SkScalar fps,
1343 const skjson::ObjectValue& json, Stats* stats)
Florin Malita094ccde2017-12-30 12:27:00 -05001344 : fVersion(std::move(version))
1345 , fSize(size)
1346 , fFrameRate(fps)
Florin Malita7d42c442018-06-14 16:16:01 -04001347 , fInPoint(ParseDefault<float>(json["ip"], 0.0f))
1348 , fOutPoint(SkTMax(ParseDefault<float>(json["op"], SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001349
1350 AssetMap assets;
Florin Malita7d42c442018-06-14 16:16:01 -04001351 if (const skjson::ArrayValue* jassets = json["assets"]) {
1352 for (const skjson::ObjectValue* asset : *jassets) {
1353 if (asset) {
1354 assets.set(ParseDefault<SkString>((*asset)["id"], SkString()), { asset, false });
1355 }
Florin Malita094ccde2017-12-30 12:27:00 -05001356 }
Florin Malita094ccde2017-12-30 12:27:00 -05001357 }
1358
Florin Malitacca86f32018-01-29 10:49:49 -05001359 sksg::AnimatorList animators;
Florin Malita5dfd0692018-07-02 15:05:26 -04001360 AttachContext ctx = { resources, assets, this->duration(), fFrameRate, animators };
Florin Malita35efaa82018-01-22 12:57:06 -05001361 auto root = AttachComposition(json, &ctx);
1362
Florin Malita6eb85a12018-04-30 10:32:18 -04001363 stats->fAnimatorCount = animators.size();
Florin Malita35efaa82018-01-22 12:57:06 -05001364
1365 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001366
Florin Malitadb385732018-01-09 12:19:32 -05001367 // In case the client calls render before the first tick.
Florin Malitaa33447d2018-05-29 13:46:54 -04001368 this->seek(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001369}
1370
1371Animation::~Animation() = default;
1372
Florin Malita35efaa82018-01-22 12:57:06 -05001373void Animation::setShowInval(bool show) {
1374 if (fScene) {
1375 fScene->setShowInval(show);
1376 }
1377}
1378
Mike Reed29859872018-01-08 08:25:27 -05001379void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001380 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001381 return;
1382
Mike Reed29859872018-01-08 08:25:27 -05001383 SkAutoCanvasRestore restore(canvas, true);
1384 const SkRect srcR = SkRect::MakeSize(this->size());
1385 if (dstR) {
1386 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1387 }
1388 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001389 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001390}
1391
Florin Malitaa33447d2018-05-29 13:46:54 -04001392void Animation::seek(SkScalar t) {
Florin Malita35efaa82018-01-22 12:57:06 -05001393 if (!fScene)
1394 return;
1395
Florin Malitaa33447d2018-05-29 13:46:54 -04001396 fScene->animate(fInPoint + SkTPin(t, 0.0f, 1.0f) * (fOutPoint - fInPoint));
Florin Malita094ccde2017-12-30 12:27:00 -05001397}
1398
Florin Malita54f65c42018-01-16 17:04:30 -05001399} // namespace skottie