Benau 53cafa9f3d
Convert .tgs with go libraries (and cgo) (telegram) (#1569)
This commit adds support for go/cgo tgs conversion when building with the -tags `cgo`
The default binaries are still "pure" go and uses the old way of converting.

* Move lottie_convert.py conversion code to its own file

* Add optional libtgsconverter

* Update vendor

* Apply suggestions from code review

* Update bridge/helper/libtgsconverter.go

Co-authored-by: Wim <wim@42.be>
2021-08-24 22:32:50 +02:00

1149 lines
35 KiB
C++

/*
* Copyright (c) 2020 Samsung Electronics Co., Ltd. All rights reserved.
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#ifndef LOTModel_H
#define LOTModel_H
#include <algorithm>
#include <cmath>
#include <cstring>
#include <functional>
#include <memory>
#include <unordered_map>
#include <vector>
#include "vector_varenaalloc.h"
#include "vector_vbezier.h"
#include "vector_vbrush.h"
#include "vector_vinterpolator.h"
#include "vector_vmatrix.h"
#include "vector_vpath.h"
#include "vector_vpoint.h"
#include "vector_vrect.h"
V_USE_NAMESPACE
namespace rlottie {
namespace internal {
using Marker = std::tuple<std::string, int, int>;
using LayerInfo = Marker;
template <typename T>
inline T lerp(const T &start, const T &end, float t)
{
return start + t * (end - start);
}
namespace model {
enum class MatteType : uchar { None = 0, Alpha = 1, AlphaInv, Luma, LumaInv };
enum class BlendMode : uchar {
Normal = 0,
Multiply = 1,
Screen = 2,
OverLay = 3
};
class Color {
public:
Color() = default;
Color(float red, float green, float blue) : r(red), g(green), b(blue) {}
VColor toColor(float a = 1)
{
return VColor(uchar(255 * r), uchar(255 * g), uchar(255 * b),
uchar(255 * a));
}
friend inline Color operator+(const Color &c1, const Color &c2);
friend inline Color operator-(const Color &c1, const Color &c2);
public:
float r{1};
float g{1};
float b{1};
};
inline Color operator-(const Color &c1, const Color &c2)
{
return Color(c1.r - c2.r, c1.g - c2.g, c1.b - c2.b);
}
inline Color operator+(const Color &c1, const Color &c2)
{
return Color(c1.r + c2.r, c1.g + c2.g, c1.b + c2.b);
}
inline const Color operator*(const Color &c, float m)
{
return Color(c.r * m, c.g * m, c.b * m);
}
inline const Color operator*(float m, const Color &c)
{
return Color(c.r * m, c.g * m, c.b * m);
}
struct PathData {
std::vector<VPointF> mPoints;
bool mClosed = false; /* "c" */
void reserve(size_t size) { mPoints.reserve(mPoints.size() + size); }
static void lerp(const PathData &start, const PathData &end, float t,
VPath &result)
{
result.reset();
// test for empty animation data.
if (start.mPoints.empty() || end.mPoints.empty())
{
return;
}
auto size = std::min(start.mPoints.size(), end.mPoints.size());
/* reserve exact memory requirement at once
* ptSize = size + 1(size + close)
* elmSize = size/3 cubic + 1 move + 1 close
*/
result.reserve(size + 1, size / 3 + 2);
result.moveTo(start.mPoints[0] +
t * (end.mPoints[0] - start.mPoints[0]));
for (size_t i = 1; i < size; i += 3) {
result.cubicTo(
start.mPoints[i] + t * (end.mPoints[i] - start.mPoints[i]),
start.mPoints[i + 1] +
t * (end.mPoints[i + 1] - start.mPoints[i + 1]),
start.mPoints[i + 2] +
t * (end.mPoints[i + 2] - start.mPoints[i + 2]));
}
if (start.mClosed) result.close();
}
void toPath(VPath &path) const
{
path.reset();
if (mPoints.empty()) return;
auto size = mPoints.size();
auto points = mPoints.data();
/* reserve exact memory requirement at once
* ptSize = size + 1(size + close)
* elmSize = size/3 cubic + 1 move + 1 close
*/
path.reserve(size + 1, size / 3 + 2);
path.moveTo(points[0]);
for (size_t i = 1; i < size; i += 3) {
path.cubicTo(points[i], points[i + 1], points[i + 2]);
}
if (mClosed) path.close();
}
};
template <typename T, typename Tag = void>
struct Value {
T start_;
T end_;
T at(float t) const { return lerp(start_, end_, t); }
float angle(float) const { return 0; }
void cache() {}
};
struct Position;
template <typename T>
struct Value<T, Position> {
T start_;
T end_;
T inTangent_;
T outTangent_;
float length_{0};
bool hasTangent_{false};
void cache()
{
if (hasTangent_) {
inTangent_ = end_ + inTangent_;
outTangent_ = start_ + outTangent_;
length_ = VBezier::fromPoints(start_, outTangent_, inTangent_, end_)
.length();
if (vIsZero(length_)) {
// this segment has zero length.
// so disable expensive path computaion.
hasTangent_ = false;
}
}
}
T at(float t) const
{
if (hasTangent_) {
/*
* position along the path calcualated
* using bezier at progress length (t * bezlen)
*/
VBezier b =
VBezier::fromPoints(start_, outTangent_, inTangent_, end_);
return b.pointAt(b.tAtLength(t * length_, length_));
}
return lerp(start_, end_, t);
}
float angle(float t) const
{
if (hasTangent_) {
VBezier b =
VBezier::fromPoints(start_, outTangent_, inTangent_, end_);
return b.angleAt(b.tAtLength(t * length_, length_));
}
return 0;
}
};
template <typename T, typename Tag>
class KeyFrames {
public:
struct Frame {
float progress(int frameNo) const
{
return interpolator_ ? interpolator_->value((frameNo - start_) /
(end_ - start_))
: 0;
}
T value(int frameNo) const { return value_.at(progress(frameNo)); }
float angle(int frameNo) const
{
return value_.angle(progress(frameNo));
}
float start_{0};
float end_{0};
VInterpolator *interpolator_{nullptr};
Value<T, Tag> value_;
};
T value(int frameNo) const
{
if (frames_.front().start_ >= frameNo)
return frames_.front().value_.start_;
if (frames_.back().end_ <= frameNo) return frames_.back().value_.end_;
for (const auto &keyFrame : frames_) {
if (frameNo >= keyFrame.start_ && frameNo < keyFrame.end_)
return keyFrame.value(frameNo);
}
return {};
}
float angle(int frameNo) const
{
if ((frames_.front().start_ >= frameNo) ||
(frames_.back().end_ <= frameNo))
return 0;
for (const auto &frame : frames_) {
if (frameNo >= frame.start_ && frameNo < frame.end_)
return frame.angle(frameNo);
}
return 0;
}
bool changed(int prevFrame, int curFrame) const
{
auto first = frames_.front().start_;
auto last = frames_.back().end_;
return !((first > prevFrame && first > curFrame) ||
(last < prevFrame && last < curFrame));
}
void cache()
{
for (auto &e : frames_) e.value_.cache();
}
public:
std::vector<Frame> frames_;
};
template <typename T, typename Tag = void>
class Property {
public:
using Animation = KeyFrames<T, Tag>;
Property() { construct(impl_.value_, {}); }
explicit Property(T value) { construct(impl_.value_, std::move(value)); }
const Animation &animation() const { return *(impl_.animation_.get()); }
const T & value() const { return impl_.value_; }
Animation &animation()
{
if (isValue_) {
destroy();
construct(impl_.animation_, std::make_unique<Animation>());
isValue_ = false;
}
return *(impl_.animation_.get());
}
T &value()
{
assert(isValue_);
return impl_.value_;
}
Property(Property &&other) noexcept
{
if (!other.isValue_) {
construct(impl_.animation_, std::move(other.impl_.animation_));
isValue_ = false;
} else {
construct(impl_.value_, std::move(other.impl_.value_));
isValue_ = true;
}
}
// delete special member functions
Property(const Property &) = delete;
Property &operator=(const Property &) = delete;
Property &operator=(Property &&) = delete;
~Property() { destroy(); }
bool isStatic() const { return isValue_; }
T value(int frameNo) const
{
return isStatic() ? value() : animation().value(frameNo);
}
// special function only for type T=PathData
template <typename forT = PathData>
auto value(int frameNo, VPath &path) const ->
typename std::enable_if_t<std::is_same<T, forT>::value, void>
{
if (isStatic()) {
value().toPath(path);
} else {
const auto &vec = animation().frames_;
if (vec.front().start_ >= frameNo)
return vec.front().value_.start_.toPath(path);
if (vec.back().end_ <= frameNo)
return vec.back().value_.end_.toPath(path);
for (const auto &keyFrame : vec) {
if (frameNo >= keyFrame.start_ && frameNo < keyFrame.end_) {
T::lerp(keyFrame.value_.start_, keyFrame.value_.end_,
keyFrame.progress(frameNo), path);
}
}
}
}
float angle(int frameNo) const
{
return isStatic() ? 0 : animation().angle(frameNo);
}
bool changed(int prevFrame, int curFrame) const
{
return isStatic() ? false : animation().changed(prevFrame, curFrame);
}
void cache()
{
if (!isStatic()) animation().cache();
}
private:
template <typename Tp>
void construct(Tp &member, Tp &&val)
{
new (&member) Tp(std::move(val));
}
void destroy()
{
if (isValue_) {
impl_.value_.~T();
} else {
using std::unique_ptr;
impl_.animation_.~unique_ptr<Animation>();
}
}
union details {
std::unique_ptr<Animation> animation_;
T value_;
details(){};
details(const details &) = delete;
details(details &&) = delete;
details &operator=(details &&) = delete;
details &operator=(const details &) = delete;
~details() noexcept {};
} impl_;
bool isValue_{true};
};
class Path;
struct PathData;
struct Dash {
std::vector<Property<float>> mData;
bool empty() const { return mData.empty(); }
size_t size() const { return mData.size(); }
bool isStatic() const
{
for (const auto &elm : mData)
if (!elm.isStatic()) return false;
return true;
}
void getDashInfo(int frameNo, std::vector<float> &result) const;
};
class Mask {
public:
enum class Mode { None, Add, Substarct, Intersect, Difference };
float opacity(int frameNo) const
{
return mOpacity.value(frameNo) / 100.0f;
}
bool isStatic() const { return mIsStatic; }
public:
Property<PathData> mShape;
Property<float> mOpacity{100};
bool mInv{false};
bool mIsStatic{true};
Mask::Mode mMode;
};
class Object {
public:
enum class Type : unsigned char {
Composition = 1,
Layer,
Group,
Transform,
Fill,
Stroke,
GFill,
GStroke,
Rect,
Ellipse,
Path,
Polystar,
Trim,
Repeater,
RoundedCorner
};
explicit Object(Object::Type type) : mPtr(nullptr)
{
mData._type = type;
mData._static = true;
mData._shortString = true;
mData._hidden = false;
}
~Object() noexcept
{
if (!shortString() && mPtr) free(mPtr);
}
Object(const Object &) = delete;
Object &operator=(const Object &) = delete;
void setStatic(bool value) { mData._static = value; }
bool isStatic() const { return mData._static; }
bool hidden() const { return mData._hidden; }
void setHidden(bool value) { mData._hidden = value; }
void setType(Object::Type type) { mData._type = type; }
Object::Type type() const { return mData._type; }
void setName(const char *name)
{
if (name) {
auto len = strlen(name);
if (len < maxShortStringLength) {
setShortString(true);
strncpy(mData._buffer, name, len + 1);
} else {
setShortString(false);
mPtr = strdup(name);
}
}
}
const char *name() const { return shortString() ? mData._buffer : mPtr; }
private:
static constexpr unsigned char maxShortStringLength = 14;
void setShortString(bool value) { mData._shortString = value; }
bool shortString() const { return mData._shortString; }
struct Data {
char _buffer[maxShortStringLength];
Object::Type _type;
bool _static : 1;
bool _hidden : 1;
bool _shortString : 1;
};
union {
Data mData;
char *mPtr{nullptr};
};
};
struct Asset {
enum class Type : unsigned char { Precomp, Image, Char };
bool isStatic() const { return mStatic; }
void setStatic(bool value) { mStatic = value; }
VBitmap bitmap() const { return mBitmap; }
void loadImageData(std::string data);
void loadImagePath(std::string Path);
Type mAssetType{Type::Precomp};
bool mStatic{true};
std::string mRefId; // ref id
std::vector<Object *> mLayers;
// image asset data
int mWidth{0};
int mHeight{0};
VBitmap mBitmap;
};
class Layer;
class Composition : public Object {
public:
Composition() : Object(Object::Type::Composition) {}
std::vector<LayerInfo> layerInfoList() const;
const std::vector<Marker> &markers() const { return mMarkers; }
double duration() const
{
return frameDuration() / frameRate(); // in second
}
size_t frameAtPos(double pos) const
{
if (pos < 0) pos = 0;
if (pos > 1) pos = 1;
return size_t(round(pos * frameDuration()));
}
long frameAtTime(double timeInSec) const
{
return long(frameAtPos(timeInSec / duration()));
}
size_t totalFrame() const { return mEndFrame - mStartFrame; }
long frameDuration() const { return mEndFrame - mStartFrame - 1; }
float frameRate() const { return mFrameRate; }
size_t startFrame() const { return mStartFrame; }
size_t endFrame() const { return mEndFrame; }
VSize size() const { return mSize; }
void processRepeaterObjects();
void updateStats();
public:
struct Stats {
uint16_t precompLayerCount{0};
uint16_t solidLayerCount{0};
uint16_t shapeLayerCount{0};
uint16_t imageLayerCount{0};
uint16_t nullLayerCount{0};
};
public:
std::string mVersion;
VSize mSize;
long mStartFrame{0};
long mEndFrame{0};
float mFrameRate{60};
BlendMode mBlendMode{BlendMode::Normal};
Layer * mRootLayer{nullptr};
std::unordered_map<std::string, Asset *> mAssets;
std::vector<Marker> mMarkers;
VArenaAlloc mArenaAlloc{2048};
Stats mStats;
};
class Transform : public Object {
public:
struct Data {
struct Extra {
Property<float> m3DRx{0};
Property<float> m3DRy{0};
Property<float> m3DRz{0};
Property<float> mSeparateX{0};
Property<float> mSeparateY{0};
bool mSeparate{false};
bool m3DData{false};
};
VMatrix matrix(int frameNo, bool autoOrient = false) const;
float opacity(int frameNo) const
{
return mOpacity.value(frameNo) / 100.0f;
}
void createExtraData()
{
if (!mExtra) mExtra = std::make_unique<Extra>();
}
Property<float> mRotation{0}; /* "r" */
Property<VPointF> mScale{{100, 100}}; /* "s" */
Property<VPointF, Position> mPosition; /* "p" */
Property<VPointF> mAnchor; /* "a" */
Property<float> mOpacity{100}; /* "o" */
std::unique_ptr<Extra> mExtra;
};
Transform() : Object(Object::Type::Transform) {}
void set(Transform::Data *data, bool staticFlag)
{
setStatic(staticFlag);
if (isStatic()) {
new (&impl.mStaticData)
StaticData(data->matrix(0), data->opacity(0));
} else {
impl.mData = data;
}
}
VMatrix matrix(int frameNo, bool autoOrient = false) const
{
if (isStatic()) return impl.mStaticData.mMatrix;
return impl.mData->matrix(frameNo, autoOrient);
}
float opacity(int frameNo) const
{
if (isStatic()) return impl.mStaticData.mOpacity;
return impl.mData->opacity(frameNo);
}
Transform(const Transform &) = delete;
Transform(Transform &&) = delete;
Transform &operator=(Transform &) = delete;
Transform &operator=(Transform &&) = delete;
~Transform() noexcept { destroy(); }
private:
void destroy()
{
if (isStatic()) {
impl.mStaticData.~StaticData();
}
}
struct StaticData {
StaticData(VMatrix &&m, float opacity)
: mOpacity(opacity), mMatrix(std::move(m))
{
}
float mOpacity;
VMatrix mMatrix;
};
union details {
Data * mData{nullptr};
StaticData mStaticData;
details(){};
details(const details &) = delete;
details(details &&) = delete;
details &operator=(details &&) = delete;
details &operator=(const details &) = delete;
~details() noexcept {};
} impl;
};
class Group : public Object {
public:
Group() : Object(Object::Type::Group) {}
explicit Group(Object::Type type) : Object(type) {}
public:
std::vector<Object *> mChildren;
Transform * mTransform{nullptr};
};
class Layer : public Group {
public:
enum class Type : uchar {
Precomp = 0,
Solid = 1,
Image = 2,
Null = 3,
Shape = 4,
Text = 5
};
Layer() : Group(Object::Type::Layer) {}
bool hasRoundedCorner() const noexcept { return mHasRoundedCorner; }
bool hasPathOperator() const noexcept { return mHasPathOperator; }
bool hasGradient() const noexcept { return mHasGradient; }
bool hasMask() const noexcept { return mHasMask; }
bool hasRepeater() const noexcept { return mHasRepeater; }
int id() const noexcept { return mId; }
int parentId() const noexcept { return mParentId; }
bool hasParent() const noexcept { return mParentId != -1; }
int inFrame() const noexcept { return mInFrame; }
int outFrame() const noexcept { return mOutFrame; }
int startFrame() const noexcept { return mStartFrame; }
Color solidColor() const noexcept
{
return mExtra ? mExtra->mSolidColor : Color();
}
bool autoOrient() const noexcept { return mAutoOrient; }
int timeRemap(int frameNo) const;
VSize layerSize() const { return mLayerSize; }
bool precompLayer() const { return mLayerType == Type::Precomp; }
VMatrix matrix(int frameNo) const
{
return mTransform ? mTransform->matrix(frameNo, autoOrient())
: VMatrix{};
}
float opacity(int frameNo) const
{
return mTransform ? mTransform->opacity(frameNo) : 1.0f;
}
Asset *asset() const { return mExtra ? mExtra->mAsset : nullptr; }
struct Extra {
Color mSolidColor;
std::string mPreCompRefId;
Property<float> mTimeRemap; /* "tm" */
Composition * mCompRef{nullptr};
Asset * mAsset{nullptr};
std::vector<Mask *> mMasks;
};
Layer::Extra *extra()
{
if (!mExtra) mExtra = std::make_unique<Layer::Extra>();
return mExtra.get();
}
public:
MatteType mMatteType{MatteType::None};
Type mLayerType{Layer::Type::Null};
BlendMode mBlendMode{BlendMode::Normal};
bool mHasRoundedCorner{false};
bool mHasPathOperator{false};
bool mHasMask{false};
bool mHasRepeater{false};
bool mHasGradient{false};
bool mAutoOrient{false};
VSize mLayerSize;
int mParentId{-1}; // Lottie the id of the parent in the composition
int mId{-1}; // Lottie the group id used for parenting.
float mTimeStreatch{1.0f};
int mInFrame{0};
int mOutFrame{0};
int mStartFrame{0};
std::unique_ptr<Extra> mExtra{nullptr};
};
/**
* TimeRemap has the value in time domain(in sec)
* To get the proper mapping first we get the mapped time at the current frame
* Number then we need to convert mapped time to frame number using the
* composition time line Ex: at frame 10 the mappend time is 0.5(500 ms) which
* will be convert to frame number 30 if the frame rate is 60. or will result to
* frame number 15 if the frame rate is 30.
*/
inline int Layer::timeRemap(int frameNo) const
{
/*
* only consider startFrame() when there is no timeRemap.
* when a layer has timeremap bodymovin updates the startFrame()
* of all child layer so we don't have to take care of it.
*/
if (!mExtra || mExtra->mTimeRemap.isStatic())
frameNo = frameNo - startFrame();
else
frameNo =
mExtra->mCompRef->frameAtTime(mExtra->mTimeRemap.value(frameNo));
/* Apply time streatch if it has any.
* Time streatch is just a factor by which the animation will speedup or
* slow down with respect to the overal animation. Time streach factor is
* already applied to the layers inFrame and outFrame.
* @TODO need to find out if timestreatch also affects the in and out frame
* of the child layers or not. */
return int(frameNo / mTimeStreatch);
}
class Stroke : public Object {
public:
Stroke() : Object(Object::Type::Stroke) {}
Color color(int frameNo) const { return mColor.value(frameNo); }
float opacity(int frameNo) const
{
return mOpacity.value(frameNo) / 100.0f;
}
float strokeWidth(int frameNo) const { return mWidth.value(frameNo); }
CapStyle capStyle() const { return mCapStyle; }
JoinStyle joinStyle() const { return mJoinStyle; }
float miterLimit() const { return mMiterLimit; }
bool hasDashInfo() const { return !mDash.empty(); }
void getDashInfo(int frameNo, std::vector<float> &result) const
{
return mDash.getDashInfo(frameNo, result);
}
public:
Property<Color> mColor; /* "c" */
Property<float> mOpacity{100}; /* "o" */
Property<float> mWidth{0}; /* "w" */
CapStyle mCapStyle{CapStyle::Flat}; /* "lc" */
JoinStyle mJoinStyle{JoinStyle::Miter}; /* "lj" */
float mMiterLimit{0}; /* "ml" */
Dash mDash;
bool mEnabled{true}; /* "fillEnabled" */
};
class Gradient : public Object {
public:
class Data {
public:
friend inline Gradient::Data operator+(const Gradient::Data &g1,
const Gradient::Data &g2);
friend inline Gradient::Data operator-(const Gradient::Data &g1,
const Gradient::Data &g2);
friend inline Gradient::Data operator*(float m,
const Gradient::Data &g);
public:
std::vector<float> mGradient;
};
explicit Gradient(Object::Type type) : Object(type) {}
inline float opacity(int frameNo) const
{
return mOpacity.value(frameNo) / 100.0f;
}
void update(std::unique_ptr<VGradient> &grad, int frameNo);
private:
void populate(VGradientStops &stops, int frameNo);
public:
int mGradientType{1}; /* "t" Linear=1 , Radial = 2*/
Property<VPointF> mStartPoint; /* "s" */
Property<VPointF> mEndPoint; /* "e" */
Property<float> mHighlightLength{0}; /* "h" */
Property<float> mHighlightAngle{0}; /* "a" */
Property<float> mOpacity{100}; /* "o" */
Property<Gradient::Data> mGradient; /* "g" */
int mColorPoints{-1};
bool mEnabled{true}; /* "fillEnabled" */
};
class GradientStroke : public Gradient {
public:
GradientStroke() : Gradient(Object::Type::GStroke) {}
float width(int frameNo) const { return mWidth.value(frameNo); }
CapStyle capStyle() const { return mCapStyle; }
JoinStyle joinStyle() const { return mJoinStyle; }
float miterLimit() const { return mMiterLimit; }
bool hasDashInfo() const { return !mDash.empty(); }
void getDashInfo(int frameNo, std::vector<float> &result) const
{
return mDash.getDashInfo(frameNo, result);
}
public:
Property<float> mWidth; /* "w" */
CapStyle mCapStyle{CapStyle::Flat}; /* "lc" */
JoinStyle mJoinStyle{JoinStyle::Miter}; /* "lj" */
float mMiterLimit{0}; /* "ml" */
Dash mDash;
};
class GradientFill : public Gradient {
public:
GradientFill() : Gradient(Object::Type::GFill) {}
FillRule fillRule() const { return mFillRule; }
public:
FillRule mFillRule{FillRule::Winding}; /* "r" */
};
class Fill : public Object {
public:
Fill() : Object(Object::Type::Fill) {}
Color color(int frameNo) const { return mColor.value(frameNo); }
float opacity(int frameNo) const
{
return mOpacity.value(frameNo) / 100.0f;
}
FillRule fillRule() const { return mFillRule; }
public:
FillRule mFillRule{FillRule::Winding}; /* "r" */
bool mEnabled{true}; /* "fillEnabled" */
Property<Color> mColor; /* "c" */
Property<float> mOpacity{100}; /* "o" */
};
class Shape : public Object {
public:
explicit Shape(Object::Type type) : Object(type) {}
VPath::Direction direction()
{
return (mDirection == 3) ? VPath::Direction::CCW : VPath::Direction::CW;
}
public:
int mDirection{1};
};
class Path : public Shape {
public:
Path() : Shape(Object::Type::Path) {}
public:
Property<PathData> mShape;
};
class RoundedCorner : public Object {
public:
RoundedCorner() : Object(Object::Type::RoundedCorner) {}
float radius(int frameNo) const { return mRadius.value(frameNo);}
public:
Property<float> mRadius{0};
};
class Rect : public Shape {
public:
Rect() : Shape(Object::Type::Rect) {}
float roundness(int frameNo)
{
return mRoundedCorner ? mRoundedCorner->radius(frameNo) :
mRound.value(frameNo);
}
bool roundnessChanged(int prevFrame, int curFrame)
{
return mRoundedCorner ? mRoundedCorner->mRadius.changed(prevFrame, curFrame) :
mRound.changed(prevFrame, curFrame);
}
public:
RoundedCorner* mRoundedCorner{nullptr};
Property<VPointF> mPos;
Property<VPointF> mSize;
Property<float> mRound{0};
};
class Ellipse : public Shape {
public:
Ellipse() : Shape(Object::Type::Ellipse) {}
public:
Property<VPointF> mPos;
Property<VPointF> mSize;
};
class Polystar : public Shape {
public:
enum class PolyType { Star = 1, Polygon = 2 };
Polystar() : Shape(Object::Type::Polystar) {}
public:
Polystar::PolyType mPolyType{PolyType::Polygon};
Property<VPointF> mPos;
Property<float> mPointCount{0};
Property<float> mInnerRadius{0};
Property<float> mOuterRadius{0};
Property<float> mInnerRoundness{0};
Property<float> mOuterRoundness{0};
Property<float> mRotation{0};
};
class Repeater : public Object {
public:
struct Transform {
VMatrix matrix(int frameNo, float multiplier) const;
float startOpacity(int frameNo) const
{
return mStartOpacity.value(frameNo) / 100;
}
float endOpacity(int frameNo) const
{
return mEndOpacity.value(frameNo) / 100;
}
bool isStatic() const
{
return mRotation.isStatic() && mScale.isStatic() &&
mPosition.isStatic() && mAnchor.isStatic() &&
mStartOpacity.isStatic() && mEndOpacity.isStatic();
}
Property<float> mRotation{0}; /* "r" */
Property<VPointF> mScale{{100, 100}}; /* "s" */
Property<VPointF> mPosition; /* "p" */
Property<VPointF> mAnchor; /* "a" */
Property<float> mStartOpacity{100}; /* "so" */
Property<float> mEndOpacity{100}; /* "eo" */
};
Repeater() : Object(Object::Type::Repeater) {}
Group *content() const { return mContent ? mContent : nullptr; }
void setContent(Group *content) { mContent = content; }
int maxCopies() const { return int(mMaxCopies); }
float copies(int frameNo) const { return mCopies.value(frameNo); }
float offset(int frameNo) const { return mOffset.value(frameNo); }
bool processed() const { return mProcessed; }
void markProcessed() { mProcessed = true; }
public:
Group * mContent{nullptr};
Transform mTransform;
Property<float> mCopies{0};
Property<float> mOffset{0};
float mMaxCopies{0.0};
bool mProcessed{false};
};
class Trim : public Object {
public:
struct Segment {
float start{0};
float end{0};
Segment() = default;
explicit Segment(float s, float e) : start(s), end(e) {}
};
enum class TrimType { Simultaneously, Individually };
Trim() : Object(Object::Type::Trim) {}
/*
* if start > end vector trims the path as a loop ( 2 segment)
* if start < end vector trims the path without loop ( 1 segment).
* if no offset then there is no loop.
*/
Segment segment(int frameNo) const
{
float start = mStart.value(frameNo) / 100.0f;
float end = mEnd.value(frameNo) / 100.0f;
float offset = std::fmod(mOffset.value(frameNo), 360.0f) / 360.0f;
float diff = std::abs(start - end);
if (vCompare(diff, 0.0f)) return Segment(0, 0);
if (vCompare(diff, 1.0f)) return Segment(0, 1);
if (offset > 0) {
start += offset;
end += offset;
if (start <= 1 && end <= 1) {
return noloop(start, end);
} else if (start > 1 && end > 1) {
return noloop(start - 1, end - 1);
} else {
return (start > 1) ? loop(start - 1, end)
: loop(start, end - 1);
}
} else {
start += offset;
end += offset;
if (start >= 0 && end >= 0) {
return noloop(start, end);
} else if (start < 0 && end < 0) {
return noloop(1 + start, 1 + end);
} else {
return (start < 0) ? loop(1 + start, end)
: loop(start, 1 + end);
}
}
}
Trim::TrimType type() const { return mTrimType; }
private:
Segment noloop(float start, float end) const
{
assert(start >= 0);
assert(end >= 0);
Segment s;
s.start = std::min(start, end);
s.end = std::max(start, end);
return s;
}
Segment loop(float start, float end) const
{
assert(start >= 0);
assert(end >= 0);
Segment s;
s.start = std::max(start, end);
s.end = std::min(start, end);
return s;
}
public:
Property<float> mStart{0};
Property<float> mEnd{0};
Property<float> mOffset{0};
Trim::TrimType mTrimType{TrimType::Simultaneously};
};
inline Gradient::Data operator+(const Gradient::Data &g1,
const Gradient::Data &g2)
{
if (g1.mGradient.size() != g2.mGradient.size()) return g1;
Gradient::Data newG;
newG.mGradient = g1.mGradient;
auto g2It = g2.mGradient.begin();
for (auto &i : newG.mGradient) {
i = i + *g2It;
g2It++;
}
return newG;
}
inline Gradient::Data operator-(const Gradient::Data &g1,
const Gradient::Data &g2)
{
if (g1.mGradient.size() != g2.mGradient.size()) return g1;
Gradient::Data newG;
newG.mGradient = g1.mGradient;
auto g2It = g2.mGradient.begin();
for (auto &i : newG.mGradient) {
i = i - *g2It;
g2It++;
}
return newG;
}
inline Gradient::Data operator*(float m, const Gradient::Data &g)
{
Gradient::Data newG;
newG.mGradient = g.mGradient;
for (auto &i : newG.mGradient) {
i = i * m;
}
return newG;
}
using ColorFilter = std::function<void(float &, float &, float &)>;
void configureModelCacheSize(size_t cacheSize);
std::shared_ptr<model::Composition> loadFromFile(const std::string &filePath,
bool cachePolicy);
std::shared_ptr<model::Composition> loadFromData(std::string jsonData,
const std::string &key,
std::string resourcePath,
bool cachePolicy);
std::shared_ptr<model::Composition> loadFromData(std::string jsonData,
std::string resourcePath,
ColorFilter filter);
std::shared_ptr<model::Composition> parse(char *str, std::string dir_path,
ColorFilter filter = {});
} // namespace model
} // namespace internal
} // namespace rlottie
#endif // LOTModel_H