2020-06-08 16:03:48 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2020 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef SkRive_DEFINED
|
|
|
|
#define SkRive_DEFINED
|
|
|
|
|
2020-06-18 18:56:14 +00:00
|
|
|
#include "include/core/SkBlendMode.h"
|
2020-06-19 18:21:33 +00:00
|
|
|
#include "include/core/SkColor.h"
|
2020-06-09 18:00:54 +00:00
|
|
|
#include "include/core/SkM44.h"
|
2020-06-23 03:39:46 +00:00
|
|
|
#include "include/core/SkPaint.h"
|
|
|
|
#include "include/core/SkPathTypes.h"
|
2020-06-09 18:00:54 +00:00
|
|
|
#include "include/core/SkRefCnt.h"
|
|
|
|
#include "include/core/SkString.h"
|
|
|
|
|
|
|
|
#include <memory>
|
2020-06-19 18:21:33 +00:00
|
|
|
#include <type_traits>
|
2020-06-09 18:00:54 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
class SkCanvas;
|
2020-06-23 03:39:46 +00:00
|
|
|
class SkPaint;
|
2020-06-09 18:00:54 +00:00
|
|
|
class SkStreamAsset;
|
|
|
|
|
|
|
|
namespace skrive {
|
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
#define ACTOR_ATTR(attr_name, attr_type, attr_default) \
|
|
|
|
private: \
|
|
|
|
attr_type f##attr_name = attr_default; \
|
|
|
|
public: \
|
|
|
|
const attr_type& get##attr_name() const { return f##attr_name; } \
|
|
|
|
void set##attr_name(const attr_type& v) { \
|
|
|
|
if (f##attr_name == v) return; \
|
|
|
|
f##attr_name = v; \
|
|
|
|
this->invalidate(); \
|
|
|
|
} \
|
|
|
|
void set##attr_name(attr_type&& v) { \
|
|
|
|
if (f##attr_name == v) return; \
|
|
|
|
f##attr_name = std::move(v); \
|
|
|
|
this->invalidate(); \
|
|
|
|
}
|
|
|
|
|
|
|
|
class Node;
|
|
|
|
|
|
|
|
class Component : public SkRefCnt {
|
2020-06-18 18:56:14 +00:00
|
|
|
public:
|
2020-06-19 18:21:33 +00:00
|
|
|
ACTOR_ATTR(Name, SkString, SkString())
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
std::enable_if_t<std::is_base_of<Component, T>::value, bool>
|
|
|
|
is() const {
|
|
|
|
if constexpr(std::is_same<Component, T>::value) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return is_base_of<T>(fType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
operator const T*() const {
|
|
|
|
return this->is<T>() ? reinterpret_cast<const T*>(this) : nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
operator T*() {
|
|
|
|
return this->is<T>() ? reinterpret_cast<T*>(this) : nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void revalidate();
|
2020-06-18 18:56:14 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
enum class Type : uint32_t {
|
2020-06-19 18:21:33 +00:00
|
|
|
kNode,
|
|
|
|
kShape,
|
2020-06-23 03:39:46 +00:00
|
|
|
kColorPaint,
|
2020-06-24 01:42:25 +00:00
|
|
|
kEllipse,
|
2020-06-18 18:56:14 +00:00
|
|
|
};
|
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
explicit Component(Type t) : fType(t) {}
|
|
|
|
|
|
|
|
void invalidate();
|
2020-06-18 18:56:14 +00:00
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
bool hasInval() const { return fDirty; }
|
2020-06-18 18:56:14 +00:00
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
virtual void onRevalidate() = 0;
|
2020-06-18 18:56:14 +00:00
|
|
|
|
|
|
|
private:
|
2020-06-19 18:21:33 +00:00
|
|
|
friend class Node; // parent access
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
static constexpr bool is_base_of(Type t);
|
2020-06-18 18:56:14 +00:00
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
const Type fType;
|
2020-06-18 18:56:14 +00:00
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
Node* fParent = nullptr;
|
|
|
|
bool fDirty = true;
|
2020-06-18 18:56:14 +00:00
|
|
|
};
|
|
|
|
|
2020-06-24 01:42:25 +00:00
|
|
|
class TransformableComponent : public Component {
|
2020-06-18 18:56:14 +00:00
|
|
|
public:
|
2020-06-19 18:21:33 +00:00
|
|
|
ACTOR_ATTR(Translation , SkV2 , SkV2({0, 0}))
|
|
|
|
ACTOR_ATTR(Scale , SkV2 , SkV2({1, 1}))
|
|
|
|
ACTOR_ATTR(Rotation , float, 0 )
|
|
|
|
ACTOR_ATTR(Opacity , float, 1 )
|
2020-06-24 01:42:25 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
explicit TransformableComponent(Type t) : INHERITED(t) {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
using INHERITED = Component;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Node : public TransformableComponent {
|
|
|
|
public:
|
|
|
|
Node() : INHERITED(Type::kNode) {}
|
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
ACTOR_ATTR(CollapsedVisibility, bool , false )
|
|
|
|
|
|
|
|
void addChild(sk_sp<Component>);
|
|
|
|
|
|
|
|
protected:
|
|
|
|
explicit Node(Type t) : INHERITED(t) {}
|
2020-06-18 18:56:14 +00:00
|
|
|
|
|
|
|
private:
|
2020-06-19 18:21:33 +00:00
|
|
|
void onRevalidate() override;
|
|
|
|
|
|
|
|
std::vector<sk_sp<Component>> fChildren;
|
|
|
|
|
2020-06-24 01:42:25 +00:00
|
|
|
using INHERITED = TransformableComponent;
|
2020-06-19 18:21:33 +00:00
|
|
|
};
|
2020-06-18 18:56:14 +00:00
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
class Drawable : public Node {
|
|
|
|
public:
|
|
|
|
ACTOR_ATTR(DrawOrder, size_t , 0 )
|
|
|
|
ACTOR_ATTR(BlendMode, SkBlendMode, SkBlendMode::kSrcOver)
|
|
|
|
ACTOR_ATTR(IsHidden , bool , false )
|
|
|
|
|
|
|
|
protected:
|
|
|
|
explicit Drawable(Type t) : INHERITED(t) {}
|
|
|
|
|
|
|
|
private:
|
2020-06-18 18:56:14 +00:00
|
|
|
using INHERITED = Node;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Shape final : public Drawable {
|
|
|
|
public:
|
2020-06-19 18:21:33 +00:00
|
|
|
Shape() : INHERITED(Type::kShape) {}
|
|
|
|
|
|
|
|
ACTOR_ATTR(TransformAffectsStroke, bool, true)
|
2020-06-18 18:56:14 +00:00
|
|
|
|
|
|
|
private:
|
2020-06-19 18:21:33 +00:00
|
|
|
void onRevalidate() override;
|
2020-06-18 18:56:14 +00:00
|
|
|
|
|
|
|
using INHERITED = Drawable;
|
|
|
|
};
|
|
|
|
|
2020-06-23 03:39:46 +00:00
|
|
|
class Paint : public Component {
|
|
|
|
public:
|
|
|
|
ACTOR_ATTR(Opacity , float , 1 )
|
|
|
|
ACTOR_ATTR(FillRule , SkPathFillType, SkPathFillType::kWinding )
|
|
|
|
ACTOR_ATTR(StrokeWidth, float , 1 )
|
|
|
|
ACTOR_ATTR(StrokeCap , SkPaint::Cap , SkPaint::Cap::kButt_Cap )
|
|
|
|
ACTOR_ATTR(StrokeJoin , SkPaint::Join , SkPaint::Join::kMiter_Join)
|
|
|
|
|
|
|
|
enum class StrokeTrim : uint8_t { kOff, kSequential, kSynced };
|
|
|
|
ACTOR_ATTR(StrokeTrim , StrokeTrim, StrokeTrim::kOff)
|
|
|
|
ACTOR_ATTR(StrokeTrimStart , float , 0)
|
|
|
|
ACTOR_ATTR(StrokeTrimEnd , float , 0)
|
|
|
|
ACTOR_ATTR(StrokeTrimOffset, float , 0)
|
|
|
|
|
|
|
|
void apply(SkPaint*) const;
|
|
|
|
|
|
|
|
SkPaint::Style style() const { return fStyle; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Paint(Type t, SkPaint::Style style) : INHERITED(t), fStyle(style) {}
|
|
|
|
|
|
|
|
virtual void onApply(SkPaint*) const = 0;
|
|
|
|
|
|
|
|
private:
|
|
|
|
const SkPaint::Style fStyle;
|
|
|
|
|
|
|
|
using INHERITED = Component;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ColorPaint final : public Paint {
|
|
|
|
public:
|
|
|
|
explicit ColorPaint(SkPaint::Style style) : INHERITED(Type::kColorPaint, style) {}
|
|
|
|
|
|
|
|
ACTOR_ATTR(Color, SkColor4f, SkColors::kBlack)
|
|
|
|
|
|
|
|
private:
|
|
|
|
void onRevalidate() override;
|
|
|
|
|
|
|
|
void onApply(SkPaint*) const override;
|
|
|
|
|
|
|
|
using INHERITED = Paint;
|
|
|
|
};
|
|
|
|
|
2020-06-24 01:42:25 +00:00
|
|
|
class Geometry : public TransformableComponent {
|
|
|
|
protected:
|
|
|
|
explicit Geometry(Type t) : INHERITED(t) {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
using INHERITED = TransformableComponent;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Ellipse final : public Geometry {
|
|
|
|
public:
|
|
|
|
Ellipse() : INHERITED(Type::kEllipse) {}
|
|
|
|
|
|
|
|
ACTOR_ATTR(Width , float, 0)
|
|
|
|
ACTOR_ATTR(Height, float, 0)
|
|
|
|
|
|
|
|
private:
|
|
|
|
void onRevalidate() override;
|
|
|
|
|
|
|
|
using INHERITED = Geometry;
|
|
|
|
};
|
|
|
|
|
2020-06-19 18:21:33 +00:00
|
|
|
template <typename T>
|
|
|
|
constexpr bool Component::is_base_of(Type t) {
|
2020-06-23 03:39:46 +00:00
|
|
|
if (t == Type::kNode ) return std::is_base_of<T, Node >::value;
|
|
|
|
if (t == Type::kShape ) return std::is_base_of<T, Shape >::value;
|
|
|
|
if (t == Type::kColorPaint) return std::is_base_of<T, ColorPaint>::value;
|
2020-06-24 01:42:25 +00:00
|
|
|
if (t == Type::kEllipse ) return std::is_base_of<T, Ellipse >::value;
|
2020-06-19 18:21:33 +00:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
class Artboard final : public SkRefCnt {
|
2020-06-09 18:00:54 +00:00
|
|
|
public:
|
2020-06-24 01:42:25 +00:00
|
|
|
ACTOR_ATTR(Root , sk_sp<Node>, nullptr )
|
|
|
|
ACTOR_ATTR(Name , SkString , SkString() )
|
|
|
|
ACTOR_ATTR(Color , SkColor4f , SkColors::kBlack)
|
|
|
|
ACTOR_ATTR(Size , SkV2 , SkV2({0,0}) )
|
|
|
|
ACTOR_ATTR(Origin , SkV2 , SkV2({0,0}) )
|
|
|
|
ACTOR_ATTR(Translation , SkV2 , SkV2({0,0}) )
|
|
|
|
ACTOR_ATTR(ClipContents, bool , false )
|
2020-06-19 18:21:33 +00:00
|
|
|
|
|
|
|
void render(SkCanvas*) const;
|
2020-06-09 18:00:54 +00:00
|
|
|
|
|
|
|
private:
|
2020-06-19 18:21:33 +00:00
|
|
|
void invalidate() {}
|
2020-06-09 18:00:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class SK_API SkRive final : public SkNVRefCnt<SkRive> {
|
|
|
|
public:
|
|
|
|
class Builder final {
|
|
|
|
public:
|
|
|
|
sk_sp<SkRive> make(std::unique_ptr<SkStreamAsset>);
|
|
|
|
};
|
|
|
|
|
|
|
|
const std::vector<sk_sp<Artboard>>& artboards() const { return fArtboards; }
|
|
|
|
std::vector<sk_sp<Artboard>>& artboards() { return fArtboards; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<sk_sp<Artboard>> fArtboards;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // skrive
|
|
|
|
|
2020-06-08 16:03:48 +00:00
|
|
|
#endif // SkRive_DEFINED
|