blob: 0ac1b4ed714b19ae5b4eac66d6c6a3f9538fb7e7 [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 Malitafa7e9a82018-05-04 15:10:54 -040055using AssetMap = SkTHashMap<SkString, json::ValueRef>;
Florin Malita094ccde2017-12-30 12:27:00 -050056
57struct AttachContext {
Florin Malitacca86f32018-01-29 10:49:49 -050058 const ResourceProvider& fResources;
59 const AssetMap& fAssets;
Florin Malita1022f742018-02-23 11:10:22 -050060 const float fFrameRate;
Florin Malitacca86f32018-01-29 10:49:49 -050061 sksg::AnimatorList& fAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -050062};
63
Florin Malitafa7e9a82018-05-04 15:10:54 -040064bool LogFail(const json::ValueRef& json, const char* msg) {
65 const auto dump = json.toString();
Florin Malita20880782018-05-09 11:35:00 -040066 LOG("!! %s: %s\n", msg, dump.c_str());
Florin Malita094ccde2017-12-30 12:27:00 -050067 return false;
68}
69
Florin Malitafa7e9a82018-05-04 15:10:54 -040070sk_sp<sksg::Matrix> AttachMatrix(const json::ValueRef& t, AttachContext* ctx,
Florin Malita18eafd92018-01-04 21:11:55 -050071 sk_sp<sksg::Matrix> parentMatrix) {
72 if (!t.isObject())
73 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -050074
Florin Malita18eafd92018-01-04 21:11:55 -050075 auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
Florin Malitaa6e30f72018-03-23 13:41:58 -040076 auto adapter = sk_make_sp<TransformAdapter>(matrix);
Florin Malitafc807c82018-01-25 22:35:09 -050077 auto anchor_attached = BindProperty<VectorValue>(t["a"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040078 [adapter](const VectorValue& a) {
79 adapter->setAnchorPoint(ValueTraits<VectorValue>::As<SkPoint>(a));
Florin Malita094ccde2017-12-30 12:27:00 -050080 });
Florin Malitafc807c82018-01-25 22:35:09 -050081 auto position_attached = BindProperty<VectorValue>(t["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040082 [adapter](const VectorValue& p) {
83 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malita094ccde2017-12-30 12:27:00 -050084 });
Florin Malitafc807c82018-01-25 22:35:09 -050085 auto scale_attached = BindProperty<VectorValue>(t["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040086 [adapter](const VectorValue& s) {
87 adapter->setScale(ValueTraits<VectorValue>::As<SkVector>(s));
Florin Malita094ccde2017-12-30 12:27:00 -050088 });
Florin Malita1eb98db2018-01-26 15:03:38 -050089
Florin Malitafa7e9a82018-05-04 15:10:54 -040090 auto jrotation = t["r"];
91 if (jrotation.isNull()) {
Florin Malita1eb98db2018-01-26 15:03:38 -050092 // 3d rotations have separate rx,ry,rz components. While we don't fully support them,
93 // we can still make use of rz.
Florin Malitafa7e9a82018-05-04 15:10:54 -040094 jrotation = t["rz"];
Florin Malita1eb98db2018-01-26 15:03:38 -050095 }
Florin Malitafa7e9a82018-05-04 15:10:54 -040096 auto rotation_attached = BindProperty<ScalarValue>(jrotation, &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -040097 [adapter](const ScalarValue& r) {
98 adapter->setRotation(r);
Florin Malita094ccde2017-12-30 12:27:00 -050099 });
Florin Malitafc807c82018-01-25 22:35:09 -0500100 auto skew_attached = BindProperty<ScalarValue>(t["sk"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400101 [adapter](const ScalarValue& sk) {
102 adapter->setSkew(sk);
Florin Malita094ccde2017-12-30 12:27:00 -0500103 });
Florin Malitafc807c82018-01-25 22:35:09 -0500104 auto skewaxis_attached = BindProperty<ScalarValue>(t["sa"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400105 [adapter](const ScalarValue& sa) {
106 adapter->setSkewAxis(sa);
Florin Malita094ccde2017-12-30 12:27:00 -0500107 });
108
109 if (!anchor_attached &&
110 !position_attached &&
111 !scale_attached &&
112 !rotation_attached &&
113 !skew_attached &&
114 !skewaxis_attached) {
115 LogFail(t, "Could not parse transform");
Florin Malita18eafd92018-01-04 21:11:55 -0500116 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500117 }
118
Florin Malita18eafd92018-01-04 21:11:55 -0500119 return matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500120}
121
Florin Malitafa7e9a82018-05-04 15:10:54 -0400122sk_sp<sksg::RenderNode> AttachOpacity(const json::ValueRef& jtransform, AttachContext* ctx,
Florin Malitac0034172018-01-08 16:42:59 -0500123 sk_sp<sksg::RenderNode> childNode) {
124 if (!jtransform.isObject() || !childNode)
125 return childNode;
126
127 // This is more peeky than other attachers, because we want to avoid redundant opacity
128 // nodes for the extremely common case of static opaciy == 100.
Florin Malita20880782018-05-09 11:35:00 -0400129 const auto opacity = jtransform["o"];
130 if (!opacity["a"].toDefault(true) &&
Florin Malitafa7e9a82018-05-04 15:10:54 -0400131 opacity["k"].toDefault<int>(-1) == 100) {
Florin Malitac0034172018-01-08 16:42:59 -0500132 // Ignoring static full opacity.
133 return childNode;
134 }
135
136 auto opacityNode = sksg::OpacityEffect::Make(childNode);
Florin Malitafc807c82018-01-25 22:35:09 -0500137 BindProperty<ScalarValue>(opacity, &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500138 [opacityNode](const ScalarValue& o) {
Florin Malitac0034172018-01-08 16:42:59 -0500139 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500140 opacityNode->setOpacity(o * 0.01f);
Florin Malitac0034172018-01-08 16:42:59 -0500141 });
142
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400143 return std::move(opacityNode);
Florin Malitac0034172018-01-08 16:42:59 -0500144}
145
Florin Malitafa7e9a82018-05-04 15:10:54 -0400146sk_sp<sksg::RenderNode> AttachComposition(const json::ValueRef&, AttachContext* ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500147
Florin Malitafa7e9a82018-05-04 15:10:54 -0400148sk_sp<sksg::Path> AttachPath(const json::ValueRef& jpath, AttachContext* ctx) {
Florin Malita25366fa2018-01-23 13:37:59 -0500149 auto path_node = sksg::Path::Make();
Florin Malitafc807c82018-01-25 22:35:09 -0500150 return BindProperty<ShapeValue>(jpath, &ctx->fAnimators,
Florin Malitac353ee22018-04-30 21:49:41 -0400151 [path_node](const ShapeValue& p) {
152 path_node->setPath(ValueTraits<ShapeValue>::As<SkPath>(p));
153 })
Florin Malita25366fa2018-01-23 13:37:59 -0500154 ? path_node
155 : nullptr;
156}
157
Florin Malitafa7e9a82018-05-04 15:10:54 -0400158sk_sp<sksg::GeometryNode> AttachPathGeometry(const json::ValueRef& jpath, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500159 SkASSERT(jpath.isObject());
160
Florin Malita25366fa2018-01-23 13:37:59 -0500161 return AttachPath(jpath["ks"], ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500162}
163
Florin Malitafa7e9a82018-05-04 15:10:54 -0400164sk_sp<sksg::GeometryNode> AttachRRectGeometry(const json::ValueRef& jrect, AttachContext* ctx) {
Florin Malita2e1d7e22018-01-02 10:40:00 -0500165 SkASSERT(jrect.isObject());
166
167 auto rect_node = sksg::RRect::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400168 auto adapter = sk_make_sp<RRectAdapter>(rect_node);
Florin Malita2e1d7e22018-01-02 10:40:00 -0500169
Florin Malitafc807c82018-01-25 22:35:09 -0500170 auto p_attached = BindProperty<VectorValue>(jrect["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400171 [adapter](const VectorValue& p) {
172 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500173 });
Florin Malitafc807c82018-01-25 22:35:09 -0500174 auto s_attached = BindProperty<VectorValue>(jrect["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400175 [adapter](const VectorValue& s) {
176 adapter->setSize(ValueTraits<VectorValue>::As<SkSize>(s));
Florin Malitaf9590922018-01-09 11:56:09 -0500177 });
Florin Malitafc807c82018-01-25 22:35:09 -0500178 auto r_attached = BindProperty<ScalarValue>(jrect["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400179 [adapter](const ScalarValue& r) {
180 adapter->setRadius(SkSize::Make(r, r));
Florin Malitaf9590922018-01-09 11:56:09 -0500181 });
Florin Malita2e1d7e22018-01-02 10:40:00 -0500182
183 if (!p_attached && !s_attached && !r_attached) {
184 return nullptr;
185 }
186
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400187 return std::move(rect_node);
Florin Malitafbc13f12018-01-04 10:26:35 -0500188}
189
Florin Malitafa7e9a82018-05-04 15:10:54 -0400190sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const json::ValueRef& jellipse, AttachContext* ctx) {
Florin Malitafbc13f12018-01-04 10:26:35 -0500191 SkASSERT(jellipse.isObject());
192
193 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 Malitafa7e9a82018-05-04 15:10:54 -0400214sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const json::ValueRef& jstar, AttachContext* ctx) {
Florin Malita02a32b02018-01-04 11:27:09 -0500215 SkASSERT(jstar.isObject());
216
Florin Malitaa6e30f72018-03-23 13:41:58 -0400217 static constexpr PolyStarAdapter::Type gTypes[] = {
218 PolyStarAdapter::Type::kStar, // "sy": 1
219 PolyStarAdapter::Type::kPoly, // "sy": 2
Florin Malita02a32b02018-01-04 11:27:09 -0500220 };
221
Florin Malitafa7e9a82018-05-04 15:10:54 -0400222 const auto type = jstar["sy"].toDefault<int>(0) - 1;
Florin Malita02a32b02018-01-04 11:27:09 -0500223 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
224 LogFail(jstar, "Unknown polystar type");
225 return nullptr;
226 }
227
228 auto path_node = sksg::Path::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400229 auto adapter = sk_make_sp<PolyStarAdapter>(path_node, gTypes[type]);
Florin Malita02a32b02018-01-04 11:27:09 -0500230
Florin Malitafc807c82018-01-25 22:35:09 -0500231 BindProperty<VectorValue>(jstar["p"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400232 [adapter](const VectorValue& p) {
233 adapter->setPosition(ValueTraits<VectorValue>::As<SkPoint>(p));
Florin Malitaf9590922018-01-09 11:56:09 -0500234 });
Florin Malitafc807c82018-01-25 22:35:09 -0500235 BindProperty<ScalarValue>(jstar["pt"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400236 [adapter](const ScalarValue& pt) {
237 adapter->setPointCount(pt);
Florin Malitaf9590922018-01-09 11:56:09 -0500238 });
Florin Malitafc807c82018-01-25 22:35:09 -0500239 BindProperty<ScalarValue>(jstar["ir"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400240 [adapter](const ScalarValue& ir) {
241 adapter->setInnerRadius(ir);
Florin Malitaf9590922018-01-09 11:56:09 -0500242 });
Florin Malitafc807c82018-01-25 22:35:09 -0500243 BindProperty<ScalarValue>(jstar["or"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400244 [adapter](const ScalarValue& otr) {
245 adapter->setOuterRadius(otr);
Florin Malita9661b982018-01-06 14:25:49 -0500246 });
Florin Malitafc807c82018-01-25 22:35:09 -0500247 BindProperty<ScalarValue>(jstar["is"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400248 [adapter](const ScalarValue& is) {
249 adapter->setInnerRoundness(is);
Florin Malita9661b982018-01-06 14:25:49 -0500250 });
Florin Malitafc807c82018-01-25 22:35:09 -0500251 BindProperty<ScalarValue>(jstar["os"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400252 [adapter](const ScalarValue& os) {
253 adapter->setOuterRoundness(os);
Florin Malita9661b982018-01-06 14:25:49 -0500254 });
Florin Malitafc807c82018-01-25 22:35:09 -0500255 BindProperty<ScalarValue>(jstar["r"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400256 [adapter](const ScalarValue& r) {
257 adapter->setRotation(r);
Florin Malitaf9590922018-01-09 11:56:09 -0500258 });
Florin Malita02a32b02018-01-04 11:27:09 -0500259
Kevin Lubickf7621cb2018-04-16 15:51:44 -0400260 return std::move(path_node);
Florin Malita02a32b02018-01-04 11:27:09 -0500261}
262
Florin Malitafa7e9a82018-05-04 15:10:54 -0400263sk_sp<sksg::Color> AttachColor(const json::ValueRef& obj, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -0500264 SkASSERT(obj.isObject());
265
266 auto color_node = sksg::Color::Make(SK_ColorBLACK);
Florin Malitafc807c82018-01-25 22:35:09 -0500267 auto color_attached = BindProperty<VectorValue>(obj["c"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500268 [color_node](const VectorValue& c) {
269 color_node->setColor(ValueTraits<VectorValue>::As<SkColor>(c));
Florin Malitaf9590922018-01-09 11:56:09 -0500270 });
Florin Malita094ccde2017-12-30 12:27:00 -0500271
Florin Malita1586d852018-01-12 14:27:39 -0500272 return color_attached ? color_node : nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500273}
274
Florin Malitafa7e9a82018-05-04 15:10:54 -0400275sk_sp<sksg::Gradient> AttachGradient(const json::ValueRef& obj, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500276 SkASSERT(obj.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500277
Florin Malita20880782018-05-09 11:35:00 -0400278 const auto stops = obj["g"];
Florin Malita6aaee592018-01-12 12:25:09 -0500279 if (!stops.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -0500280 return nullptr;
281
Florin Malitafa7e9a82018-05-04 15:10:54 -0400282 const auto stopCount = stops["p"].toDefault<int>(-1);
Florin Malita6aaee592018-01-12 12:25:09 -0500283 if (stopCount < 0)
284 return nullptr;
285
286 sk_sp<sksg::Gradient> gradient_node;
Florin Malitaa6e30f72018-03-23 13:41:58 -0400287 sk_sp<GradientAdapter> adapter;
Florin Malita6aaee592018-01-12 12:25:09 -0500288
Florin Malitafa7e9a82018-05-04 15:10:54 -0400289 if (obj["t"].toDefault<int>(1) == 1) {
Florin Malita6aaee592018-01-12 12:25:09 -0500290 auto linear_node = sksg::LinearGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400291 adapter = sk_make_sp<LinearGradientAdapter>(linear_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500292 gradient_node = std::move(linear_node);
293 } else {
294 auto radial_node = sksg::RadialGradient::Make();
Florin Malitaa6e30f72018-03-23 13:41:58 -0400295 adapter = sk_make_sp<RadialGradientAdapter>(radial_node, stopCount);
Florin Malita6aaee592018-01-12 12:25:09 -0500296
297 // TODO: highlight, angle
298 gradient_node = std::move(radial_node);
299 }
300
Florin Malitafc807c82018-01-25 22:35:09 -0500301 BindProperty<VectorValue>(stops["k"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400302 [adapter](const VectorValue& stops) {
303 adapter->setColorStops(stops);
Florin Malita6aaee592018-01-12 12:25:09 -0500304 });
Florin Malitafc807c82018-01-25 22:35:09 -0500305 BindProperty<VectorValue>(obj["s"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400306 [adapter](const VectorValue& s) {
307 adapter->setStartPoint(ValueTraits<VectorValue>::As<SkPoint>(s));
Florin Malita6aaee592018-01-12 12:25:09 -0500308 });
Florin Malitafc807c82018-01-25 22:35:09 -0500309 BindProperty<VectorValue>(obj["e"], &ctx->fAnimators,
Florin Malitaa6e30f72018-03-23 13:41:58 -0400310 [adapter](const VectorValue& e) {
311 adapter->setEndPoint(ValueTraits<VectorValue>::As<SkPoint>(e));
Florin Malita6aaee592018-01-12 12:25:09 -0500312 });
313
314 return gradient_node;
315}
316
Florin Malitafa7e9a82018-05-04 15:10:54 -0400317sk_sp<sksg::PaintNode> AttachPaint(const json::ValueRef& jpaint, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500318 sk_sp<sksg::PaintNode> paint_node) {
319 if (paint_node) {
320 paint_node->setAntiAlias(true);
321
Florin Malitafc807c82018-01-25 22:35:09 -0500322 BindProperty<ScalarValue>(jpaint["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500323 [paint_node](const ScalarValue& o) {
Florin Malita1586d852018-01-12 14:27:39 -0500324 // BM opacity is [0..100]
Florin Malita2518a0a2018-01-24 18:29:00 -0500325 paint_node->setOpacity(o * 0.01f);
Florin Malita1586d852018-01-12 14:27:39 -0500326 });
Florin Malita6aaee592018-01-12 12:25:09 -0500327 }
328
329 return paint_node;
330}
331
Florin Malitafa7e9a82018-05-04 15:10:54 -0400332sk_sp<sksg::PaintNode> AttachStroke(const json::ValueRef& jstroke, AttachContext* ctx,
Florin Malita6aaee592018-01-12 12:25:09 -0500333 sk_sp<sksg::PaintNode> stroke_node) {
334 SkASSERT(jstroke.isObject());
335
336 if (!stroke_node)
337 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -0500338
339 stroke_node->setStyle(SkPaint::kStroke_Style);
340
Florin Malitafc807c82018-01-25 22:35:09 -0500341 auto width_attached = BindProperty<ScalarValue>(jstroke["w"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -0500342 [stroke_node](const ScalarValue& w) {
343 stroke_node->setStrokeWidth(w);
Florin Malitaf9590922018-01-09 11:56:09 -0500344 });
Florin Malita094ccde2017-12-30 12:27:00 -0500345 if (!width_attached)
346 return nullptr;
347
Florin Malitafa7e9a82018-05-04 15:10:54 -0400348 stroke_node->setStrokeMiter(jstroke["ml"].toDefault(4.0f));
Florin Malita094ccde2017-12-30 12:27:00 -0500349
350 static constexpr SkPaint::Join gJoins[] = {
351 SkPaint::kMiter_Join,
352 SkPaint::kRound_Join,
353 SkPaint::kBevel_Join,
354 };
Florin Malitafa7e9a82018-05-04 15:10:54 -0400355 stroke_node->setStrokeJoin(gJoins[SkTPin<int>(jstroke["lj"].toDefault<int>(1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500356 0, SK_ARRAY_COUNT(gJoins) - 1)]);
357
358 static constexpr SkPaint::Cap gCaps[] = {
359 SkPaint::kButt_Cap,
360 SkPaint::kRound_Cap,
361 SkPaint::kSquare_Cap,
362 };
Florin Malitafa7e9a82018-05-04 15:10:54 -0400363 stroke_node->setStrokeCap(gCaps[SkTPin<int>(jstroke["lc"].toDefault<int>(1) - 1,
Florin Malita094ccde2017-12-30 12:27:00 -0500364 0, SK_ARRAY_COUNT(gCaps) - 1)]);
365
366 return stroke_node;
367}
368
Florin Malitafa7e9a82018-05-04 15:10:54 -0400369sk_sp<sksg::PaintNode> AttachColorFill(const json::ValueRef& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500370 SkASSERT(jfill.isObject());
371
372 return AttachPaint(jfill, ctx, AttachColor(jfill, ctx));
373}
374
Florin Malitafa7e9a82018-05-04 15:10:54 -0400375sk_sp<sksg::PaintNode> AttachGradientFill(const json::ValueRef& jfill, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500376 SkASSERT(jfill.isObject());
377
378 return AttachPaint(jfill, ctx, AttachGradient(jfill, ctx));
379}
380
Florin Malitafa7e9a82018-05-04 15:10:54 -0400381sk_sp<sksg::PaintNode> AttachColorStroke(const json::ValueRef& jstroke, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500382 SkASSERT(jstroke.isObject());
383
384 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachColor(jstroke, ctx)));
385}
386
Florin Malitafa7e9a82018-05-04 15:10:54 -0400387sk_sp<sksg::PaintNode> AttachGradientStroke(const json::ValueRef& jstroke, AttachContext* ctx) {
Florin Malita6aaee592018-01-12 12:25:09 -0500388 SkASSERT(jstroke.isObject());
389
390 return AttachStroke(jstroke, ctx, AttachPaint(jstroke, ctx, AttachGradient(jstroke, ctx)));
391}
392
Florin Malitae6345d92018-01-03 23:37:54 -0500393std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400394 const json::ValueRef& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malitae6345d92018-01-03 23:37:54 -0500395 std::vector<sk_sp<sksg::GeometryNode>> merged;
396
397 static constexpr sksg::Merge::Mode gModes[] = {
398 sksg::Merge::Mode::kMerge, // "mm": 1
399 sksg::Merge::Mode::kUnion, // "mm": 2
400 sksg::Merge::Mode::kDifference, // "mm": 3
401 sksg::Merge::Mode::kIntersect, // "mm": 4
402 sksg::Merge::Mode::kXOR , // "mm": 5
403 };
404
Florin Malitafa7e9a82018-05-04 15:10:54 -0400405 const auto mode = gModes[SkTPin<int>(jmerge["mm"].toDefault(1) - 1,
Florin Malita51b8c892018-01-07 08:54:24 -0500406 0, SK_ARRAY_COUNT(gModes) - 1)];
Florin Malitae6345d92018-01-03 23:37:54 -0500407 merged.push_back(sksg::Merge::Make(std::move(geos), mode));
408
Florin Malitae6345d92018-01-03 23:37:54 -0500409 return merged;
410}
411
Florin Malita51b8c892018-01-07 08:54:24 -0500412std::vector<sk_sp<sksg::GeometryNode>> AttachTrimGeometryEffect(
Florin Malitafa7e9a82018-05-04 15:10:54 -0400413 const json::ValueRef& jtrim, AttachContext* ctx, 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 Malitafa7e9a82018-05-04 15:10:54 -0400420 const auto mode = gModes[SkTPin<int>(jtrim["m"].toDefault(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) {
425 inputs.push_back(sksg::Merge::Make(std::move(geos), sksg::Merge::Mode::kMerge));
426 } 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 Malitafa7e9a82018-05-04 15:10:54 -0400455 const json::ValueRef& jtrim, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
Florin Malita41dff6e2018-04-30 23:08:15 -0400456
457 std::vector<sk_sp<sksg::GeometryNode>> rounded;
458 rounded.reserve(geos.size());
459
460 for (const auto& g : geos) {
461 const auto roundEffect = sksg::RoundEffect::Make(std::move(g));
462 rounded.push_back(roundEffect);
463
464 BindProperty<ScalarValue>(jtrim["r"], &ctx->fAnimators,
465 [roundEffect](const ScalarValue& r) {
466 roundEffect->setRadius(r);
467 });
468 }
469
470 return rounded;
471}
472
Florin Malitafa7e9a82018-05-04 15:10:54 -0400473using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const json::ValueRef&, AttachContext*);
Florin Malita094ccde2017-12-30 12:27:00 -0500474static constexpr GeometryAttacherT gGeometryAttachers[] = {
475 AttachPathGeometry,
Florin Malita2e1d7e22018-01-02 10:40:00 -0500476 AttachRRectGeometry,
Florin Malitafbc13f12018-01-04 10:26:35 -0500477 AttachEllipseGeometry,
Florin Malita02a32b02018-01-04 11:27:09 -0500478 AttachPolystarGeometry,
Florin Malita094ccde2017-12-30 12:27:00 -0500479};
480
Florin Malitafa7e9a82018-05-04 15:10:54 -0400481using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const json::ValueRef&, AttachContext*);
Florin Malita094ccde2017-12-30 12:27:00 -0500482static constexpr PaintAttacherT gPaintAttachers[] = {
Florin Malita6aaee592018-01-12 12:25:09 -0500483 AttachColorFill,
484 AttachColorStroke,
485 AttachGradientFill,
486 AttachGradientStroke,
Florin Malita094ccde2017-12-30 12:27:00 -0500487};
488
Florin Malitae6345d92018-01-03 23:37:54 -0500489using GeometryEffectAttacherT =
Florin Malitafa7e9a82018-05-04 15:10:54 -0400490 std::vector<sk_sp<sksg::GeometryNode>> (*)(const json::ValueRef&,
Florin Malitae6345d92018-01-03 23:37:54 -0500491 AttachContext*,
492 std::vector<sk_sp<sksg::GeometryNode>>&&);
493static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
494 AttachMergeGeometryEffect,
Florin Malita51b8c892018-01-07 08:54:24 -0500495 AttachTrimGeometryEffect,
Florin Malita41dff6e2018-04-30 23:08:15 -0400496 AttachRoundGeometryEffect,
Florin Malitae6345d92018-01-03 23:37:54 -0500497};
498
Florin Malita094ccde2017-12-30 12:27:00 -0500499enum class ShapeType {
500 kGeometry,
Florin Malitae6345d92018-01-03 23:37:54 -0500501 kGeometryEffect,
Florin Malita094ccde2017-12-30 12:27:00 -0500502 kPaint,
503 kGroup,
Florin Malitadacc02b2017-12-31 09:12:31 -0500504 kTransform,
Florin Malita094ccde2017-12-30 12:27:00 -0500505};
506
507struct ShapeInfo {
508 const char* fTypeString;
509 ShapeType fShapeType;
510 uint32_t fAttacherIndex; // index into respective attacher tables
511};
512
Florin Malitafa7e9a82018-05-04 15:10:54 -0400513const ShapeInfo* FindShapeInfo(const json::ValueRef& shape) {
Florin Malita094ccde2017-12-30 12:27:00 -0500514 static constexpr ShapeInfo gShapeInfo[] = {
Florin Malitafbc13f12018-01-04 10:26:35 -0500515 { "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500516 { "fl", ShapeType::kPaint , 0 }, // fill -> AttachColorFill
517 { "gf", ShapeType::kPaint , 2 }, // gfill -> AttachGradientFill
Florin Malita16d0ad02018-01-19 15:07:29 -0500518 { "gr", ShapeType::kGroup , 0 }, // group -> Inline handler
Florin Malita6aaee592018-01-12 12:25:09 -0500519 { "gs", ShapeType::kPaint , 3 }, // gstroke -> AttachGradientStroke
Florin Malitae6345d92018-01-03 23:37:54 -0500520 { "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
Florin Malita02a32b02018-01-04 11:27:09 -0500521 { "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
Florin Malita41dff6e2018-04-30 23:08:15 -0400522 { "rd", ShapeType::kGeometryEffect, 2 }, // round -> AttachRoundGeometryEffect
Florin Malitae6345d92018-01-03 23:37:54 -0500523 { "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
Florin Malita02a32b02018-01-04 11:27:09 -0500524 { "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
Florin Malita6aaee592018-01-12 12:25:09 -0500525 { "st", ShapeType::kPaint , 1 }, // stroke -> AttachColorStroke
Florin Malita51b8c892018-01-07 08:54:24 -0500526 { "tm", ShapeType::kGeometryEffect, 1 }, // trim -> AttachTrimGeometryEffect
Florin Malita16d0ad02018-01-19 15:07:29 -0500527 { "tr", ShapeType::kTransform , 0 }, // transform -> Inline handler
Florin Malita094ccde2017-12-30 12:27:00 -0500528 };
529
Florin Malita20880782018-05-09 11:35:00 -0400530 SkString type;
531 if (!shape["ty"].to(&type) || type.isEmpty())
Florin Malita094ccde2017-12-30 12:27:00 -0500532 return nullptr;
533
Florin Malitafa7e9a82018-05-04 15:10:54 -0400534 const auto* info = bsearch(type.c_str(),
Florin Malita094ccde2017-12-30 12:27:00 -0500535 gShapeInfo,
536 SK_ARRAY_COUNT(gShapeInfo),
537 sizeof(ShapeInfo),
538 [](const void* key, const void* info) {
539 return strcmp(static_cast<const char*>(key),
540 static_cast<const ShapeInfo*>(info)->fTypeString);
541 });
542
543 return static_cast<const ShapeInfo*>(info);
544}
545
Florin Malita16d0ad02018-01-19 15:07:29 -0500546struct GeometryEffectRec {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400547 const json::ValueRef fJson;
Florin Malita16d0ad02018-01-19 15:07:29 -0500548 GeometryEffectAttacherT fAttach;
549};
550
Florin Malitaca4439f2018-01-23 10:31:59 -0500551struct AttachShapeContext {
552 AttachShapeContext(AttachContext* ctx,
553 std::vector<sk_sp<sksg::GeometryNode>>* geos,
554 std::vector<GeometryEffectRec>* effects,
555 size_t committedAnimators)
556 : fCtx(ctx)
557 , fGeometryStack(geos)
558 , fGeometryEffectStack(effects)
559 , fCommittedAnimators(committedAnimators) {}
560
561 AttachContext* fCtx;
562 std::vector<sk_sp<sksg::GeometryNode>>* fGeometryStack;
563 std::vector<GeometryEffectRec>* fGeometryEffectStack;
564 size_t fCommittedAnimators;
565};
566
Florin Malitafa7e9a82018-05-04 15:10:54 -0400567sk_sp<sksg::RenderNode> AttachShape(const json::ValueRef& jshape, AttachShapeContext* shapeCtx) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500568 if (!jshape.isArray())
Florin Malita094ccde2017-12-30 12:27:00 -0500569 return nullptr;
570
Florin Malitaca4439f2018-01-23 10:31:59 -0500571 SkDEBUGCODE(const auto initialGeometryEffects = shapeCtx->fGeometryEffectStack->size();)
Florin Malita094ccde2017-12-30 12:27:00 -0500572
Florin Malita16d0ad02018-01-19 15:07:29 -0500573 sk_sp<sksg::Group> shape_group = sksg::Group::Make();
574 sk_sp<sksg::RenderNode> shape_wrapper = shape_group;
575 sk_sp<sksg::Matrix> shape_matrix;
Florin Malita094ccde2017-12-30 12:27:00 -0500576
Florin Malita16d0ad02018-01-19 15:07:29 -0500577 struct ShapeRec {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400578 const json::ValueRef fJson;
Florin Malita16d0ad02018-01-19 15:07:29 -0500579 const ShapeInfo& fInfo;
580 };
581
582 // First pass (bottom->top):
583 //
584 // * pick up the group transform and opacity
585 // * push local geometry effects onto the stack
586 // * store recs for next pass
587 //
588 std::vector<ShapeRec> recs;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400589 for (size_t i = 0; i < jshape.size(); ++i) {
590 const auto s = jshape[jshape.size() - 1 - i];
Florin Malita094ccde2017-12-30 12:27:00 -0500591 const auto* info = FindShapeInfo(s);
592 if (!info) {
Florin Malita20880782018-05-09 11:35:00 -0400593 LogFail(s["ty"], "Unknown shape");
Florin Malita094ccde2017-12-30 12:27:00 -0500594 continue;
595 }
596
Florin Malita16d0ad02018-01-19 15:07:29 -0500597 recs.push_back({ s, *info });
598
Florin Malita094ccde2017-12-30 12:27:00 -0500599 switch (info->fShapeType) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500600 case ShapeType::kTransform:
Florin Malitaca4439f2018-01-23 10:31:59 -0500601 if ((shape_matrix = AttachMatrix(s, shapeCtx->fCtx, nullptr))) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500602 shape_wrapper = sksg::Transform::Make(std::move(shape_wrapper), shape_matrix);
603 }
Florin Malitaca4439f2018-01-23 10:31:59 -0500604 shape_wrapper = AttachOpacity(s, shapeCtx->fCtx, std::move(shape_wrapper));
Florin Malita16d0ad02018-01-19 15:07:29 -0500605 break;
606 case ShapeType::kGeometryEffect:
607 SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500608 shapeCtx->fGeometryEffectStack->push_back(
Florin Malita16d0ad02018-01-19 15:07:29 -0500609 { s, gGeometryEffectAttachers[info->fAttacherIndex] });
610 break;
611 default:
612 break;
613 }
614 }
615
616 // Second pass (top -> bottom, after 2x reverse):
617 //
618 // * track local geometry
619 // * emit local paints
620 //
621 std::vector<sk_sp<sksg::GeometryNode>> geos;
622 std::vector<sk_sp<sksg::RenderNode >> draws;
623 for (auto rec = recs.rbegin(); rec != recs.rend(); ++rec) {
624 switch (rec->fInfo.fShapeType) {
Florin Malita094ccde2017-12-30 12:27:00 -0500625 case ShapeType::kGeometry: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500626 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500627 if (auto geo = gGeometryAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
628 shapeCtx->fCtx)) {
Florin Malita094ccde2017-12-30 12:27:00 -0500629 geos.push_back(std::move(geo));
630 }
631 } break;
Florin Malitae6345d92018-01-03 23:37:54 -0500632 case ShapeType::kGeometryEffect: {
Florin Malita16d0ad02018-01-19 15:07:29 -0500633 // Apply the current effect and pop from the stack.
634 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
Florin Malitaee6de4b2018-01-21 11:13:17 -0500635 if (!geos.empty()) {
636 geos = gGeometryEffectAttachers[rec->fInfo.fAttacherIndex](rec->fJson,
Florin Malitaca4439f2018-01-23 10:31:59 -0500637 shapeCtx->fCtx,
Florin Malitaee6de4b2018-01-21 11:13:17 -0500638 std::move(geos));
639 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500640
Florin Malitaca4439f2018-01-23 10:31:59 -0500641 SkASSERT(shapeCtx->fGeometryEffectStack->back().fJson == rec->fJson);
642 SkASSERT(shapeCtx->fGeometryEffectStack->back().fAttach ==
Florin Malita16d0ad02018-01-19 15:07:29 -0500643 gGeometryEffectAttachers[rec->fInfo.fAttacherIndex]);
Florin Malitaca4439f2018-01-23 10:31:59 -0500644 shapeCtx->fGeometryEffectStack->pop_back();
Florin Malita094ccde2017-12-30 12:27:00 -0500645 } break;
646 case ShapeType::kGroup: {
Florin Malitaca4439f2018-01-23 10:31:59 -0500647 AttachShapeContext groupShapeCtx(shapeCtx->fCtx,
648 &geos,
649 shapeCtx->fGeometryEffectStack,
650 shapeCtx->fCommittedAnimators);
651 if (auto subgroup = AttachShape(rec->fJson["it"], &groupShapeCtx)) {
Florin Malita16d0ad02018-01-19 15:07:29 -0500652 draws.push_back(std::move(subgroup));
Florin Malitaca4439f2018-01-23 10:31:59 -0500653 SkASSERT(groupShapeCtx.fCommittedAnimators >= shapeCtx->fCommittedAnimators);
654 shapeCtx->fCommittedAnimators = groupShapeCtx.fCommittedAnimators;
Florin Malita094ccde2017-12-30 12:27:00 -0500655 }
656 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500657 case ShapeType::kPaint: {
658 SkASSERT(rec->fInfo.fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
Florin Malitaca4439f2018-01-23 10:31:59 -0500659 auto paint = gPaintAttachers[rec->fInfo.fAttacherIndex](rec->fJson, shapeCtx->fCtx);
Florin Malita16d0ad02018-01-19 15:07:29 -0500660 if (!paint || geos.empty())
661 break;
662
663 auto drawGeos = geos;
664
665 // Apply all pending effects from the stack.
Florin Malitaca4439f2018-01-23 10:31:59 -0500666 for (auto it = shapeCtx->fGeometryEffectStack->rbegin();
667 it != shapeCtx->fGeometryEffectStack->rend(); ++it) {
668 drawGeos = it->fAttach(it->fJson, shapeCtx->fCtx, std::move(drawGeos));
Florin Malita18eafd92018-01-04 21:11:55 -0500669 }
Florin Malita16d0ad02018-01-19 15:07:29 -0500670
671 // If we still have multiple geos, reduce using 'merge'.
672 auto geo = drawGeos.size() > 1
673 ? sksg::Merge::Make(std::move(drawGeos), sksg::Merge::Mode::kMerge)
674 : drawGeos[0];
675
676 SkASSERT(geo);
677 draws.push_back(sksg::Draw::Make(std::move(geo), std::move(paint)));
Florin Malitaca4439f2018-01-23 10:31:59 -0500678 shapeCtx->fCommittedAnimators = shapeCtx->fCtx->fAnimators.size();
Florin Malitadacc02b2017-12-31 09:12:31 -0500679 } break;
Florin Malita16d0ad02018-01-19 15:07:29 -0500680 default:
681 break;
Florin Malita094ccde2017-12-30 12:27:00 -0500682 }
683 }
684
Florin Malita16d0ad02018-01-19 15:07:29 -0500685 // By now we should have popped all local geometry effects.
Florin Malitaca4439f2018-01-23 10:31:59 -0500686 SkASSERT(shapeCtx->fGeometryEffectStack->size() == initialGeometryEffects);
Florin Malita16d0ad02018-01-19 15:07:29 -0500687
688 // Push transformed local geometries to parent list, for subsequent paints.
689 for (const auto& geo : geos) {
Florin Malitaca4439f2018-01-23 10:31:59 -0500690 shapeCtx->fGeometryStack->push_back(shape_matrix
Florin Malita16d0ad02018-01-19 15:07:29 -0500691 ? sksg::GeometryTransform::Make(std::move(geo), shape_matrix)
692 : std::move(geo));
Florin Malita094ccde2017-12-30 12:27:00 -0500693 }
694
Florin Malita16d0ad02018-01-19 15:07:29 -0500695 // Emit local draws reversed (bottom->top, per spec).
696 for (auto it = draws.rbegin(); it != draws.rend(); ++it) {
697 shape_group->addChild(std::move(*it));
Florin Malita2a8275b2018-01-02 12:52:43 -0500698 }
699
Florin Malita16d0ad02018-01-19 15:07:29 -0500700 return draws.empty() ? nullptr : shape_wrapper;
Florin Malita094ccde2017-12-30 12:27:00 -0500701}
702
Florin Malita1022f742018-02-23 11:10:22 -0500703sk_sp<sksg::RenderNode> AttachNestedAnimation(const char* path, AttachContext* ctx) {
704 class SkottieSGAdapter final : public sksg::RenderNode {
705 public:
706 explicit SkottieSGAdapter(sk_sp<Animation> animation)
707 : fAnimation(std::move(animation)) {
708 SkASSERT(fAnimation);
709 }
710
711 protected:
712 SkRect onRevalidate(sksg::InvalidationController*, const SkMatrix&) override {
713 return SkRect::MakeSize(fAnimation->size());
714 }
715
716 void onRender(SkCanvas* canvas) const override {
717 fAnimation->render(canvas);
718 }
719
720 private:
721 const sk_sp<Animation> fAnimation;
722 };
723
724 class SkottieAnimatorAdapter final : public sksg::Animator {
725 public:
726 SkottieAnimatorAdapter(sk_sp<Animation> animation, float frameRate)
727 : fAnimation(std::move(animation))
728 , fFrameRate(frameRate) {
729 SkASSERT(fAnimation);
730 SkASSERT(fFrameRate > 0);
731 }
732
733 protected:
734 void onTick(float t) {
735 // map back from frame # to ms.
736 const auto t_ms = t * 1000 / fFrameRate;
737 fAnimation->animationTick(t_ms);
738 }
739
740 private:
741 const sk_sp<Animation> fAnimation;
742 const float fFrameRate;
743 };
744
745 const auto resStream = ctx->fResources.openStream(path);
746 if (!resStream || !resStream->hasLength()) {
747 LOG("!! Could not open: %s\n", path);
748 return nullptr;
749 }
750
751 auto animation = Animation::Make(resStream.get(), ctx->fResources);
752 if (!animation) {
753 LOG("!! Could not load nested animation: %s\n", path);
754 return nullptr;
755 }
756
757 ctx->fAnimators.push_back(skstd::make_unique<SkottieAnimatorAdapter>(animation,
758 ctx->fFrameRate));
759
760 return sk_make_sp<SkottieSGAdapter>(std::move(animation));
761}
762
Florin Malitafa7e9a82018-05-04 15:10:54 -0400763sk_sp<sksg::RenderNode> AttachCompLayer(const json::ValueRef& jlayer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500764 float* time_bias, float* time_scale) {
765 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500766
Florin Malitacf8ed522018-01-25 15:27:33 -0500767 SkString refId;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400768 if (!jlayer["refId"].to(&refId) || refId.isEmpty()) {
Florin Malitadd22cf962018-01-29 15:42:01 +0000769 LOG("!! Comp layer missing refId\n");
Florin Malita094ccde2017-12-30 12:27:00 -0500770 return nullptr;
771 }
772
Florin Malitafa7e9a82018-05-04 15:10:54 -0400773 const auto start_time = jlayer["st"].toDefault(0.0f),
774 stretch_time = jlayer["sr"].toDefault(1.0f);
Florin Malitaeb87d672018-01-29 15:28:24 -0500775
776 *time_bias = -start_time;
777 *time_scale = 1 / stretch_time;
778 if (SkScalarIsNaN(*time_scale)) {
779 *time_scale = 1;
780 }
781
Florin Malita1022f742018-02-23 11:10:22 -0500782 if (refId.startsWith("$")) {
783 return AttachNestedAnimation(refId.c_str() + 1, ctx);
784 }
785
786 const auto* comp = ctx->fAssets.find(refId);
787 if (!comp) {
788 LOG("!! Pre-comp not found: '%s'\n", refId.c_str());
789 return nullptr;
790 }
791
Florin Malitadd22cf962018-01-29 15:42:01 +0000792 // TODO: cycle detection
Florin Malitafa7e9a82018-05-04 15:10:54 -0400793 return AttachComposition(*comp, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500794}
795
Florin Malitafa7e9a82018-05-04 15:10:54 -0400796sk_sp<sksg::RenderNode> AttachSolidLayer(const json::ValueRef& jlayer, AttachContext*,
Florin Malitaeb87d672018-01-29 15:28:24 -0500797 float*, float*) {
Florin Malita0e66fba2018-01-09 17:10:18 -0500798 SkASSERT(jlayer.isObject());
Florin Malita094ccde2017-12-30 12:27:00 -0500799
Florin Malitafa7e9a82018-05-04 15:10:54 -0400800 const auto size = SkSize::Make(jlayer["sw"].toDefault(0.0f),
801 jlayer["sh"].toDefault(0.0f));
802 const auto hex = jlayer["sc"].toDefault(SkString());
Florin Malita0e66fba2018-01-09 17:10:18 -0500803 uint32_t c;
804 if (size.isEmpty() ||
805 !hex.startsWith("#") ||
806 !SkParse::FindHex(hex.c_str() + 1, &c)) {
807 LogFail(jlayer, "Could not parse solid layer");
808 return nullptr;
809 }
810
811 const SkColor color = 0xff000000 | c;
812
813 return sksg::Draw::Make(sksg::Rect::Make(SkRect::MakeSize(size)),
814 sksg::Color::Make(color));
Florin Malita094ccde2017-12-30 12:27:00 -0500815}
816
Florin Malitafa7e9a82018-05-04 15:10:54 -0400817sk_sp<sksg::RenderNode> AttachImageAsset(const json::ValueRef& jimage, AttachContext* ctx) {
Florin Malita49328072018-01-08 12:51:12 -0500818 SkASSERT(jimage.isObject());
819
Florin Malitafa7e9a82018-05-04 15:10:54 -0400820 const auto name = jimage["p"].toDefault(SkString()),
821 path = jimage["u"].toDefault(SkString());
Florin Malita49328072018-01-08 12:51:12 -0500822 if (name.isEmpty())
823 return nullptr;
824
825 // TODO: plumb resource paths explicitly to ResourceProvider?
826 const auto resName = path.isEmpty() ? name : SkOSPath::Join(path.c_str(), name.c_str());
827 const auto resStream = ctx->fResources.openStream(resName.c_str());
828 if (!resStream || !resStream->hasLength()) {
829 LOG("!! Could not load image resource: %s\n", resName.c_str());
830 return nullptr;
831 }
832
833 // TODO: non-intrisic image sizing
834 return sksg::Image::Make(
835 SkImage::MakeFromEncoded(SkData::MakeFromStream(resStream.get(), resStream->getLength())));
836}
837
Florin Malitafa7e9a82018-05-04 15:10:54 -0400838sk_sp<sksg::RenderNode> AttachImageLayer(const json::ValueRef& layer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500839 float*, float*) {
Florin Malitadd22cf962018-01-29 15:42:01 +0000840 SkASSERT(layer.isObject());
841
842 SkString refId;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400843 if (!layer["refId"].to(&refId) || refId.isEmpty()) {
Florin Malitadd22cf962018-01-29 15:42:01 +0000844 LOG("!! Image layer missing refId\n");
845 return nullptr;
846 }
847
848 const auto* jimage = ctx->fAssets.find(refId);
849 if (!jimage) {
850 LOG("!! Image asset not found: '%s'\n", refId.c_str());
851 return nullptr;
852 }
853
Florin Malitafa7e9a82018-05-04 15:10:54 -0400854 return AttachImageAsset(*jimage, ctx);
Florin Malita094ccde2017-12-30 12:27:00 -0500855}
856
Florin Malitafa7e9a82018-05-04 15:10:54 -0400857sk_sp<sksg::RenderNode> AttachNullLayer(const json::ValueRef& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500858 SkASSERT(layer.isObject());
859
Florin Malita18eafd92018-01-04 21:11:55 -0500860 // Null layers are used solely to drive dependent transforms,
861 // but we use free-floating sksg::Matrices for that purpose.
Florin Malita094ccde2017-12-30 12:27:00 -0500862 return nullptr;
863}
864
Florin Malitafa7e9a82018-05-04 15:10:54 -0400865sk_sp<sksg::RenderNode> AttachShapeLayer(const json::ValueRef& layer, AttachContext* ctx,
Florin Malitaeb87d672018-01-29 15:28:24 -0500866 float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500867 SkASSERT(layer.isObject());
868
Florin Malita16d0ad02018-01-19 15:07:29 -0500869 std::vector<sk_sp<sksg::GeometryNode>> geometryStack;
870 std::vector<GeometryEffectRec> geometryEffectStack;
Florin Malitaca4439f2018-01-23 10:31:59 -0500871 AttachShapeContext shapeCtx(ctx, &geometryStack, &geometryEffectStack, ctx->fAnimators.size());
872 auto shapeNode = AttachShape(layer["shapes"], &shapeCtx);
873
874 // Trim uncommitted animators: AttachShape consumes effects on the fly, and greedily attaches
875 // geometries => at the end, we can end up with unused geometries, which are nevertheless alive
876 // due to attached animators. To avoid this, we track committed animators and discard the
877 // orphans here.
878 SkASSERT(shapeCtx.fCommittedAnimators <= ctx->fAnimators.size());
879 ctx->fAnimators.resize(shapeCtx.fCommittedAnimators);
880
881 return shapeNode;
Florin Malita094ccde2017-12-30 12:27:00 -0500882}
883
Florin Malitafa7e9a82018-05-04 15:10:54 -0400884sk_sp<sksg::RenderNode> AttachTextLayer(const json::ValueRef& layer, AttachContext*, float*, float*) {
Florin Malita094ccde2017-12-30 12:27:00 -0500885 SkASSERT(layer.isObject());
886
887 LOG("?? Text layer stub\n");
888 return nullptr;
889}
890
Florin Malita18eafd92018-01-04 21:11:55 -0500891struct AttachLayerContext {
Florin Malitafa7e9a82018-05-04 15:10:54 -0400892 AttachLayerContext(const json::ValueRef& jlayers, AttachContext* ctx)
Florin Malita18eafd92018-01-04 21:11:55 -0500893 : fLayerList(jlayers), fCtx(ctx) {}
894
Florin Malitafa7e9a82018-05-04 15:10:54 -0400895 const json::ValueRef fLayerList;
Florin Malita4a490682018-01-28 14:27:51 -0500896 AttachContext* fCtx;
897 SkTHashMap<int, sk_sp<sksg::Matrix>> fLayerMatrixMap;
898 sk_sp<sksg::RenderNode> fCurrentMatte;
Florin Malita18eafd92018-01-04 21:11:55 -0500899
Florin Malitafa7e9a82018-05-04 15:10:54 -0400900 sk_sp<sksg::Matrix> AttachParentLayerMatrix(const json::ValueRef& jlayer) {
Florin Malita4a490682018-01-28 14:27:51 -0500901 SkASSERT(jlayer.isObject());
Florin Malita18eafd92018-01-04 21:11:55 -0500902 SkASSERT(fLayerList.isArray());
903
Florin Malitafa7e9a82018-05-04 15:10:54 -0400904 const auto parent_index = jlayer["parent"].toDefault<int>(-1);
Florin Malita4a490682018-01-28 14:27:51 -0500905 if (parent_index < 0)
Florin Malita18eafd92018-01-04 21:11:55 -0500906 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500907
Florin Malita4a490682018-01-28 14:27:51 -0500908 if (auto* m = fLayerMatrixMap.find(parent_index))
909 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500910
Florin Malitafa7e9a82018-05-04 15:10:54 -0400911 sk_sp<sksg::Matrix> matrix;
912 for (const json::ValueRef l : fLayerList) {
913 if (l["ind"].toDefault<int>(-1) == parent_index) {
914 matrix = this->AttachLayerMatrix(l);
915 break;
Florin Malita18eafd92018-01-04 21:11:55 -0500916 }
917 }
918
919 return nullptr;
920 }
921
Florin Malitafa7e9a82018-05-04 15:10:54 -0400922 sk_sp<sksg::Matrix> AttachLayerMatrix(const json::ValueRef& jlayer) {
Florin Malita18eafd92018-01-04 21:11:55 -0500923 SkASSERT(jlayer.isObject());
924
Florin Malitafa7e9a82018-05-04 15:10:54 -0400925 const auto layer_index = jlayer["ind"].toDefault<int>(-1);
Florin Malita4a490682018-01-28 14:27:51 -0500926 if (layer_index < 0)
927 return nullptr;
Florin Malita18eafd92018-01-04 21:11:55 -0500928
Florin Malita4a490682018-01-28 14:27:51 -0500929 if (auto* m = fLayerMatrixMap.find(layer_index))
930 return *m;
Florin Malita18eafd92018-01-04 21:11:55 -0500931
Florin Malita4a490682018-01-28 14:27:51 -0500932 // Add a stub entry to break recursion cycles.
933 fLayerMatrixMap.set(layer_index, nullptr);
Florin Malita18eafd92018-01-04 21:11:55 -0500934
Florin Malita4a490682018-01-28 14:27:51 -0500935 auto parent_matrix = this->AttachParentLayerMatrix(jlayer);
Florin Malita18eafd92018-01-04 21:11:55 -0500936
Florin Malita4a490682018-01-28 14:27:51 -0500937 return *fLayerMatrixMap.set(layer_index,
938 AttachMatrix(jlayer["ks"],
939 fCtx,
940 this->AttachParentLayerMatrix(jlayer)));
Florin Malita18eafd92018-01-04 21:11:55 -0500941 }
942};
943
Florin Malita25366fa2018-01-23 13:37:59 -0500944SkBlendMode MaskBlendMode(char mode) {
945 switch (mode) {
946 case 'a': return SkBlendMode::kSrcOver; // Additive
947 case 's': return SkBlendMode::kExclusion; // Subtract
948 case 'i': return SkBlendMode::kDstIn; // Intersect
949 case 'l': return SkBlendMode::kLighten; // Lighten
950 case 'd': return SkBlendMode::kDarken; // Darken
951 case 'f': return SkBlendMode::kDifference; // Difference
952 default: break;
953 }
954
955 return SkBlendMode::kSrcOver;
956}
957
Florin Malitafa7e9a82018-05-04 15:10:54 -0400958sk_sp<sksg::RenderNode> AttachMask(const json::ValueRef& jmask,
Florin Malita25366fa2018-01-23 13:37:59 -0500959 AttachContext* ctx,
960 sk_sp<sksg::RenderNode> childNode) {
961 if (!jmask.isArray())
962 return childNode;
963
Florin Malita0c51c212018-04-26 14:13:14 -0400964 struct MaskRecord {
965 sk_sp<sksg::Path> mask_path;
966 sk_sp<sksg::Color> mask_paint;
967 };
968
969 SkSTArray<4, MaskRecord, true> mask_stack;
970
971 bool opaque_mask = true;
Florin Malita25366fa2018-01-23 13:37:59 -0500972
Florin Malitafa7e9a82018-05-04 15:10:54 -0400973 for (const json::ValueRef m : jmask) {
Florin Malita25366fa2018-01-23 13:37:59 -0500974 if (!m.isObject())
975 continue;
976
Florin Malita25366fa2018-01-23 13:37:59 -0500977 auto mask_path = AttachPath(m["pt"], ctx);
978 if (!mask_path) {
979 LogFail(m, "Could not parse mask path");
980 continue;
981 }
982
Florin Malitafa7e9a82018-05-04 15:10:54 -0400983 mask_path->setFillType(m["inv"].toDefault(false)
Florin Malitacd05b192018-04-25 21:43:03 -0400984 ? SkPath::kInverseWinding_FillType
985 : SkPath::kWinding_FillType);
986
Florin Malitacf8ed522018-01-25 15:27:33 -0500987 SkString mode;
Florin Malitafa7e9a82018-05-04 15:10:54 -0400988 if (!m["mode"].to(&mode) ||
Florin Malitacf8ed522018-01-25 15:27:33 -0500989 mode.size() != 1 ||
990 !strcmp(mode.c_str(), "n")) { // "None" masks have no effect.
Florin Malita25366fa2018-01-23 13:37:59 -0500991 continue;
Florin Malitacf8ed522018-01-25 15:27:33 -0500992 }
Florin Malita25366fa2018-01-23 13:37:59 -0500993
994 auto mask_paint = sksg::Color::Make(SK_ColorBLACK);
Florin Malitaaa71c892018-01-30 09:27:01 -0500995 mask_paint->setAntiAlias(true);
Florin Malita25366fa2018-01-23 13:37:59 -0500996 mask_paint->setBlendMode(MaskBlendMode(mode.c_str()[0]));
Florin Malita0c51c212018-04-26 14:13:14 -0400997
998 const auto animator_count = ctx->fAnimators.size();
Florin Malitafc807c82018-01-25 22:35:09 -0500999 BindProperty<ScalarValue>(m["o"], &ctx->fAnimators,
Florin Malita2518a0a2018-01-24 18:29:00 -05001000 [mask_paint](const ScalarValue& o) { mask_paint->setOpacity(o * 0.01f); });
Florin Malita25366fa2018-01-23 13:37:59 -05001001
Florin Malita0c51c212018-04-26 14:13:14 -04001002 opaque_mask &= (animator_count == ctx->fAnimators.size() && mask_paint->getOpacity() >= 1);
1003
1004 mask_stack.push_back({mask_path, mask_paint});
Florin Malita25366fa2018-01-23 13:37:59 -05001005 }
1006
Florin Malita0c51c212018-04-26 14:13:14 -04001007 if (mask_stack.empty())
1008 return childNode;
1009
1010 if (mask_stack.count() == 1 && opaque_mask) {
1011 // Single opaque mask => clip path.
1012 return sksg::ClipEffect::Make(std::move(childNode),
1013 std::move(mask_stack.front().mask_path),
1014 true);
1015 }
1016
1017 auto mask_group = sksg::Group::Make();
1018 for (const auto& rec : mask_stack) {
1019 mask_group->addChild(sksg::Draw::Make(std::move(rec.mask_path),
1020 std::move(rec.mask_paint)));
1021
1022 }
1023
1024 return sksg::MaskEffect::Make(std::move(childNode), std::move(mask_group));
Florin Malita25366fa2018-01-23 13:37:59 -05001025}
1026
Florin Malitafa7e9a82018-05-04 15:10:54 -04001027sk_sp<sksg::RenderNode> AttachLayer(const json::ValueRef& jlayer, AttachLayerContext* layerCtx) {
Florin Malita18eafd92018-01-04 21:11:55 -05001028 if (!jlayer.isObject())
Florin Malita094ccde2017-12-30 12:27:00 -05001029 return nullptr;
1030
Florin Malitafa7e9a82018-05-04 15:10:54 -04001031 using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const json::ValueRef&, AttachContext*,
Florin Malitaeb87d672018-01-29 15:28:24 -05001032 float* time_bias, float* time_scale);
Florin Malita094ccde2017-12-30 12:27:00 -05001033 static constexpr LayerAttacher gLayerAttachers[] = {
1034 AttachCompLayer, // 'ty': 0
1035 AttachSolidLayer, // 'ty': 1
1036 AttachImageLayer, // 'ty': 2
1037 AttachNullLayer, // 'ty': 3
1038 AttachShapeLayer, // 'ty': 4
1039 AttachTextLayer, // 'ty': 5
1040 };
1041
Florin Malitafa7e9a82018-05-04 15:10:54 -04001042 int type = jlayer["ty"].toDefault<int>(-1);
Florin Malita094ccde2017-12-30 12:27:00 -05001043 if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
1044 return nullptr;
1045 }
1046
Florin Malitacca86f32018-01-29 10:49:49 -05001047 sksg::AnimatorList layer_animators;
Florin Malita1022f742018-02-23 11:10:22 -05001048 AttachContext local_ctx = { layerCtx->fCtx->fResources,
1049 layerCtx->fCtx->fAssets,
1050 layerCtx->fCtx->fFrameRate,
1051 layer_animators};
Florin Malitacca86f32018-01-29 10:49:49 -05001052
Florin Malitaeb87d672018-01-29 15:28:24 -05001053 // Layer attachers may adjust these.
1054 float time_bias = 0,
1055 time_scale = 1;
1056
Florin Malita71cba8f2018-01-09 08:07:14 -05001057 // Layer content.
Florin Malitaeb87d672018-01-29 15:28:24 -05001058 auto layer = gLayerAttachers[type](jlayer, &local_ctx, &time_bias, &time_scale);
Florin Malita38ea40e2018-01-29 16:31:14 -05001059
1060 // Clip layers with explicit dimensions.
Florin Malitafa7e9a82018-05-04 15:10:54 -04001061 float w = 0, h = 0;
1062 if (jlayer["w"].to(&w) && jlayer["h"].to(&h)) {
Florin Malita38ea40e2018-01-29 16:31:14 -05001063 layer = sksg::ClipEffect::Make(std::move(layer),
1064 sksg::Rect::Make(SkRect::MakeWH(w, h)),
1065 true);
1066 }
1067
Florin Malita25366fa2018-01-23 13:37:59 -05001068 // Optional layer mask.
Florin Malitacca86f32018-01-29 10:49:49 -05001069 layer = AttachMask(jlayer["masksProperties"], &local_ctx, std::move(layer));
Florin Malita38ea40e2018-01-29 16:31:14 -05001070
Florin Malita25366fa2018-01-23 13:37:59 -05001071 // Optional layer transform.
Florin Malita71cba8f2018-01-09 08:07:14 -05001072 if (auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer)) {
Florin Malita71cba8f2018-01-09 08:07:14 -05001073 layer = sksg::Transform::Make(std::move(layer), std::move(layerMatrix));
1074 }
Florin Malita38ea40e2018-01-29 16:31:14 -05001075
Florin Malita71cba8f2018-01-09 08:07:14 -05001076 // Optional layer opacity.
Florin Malitacca86f32018-01-29 10:49:49 -05001077 layer = AttachOpacity(jlayer["ks"], &local_ctx, std::move(layer));
Florin Malita18eafd92018-01-04 21:11:55 -05001078
Florin Malitaeb87d672018-01-29 15:28:24 -05001079 class LayerController final : public sksg::GroupAnimator {
Florin Malita71cba8f2018-01-09 08:07:14 -05001080 public:
Florin Malitaeb87d672018-01-29 15:28:24 -05001081 LayerController(sksg::AnimatorList&& layer_animators,
1082 sk_sp<sksg::OpacityEffect> controlNode,
1083 float in, float out,
1084 float time_bias, float time_scale)
Florin Malitacca86f32018-01-29 10:49:49 -05001085 : INHERITED(std::move(layer_animators))
1086 , fControlNode(std::move(controlNode))
Florin Malita71cba8f2018-01-09 08:07:14 -05001087 , fIn(in)
Florin Malitaeb87d672018-01-29 15:28:24 -05001088 , fOut(out)
1089 , fTimeBias(time_bias)
1090 , fTimeScale(time_scale) {}
Florin Malita71cba8f2018-01-09 08:07:14 -05001091
Florin Malita35efaa82018-01-22 12:57:06 -05001092 void onTick(float t) override {
Florin Malitacca86f32018-01-29 10:49:49 -05001093 const auto active = (t >= fIn && t <= fOut);
1094
Florin Malita71cba8f2018-01-09 08:07:14 -05001095 // Keep the layer fully transparent except for its [in..out] lifespan.
1096 // (note: opacity == 0 disables rendering, while opacity == 1 is a noop)
Florin Malitacca86f32018-01-29 10:49:49 -05001097 fControlNode->setOpacity(active ? 1 : 0);
1098
1099 // Dispatch ticks only while active.
1100 if (active)
Florin Malitaeb87d672018-01-29 15:28:24 -05001101 this->INHERITED::onTick((t + fTimeBias) * fTimeScale);
Florin Malita71cba8f2018-01-09 08:07:14 -05001102 }
1103
1104 private:
1105 const sk_sp<sksg::OpacityEffect> fControlNode;
1106 const float fIn,
Florin Malitaeb87d672018-01-29 15:28:24 -05001107 fOut,
1108 fTimeBias,
1109 fTimeScale;
Florin Malitacca86f32018-01-29 10:49:49 -05001110
1111 using INHERITED = sksg::GroupAnimator;
Florin Malita71cba8f2018-01-09 08:07:14 -05001112 };
1113
Florin Malitaeb87d672018-01-29 15:28:24 -05001114 auto controller_node = sksg::OpacityEffect::Make(std::move(layer));
Florin Malitafa7e9a82018-05-04 15:10:54 -04001115 const auto in = jlayer["ip"].toDefault(0.0f),
1116 out = jlayer["op"].toDefault(in);
Florin Malita71cba8f2018-01-09 08:07:14 -05001117
Florin Malitaeb87d672018-01-29 15:28:24 -05001118 if (!jlayer["tm"].isNull()) {
1119 LogFail(jlayer["tm"], "Unsupported time remapping");
1120 }
1121
1122 if (in >= out || !controller_node)
Florin Malita71cba8f2018-01-09 08:07:14 -05001123 return nullptr;
1124
Florin Malitacca86f32018-01-29 10:49:49 -05001125 layerCtx->fCtx->fAnimators.push_back(
Florin Malitaeb87d672018-01-29 15:28:24 -05001126 skstd::make_unique<LayerController>(std::move(layer_animators),
1127 controller_node,
1128 in,
1129 out,
1130 time_bias,
1131 time_scale));
Florin Malita71cba8f2018-01-09 08:07:14 -05001132
Florin Malitafa7e9a82018-05-04 15:10:54 -04001133 if (jlayer["td"].toDefault(false)) {
Florin Malita5f9102f2018-01-10 13:36:22 -05001134 // This layer is a matte. We apply it as a mask to the next layer.
Florin Malitaeb87d672018-01-29 15:28:24 -05001135 layerCtx->fCurrentMatte = std::move(controller_node);
Florin Malita5f9102f2018-01-10 13:36:22 -05001136 return nullptr;
1137 }
1138
1139 if (layerCtx->fCurrentMatte) {
1140 // There is a pending matte. Apply and reset.
Florin Malitaa016be92018-03-05 14:01:41 -05001141 static constexpr sksg::MaskEffect::Mode gMaskModes[] = {
1142 sksg::MaskEffect::Mode::kNormal, // tt: 1
1143 sksg::MaskEffect::Mode::kInvert, // tt: 2
1144 };
Florin Malitafa7e9a82018-05-04 15:10:54 -04001145 const auto matteType = jlayer["tt"].toDefault<int>(1) - 1;
Florin Malitaa016be92018-03-05 14:01:41 -05001146
1147 if (matteType >= 0 && matteType < SkTo<int>(SK_ARRAY_COUNT(gMaskModes))) {
1148 return sksg::MaskEffect::Make(std::move(controller_node),
1149 std::move(layerCtx->fCurrentMatte),
1150 gMaskModes[matteType]);
1151 }
1152 layerCtx->fCurrentMatte.reset();
Florin Malita5f9102f2018-01-10 13:36:22 -05001153 }
1154
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001155 return std::move(controller_node);
Florin Malita094ccde2017-12-30 12:27:00 -05001156}
1157
Florin Malitafa7e9a82018-05-04 15:10:54 -04001158sk_sp<sksg::RenderNode> AttachComposition(const json::ValueRef& comp, AttachContext* ctx) {
Florin Malita094ccde2017-12-30 12:27:00 -05001159 if (!comp.isObject())
1160 return nullptr;
1161
Florin Malitafa7e9a82018-05-04 15:10:54 -04001162 const auto jlayers = comp["layers"];
Florin Malita18eafd92018-01-04 21:11:55 -05001163 if (!jlayers.isArray())
1164 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001165
Florin Malita18eafd92018-01-04 21:11:55 -05001166 SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
1167 AttachLayerContext layerCtx(jlayers, ctx);
1168
Florin Malitafa7e9a82018-05-04 15:10:54 -04001169 for (const json::ValueRef l : jlayers) {
Florin Malita18eafd92018-01-04 21:11:55 -05001170 if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
Florin Malita2a8275b2018-01-02 12:52:43 -05001171 layers.push_back(std::move(layer_fragment));
Florin Malita094ccde2017-12-30 12:27:00 -05001172 }
1173 }
1174
Florin Malita2a8275b2018-01-02 12:52:43 -05001175 if (layers.empty()) {
1176 return nullptr;
1177 }
1178
1179 // Layers are painted in bottom->top order.
1180 auto comp_group = sksg::Group::Make();
1181 for (int i = layers.count() - 1; i >= 0; --i) {
1182 comp_group->addChild(std::move(layers[i]));
1183 }
1184
Kevin Lubickf7621cb2018-04-16 15:51:44 -04001185 return std::move(comp_group);
Florin Malita094ccde2017-12-30 12:27:00 -05001186}
1187
1188} // namespace
1189
Florin Malita6eb85a12018-04-30 10:32:18 -04001190sk_sp<Animation> Animation::Make(SkStream* stream, const ResourceProvider& res, Stats* stats) {
1191 Stats stats_storage;
1192 if (!stats)
1193 stats = &stats_storage;
1194 memset(stats, 0, sizeof(struct Stats));
1195
Florin Malita094ccde2017-12-30 12:27:00 -05001196 if (!stream->hasLength()) {
1197 // TODO: handle explicit buffering?
1198 LOG("!! cannot parse streaming content\n");
1199 return nullptr;
1200 }
1201
Florin Malitafa7e9a82018-05-04 15:10:54 -04001202 stats->fJsonSize = stream->getLength();
Florin Malita6eb85a12018-04-30 10:32:18 -04001203 const auto t0 = SkTime::GetMSecs();
1204
Florin Malitafa7e9a82018-05-04 15:10:54 -04001205 const json::Document doc(stream);
1206 const auto json = doc.root();
1207 if (!json.isObject())
1208 return nullptr;
Florin Malita094ccde2017-12-30 12:27:00 -05001209
Florin Malita6eb85a12018-04-30 10:32:18 -04001210 const auto t1 = SkTime::GetMSecs();
1211 stats->fJsonParseTimeMS = t1 - t0;
1212
Florin Malitafa7e9a82018-05-04 15:10:54 -04001213 const auto version = json["v"].toDefault(SkString());
1214 const auto size = SkSize::Make(json["w"].toDefault(0.0f),
1215 json["h"].toDefault(0.0f));
1216 const auto fps = json["fr"].toDefault(-1.0f);
Florin Malita094ccde2017-12-30 12:27:00 -05001217
Florin Malita1022f742018-02-23 11:10:22 -05001218 if (size.isEmpty() || version.isEmpty() || fps <= 0) {
Florin Malita094ccde2017-12-30 12:27:00 -05001219 LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
1220 version.c_str(), size.width(), size.height(), fps);
1221 return nullptr;
1222 }
1223
Florin Malita6eb85a12018-04-30 10:32:18 -04001224 const auto anim =
1225 sk_sp<Animation>(new Animation(res, std::move(version), size, fps, json, stats));
1226 const auto t2 = SkTime::GetMSecs();
1227 stats->fSceneParseTimeMS = t2 - t1;
1228 stats->fTotalLoadTimeMS = t2 - t0;
1229
1230 return anim;
Florin Malita094ccde2017-12-30 12:27:00 -05001231}
1232
Florin Malita6eb85a12018-04-30 10:32:18 -04001233sk_sp<Animation> Animation::MakeFromFile(const char path[], const ResourceProvider* res,
1234 Stats* stats) {
Florin Malita49328072018-01-08 12:51:12 -05001235 class DirectoryResourceProvider final : public ResourceProvider {
1236 public:
1237 explicit DirectoryResourceProvider(SkString dir) : fDir(std::move(dir)) {}
1238
1239 std::unique_ptr<SkStream> openStream(const char resource[]) const override {
1240 const auto resPath = SkOSPath::Join(fDir.c_str(), resource);
1241 return SkStream::MakeFromFile(resPath.c_str());
1242 }
1243
1244 private:
1245 const SkString fDir;
1246 };
1247
1248 const auto jsonStream = SkStream::MakeFromFile(path);
1249 if (!jsonStream)
1250 return nullptr;
1251
1252 std::unique_ptr<ResourceProvider> defaultProvider;
1253 if (!res) {
1254 defaultProvider = skstd::make_unique<DirectoryResourceProvider>(SkOSPath::Dirname(path));
1255 }
1256
Florin Malita6eb85a12018-04-30 10:32:18 -04001257 return Make(jsonStream.get(), res ? *res : *defaultProvider, stats);
Florin Malita49328072018-01-08 12:51:12 -05001258}
1259
1260Animation::Animation(const ResourceProvider& resources,
Florin Malitafa7e9a82018-05-04 15:10:54 -04001261 SkString version, const SkSize& size, SkScalar fps, const json::ValueRef& json,
Florin Malita6eb85a12018-04-30 10:32:18 -04001262 Stats* stats)
Florin Malita094ccde2017-12-30 12:27:00 -05001263 : fVersion(std::move(version))
1264 , fSize(size)
1265 , fFrameRate(fps)
Florin Malitafa7e9a82018-05-04 15:10:54 -04001266 , fInPoint(json["ip"].toDefault(0.0f))
1267 , fOutPoint(SkTMax(json["op"].toDefault(SK_ScalarMax), fInPoint)) {
Florin Malita094ccde2017-12-30 12:27:00 -05001268
1269 AssetMap assets;
Florin Malitafa7e9a82018-05-04 15:10:54 -04001270 for (const json::ValueRef asset : json["assets"]) {
1271 if (asset.isObject()) {
1272 assets.set(asset["id"].toDefault(SkString()), asset);
Florin Malita094ccde2017-12-30 12:27:00 -05001273 }
Florin Malita094ccde2017-12-30 12:27:00 -05001274 }
1275
Florin Malitacca86f32018-01-29 10:49:49 -05001276 sksg::AnimatorList animators;
Florin Malita1022f742018-02-23 11:10:22 -05001277 AttachContext ctx = { resources, assets, fFrameRate, animators };
Florin Malita35efaa82018-01-22 12:57:06 -05001278 auto root = AttachComposition(json, &ctx);
1279
Florin Malita6eb85a12018-04-30 10:32:18 -04001280 stats->fAnimatorCount = animators.size();
Florin Malita35efaa82018-01-22 12:57:06 -05001281
1282 fScene = sksg::Scene::Make(std::move(root), std::move(animators));
Florin Malita094ccde2017-12-30 12:27:00 -05001283
Florin Malitadb385732018-01-09 12:19:32 -05001284 // In case the client calls render before the first tick.
1285 this->animationTick(0);
Florin Malita094ccde2017-12-30 12:27:00 -05001286}
1287
1288Animation::~Animation() = default;
1289
Florin Malita35efaa82018-01-22 12:57:06 -05001290void Animation::setShowInval(bool show) {
1291 if (fScene) {
1292 fScene->setShowInval(show);
1293 }
1294}
1295
Mike Reed29859872018-01-08 08:25:27 -05001296void Animation::render(SkCanvas* canvas, const SkRect* dstR) const {
Florin Malita35efaa82018-01-22 12:57:06 -05001297 if (!fScene)
Florin Malita094ccde2017-12-30 12:27:00 -05001298 return;
1299
Mike Reed29859872018-01-08 08:25:27 -05001300 SkAutoCanvasRestore restore(canvas, true);
1301 const SkRect srcR = SkRect::MakeSize(this->size());
1302 if (dstR) {
1303 canvas->concat(SkMatrix::MakeRectToRect(srcR, *dstR, SkMatrix::kCenter_ScaleToFit));
1304 }
1305 canvas->clipRect(srcR);
Florin Malita35efaa82018-01-22 12:57:06 -05001306 fScene->render(canvas);
Florin Malita094ccde2017-12-30 12:27:00 -05001307}
1308
1309void Animation::animationTick(SkMSec ms) {
Florin Malita35efaa82018-01-22 12:57:06 -05001310 if (!fScene)
1311 return;
1312
Florin Malita094ccde2017-12-30 12:27:00 -05001313 // 't' in the BM model really means 'frame #'
1314 auto t = static_cast<float>(ms) * fFrameRate / 1000;
1315
1316 t = fInPoint + std::fmod(t, fOutPoint - fInPoint);
1317
Florin Malita35efaa82018-01-22 12:57:06 -05001318 fScene->animate(t);
Florin Malita094ccde2017-12-30 12:27:00 -05001319}
1320
Florin Malita54f65c42018-01-16 17:04:30 -05001321} // namespace skottie