2017-12-30 17:27:00 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Skotty.h"
|
|
|
|
|
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkottyAnimator.h"
|
|
|
|
#include "SkottyPriv.h"
|
|
|
|
#include "SkottyProperties.h"
|
|
|
|
#include "SkData.h"
|
|
|
|
#include "SkMakeUnique.h"
|
|
|
|
#include "SkPaint.h"
|
|
|
|
#include "SkPath.h"
|
|
|
|
#include "SkPoint.h"
|
|
|
|
#include "SkSGColor.h"
|
|
|
|
#include "SkSGDraw.h"
|
|
|
|
#include "SkSGInvalidationController.h"
|
|
|
|
#include "SkSGGroup.h"
|
2018-01-04 04:37:54 +00:00
|
|
|
#include "SkSGMerge.h"
|
2017-12-30 17:27:00 +00:00
|
|
|
#include "SkSGPath.h"
|
2018-01-02 15:40:00 +00:00
|
|
|
#include "SkSGRect.h"
|
2017-12-30 17:27:00 +00:00
|
|
|
#include "SkSGTransform.h"
|
|
|
|
#include "SkStream.h"
|
|
|
|
#include "SkTArray.h"
|
|
|
|
#include "SkTHash.h"
|
|
|
|
|
|
|
|
#include <cmath>
|
2018-01-05 02:11:55 +00:00
|
|
|
#include <unordered_map>
|
2018-01-04 04:37:54 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2017-12-30 17:27:00 +00:00
|
|
|
#include "stdlib.h"
|
|
|
|
|
|
|
|
namespace skotty {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
using AssetMap = SkTHashMap<SkString, const Json::Value*>;
|
|
|
|
|
|
|
|
struct AttachContext {
|
|
|
|
const AssetMap& fAssets;
|
|
|
|
SkTArray<std::unique_ptr<AnimatorBase>>& fAnimators;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool LogFail(const Json::Value& json, const char* msg) {
|
|
|
|
const auto dump = json.toStyledString();
|
|
|
|
LOG("!! %s: %s", msg, dump.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is the workhorse for binding properties: depending on whether the property is animated,
|
|
|
|
// it will either apply immediately or instantiate and attach a keyframe animator.
|
|
|
|
template <typename ValueT, typename AttrT, typename NodeT, typename ApplyFuncT>
|
|
|
|
bool AttachProperty(const Json::Value& jprop, AttachContext* ctx, const sk_sp<NodeT>& node,
|
|
|
|
ApplyFuncT&& apply) {
|
|
|
|
if (!jprop.isObject())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!ParseBool(jprop["a"], false)) {
|
|
|
|
// Static property.
|
|
|
|
ValueT val;
|
|
|
|
if (!ValueT::Parse(jprop["k"], &val)) {
|
|
|
|
return LogFail(jprop, "Could not parse static property");
|
|
|
|
}
|
|
|
|
|
|
|
|
apply(node, val.template as<AttrT>());
|
|
|
|
} else {
|
|
|
|
// Keyframe property.
|
|
|
|
using AnimatorT = Animator<ValueT, AttrT, NodeT>;
|
|
|
|
auto animator = AnimatorT::Make(jprop["k"], node, std::move(apply));
|
|
|
|
|
|
|
|
if (!animator) {
|
|
|
|
return LogFail(jprop, "Could not instantiate keyframe animator");
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx->fAnimators.push_back(std::move(animator));
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
sk_sp<sksg::Matrix> AttachMatrix(const Json::Value& t, AttachContext* ctx,
|
|
|
|
sk_sp<sksg::Matrix> parentMatrix) {
|
|
|
|
if (!t.isObject())
|
|
|
|
return nullptr;
|
2017-12-30 17:27:00 +00:00
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
auto matrix = sksg::Matrix::Make(SkMatrix::I(), std::move(parentMatrix));
|
|
|
|
auto composite = sk_make_sp<CompositeTransform>(matrix);
|
|
|
|
auto anchor_attached = AttachProperty<VectorValue, SkPoint>(t["a"], ctx, composite,
|
2017-12-30 17:27:00 +00:00
|
|
|
[](const sk_sp<CompositeTransform>& node, const SkPoint& a) {
|
|
|
|
node->setAnchorPoint(a);
|
|
|
|
});
|
2018-01-05 02:11:55 +00:00
|
|
|
auto position_attached = AttachProperty<VectorValue, SkPoint>(t["p"], ctx, composite,
|
2017-12-30 17:27:00 +00:00
|
|
|
[](const sk_sp<CompositeTransform>& node, const SkPoint& p) {
|
|
|
|
node->setPosition(p);
|
|
|
|
});
|
2018-01-05 02:11:55 +00:00
|
|
|
auto scale_attached = AttachProperty<VectorValue, SkVector>(t["s"], ctx, composite,
|
2017-12-30 17:27:00 +00:00
|
|
|
[](const sk_sp<CompositeTransform>& node, const SkVector& s) {
|
|
|
|
node->setScale(s);
|
|
|
|
});
|
2018-01-05 02:11:55 +00:00
|
|
|
auto rotation_attached = AttachProperty<ScalarValue, SkScalar>(t["r"], ctx, composite,
|
2017-12-30 17:27:00 +00:00
|
|
|
[](const sk_sp<CompositeTransform>& node, SkScalar r) {
|
|
|
|
node->setRotation(r);
|
|
|
|
});
|
2018-01-05 02:11:55 +00:00
|
|
|
auto skew_attached = AttachProperty<ScalarValue, SkScalar>(t["sk"], ctx, composite,
|
2017-12-30 17:27:00 +00:00
|
|
|
[](const sk_sp<CompositeTransform>& node, SkScalar sk) {
|
|
|
|
node->setSkew(sk);
|
|
|
|
});
|
2018-01-05 02:11:55 +00:00
|
|
|
auto skewaxis_attached = AttachProperty<ScalarValue, SkScalar>(t["sa"], ctx, composite,
|
2017-12-30 17:27:00 +00:00
|
|
|
[](const sk_sp<CompositeTransform>& node, SkScalar sa) {
|
|
|
|
node->setSkewAxis(sa);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!anchor_attached &&
|
|
|
|
!position_attached &&
|
|
|
|
!scale_attached &&
|
|
|
|
!rotation_attached &&
|
|
|
|
!skew_attached &&
|
|
|
|
!skewaxis_attached) {
|
|
|
|
LogFail(t, "Could not parse transform");
|
2018-01-05 02:11:55 +00:00
|
|
|
return nullptr;
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
return matrix;
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachShape(const Json::Value&, AttachContext* ctx);
|
|
|
|
sk_sp<sksg::RenderNode> AttachComposition(const Json::Value&, AttachContext* ctx);
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachShapeGroup(const Json::Value& jgroup, AttachContext* ctx) {
|
|
|
|
SkASSERT(jgroup.isObject());
|
|
|
|
|
|
|
|
return AttachShape(jgroup["it"], ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::GeometryNode> AttachPathGeometry(const Json::Value& jpath, AttachContext* ctx) {
|
|
|
|
SkASSERT(jpath.isObject());
|
|
|
|
|
|
|
|
auto path_node = sksg::Path::Make();
|
|
|
|
auto path_attached = AttachProperty<ShapeValue, SkPath>(jpath["ks"], ctx, path_node,
|
|
|
|
[](const sk_sp<sksg::Path>& node, const SkPath& p) { node->setPath(p); });
|
|
|
|
|
|
|
|
if (path_attached)
|
|
|
|
LOG("** Attached path geometry - verbs: %d\n", path_node->getPath().countVerbs());
|
|
|
|
|
|
|
|
return path_attached ? path_node : nullptr;
|
|
|
|
}
|
|
|
|
|
2018-01-02 15:40:00 +00:00
|
|
|
sk_sp<sksg::GeometryNode> AttachRRectGeometry(const Json::Value& jrect, AttachContext* ctx) {
|
|
|
|
SkASSERT(jrect.isObject());
|
|
|
|
|
|
|
|
auto rect_node = sksg::RRect::Make();
|
|
|
|
auto composite = sk_make_sp<CompositeRRect>(rect_node);
|
|
|
|
|
|
|
|
auto p_attached = AttachProperty<VectorValue, SkPoint>(jrect["p"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositeRRect>& node, const SkPoint& pos) { node->setPosition(pos); });
|
|
|
|
auto s_attached = AttachProperty<VectorValue, SkSize>(jrect["s"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositeRRect>& node, const SkSize& sz) { node->setSize(sz); });
|
|
|
|
auto r_attached = AttachProperty<ScalarValue, SkScalar>(jrect["r"], ctx, composite,
|
2018-01-04 15:26:35 +00:00
|
|
|
[](const sk_sp<CompositeRRect>& node, SkScalar radius) {
|
|
|
|
node->setRadius(SkSize::Make(radius, radius));
|
|
|
|
});
|
2018-01-02 15:40:00 +00:00
|
|
|
|
|
|
|
if (!p_attached && !s_attached && !r_attached) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-01-04 15:26:35 +00:00
|
|
|
LOG("** Attached (r)rect geometry\n");
|
|
|
|
|
|
|
|
return rect_node;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::GeometryNode> AttachEllipseGeometry(const Json::Value& jellipse, AttachContext* ctx) {
|
|
|
|
SkASSERT(jellipse.isObject());
|
|
|
|
|
|
|
|
auto rect_node = sksg::RRect::Make();
|
|
|
|
auto composite = sk_make_sp<CompositeRRect>(rect_node);
|
|
|
|
|
|
|
|
auto p_attached = AttachProperty<VectorValue, SkPoint>(jellipse["p"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositeRRect>& node, const SkPoint& pos) { node->setPosition(pos); });
|
|
|
|
auto s_attached = AttachProperty<VectorValue, SkSize>(jellipse["s"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositeRRect>& node, const SkSize& sz) {
|
|
|
|
node->setSize(sz);
|
|
|
|
node->setRadius(SkSize::Make(sz.width() / 2, sz.height() / 2));
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!p_attached && !s_attached) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG("** Attached ellipse geometry\n");
|
|
|
|
|
2018-01-02 15:40:00 +00:00
|
|
|
return rect_node;
|
|
|
|
}
|
|
|
|
|
2018-01-04 16:27:09 +00:00
|
|
|
sk_sp<sksg::GeometryNode> AttachPolystarGeometry(const Json::Value& jstar, AttachContext* ctx) {
|
|
|
|
SkASSERT(jstar.isObject());
|
|
|
|
|
|
|
|
static constexpr CompositePolyStar::Type gTypes[] = {
|
|
|
|
CompositePolyStar::Type::kStar, // "sy": 1
|
|
|
|
CompositePolyStar::Type::kPoly, // "sy": 2
|
|
|
|
};
|
|
|
|
|
|
|
|
const auto type = ParseInt(jstar["sy"], 0) - 1;
|
|
|
|
if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gTypes))) {
|
|
|
|
LogFail(jstar, "Unknown polystar type");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto path_node = sksg::Path::Make();
|
|
|
|
auto composite = sk_make_sp<CompositePolyStar>(path_node, gTypes[type]);
|
|
|
|
|
|
|
|
AttachProperty<VectorValue, SkPoint>(jstar["p"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositePolyStar>& node, const SkPoint& p) { node->setPosition(p); });
|
|
|
|
AttachProperty<ScalarValue, SkScalar>(jstar["pt"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositePolyStar>& node, SkScalar pt) { node->setPointCount(pt); });
|
|
|
|
AttachProperty<ScalarValue, SkScalar>(jstar["ir"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositePolyStar>& node, SkScalar ir) { node->setInnerRadius(ir); });
|
|
|
|
AttachProperty<ScalarValue, SkScalar>(jstar["or"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositePolyStar>& node, SkScalar otr) { node->setOuterRadius(otr); });
|
|
|
|
AttachProperty<ScalarValue, SkScalar>(jstar["is"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositePolyStar>& node, SkScalar is) { node->setInnerRoundness(is); });
|
|
|
|
AttachProperty<ScalarValue, SkScalar>(jstar["os"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositePolyStar>& node, SkScalar os) { node->setOuterRoundness(os); });
|
|
|
|
AttachProperty<ScalarValue, SkScalar>(jstar["r"], ctx, composite,
|
|
|
|
[](const sk_sp<CompositePolyStar>& node, SkScalar r) { node->setRotation(r); });
|
|
|
|
|
|
|
|
return path_node;
|
|
|
|
}
|
|
|
|
|
2017-12-30 17:27:00 +00:00
|
|
|
sk_sp<sksg::Color> AttachColorPaint(const Json::Value& obj, AttachContext* ctx) {
|
|
|
|
SkASSERT(obj.isObject());
|
|
|
|
|
|
|
|
auto color_node = sksg::Color::Make(SK_ColorBLACK);
|
|
|
|
color_node->setAntiAlias(true);
|
|
|
|
|
|
|
|
auto color_attached = AttachProperty<VectorValue, SkColor>(obj["c"], ctx, color_node,
|
|
|
|
[](const sk_sp<sksg::Color>& node, SkColor c) { node->setColor(c); });
|
|
|
|
|
|
|
|
return color_attached ? color_node : nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::PaintNode> AttachFillPaint(const Json::Value& jfill, AttachContext* ctx) {
|
|
|
|
SkASSERT(jfill.isObject());
|
|
|
|
|
|
|
|
auto color = AttachColorPaint(jfill, ctx);
|
|
|
|
if (color) {
|
|
|
|
LOG("** Attached color fill: 0x%x\n", color->getColor());
|
|
|
|
}
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::PaintNode> AttachStrokePaint(const Json::Value& jstroke, AttachContext* ctx) {
|
|
|
|
SkASSERT(jstroke.isObject());
|
|
|
|
|
|
|
|
auto stroke_node = AttachColorPaint(jstroke, ctx);
|
|
|
|
if (!stroke_node)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
LOG("** Attached color stroke: 0x%x\n", stroke_node->getColor());
|
|
|
|
|
|
|
|
stroke_node->setStyle(SkPaint::kStroke_Style);
|
|
|
|
|
|
|
|
auto width_attached = AttachProperty<ScalarValue, SkScalar>(jstroke["w"], ctx, stroke_node,
|
|
|
|
[](const sk_sp<sksg::Color>& node, SkScalar width) { node->setStrokeWidth(width); });
|
|
|
|
if (!width_attached)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
stroke_node->setStrokeMiter(ParseScalar(jstroke["ml"], 4));
|
|
|
|
|
|
|
|
static constexpr SkPaint::Join gJoins[] = {
|
|
|
|
SkPaint::kMiter_Join,
|
|
|
|
SkPaint::kRound_Join,
|
|
|
|
SkPaint::kBevel_Join,
|
|
|
|
};
|
|
|
|
stroke_node->setStrokeJoin(gJoins[SkTPin<int>(ParseInt(jstroke["lj"], 1) - 1,
|
|
|
|
0, SK_ARRAY_COUNT(gJoins) - 1)]);
|
|
|
|
|
|
|
|
static constexpr SkPaint::Cap gCaps[] = {
|
|
|
|
SkPaint::kButt_Cap,
|
|
|
|
SkPaint::kRound_Cap,
|
|
|
|
SkPaint::kSquare_Cap,
|
|
|
|
};
|
|
|
|
stroke_node->setStrokeCap(gCaps[SkTPin<int>(ParseInt(jstroke["lc"], 1) - 1,
|
|
|
|
0, SK_ARRAY_COUNT(gCaps) - 1)]);
|
|
|
|
|
|
|
|
return stroke_node;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:37:54 +00:00
|
|
|
std::vector<sk_sp<sksg::GeometryNode>> AttachMergeGeometryEffect(
|
|
|
|
const Json::Value& jmerge, AttachContext* ctx, std::vector<sk_sp<sksg::GeometryNode>>&& geos) {
|
|
|
|
std::vector<sk_sp<sksg::GeometryNode>> merged;
|
|
|
|
|
|
|
|
static constexpr sksg::Merge::Mode gModes[] = {
|
|
|
|
sksg::Merge::Mode::kMerge, // "mm": 1
|
|
|
|
sksg::Merge::Mode::kUnion, // "mm": 2
|
|
|
|
sksg::Merge::Mode::kDifference, // "mm": 3
|
|
|
|
sksg::Merge::Mode::kIntersect, // "mm": 4
|
|
|
|
sksg::Merge::Mode::kXOR , // "mm": 5
|
|
|
|
};
|
|
|
|
|
|
|
|
const auto mode = gModes[SkTPin<int>(ParseInt(jmerge["mm"], 1) - 1, 0, SK_ARRAY_COUNT(gModes))];
|
|
|
|
merged.push_back(sksg::Merge::Make(std::move(geos), mode));
|
|
|
|
|
|
|
|
LOG("** Attached merge path effect, mode: %d\n", mode);
|
|
|
|
|
|
|
|
return merged;
|
|
|
|
}
|
|
|
|
|
2017-12-30 17:27:00 +00:00
|
|
|
using GeometryAttacherT = sk_sp<sksg::GeometryNode> (*)(const Json::Value&, AttachContext*);
|
|
|
|
static constexpr GeometryAttacherT gGeometryAttachers[] = {
|
|
|
|
AttachPathGeometry,
|
2018-01-02 15:40:00 +00:00
|
|
|
AttachRRectGeometry,
|
2018-01-04 15:26:35 +00:00
|
|
|
AttachEllipseGeometry,
|
2018-01-04 16:27:09 +00:00
|
|
|
AttachPolystarGeometry,
|
2017-12-30 17:27:00 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
using PaintAttacherT = sk_sp<sksg::PaintNode> (*)(const Json::Value&, AttachContext*);
|
|
|
|
static constexpr PaintAttacherT gPaintAttachers[] = {
|
|
|
|
AttachFillPaint,
|
|
|
|
AttachStrokePaint,
|
|
|
|
};
|
|
|
|
|
|
|
|
using GroupAttacherT = sk_sp<sksg::RenderNode> (*)(const Json::Value&, AttachContext*);
|
|
|
|
static constexpr GroupAttacherT gGroupAttachers[] = {
|
|
|
|
AttachShapeGroup,
|
|
|
|
};
|
|
|
|
|
2018-01-04 04:37:54 +00:00
|
|
|
using GeometryEffectAttacherT =
|
|
|
|
std::vector<sk_sp<sksg::GeometryNode>> (*)(const Json::Value&,
|
|
|
|
AttachContext*,
|
|
|
|
std::vector<sk_sp<sksg::GeometryNode>>&&);
|
|
|
|
static constexpr GeometryEffectAttacherT gGeometryEffectAttachers[] = {
|
|
|
|
AttachMergeGeometryEffect,
|
|
|
|
};
|
|
|
|
|
2017-12-30 17:27:00 +00:00
|
|
|
enum class ShapeType {
|
|
|
|
kGeometry,
|
2018-01-04 04:37:54 +00:00
|
|
|
kGeometryEffect,
|
2017-12-30 17:27:00 +00:00
|
|
|
kPaint,
|
|
|
|
kGroup,
|
2017-12-31 14:12:31 +00:00
|
|
|
kTransform,
|
2017-12-30 17:27:00 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ShapeInfo {
|
|
|
|
const char* fTypeString;
|
|
|
|
ShapeType fShapeType;
|
|
|
|
uint32_t fAttacherIndex; // index into respective attacher tables
|
|
|
|
};
|
|
|
|
|
|
|
|
const ShapeInfo* FindShapeInfo(const Json::Value& shape) {
|
|
|
|
static constexpr ShapeInfo gShapeInfo[] = {
|
2018-01-04 15:26:35 +00:00
|
|
|
{ "el", ShapeType::kGeometry , 2 }, // ellipse -> AttachEllipseGeometry
|
2018-01-04 04:37:54 +00:00
|
|
|
{ "fl", ShapeType::kPaint , 0 }, // fill -> AttachFillPaint
|
|
|
|
{ "gr", ShapeType::kGroup , 0 }, // group -> AttachShapeGroup
|
|
|
|
{ "mm", ShapeType::kGeometryEffect, 0 }, // merge -> AttachMergeGeometryEffect
|
2018-01-04 16:27:09 +00:00
|
|
|
{ "rc", ShapeType::kGeometry , 1 }, // rrect -> AttachRRectGeometry
|
2018-01-04 04:37:54 +00:00
|
|
|
{ "sh", ShapeType::kGeometry , 0 }, // shape -> AttachPathGeometry
|
2018-01-04 16:27:09 +00:00
|
|
|
{ "sr", ShapeType::kGeometry , 3 }, // polystar -> AttachPolyStarGeometry
|
2018-01-04 04:37:54 +00:00
|
|
|
{ "st", ShapeType::kPaint , 1 }, // stroke -> AttachStrokePaint
|
2018-01-05 02:11:55 +00:00
|
|
|
{ "tr", ShapeType::kTransform , 0 }, // transform -> In-place handler
|
2017-12-30 17:27:00 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if (!shape.isObject())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
const auto& type = shape["ty"];
|
|
|
|
if (!type.isString())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
const auto* info = bsearch(type.asCString(),
|
|
|
|
gShapeInfo,
|
|
|
|
SK_ARRAY_COUNT(gShapeInfo),
|
|
|
|
sizeof(ShapeInfo),
|
|
|
|
[](const void* key, const void* info) {
|
|
|
|
return strcmp(static_cast<const char*>(key),
|
|
|
|
static_cast<const ShapeInfo*>(info)->fTypeString);
|
|
|
|
});
|
|
|
|
|
|
|
|
return static_cast<const ShapeInfo*>(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachShape(const Json::Value& shapeArray, AttachContext* ctx) {
|
|
|
|
if (!shapeArray.isArray())
|
|
|
|
return nullptr;
|
|
|
|
|
2018-01-02 17:52:43 +00:00
|
|
|
// (https://helpx.adobe.com/after-effects/using/overview-shape-layers-paths-vector.html#groups_and_render_order_for_shapes_and_shape_attributes)
|
|
|
|
//
|
|
|
|
// Render order for shapes within a shape layer
|
|
|
|
//
|
|
|
|
// The rules for rendering a shape layer are similar to the rules for rendering a composition
|
|
|
|
// that contains nested compositions:
|
|
|
|
//
|
|
|
|
// * Within a group, the shape at the bottom of the Timeline panel stacking order is rendered
|
|
|
|
// first.
|
|
|
|
//
|
|
|
|
// * All path operations within a group are performed before paint operations. This means,
|
|
|
|
// for example, that the stroke follows the distortions in the path made by the Wiggle Paths
|
|
|
|
// path operation. Path operations within a group are performed from top to bottom.
|
|
|
|
//
|
|
|
|
// * Paint operations within a group are performed from the bottom to the top in the Timeline
|
|
|
|
// panel stacking order. This means, for example, that a stroke is rendered on top of
|
|
|
|
// (in front of) a stroke that appears after it in the Timeline panel.
|
|
|
|
//
|
2017-12-31 14:12:31 +00:00
|
|
|
sk_sp<sksg::Group> shape_group = sksg::Group::Make();
|
|
|
|
sk_sp<sksg::RenderNode> xformed_group = shape_group;
|
2017-12-30 17:27:00 +00:00
|
|
|
|
2018-01-04 04:37:54 +00:00
|
|
|
std::vector<sk_sp<sksg::GeometryNode>> geos;
|
|
|
|
std::vector<sk_sp<sksg::RenderNode>> draws;
|
2017-12-30 17:27:00 +00:00
|
|
|
|
|
|
|
for (const auto& s : shapeArray) {
|
|
|
|
const auto* info = FindShapeInfo(s);
|
|
|
|
if (!info) {
|
|
|
|
LogFail(s.isObject() ? s["ty"] : s, "Unknown shape");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (info->fShapeType) {
|
|
|
|
case ShapeType::kGeometry: {
|
|
|
|
SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryAttachers));
|
|
|
|
if (auto geo = gGeometryAttachers[info->fAttacherIndex](s, ctx)) {
|
|
|
|
geos.push_back(std::move(geo));
|
|
|
|
}
|
|
|
|
} break;
|
2018-01-04 04:37:54 +00:00
|
|
|
case ShapeType::kGeometryEffect: {
|
|
|
|
SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGeometryEffectAttachers));
|
|
|
|
geos = gGeometryEffectAttachers[info->fAttacherIndex](s, ctx, std::move(geos));
|
|
|
|
} break;
|
2017-12-30 17:27:00 +00:00
|
|
|
case ShapeType::kPaint: {
|
|
|
|
SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gPaintAttachers));
|
|
|
|
if (auto paint = gPaintAttachers[info->fAttacherIndex](s, ctx)) {
|
2018-01-02 17:52:43 +00:00
|
|
|
for (const auto& geo : geos) {
|
|
|
|
draws.push_back(sksg::Draw::Make(geo, paint));
|
|
|
|
}
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
} break;
|
|
|
|
case ShapeType::kGroup: {
|
|
|
|
SkASSERT(info->fAttacherIndex < SK_ARRAY_COUNT(gGroupAttachers));
|
|
|
|
if (auto group = gGroupAttachers[info->fAttacherIndex](s, ctx)) {
|
2018-01-02 17:52:43 +00:00
|
|
|
draws.push_back(std::move(group));
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
} break;
|
2017-12-31 14:12:31 +00:00
|
|
|
case ShapeType::kTransform: {
|
2018-01-02 17:52:43 +00:00
|
|
|
// TODO: BM appears to transform the geometry, not the draw op itself.
|
2018-01-05 02:11:55 +00:00
|
|
|
if (auto matrix = AttachMatrix(s, ctx, nullptr)) {
|
|
|
|
xformed_group = sksg::Transform::Make(std::move(xformed_group),
|
|
|
|
std::move(matrix));
|
|
|
|
}
|
2017-12-31 14:12:31 +00:00
|
|
|
} break;
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-02 17:52:43 +00:00
|
|
|
if (draws.empty()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:37:54 +00:00
|
|
|
for (auto draw = draws.rbegin(); draw != draws.rend(); ++draw) {
|
|
|
|
shape_group->addChild(std::move(*draw));
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:37:54 +00:00
|
|
|
LOG("** Attached shape: %zd draws.\n", draws.size());
|
2017-12-31 14:12:31 +00:00
|
|
|
return xformed_group;
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachCompLayer(const Json::Value& layer, AttachContext* ctx) {
|
|
|
|
SkASSERT(layer.isObject());
|
|
|
|
|
|
|
|
auto refId = ParseString(layer["refId"], "");
|
|
|
|
if (refId.isEmpty()) {
|
|
|
|
LOG("!! Comp layer missing refId\n");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto* comp = ctx->fAssets.find(refId);
|
|
|
|
if (!comp) {
|
|
|
|
LOG("!! Pre-comp not found: '%s'\n", refId.c_str());
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: cycle detection
|
|
|
|
return AttachComposition(**comp, ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachSolidLayer(const Json::Value& layer, AttachContext*) {
|
|
|
|
SkASSERT(layer.isObject());
|
|
|
|
|
|
|
|
LOG("?? Solid layer stub\n");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachImageLayer(const Json::Value& layer, AttachContext*) {
|
|
|
|
SkASSERT(layer.isObject());
|
|
|
|
|
|
|
|
LOG("?? Image layer stub\n");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachNullLayer(const Json::Value& layer, AttachContext*) {
|
|
|
|
SkASSERT(layer.isObject());
|
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
// Null layers are used solely to drive dependent transforms,
|
|
|
|
// but we use free-floating sksg::Matrices for that purpose.
|
2017-12-30 17:27:00 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachShapeLayer(const Json::Value& layer, AttachContext* ctx) {
|
|
|
|
SkASSERT(layer.isObject());
|
|
|
|
|
|
|
|
LOG("** Attaching shape layer ind: %d\n", ParseInt(layer["ind"], 0));
|
|
|
|
|
|
|
|
return AttachShape(layer["shapes"], ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachTextLayer(const Json::Value& layer, AttachContext*) {
|
|
|
|
SkASSERT(layer.isObject());
|
|
|
|
|
|
|
|
LOG("?? Text layer stub\n");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
struct AttachLayerContext {
|
|
|
|
AttachLayerContext(const Json::Value& jlayers, AttachContext* ctx)
|
|
|
|
: fLayerList(jlayers), fCtx(ctx) {}
|
|
|
|
|
|
|
|
const Json::Value& fLayerList;
|
|
|
|
AttachContext* fCtx;
|
|
|
|
std::unordered_map<const Json::Value*, sk_sp<sksg::Matrix>> fLayerMatrixCache;
|
|
|
|
std::unordered_map<int, const Json::Value*> fLayerIndexCache;
|
|
|
|
|
|
|
|
const Json::Value* findLayer(int index) {
|
|
|
|
SkASSERT(fLayerList.isArray());
|
|
|
|
|
|
|
|
if (index < 0) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto cached = fLayerIndexCache.find(index);
|
|
|
|
if (cached != fLayerIndexCache.end()) {
|
|
|
|
return cached->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto& l : fLayerList) {
|
|
|
|
if (!l.isObject()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ParseInt(l["ind"], -1) == index) {
|
|
|
|
fLayerIndexCache.insert(std::make_pair(index, &l));
|
|
|
|
return &l;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::Matrix> AttachLayerMatrix(const Json::Value& jlayer) {
|
|
|
|
SkASSERT(jlayer.isObject());
|
|
|
|
|
|
|
|
const auto cached = fLayerMatrixCache.find(&jlayer);
|
|
|
|
if (cached != fLayerMatrixCache.end()) {
|
|
|
|
return cached->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto* parentLayer = this->findLayer(ParseInt(jlayer["parent"], -1));
|
|
|
|
|
|
|
|
// TODO: cycle detection?
|
|
|
|
auto parentMatrix = (parentLayer && parentLayer != &jlayer)
|
|
|
|
? this->AttachLayerMatrix(*parentLayer) : nullptr;
|
|
|
|
|
|
|
|
auto layerMatrix = AttachMatrix(jlayer["ks"], fCtx, std::move(parentMatrix));
|
|
|
|
fLayerMatrixCache.insert(std::make_pair(&jlayer, layerMatrix));
|
|
|
|
|
|
|
|
return layerMatrix;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachLayer(const Json::Value& jlayer,
|
|
|
|
AttachLayerContext* layerCtx) {
|
|
|
|
if (!jlayer.isObject())
|
2017-12-30 17:27:00 +00:00
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
using LayerAttacher = sk_sp<sksg::RenderNode> (*)(const Json::Value&, AttachContext*);
|
|
|
|
static constexpr LayerAttacher gLayerAttachers[] = {
|
|
|
|
AttachCompLayer, // 'ty': 0
|
|
|
|
AttachSolidLayer, // 'ty': 1
|
|
|
|
AttachImageLayer, // 'ty': 2
|
|
|
|
AttachNullLayer, // 'ty': 3
|
|
|
|
AttachShapeLayer, // 'ty': 4
|
|
|
|
AttachTextLayer, // 'ty': 5
|
|
|
|
};
|
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
int type = ParseInt(jlayer["ty"], -1);
|
2017-12-30 17:27:00 +00:00
|
|
|
if (type < 0 || type >= SkTo<int>(SK_ARRAY_COUNT(gLayerAttachers))) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
auto layer = gLayerAttachers[type](jlayer, layerCtx->fCtx);
|
|
|
|
auto layerMatrix = layerCtx->AttachLayerMatrix(jlayer);
|
|
|
|
|
|
|
|
return layerMatrix
|
|
|
|
? sksg::Transform::Make(std::move(layer), std::move(layerMatrix))
|
|
|
|
: layer;
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<sksg::RenderNode> AttachComposition(const Json::Value& comp, AttachContext* ctx) {
|
|
|
|
if (!comp.isObject())
|
|
|
|
return nullptr;
|
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
const auto& jlayers = comp["layers"];
|
|
|
|
if (!jlayers.isArray())
|
|
|
|
return nullptr;
|
|
|
|
|
2018-01-02 17:52:43 +00:00
|
|
|
SkSTArray<16, sk_sp<sksg::RenderNode>, true> layers;
|
2018-01-05 02:11:55 +00:00
|
|
|
AttachLayerContext layerCtx(jlayers, ctx);
|
2017-12-30 17:27:00 +00:00
|
|
|
|
2018-01-05 02:11:55 +00:00
|
|
|
for (const auto& l : jlayers) {
|
|
|
|
if (auto layer_fragment = AttachLayer(l, &layerCtx)) {
|
2018-01-02 17:52:43 +00:00
|
|
|
layers.push_back(std::move(layer_fragment));
|
2017-12-30 17:27:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-02 17:52:43 +00:00
|
|
|
if (layers.empty()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Layers are painted in bottom->top order.
|
|
|
|
auto comp_group = sksg::Group::Make();
|
|
|
|
for (int i = layers.count() - 1; i >= 0; --i) {
|
|
|
|
comp_group->addChild(std::move(layers[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG("** Attached composition '%s': %d layers.\n",
|
|
|
|
ParseString(comp["id"], "").c_str(), layers.count());
|
|
|
|
|
2017-12-30 17:27:00 +00:00
|
|
|
return comp_group;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
std::unique_ptr<Animation> Animation::Make(SkStream* stream) {
|
|
|
|
if (!stream->hasLength()) {
|
|
|
|
// TODO: handle explicit buffering?
|
|
|
|
LOG("!! cannot parse streaming content\n");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
Json::Value json;
|
|
|
|
{
|
|
|
|
auto data = SkData::MakeFromStream(stream, stream->getLength());
|
|
|
|
if (!data) {
|
|
|
|
LOG("!! could not read stream\n");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
Json::Reader reader;
|
|
|
|
|
|
|
|
auto dataStart = static_cast<const char*>(data->data());
|
|
|
|
if (!reader.parse(dataStart, dataStart + data->size(), json, false) || !json.isObject()) {
|
|
|
|
LOG("!! failed to parse json: %s\n", reader.getFormattedErrorMessages().c_str());
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto version = ParseString(json["v"], "");
|
|
|
|
const auto size = SkSize::Make(ParseScalar(json["w"], -1), ParseScalar(json["h"], -1));
|
|
|
|
const auto fps = ParseScalar(json["fr"], -1);
|
|
|
|
|
|
|
|
if (size.isEmpty() || version.isEmpty() || fps < 0) {
|
|
|
|
LOG("!! invalid animation params (version: %s, size: [%f %f], frame rate: %f)",
|
|
|
|
version.c_str(), size.width(), size.height(), fps);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::unique_ptr<Animation>(new Animation(std::move(version), size, fps, json));
|
|
|
|
}
|
|
|
|
|
|
|
|
Animation::Animation(SkString version, const SkSize& size, SkScalar fps, const Json::Value& json)
|
|
|
|
: fVersion(std::move(version))
|
|
|
|
, fSize(size)
|
|
|
|
, fFrameRate(fps)
|
|
|
|
, fInPoint(ParseScalar(json["ip"], 0))
|
|
|
|
, fOutPoint(SkTMax(ParseScalar(json["op"], SK_ScalarMax), fInPoint)) {
|
|
|
|
|
|
|
|
AssetMap assets;
|
|
|
|
for (const auto& asset : json["assets"]) {
|
|
|
|
if (!asset.isObject()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
assets.set(ParseString(asset["id"], ""), &asset);
|
|
|
|
}
|
|
|
|
|
|
|
|
AttachContext ctx = { assets, fAnimators };
|
|
|
|
fDom = AttachComposition(json, &ctx);
|
|
|
|
|
|
|
|
LOG("** Attached %d animators\n", fAnimators.count());
|
|
|
|
}
|
|
|
|
|
|
|
|
Animation::~Animation() = default;
|
|
|
|
|
|
|
|
void Animation::render(SkCanvas* canvas) const {
|
|
|
|
if (!fDom)
|
|
|
|
return;
|
|
|
|
|
|
|
|
sksg::InvalidationController ic;
|
|
|
|
fDom->revalidate(&ic, SkMatrix::I());
|
|
|
|
|
|
|
|
// TODO: proper inval
|
|
|
|
fDom->render(canvas);
|
|
|
|
|
|
|
|
if (!fShowInval)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SkPaint fill, stroke;
|
|
|
|
fill.setAntiAlias(true);
|
|
|
|
fill.setColor(0x40ff0000);
|
|
|
|
stroke.setAntiAlias(true);
|
|
|
|
stroke.setColor(0xffff0000);
|
|
|
|
stroke.setStyle(SkPaint::kStroke_Style);
|
|
|
|
|
|
|
|
for (const auto& r : ic) {
|
|
|
|
canvas->drawRect(r, fill);
|
|
|
|
canvas->drawRect(r, stroke);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::animationTick(SkMSec ms) {
|
|
|
|
// 't' in the BM model really means 'frame #'
|
|
|
|
auto t = static_cast<float>(ms) * fFrameRate / 1000;
|
|
|
|
|
|
|
|
t = fInPoint + std::fmod(t, fOutPoint - fInPoint);
|
|
|
|
|
|
|
|
// TODO: this can be optimized quite a bit with some sorting/state tracking.
|
|
|
|
for (const auto& a : fAnimators) {
|
|
|
|
a->tick(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace skotty
|