kaliber/src/engine/animator.cc

270 lines
7.7 KiB
C++

#include "engine/animator.h"
#include "base/interpolation.h"
#include "base/log.h"
#include "engine/animatable.h"
#include "engine/engine.h"
using namespace base;
namespace eng {
Animator::Animator() {
Engine::Get().AddAnimator(this);
}
Animator::~Animator() {
Engine::Get().RemoveAnimator(this);
}
void Animator::Attach(Animatable* animatable) {
elements_.push_back({animatable,
{0, 0},
0,
animatable->GetColor(),
(int)animatable->GetFrame()});
}
void Animator::Play(int animation, bool loop) {
play_flags_ |= animation;
if (loop)
loop_flags_ |= animation;
else
loop_flags_ &= ~animation;
}
void Animator::Pause(int animation) {
play_flags_ &= ~animation;
}
void Animator::Stop(int animation) {
if ((animation & kMovement) != 0)
movement_time_ = 0;
if ((animation & kRotation) != 0)
rotation_time_ = 0;
if ((animation & kBlending) != 0)
blending_time_ = 0;
if ((animation & kFrames) != 0)
frame_time_ = 0;
if ((animation & kTimer) != 0)
timer_time_ = 0;
play_flags_ |= animation;
Evaluate(0);
play_flags_ &= ~animation;
loop_flags_ &= ~animation;
}
void Animator::PauseOrResumeAll(bool pause) {
if (pause) {
resume_flags_ = play_flags_;
play_flags_ = 0;
} else {
play_flags_ = resume_flags_;
resume_flags_ = 0;
}
}
float Animator::GetTime(int animation) {
if ((animation & kMovement) != 0)
return movement_time_;
if ((animation & kRotation) != 0)
return rotation_time_;
if ((animation & kBlending) != 0)
return blending_time_;
if ((animation & kFrames) != 0)
return frame_time_;
return timer_time_;
}
void Animator::SetTime(int animation, float time, bool force_update) {
DCHECK(time >= 0 && time <= 1);
if ((animation & kMovement) != 0)
movement_time_ = time;
if ((animation & kRotation) != 0)
rotation_time_ = time;
if ((animation & kBlending) != 0)
blending_time_ = time;
if ((animation & kFrames) != 0)
frame_time_ = time;
if ((animation & kTimer) != 0)
timer_time_ = time;
if (force_update) {
unsigned play_flags = play_flags_;
play_flags_ = animation;
Evaluate(0);
play_flags_ = play_flags;
}
}
void Animator::SetEndCallback(int animation, base::Closure cb) {
if ((inside_cb_ & animation) != 0) {
has_pending_cb_ = true;
pending_cb_ = std::move(cb);
} else if ((animation & kMovement) != 0)
movement_cb_ = std::move(cb);
else if ((animation & kRotation) != 0)
rotation_cb_ = std::move(cb);
else if ((animation & kBlending) != 0)
blending_cb_ = std::move(cb);
else if ((animation & kFrames) != 0)
frame_cb_ = std::move(cb);
else if ((animation & kTimer) != 0)
timer_cb_ = std::move(cb);
}
void Animator::SetMovement(Vector2f direction,
float duration,
Interpolator interpolator) {
movement_direction_ = direction;
movement_speed_ = 1.0f / duration;
movement_interpolator_ = std::move(interpolator);
for (auto& a : elements_)
a.movement_last_pos = {0, 0};
}
void Animator::SetRotation(float target,
float duration,
Interpolator interpolator) {
rotation_target_ = target;
rotation_speed_ = 1.0f / duration;
rotation_interpolator_ = std::move(interpolator);
for (auto& a : elements_)
a.rotation_last_theta = 0;
}
void Animator::SetBlending(Vector4f target,
float duration,
Interpolator interpolator) {
blending_target_ = target;
blending_speed_ = 1.0f / duration;
for (auto& a : elements_)
a.blending_start = a.animatable->GetColor();
blending_interpolator_ = std::move(interpolator);
}
void Animator::SetFrames(int count,
int frames_per_second,
Interpolator interpolator) {
frame_count_ = count;
frame_speed_ = (float)frames_per_second / (float)count;
for (auto& a : elements_)
a.frame_start = a.animatable->GetFrame();
frame_interpolator_ = std::move(interpolator);
}
void Animator::SetTimer(float duration) {
timer_speed_ = 1.0f / duration;
}
void Animator::SetVisible(bool visible) {
for (auto& a : elements_)
a.animatable->SetVisible(visible);
}
void Animator::Update(float delta_time) {
if (play_flags_ & kMovement)
UpdateAnimTime(delta_time, kMovement, movement_speed_, movement_time_,
movement_cb_);
if (play_flags_ & kRotation)
UpdateAnimTime(delta_time, kRotation, rotation_speed_, rotation_time_,
rotation_cb_);
if (play_flags_ & kBlending)
UpdateAnimTime(delta_time, kBlending, blending_speed_, blending_time_,
blending_cb_);
if (play_flags_ & kFrames)
UpdateAnimTime(delta_time, kFrames, frame_speed_, frame_time_, frame_cb_);
if (play_flags_ & kTimer)
UpdateAnimTime(delta_time, kTimer, timer_speed_, timer_time_, timer_cb_);
}
void Animator::Evaluate(float frame_frac) {
if (play_flags_ & kMovement)
EvaluateAnimation<Vector2f, true>(
frame_frac, movement_speed_, movement_time_, movement_interpolator_,
movement_direction_, [](Element& a, Vector2f pos) {
a.animatable->Translate(pos - a.movement_last_pos);
a.movement_last_pos = pos;
});
if (play_flags_ & kRotation)
EvaluateAnimation<float, true>(
frame_frac, rotation_speed_, rotation_time_, rotation_interpolator_,
rotation_target_, [](Element& a, float theta) {
a.animatable->Rotate(theta - a.rotation_last_theta);
a.rotation_last_theta = theta;
});
if (play_flags_ & kBlending)
EvaluateAnimation<float, false>(
frame_frac, blending_speed_, blending_time_, blending_interpolator_, {},
[&](Element& a, float interpolated_time) {
Vector4f blending =
base::Lerp(a.blending_start, blending_target_, interpolated_time);
a.animatable->SetColor(blending);
});
if (play_flags_ & kFrames)
EvaluateAnimation<float, false>(
frame_frac, frame_speed_, frame_time_, frame_interpolator_, {},
[&](Element& a, float interpolated_time) {
int target = a.frame_start + frame_count_;
int frame = base::Lerp(a.frame_start, target, interpolated_time);
if (frame < target)
a.animatable->SetFrame(frame);
});
}
void Animator::UpdateAnimTime(float delta_time,
int anim,
float anim_speed,
float& anim_time,
base::Closure& cb) {
anim_time += anim_speed * delta_time;
if (anim_time > 1.0f) {
if (loop_flags_ & anim) {
anim_time = fmod(anim_time, 1.0f);
} else {
anim_time = 1.0f;
Evaluate(0);
anim_time = 0;
play_flags_ &= ~anim;
if (cb) {
inside_cb_ = (Flags)anim;
cb();
inside_cb_ = kNone;
if (has_pending_cb_) {
has_pending_cb_ = false;
cb = std::move(pending_cb_);
}
}
}
}
}
template <typename T, bool relative>
void Animator::EvaluateAnimation(float frame_frac,
float anim_speed,
float anim_time,
Interpolator& interpolator,
T target,
ResultSetter<T> set_result) {
float time = anim_time + anim_speed * frame_frac;
float interpolated_time = interpolator ? interpolator(time) : time;
if constexpr (relative) {
T result = base::Lerp(T{0}, target, interpolated_time);
for (auto& a : elements_)
set_result(a, result);
} else {
for (auto& a : elements_)
set_result(a, interpolated_time);
}
}
} // namespace eng