AnimChain

This commit is contained in:
2025-05-13 17:23:59 +03:00
parent 77e4385c14
commit cfc67b8138

157
Tween.h
View File

@ -2,6 +2,9 @@
#include "Easing.h" #include "Easing.h"
#include "TimeWarp.h" #include "TimeWarp.h"
//----------------------------------------------------------------------
// EASING HELPERS
//----------------------------------------------------------------------
namespace EasingFunc { namespace EasingFunc {
// typedef float (*eFunc)(float); // тип easing-функции, как у Вас // typedef float (*eFunc)(float); // тип easing-функции, как у Вас
@ -26,6 +29,21 @@ namespace EasingFunc {
}; // namespace EasingFunc }; // namespace EasingFunc
//----------------------------------------------------------------------
// LISTENERИНТЕРФЕЙС
//----------------------------------------------------------------------
class Tween; // forward
class TweenListener {
public:
virtual void onTweenFinished(Tween& tween) = 0;
virtual void onTweenUpdate(Tween& tween) = 0;
virtual ~TweenListener() = default;
};
//----------------------------------------------------------------------
// TWEEN
//----------------------------------------------------------------------
class Tween { class Tween {
private: private:
static inline Tween* head = nullptr; static inline Tween* head = nullptr;
@ -33,6 +51,8 @@ private:
static inline uint32_t frameRateTimer; static inline uint32_t frameRateTimer;
Tween* next; Tween* next;
TweenListener* listener = nullptr;
public: public:
Tween(uint16_t fps = 30) { Tween(uint16_t fps = 30) {
setFps(fps); setFps(fps);
@ -53,6 +73,9 @@ public:
} }
} }
void setListener(TweenListener* l) { listener = l; }
TweenListener* getListener() const { return listener; }
/////// ///////
private: private:
@ -86,10 +109,11 @@ public:
if (!isPlayingF && triggerLastTick) { if (!isPlayingF && triggerLastTick) {
triggerLastTick = false; triggerLastTick = false;
current = to; current = to;
if (listener != nullptr) listener->onTweenFinished(*this);
return; return;
} }
if (!isPlayingF || easing == nullptr) return; if (!isPlayingF) return;
progress = constrain(progress + dt, 0, duration); progress = constrain(progress + dt, 0, duration);
float normProgress = constrain(progress / duration, 0, 1); float normProgress = constrain(progress / duration, 0, 1);
@ -102,6 +126,8 @@ public:
isPlayingF = false; isPlayingF = false;
} }
if (listener != nullptr) listener->onTweenUpdate(*this);
frameRateTimer = millis(); frameRateTimer = millis();
} }
} }
@ -110,20 +136,15 @@ public:
return constrain(progress / duration, 0, 1); return constrain(progress / duration, 0, 1);
} }
void start(float from, float to, uint16_t duration, EasingFunc::eFunc easing) { void start(float from_, float to_, uint16_t durationMs, EasingFunc::eFunc easing_ = nullptr) {
current = from; // если easing не задан, используем линейную
if (from == to) { easing = (easing_ != nullptr) ? easing_ : EasingFunc::easeLinear;
current = to;
isPlayingF = false;
triggerLastTick = true;
return;
}
this->from = from; from = from_;
this->to = to; to = to_;
this->duration = duration / 1000.0; duration = durationMs / 1000.0f;
this->easing = easing;
progress = 0; progress = 0;
current = from;
oldMillis = millis(); oldMillis = millis();
isPlayingF = true; isPlayingF = true;
triggerLastTick = false; triggerLastTick = false;
@ -178,4 +199,112 @@ private:
// else if (value > b) return b; // else if (value > b) return b;
// else return value; // else return value;
// } // }
}; };
//----------------------------------------------------------------------
// ANIMATION CHAIN
//----------------------------------------------------------------------
class AnimationChain : public TweenListener {
public:
using FromFunc = std::function<float()>;
struct Anim {
float from;
FromFunc fromFunc = nullptr;
float to;
uint16_t duration; // мс
EasingFunc::eFunc easing;
};
private:
std::vector<Anim> animations;
Tween tween; // единственный Tween
size_t currentIndex = 0;
bool chainPlaying = false;
float* currentUserPtr = nullptr;
void launchCurrent() {
const Anim& a = animations[currentIndex];
tween.start(a.fromFunc ? a.fromFunc() : a.from, a.to, a.duration, a.easing);
}
public:
float current = 0; // актуальное значение наружу
explicit AnimationChain(uint16_t fps = 30) : tween(fps) {
tween.setListener(this); // подписываемся
}
//------------------------------------------------------------------
// API ЦЕПОЧКИ
//------------------------------------------------------------------
void addAnim(float from, float to, uint16_t duration, EasingFunc::eFunc easing = nullptr) {
animations.push_back({from, {}, to, duration, easing});
}
void addAnim(FromFunc fromF, float to, uint16_t duration, EasingFunc::eFunc easing = nullptr) {
animations.push_back({0, fromF, to, duration, easing});
}
void clear() { animations.clear(); }
void start() {
if (animations.empty()) return;
currentIndex = 0;
chainPlaying = true;
launchCurrent();
Serial.printf("Chain start [%d]\n", currentIndex);
}
void stop() {
chainPlaying = false;
tween.stop();
}
void setCustomParam(float* val) {
currentUserPtr = val;
}
void resetCustomParam() {
currentUserPtr = nullptr;
}
//------------------------------------------------------------------
// STATE
//------------------------------------------------------------------
bool isChainPlaying() const { return chainPlaying; }
bool isAnimPlaying(size_t index) const {
return chainPlaying && index == currentIndex && tween.isPlaying();
}
size_t getCurrentIndex() const { return currentIndex; }
// void setCurrentIndex(size_t index){ currentIndex = index; }
private:
//------------------------------------------------------------------
// LISTENER CALLBACK
//------------------------------------------------------------------
void onTweenFinished(Tween& /*t*/) override {
++currentIndex;
if (currentIndex < animations.size()) {
Serial.printf("Chain next [%d]\n", currentIndex);
launchCurrent();
} else {
Serial.printf("Chain stop [x]\n", currentIndex);
chainPlaying = false; // вся цепочка закончилась
}
}
void onTweenUpdate(Tween& /*t*/) override {
if (chainPlaying) {
current = tween.current; // даём наружу актуальное значение
if(currentUserPtr != nullptr) *currentUserPtr = current;
} else {
current = *currentUserPtr;
}
}
};