2018-08-29 20:38:13 +00:00
|
|
|
/*************************************************************************/
|
|
|
|
/* animation_tree.h */
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* GODOT ENGINE */
|
|
|
|
/* https://godotengine.org */
|
|
|
|
/*************************************************************************/
|
2022-01-03 20:27:34 +00:00
|
|
|
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
|
|
|
|
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
|
2018-08-29 20:38:13 +00:00
|
|
|
/* */
|
|
|
|
/* 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. */
|
|
|
|
/*************************************************************************/
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
#ifndef ANIMATION_GRAPH_PLAYER_H
|
|
|
|
#define ANIMATION_GRAPH_PLAYER_H
|
|
|
|
|
|
|
|
#include "animation_player.h"
|
2020-03-26 21:49:16 +00:00
|
|
|
#include "scene/3d/node_3d.h"
|
|
|
|
#include "scene/3d/skeleton_3d.h"
|
2018-06-19 01:10:48 +00:00
|
|
|
#include "scene/resources/animation.h"
|
|
|
|
|
|
|
|
class AnimationNodeBlendTree;
|
|
|
|
class AnimationPlayer;
|
2018-06-25 21:40:24 +00:00
|
|
|
class AnimationTree;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
class AnimationNode : public Resource {
|
2019-03-19 18:35:57 +00:00
|
|
|
GDCLASS(AnimationNode, Resource);
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
public:
|
|
|
|
enum FilterAction {
|
|
|
|
FILTER_IGNORE,
|
|
|
|
FILTER_PASS,
|
|
|
|
FILTER_STOP,
|
|
|
|
FILTER_BLEND
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Input {
|
|
|
|
String name;
|
|
|
|
};
|
|
|
|
|
|
|
|
Vector<Input> inputs;
|
|
|
|
|
2018-06-25 21:40:24 +00:00
|
|
|
friend class AnimationTree;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
struct AnimationState {
|
|
|
|
Ref<Animation> animation;
|
2021-05-21 06:42:37 +00:00
|
|
|
double time = 0.0;
|
|
|
|
double delta = 0.0;
|
2021-08-09 22:15:17 +00:00
|
|
|
const Vector<real_t> *track_blends = nullptr;
|
|
|
|
real_t blend = 0.0;
|
2021-02-07 21:29:31 +00:00
|
|
|
bool seeked = false;
|
2021-10-15 13:25:00 +00:00
|
|
|
int pingponged = 0;
|
2018-06-19 01:10:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct State {
|
2021-02-07 21:29:31 +00:00
|
|
|
int track_count = 0;
|
2018-06-19 01:10:48 +00:00
|
|
|
HashMap<NodePath, int> track_map;
|
|
|
|
List<AnimationState> animation_states;
|
2021-02-07 21:29:31 +00:00
|
|
|
bool valid = false;
|
|
|
|
AnimationPlayer *player = nullptr;
|
|
|
|
AnimationTree *tree = nullptr;
|
2018-06-19 01:10:48 +00:00
|
|
|
String invalid_reasons;
|
2021-02-07 21:29:31 +00:00
|
|
|
uint64_t last_pass = 0;
|
2018-06-19 01:10:48 +00:00
|
|
|
};
|
|
|
|
|
2021-08-09 22:15:17 +00:00
|
|
|
Vector<real_t> blends;
|
2021-02-07 21:29:31 +00:00
|
|
|
State *state = nullptr;
|
2018-08-23 19:44:10 +00:00
|
|
|
|
2022-01-28 20:07:30 +00:00
|
|
|
double _pre_process(const StringName &p_base_path, AnimationNode *p_parent, State *p_state, double p_time, bool p_seek, const Vector<StringName> &p_connections);
|
2018-06-19 01:10:48 +00:00
|
|
|
|
2018-08-20 16:38:18 +00:00
|
|
|
//all this is temporary
|
|
|
|
StringName base_path;
|
|
|
|
Vector<StringName> connections;
|
2021-02-07 21:29:31 +00:00
|
|
|
AnimationNode *parent = nullptr;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
HashMap<NodePath, bool> filter;
|
2021-02-07 21:29:31 +00:00
|
|
|
bool filter_enabled = false;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
Array _get_filters() const;
|
|
|
|
void _set_filters(const Array &p_filters);
|
2018-08-20 16:38:18 +00:00
|
|
|
friend class AnimationNodeBlendTree;
|
2022-01-28 20:07:30 +00:00
|
|
|
double _blend_node(const StringName &p_subpath, const Vector<StringName> &p_connections, AnimationNode *p_new_parent, Ref<AnimationNode> p_node, double p_time, bool p_seek, real_t p_blend, FilterAction p_filter = FILTER_IGNORE, bool p_optimize = true, real_t *r_max = nullptr);
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
protected:
|
2022-01-28 20:07:30 +00:00
|
|
|
void blend_animation(const StringName &p_animation, double p_time, double p_delta, bool p_seeked, real_t p_blend, int p_pingponged = 0);
|
|
|
|
double blend_node(const StringName &p_sub_path, Ref<AnimationNode> p_node, double p_time, bool p_seek, real_t p_blend, FilterAction p_filter = FILTER_IGNORE, bool p_optimize = true);
|
|
|
|
double blend_input(int p_input, double p_time, bool p_seek, real_t p_blend, FilterAction p_filter = FILTER_IGNORE, bool p_optimize = true);
|
2021-10-15 13:25:00 +00:00
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
void make_invalid(const String &p_reason);
|
|
|
|
|
|
|
|
static void _bind_methods();
|
|
|
|
|
2020-07-10 10:34:39 +00:00
|
|
|
void _validate_property(PropertyInfo &property) const override;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
2021-08-22 01:52:44 +00:00
|
|
|
GDVIRTUAL0RC(Dictionary, _get_child_nodes)
|
|
|
|
GDVIRTUAL0RC(Array, _get_parameter_list)
|
|
|
|
GDVIRTUAL1RC(Ref<AnimationNode>, _get_child_by_name, StringName)
|
|
|
|
GDVIRTUAL1RC(Variant, _get_parameter_default_value, StringName)
|
|
|
|
GDVIRTUAL2RC(double, _process, double, bool)
|
|
|
|
GDVIRTUAL0RC(String, _get_caption)
|
|
|
|
GDVIRTUAL0RC(bool, _has_filter)
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
public:
|
2018-08-20 16:38:18 +00:00
|
|
|
virtual void get_parameter_list(List<PropertyInfo> *r_list) const;
|
|
|
|
virtual Variant get_parameter_default_value(const StringName &p_parameter) const;
|
|
|
|
|
|
|
|
void set_parameter(const StringName &p_name, const Variant &p_value);
|
|
|
|
Variant get_parameter(const StringName &p_name) const;
|
|
|
|
|
|
|
|
struct ChildNode {
|
|
|
|
StringName name;
|
|
|
|
Ref<AnimationNode> node;
|
|
|
|
};
|
|
|
|
|
|
|
|
virtual void get_child_nodes(List<ChildNode> *r_child_nodes);
|
2018-06-19 01:10:48 +00:00
|
|
|
|
2021-05-21 06:42:37 +00:00
|
|
|
virtual double process(double p_time, bool p_seek);
|
2018-06-21 18:45:44 +00:00
|
|
|
virtual String get_caption() const;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
int get_input_count() const;
|
|
|
|
String get_input_name(int p_input);
|
|
|
|
|
|
|
|
void add_input(const String &p_name);
|
|
|
|
void set_input_name(int p_input, const String &p_name);
|
|
|
|
void remove_input(int p_index);
|
|
|
|
|
|
|
|
void set_filter_path(const NodePath &p_path, bool p_enable);
|
|
|
|
bool is_path_filtered(const NodePath &p_path) const;
|
|
|
|
|
|
|
|
void set_filter_enabled(bool p_enable);
|
|
|
|
bool is_filter_enabled() const;
|
|
|
|
|
|
|
|
virtual bool has_filter() const;
|
|
|
|
|
2018-08-20 16:38:18 +00:00
|
|
|
virtual Ref<AnimationNode> get_child_by_name(const StringName &p_name);
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
AnimationNode();
|
|
|
|
};
|
|
|
|
|
|
|
|
VARIANT_ENUM_CAST(AnimationNode::FilterAction)
|
|
|
|
|
2018-06-21 18:45:44 +00:00
|
|
|
//root node does not allow inputs
|
|
|
|
class AnimationRootNode : public AnimationNode {
|
2019-03-19 18:35:57 +00:00
|
|
|
GDCLASS(AnimationRootNode, AnimationNode);
|
|
|
|
|
2018-06-21 18:45:44 +00:00
|
|
|
public:
|
|
|
|
AnimationRootNode() {}
|
|
|
|
};
|
|
|
|
|
2018-06-25 21:40:24 +00:00
|
|
|
class AnimationTree : public Node {
|
2019-03-19 18:35:57 +00:00
|
|
|
GDCLASS(AnimationTree, Node);
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
public:
|
2021-02-18 18:52:29 +00:00
|
|
|
enum AnimationProcessCallback {
|
2018-06-19 01:10:48 +00:00
|
|
|
ANIMATION_PROCESS_PHYSICS,
|
|
|
|
ANIMATION_PROCESS_IDLE,
|
2018-08-02 07:22:24 +00:00
|
|
|
ANIMATION_PROCESS_MANUAL,
|
2018-06-19 01:10:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct TrackCache {
|
2021-02-07 21:29:31 +00:00
|
|
|
bool root_motion = false;
|
|
|
|
uint64_t setup_pass = 0;
|
|
|
|
uint64_t process_pass = 0;
|
|
|
|
Animation::TrackType type = Animation::TrackType::TYPE_ANIMATION;
|
|
|
|
Object *object = nullptr;
|
2018-06-19 01:10:48 +00:00
|
|
|
ObjectID object_id;
|
|
|
|
|
|
|
|
TrackCache() {
|
|
|
|
}
|
|
|
|
virtual ~TrackCache() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TrackCacheTransform : public TrackCache {
|
2021-03-17 23:35:42 +00:00
|
|
|
#ifndef _3D_DISABLED
|
|
|
|
Node3D *node_3d = nullptr;
|
2021-02-07 21:29:31 +00:00
|
|
|
Skeleton3D *skeleton = nullptr;
|
2021-03-17 23:35:42 +00:00
|
|
|
#endif // _3D_DISABLED
|
2021-02-07 21:29:31 +00:00
|
|
|
int bone_idx = -1;
|
Remove animation 3D transform track, replace by loc/rot/scale tracks.
* `Animation.TYPE_TRANSFORM3D` track is gone.
* Added POSITION_3D, ROTATION_3D, SCALE_3D tracks.
* GLTF2, Collada, FBX importers will only import the track types found.
* Skeleton3D bone poses are now Pos/Rot/Scale, pose matrix removed.
* AnimationPlayer and AnimationTree animate these tracks separately, only when found.
* Removed BakeReset code, is useless with these changes.
This is the first in a series of commits designed to make the animation system in Godot more useful, which includes:
* Better compatibility with Autodesk products
* Better reusability of animations across models (including retargeting).
* Proper animation compression.
* etc.
*Note* GLTF2 animation saving went broken with this PR, needs to be fixed in a subsequent one.
2021-10-11 22:20:58 +00:00
|
|
|
bool loc_used = false;
|
|
|
|
bool rot_used = false;
|
|
|
|
bool scale_used = false;
|
2022-03-15 16:35:25 +00:00
|
|
|
Vector3 init_loc = Vector3(0, 0, 0);
|
2022-04-16 19:34:28 +00:00
|
|
|
Quaternion init_rot = Quaternion(0, 0, 0, 1);
|
2022-03-15 16:35:25 +00:00
|
|
|
Vector3 init_scale = Vector3(1, 1, 1);
|
2018-06-19 01:10:48 +00:00
|
|
|
Vector3 loc;
|
2021-01-20 07:02:02 +00:00
|
|
|
Quaternion rot;
|
2018-06-19 01:10:48 +00:00
|
|
|
Vector3 scale;
|
|
|
|
|
|
|
|
TrackCacheTransform() {
|
Remove animation 3D transform track, replace by loc/rot/scale tracks.
* `Animation.TYPE_TRANSFORM3D` track is gone.
* Added POSITION_3D, ROTATION_3D, SCALE_3D tracks.
* GLTF2, Collada, FBX importers will only import the track types found.
* Skeleton3D bone poses are now Pos/Rot/Scale, pose matrix removed.
* AnimationPlayer and AnimationTree animate these tracks separately, only when found.
* Removed BakeReset code, is useless with these changes.
This is the first in a series of commits designed to make the animation system in Godot more useful, which includes:
* Better compatibility with Autodesk products
* Better reusability of animations across models (including retargeting).
* Proper animation compression.
* etc.
*Note* GLTF2 animation saving went broken with this PR, needs to be fixed in a subsequent one.
2021-10-11 22:20:58 +00:00
|
|
|
type = Animation::TYPE_POSITION_3D;
|
2018-06-19 01:10:48 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-10-15 22:04:35 +00:00
|
|
|
struct TrackCacheBlendShape : public TrackCache {
|
|
|
|
MeshInstance3D *mesh_3d = nullptr;
|
2022-02-05 05:16:43 +00:00
|
|
|
float init_value = 0;
|
2021-10-15 22:04:35 +00:00
|
|
|
float value = 0;
|
|
|
|
int shape_index = -1;
|
|
|
|
TrackCacheBlendShape() { type = Animation::TYPE_BLEND_SHAPE; }
|
|
|
|
};
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
struct TrackCacheValue : public TrackCache {
|
2022-02-05 05:16:43 +00:00
|
|
|
Variant init_value;
|
2018-06-19 01:10:48 +00:00
|
|
|
Variant value;
|
|
|
|
Vector<StringName> subpath;
|
|
|
|
TrackCacheValue() { type = Animation::TYPE_VALUE; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TrackCacheMethod : public TrackCache {
|
|
|
|
TrackCacheMethod() { type = Animation::TYPE_METHOD; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TrackCacheBezier : public TrackCache {
|
2022-02-05 05:16:43 +00:00
|
|
|
real_t init_value = 0.0;
|
2021-08-09 22:15:17 +00:00
|
|
|
real_t value = 0.0;
|
2018-06-19 01:10:48 +00:00
|
|
|
Vector<StringName> subpath;
|
|
|
|
TrackCacheBezier() {
|
|
|
|
type = Animation::TYPE_BEZIER;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TrackCacheAudio : public TrackCache {
|
2021-02-07 21:29:31 +00:00
|
|
|
bool playing = false;
|
2022-01-28 20:07:30 +00:00
|
|
|
double start = 0.0;
|
|
|
|
double len = 0.0;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
TrackCacheAudio() {
|
|
|
|
type = Animation::TYPE_AUDIO;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TrackCacheAnimation : public TrackCache {
|
2021-02-07 21:29:31 +00:00
|
|
|
bool playing = false;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
TrackCacheAnimation() {
|
|
|
|
type = Animation::TYPE_ANIMATION;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
HashMap<NodePath, TrackCache *> track_cache;
|
|
|
|
Set<TrackCache *> playing_caches;
|
|
|
|
|
|
|
|
Ref<AnimationNode> root;
|
|
|
|
|
2021-02-18 18:52:29 +00:00
|
|
|
AnimationProcessCallback process_callback = ANIMATION_PROCESS_IDLE;
|
2021-02-07 21:29:31 +00:00
|
|
|
bool active = false;
|
2018-06-19 01:10:48 +00:00
|
|
|
NodePath animation_player;
|
|
|
|
|
|
|
|
AnimationNode::State state;
|
2021-02-07 21:29:31 +00:00
|
|
|
bool cache_valid = false;
|
2018-06-19 01:10:48 +00:00
|
|
|
void _node_removed(Node *p_node);
|
|
|
|
|
|
|
|
void _clear_caches();
|
|
|
|
bool _update_caches(AnimationPlayer *player);
|
2022-01-28 20:07:30 +00:00
|
|
|
void _process_graph(double p_delta);
|
2018-06-19 01:10:48 +00:00
|
|
|
|
2021-02-07 21:29:31 +00:00
|
|
|
uint64_t setup_pass = 1;
|
|
|
|
uint64_t process_pass = 1;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
2021-02-07 21:29:31 +00:00
|
|
|
bool started = true;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
2018-06-26 22:05:11 +00:00
|
|
|
NodePath root_motion_track;
|
2020-10-17 05:08:21 +00:00
|
|
|
Transform3D root_motion_transform;
|
2018-06-26 22:05:11 +00:00
|
|
|
|
2018-08-20 16:38:18 +00:00
|
|
|
friend class AnimationNode;
|
2021-02-07 21:29:31 +00:00
|
|
|
bool properties_dirty = true;
|
2018-08-20 16:38:18 +00:00
|
|
|
void _tree_changed();
|
|
|
|
void _update_properties();
|
|
|
|
List<PropertyInfo> properties;
|
2020-03-17 06:33:00 +00:00
|
|
|
HashMap<StringName, HashMap<StringName, StringName>> property_parent_map;
|
2018-08-20 16:38:18 +00:00
|
|
|
HashMap<StringName, Variant> property_map;
|
|
|
|
|
2018-08-23 19:44:10 +00:00
|
|
|
struct Activity {
|
2021-02-07 21:29:31 +00:00
|
|
|
uint64_t last_pass = 0;
|
2021-08-09 22:15:17 +00:00
|
|
|
real_t activity = 0.0;
|
2018-08-23 19:44:10 +00:00
|
|
|
};
|
|
|
|
|
2020-03-17 06:33:00 +00:00
|
|
|
HashMap<StringName, Vector<Activity>> input_activity_map;
|
2018-08-23 19:44:10 +00:00
|
|
|
HashMap<StringName, Vector<Activity> *> input_activity_map_get;
|
|
|
|
|
2018-08-20 16:38:18 +00:00
|
|
|
void _update_properties_for_node(const String &p_base_path, Ref<AnimationNode> node);
|
|
|
|
|
2018-08-24 14:41:04 +00:00
|
|
|
ObjectID last_animation_player;
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
protected:
|
2018-08-20 16:38:18 +00:00
|
|
|
bool _set(const StringName &p_name, const Variant &p_value);
|
|
|
|
bool _get(const StringName &p_name, Variant &r_ret) const;
|
|
|
|
void _get_property_list(List<PropertyInfo> *p_list) const;
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
void _notification(int p_what);
|
|
|
|
static void _bind_methods();
|
|
|
|
|
|
|
|
public:
|
2018-06-27 06:00:08 +00:00
|
|
|
void set_tree_root(const Ref<AnimationNode> &p_root);
|
|
|
|
Ref<AnimationNode> get_tree_root() const;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
void set_active(bool p_active);
|
|
|
|
bool is_active() const;
|
|
|
|
|
2021-02-18 18:52:29 +00:00
|
|
|
void set_process_callback(AnimationProcessCallback p_mode);
|
|
|
|
AnimationProcessCallback get_process_callback() const;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
void set_animation_player(const NodePath &p_player);
|
|
|
|
NodePath get_animation_player() const;
|
|
|
|
|
2020-10-29 10:01:28 +00:00
|
|
|
TypedArray<String> get_configuration_warnings() const override;
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
bool is_state_invalid() const;
|
|
|
|
String get_invalid_state_reason() const;
|
|
|
|
|
2018-06-26 22:05:11 +00:00
|
|
|
void set_root_motion_track(const NodePath &p_track);
|
|
|
|
NodePath get_root_motion_track() const;
|
|
|
|
|
2020-10-17 05:08:21 +00:00
|
|
|
Transform3D get_root_motion_transform() const;
|
2018-06-26 22:05:11 +00:00
|
|
|
|
2021-08-09 22:15:17 +00:00
|
|
|
real_t get_connection_activity(const StringName &p_path, int p_connection) const;
|
|
|
|
void advance(real_t p_time);
|
2018-08-02 07:22:24 +00:00
|
|
|
|
2018-08-20 16:38:18 +00:00
|
|
|
void rename_parameter(const String &p_base, const String &p_new_base);
|
|
|
|
|
2018-06-19 01:10:48 +00:00
|
|
|
uint64_t get_last_process_pass() const;
|
2018-06-25 21:40:24 +00:00
|
|
|
AnimationTree();
|
|
|
|
~AnimationTree();
|
2018-06-19 01:10:48 +00:00
|
|
|
};
|
|
|
|
|
2021-02-18 18:52:29 +00:00
|
|
|
VARIANT_ENUM_CAST(AnimationTree::AnimationProcessCallback)
|
2018-06-19 01:10:48 +00:00
|
|
|
|
|
|
|
#endif // ANIMATION_GRAPH_PLAYER_H
|