mirror of https://github.com/procxx/kepka.git
Don't restart animations timer.
This commit is contained in:
parent
7808620764
commit
392d90dc1d
|
@ -14,7 +14,7 @@ namespace Ui {
|
||||||
namespace Animations {
|
namespace Animations {
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
constexpr auto kAnimationTimeout = crl::time(1000) / 60;
|
constexpr auto kAnimationTick = crl::time(1000) / 60;
|
||||||
constexpr auto kIgnoreUpdatesTimeout = crl::time(4);
|
constexpr auto kIgnoreUpdatesTimeout = crl::time(4);
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
@ -53,21 +53,22 @@ void Basic::markStopped() {
|
||||||
|
|
||||||
Manager::Manager() {
|
Manager::Manager() {
|
||||||
Core::Sandbox::Instance().widgetUpdateRequests(
|
Core::Sandbox::Instance().widgetUpdateRequests(
|
||||||
) | rpl::filter([=] {
|
/*) | rpl::filter([=] {
|
||||||
return (_lastUpdateTime + kIgnoreUpdatesTimeout < crl::now());
|
return (_lastUpdateTime + kIgnoreUpdatesTimeout < crl::now());
|
||||||
}) | rpl::start_with_next([=] {
|
}*/) | rpl::start_with_next([=] {
|
||||||
update();
|
update(UpdateSource::RepaintRequest);
|
||||||
}, _lifetime);
|
}, _lifetime);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::start(not_null<Basic*> animation) {
|
void Manager::start(not_null<Basic*> animation) {
|
||||||
_forceImmediateUpdate = true;
|
_forceUpdateProcessing = true;
|
||||||
if (_updating) {
|
if (_updating) {
|
||||||
_starting.emplace_back(animation.get());
|
_starting.emplace_back(animation.get());
|
||||||
} else {
|
return;
|
||||||
schedule();
|
|
||||||
_active.emplace_back(animation.get());
|
|
||||||
}
|
}
|
||||||
|
_active.emplace_back(animation.get());
|
||||||
|
startTimer();
|
||||||
|
updateQueued();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::stop(not_null<Basic*> animation) {
|
void Manager::stop(not_null<Basic*> animation) {
|
||||||
|
@ -89,15 +90,16 @@ void Manager::stop(not_null<Basic*> animation) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::update() {
|
void Manager::update(UpdateSource source) {
|
||||||
if (_active.empty() || _updating || _scheduled) {
|
if (_active.empty() || _updating) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const auto now = crl::now();
|
const auto now = crl::now();
|
||||||
if (_forceImmediateUpdate) {
|
if (_forceUpdateProcessing) {
|
||||||
_forceImmediateUpdate = false;
|
_forceUpdateProcessing = false;
|
||||||
|
} else if (now < _lastUpdateTime + kIgnoreUpdatesTimeout) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
schedule();
|
|
||||||
|
|
||||||
_updating = true;
|
_updating = true;
|
||||||
const auto guard = gsl::finally([&] { _updating = false; });
|
const auto guard = gsl::finally([&] { _updating = false; });
|
||||||
|
@ -108,48 +110,30 @@ void Manager::update() {
|
||||||
};
|
};
|
||||||
_active.erase(ranges::remove_if(_active, isFinished), end(_active));
|
_active.erase(ranges::remove_if(_active, isFinished), end(_active));
|
||||||
|
|
||||||
if (!empty(_starting)) {
|
if (empty(_starting)) {
|
||||||
_active.insert(
|
if (empty(_active)) {
|
||||||
end(_active),
|
stopTimer();
|
||||||
std::make_move_iterator(begin(_starting)),
|
}
|
||||||
std::make_move_iterator(end(_starting)));
|
return;
|
||||||
_starting.clear();
|
}
|
||||||
|
_active.insert(
|
||||||
|
end(_active),
|
||||||
|
std::make_move_iterator(begin(_starting)),
|
||||||
|
std::make_move_iterator(end(_starting)));
|
||||||
|
_starting.clear();
|
||||||
|
if (_forceUpdateProcessing) {
|
||||||
|
updateQueued();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::updateQueued() {
|
void Manager::updateQueued() {
|
||||||
Expects(_timerId == 0);
|
if (_queued) {
|
||||||
|
|
||||||
_timerId = -1;
|
|
||||||
crl::on_main(delayedCallGuard(), [=] {
|
|
||||||
Expects(_timerId < 0);
|
|
||||||
|
|
||||||
_timerId = 0;
|
|
||||||
update();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void Manager::schedule() {
|
|
||||||
if (_scheduled || _timerId < 0) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
stopTimer();
|
_queued = true;
|
||||||
|
crl::on_main(delayedCallGuard(), [=] {
|
||||||
_scheduled = true;
|
_queued = false;
|
||||||
Ui::PostponeCall(delayedCallGuard(), [=] {
|
update(UpdateSource::Queued);
|
||||||
_scheduled = false;
|
|
||||||
if (_forceImmediateUpdate) {
|
|
||||||
_forceImmediateUpdate = false;
|
|
||||||
updateQueued();
|
|
||||||
} else {
|
|
||||||
const auto next = _lastUpdateTime + kAnimationTimeout;
|
|
||||||
const auto now = crl::now();
|
|
||||||
if (now < next) {
|
|
||||||
_timerId = startTimer(next - now, Qt::PreciseTimer);
|
|
||||||
} else {
|
|
||||||
updateQueued();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -157,14 +141,22 @@ not_null<const QObject*> Manager::delayedCallGuard() const {
|
||||||
return static_cast<const QObject*>(this);
|
return static_cast<const QObject*>(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::stopTimer() {
|
void Manager::startTimer() {
|
||||||
if (_timerId > 0) {
|
if (_timerId) {
|
||||||
killTimer(base::take(_timerId));
|
return;
|
||||||
}
|
}
|
||||||
|
_timerId = QObject::startTimer(kAnimationTick, Qt::PreciseTimer);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Manager::stopTimer() {
|
||||||
|
if (!_timerId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
killTimer(base::take(_timerId));
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::timerEvent(QTimerEvent *e) {
|
void Manager::timerEvent(QTimerEvent *e) {
|
||||||
update();
|
update(UpdateSource::TimerEvent);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Animations
|
} // namespace Animations
|
||||||
|
|
|
@ -101,9 +101,13 @@ class Manager final : private QObject {
|
||||||
public:
|
public:
|
||||||
Manager();
|
Manager();
|
||||||
|
|
||||||
void update();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
enum class UpdateSource {
|
||||||
|
Queued,
|
||||||
|
TimerEvent,
|
||||||
|
RepaintRequest,
|
||||||
|
};
|
||||||
|
|
||||||
class ActiveBasicPointer {
|
class ActiveBasicPointer {
|
||||||
public:
|
public:
|
||||||
ActiveBasicPointer(Basic *value = nullptr) : _value(value) {
|
ActiveBasicPointer(Basic *value = nullptr) : _value(value) {
|
||||||
|
@ -151,20 +155,21 @@ private:
|
||||||
friend class Basic;
|
friend class Basic;
|
||||||
|
|
||||||
void timerEvent(QTimerEvent *e) override;
|
void timerEvent(QTimerEvent *e) override;
|
||||||
|
void update(UpdateSource source);
|
||||||
|
|
||||||
void start(not_null<Basic*> animation);
|
void start(not_null<Basic*> animation);
|
||||||
void stop(not_null<Basic*> animation);
|
void stop(not_null<Basic*> animation);
|
||||||
|
|
||||||
void schedule();
|
|
||||||
void updateQueued();
|
void updateQueued();
|
||||||
|
void startTimer();
|
||||||
void stopTimer();
|
void stopTimer();
|
||||||
not_null<const QObject*> delayedCallGuard() const;
|
not_null<const QObject*> delayedCallGuard() const;
|
||||||
|
|
||||||
crl::time _lastUpdateTime = 0;
|
crl::time _lastUpdateTime = 0;
|
||||||
int _timerId = 0;
|
int _timerId = 0;
|
||||||
bool _updating = false;
|
bool _updating = false;
|
||||||
bool _scheduled = false;
|
bool _queued = false;
|
||||||
bool _forceImmediateUpdate = false;
|
bool _forceUpdateProcessing = false;
|
||||||
std::vector<ActiveBasicPointer> _active;
|
std::vector<ActiveBasicPointer> _active;
|
||||||
std::vector<ActiveBasicPointer> _starting;
|
std::vector<ActiveBasicPointer> _starting;
|
||||||
rpl::lifetime _lifetime;
|
rpl::lifetime _lifetime;
|
||||||
|
|
Loading…
Reference in New Issue