Rework how animationsystem interoperate with an animation driver.
We need to keep track of both wall time which are used for pauses and actual animation driver time which is used for actual animations. When switching between these, we need to also maintain the temporal drift potentially introduced by the driver and also the time that has passed in wall-time from when a pause has started until an action animation takes over. This change introduces a well defined elapsed() function in QUnifiedTimer which will return the right value based on which mode we are currently in. It also introduces start/stopAnimationDriver functions which helps us maintain the temporal drift and pause-delta. Change-Id: I5b5100432a6db444a413d1bca4f2d5f800e8cf3e Reviewed-by: Michael Brasser <michael.brasser@live.com>
This commit is contained in:
parent
28add98e24
commit
dfc8f8b5d4
@ -222,7 +222,8 @@ QUnifiedTimer::QUnifiedTimer() :
|
|||||||
QObject(), defaultDriver(this), lastTick(0), timingInterval(DEFAULT_TIMER_INTERVAL),
|
QObject(), defaultDriver(this), lastTick(0), timingInterval(DEFAULT_TIMER_INTERVAL),
|
||||||
currentAnimationIdx(0), insideTick(false), insideRestart(false), consistentTiming(false), slowMode(false),
|
currentAnimationIdx(0), insideTick(false), insideRestart(false), consistentTiming(false), slowMode(false),
|
||||||
startTimersPending(false), stopTimerPending(false),
|
startTimersPending(false), stopTimerPending(false),
|
||||||
slowdownFactor(5.0f), profilerCallback(0)
|
slowdownFactor(5.0f), profilerCallback(0),
|
||||||
|
driverStartTime(0), temporalDrift(0)
|
||||||
{
|
{
|
||||||
time.invalidate();
|
time.invalidate();
|
||||||
driver = &defaultDriver;
|
driver = &defaultDriver;
|
||||||
@ -253,18 +254,56 @@ QUnifiedTimer *QUnifiedTimer::instance()
|
|||||||
|
|
||||||
void QUnifiedTimer::maybeUpdateAnimationsToCurrentTime()
|
void QUnifiedTimer::maybeUpdateAnimationsToCurrentTime()
|
||||||
{
|
{
|
||||||
qint64 elapsed = driver->elapsed();
|
if (elapsed() - lastTick > 50)
|
||||||
if (elapsed - lastTick > 50)
|
updateAnimationTimers(-1);
|
||||||
updateAnimationTimers(elapsed);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void QUnifiedTimer::updateAnimationTimers(qint64 currentTick)
|
qint64 QUnifiedTimer::elapsed() const
|
||||||
|
{
|
||||||
|
if (driver->isRunning())
|
||||||
|
return driverStartTime + driver->elapsed();
|
||||||
|
else if (time.isValid())
|
||||||
|
return time.elapsed() + temporalDrift;
|
||||||
|
|
||||||
|
// Reaching here would normally indicate that the function is called
|
||||||
|
// under the wrong circumstances as neither pauses nor actual animations
|
||||||
|
// are running and there should be no need to query for elapsed().
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void QUnifiedTimer::startAnimationDriver()
|
||||||
|
{
|
||||||
|
if (driver->isRunning()) {
|
||||||
|
qWarning("QUnifiedTimer::startAnimationDriver: driver is already running...");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Set the start time to the currently elapsed() value before starting.
|
||||||
|
// This means we get the animation system time including the temporal drift
|
||||||
|
// which is what we want.
|
||||||
|
driverStartTime = elapsed();
|
||||||
|
driver->start();
|
||||||
|
}
|
||||||
|
|
||||||
|
void QUnifiedTimer::stopAnimationDriver()
|
||||||
|
{
|
||||||
|
if (!driver->isRunning()) {
|
||||||
|
qWarning("QUnifiedTimer::stopAnimationDriver: driver is not running");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Update temporal drift. Since the driver is running, elapsed() will
|
||||||
|
// return the total animation time in driver-time. Subtract the current
|
||||||
|
// wall time to get the delta.
|
||||||
|
temporalDrift = elapsed() - time.elapsed();
|
||||||
|
driver->stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
void QUnifiedTimer::updateAnimationTimers(qint64)
|
||||||
{
|
{
|
||||||
//setCurrentTime can get this called again while we're the for loop. At least with pauseAnimations
|
//setCurrentTime can get this called again while we're the for loop. At least with pauseAnimations
|
||||||
if(insideTick)
|
if(insideTick)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
qint64 totalElapsed = currentTick >= 0 ? currentTick : driver->elapsed();
|
qint64 totalElapsed = elapsed();
|
||||||
|
|
||||||
// ignore consistentTiming in case the pause timer is active
|
// ignore consistentTiming in case the pause timer is active
|
||||||
qint64 delta = (consistentTiming && !pauseTimer.isActive()) ?
|
qint64 delta = (consistentTiming && !pauseTimer.isActive()) ?
|
||||||
@ -323,8 +362,7 @@ void QUnifiedTimer::localRestart()
|
|||||||
} else if (!driver->isRunning()) {
|
} else if (!driver->isRunning()) {
|
||||||
if (pauseTimer.isActive())
|
if (pauseTimer.isActive())
|
||||||
pauseTimer.stop();
|
pauseTimer.stop();
|
||||||
driver->setStartTime(time.isValid() ? time.elapsed() : 0);
|
startAnimationDriver();
|
||||||
driver->start();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -345,35 +383,39 @@ void QUnifiedTimer::setTimingInterval(int interval)
|
|||||||
|
|
||||||
if (driver->isRunning() && !pauseTimer.isActive()) {
|
if (driver->isRunning() && !pauseTimer.isActive()) {
|
||||||
//we changed the timing interval
|
//we changed the timing interval
|
||||||
driver->stop();
|
stopAnimationDriver();
|
||||||
driver->setStartTime(time.isValid() ? time.elapsed() : 0);
|
startAnimationDriver();
|
||||||
driver->start();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void QUnifiedTimer::startTimers()
|
void QUnifiedTimer::startTimers()
|
||||||
{
|
{
|
||||||
startTimersPending = false;
|
startTimersPending = false;
|
||||||
|
|
||||||
|
// Initialize the wall clock right away as we need this for
|
||||||
|
// both localRestart and updateAnimationTimers() down below..
|
||||||
|
if (!time.isValid()) {
|
||||||
|
lastTick = 0;
|
||||||
|
time.start();
|
||||||
|
temporalDrift = 0;
|
||||||
|
driverStartTime = 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (!animationTimers.isEmpty())
|
if (!animationTimers.isEmpty())
|
||||||
updateAnimationTimers(-1);
|
updateAnimationTimers(-1);
|
||||||
|
|
||||||
//we transfer the waiting animations into the "really running" state
|
//we transfer the waiting animations into the "really running" state
|
||||||
animationTimers += animationTimersToStart;
|
animationTimers += animationTimersToStart;
|
||||||
animationTimersToStart.clear();
|
animationTimersToStart.clear();
|
||||||
if (!animationTimers.isEmpty()) {
|
if (!animationTimers.isEmpty())
|
||||||
localRestart();
|
localRestart();
|
||||||
if (!time.isValid()) {
|
|
||||||
lastTick = 0;
|
|
||||||
time.start();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void QUnifiedTimer::stopTimer()
|
void QUnifiedTimer::stopTimer()
|
||||||
{
|
{
|
||||||
stopTimerPending = false;
|
stopTimerPending = false;
|
||||||
if (animationTimers.isEmpty()) {
|
if (animationTimers.isEmpty()) {
|
||||||
driver->stop();
|
stopAnimationDriver();
|
||||||
pauseTimer.stop();
|
pauseTimer.stop();
|
||||||
// invalidate the start reference time
|
// invalidate the start reference time
|
||||||
time.invalidate();
|
time.invalidate();
|
||||||
@ -483,14 +525,12 @@ void QUnifiedTimer::installAnimationDriver(QAnimationDriver *d)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (driver->isRunning()) {
|
bool running = driver->isRunning();
|
||||||
driver->stop();
|
if (running)
|
||||||
d->setStartTime(time.isValid() ? time.elapsed() : 0);
|
stopAnimationDriver();
|
||||||
d->start();
|
|
||||||
}
|
|
||||||
|
|
||||||
driver = d;
|
driver = d;
|
||||||
|
if (running)
|
||||||
|
startAnimationDriver();
|
||||||
}
|
}
|
||||||
|
|
||||||
void QUnifiedTimer::uninstallAnimationDriver(QAnimationDriver *d)
|
void QUnifiedTimer::uninstallAnimationDriver(QAnimationDriver *d)
|
||||||
@ -500,13 +540,12 @@ void QUnifiedTimer::uninstallAnimationDriver(QAnimationDriver *d)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool running = driver->isRunning();
|
||||||
|
if (running)
|
||||||
|
stopAnimationDriver();
|
||||||
driver = &defaultDriver;
|
driver = &defaultDriver;
|
||||||
|
if (running)
|
||||||
if (d->isRunning()) {
|
startAnimationDriver();
|
||||||
d->stop();
|
|
||||||
driver->setStartTime(time.isValid() ? time.elapsed() : 0);
|
|
||||||
driver->start();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@ -604,6 +643,7 @@ void QAnimationTimer::restartAnimationTimer()
|
|||||||
void QAnimationTimer::startAnimations()
|
void QAnimationTimer::startAnimations()
|
||||||
{
|
{
|
||||||
startAnimationPending = false;
|
startAnimationPending = false;
|
||||||
|
|
||||||
//force timer to update, which prevents large deltas for our newly added animations
|
//force timer to update, which prevents large deltas for our newly added animations
|
||||||
if (!animations.isEmpty())
|
if (!animations.isEmpty())
|
||||||
QUnifiedTimer::instance()->maybeUpdateAnimationsToCurrentTime();
|
QUnifiedTimer::instance()->maybeUpdateAnimationsToCurrentTime();
|
||||||
@ -749,20 +789,25 @@ QAnimationDriver::~QAnimationDriver()
|
|||||||
This is to take into account that pauses can occur in running
|
This is to take into account that pauses can occur in running
|
||||||
animations which will stop the driver, but the time still
|
animations which will stop the driver, but the time still
|
||||||
increases.
|
increases.
|
||||||
|
|
||||||
|
\obsolete
|
||||||
|
|
||||||
|
This logic is now handled internally in the animation system.
|
||||||
*/
|
*/
|
||||||
void QAnimationDriver::setStartTime(qint64 startTime)
|
void QAnimationDriver::setStartTime(qint64)
|
||||||
{
|
{
|
||||||
Q_D(QAnimationDriver);
|
|
||||||
d->startTime = startTime;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
Returns the start time of the animation.
|
Returns the start time of the animation.
|
||||||
|
|
||||||
|
\obsolete
|
||||||
|
|
||||||
|
This logic is now handled internally in the animation system.
|
||||||
*/
|
*/
|
||||||
qint64 QAnimationDriver::startTime() const
|
qint64 QAnimationDriver::startTime() const
|
||||||
{
|
{
|
||||||
Q_D(const QAnimationDriver);
|
return 0;
|
||||||
return d->startTime;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -772,6 +817,10 @@ qint64 QAnimationDriver::startTime() const
|
|||||||
|
|
||||||
If \a timeStep is positive, it will be used as the current time in the
|
If \a timeStep is positive, it will be used as the current time in the
|
||||||
calculations; otherwise, the current clock time will be used.
|
calculations; otherwise, the current clock time will be used.
|
||||||
|
|
||||||
|
Since 5.4, the timeStep argument is ignored and elapsed() will be
|
||||||
|
used instead in combination with the internal time offsets of the
|
||||||
|
animation system.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
void QAnimationDriver::advanceAnimation(qint64 timeStep)
|
void QAnimationDriver::advanceAnimation(qint64 timeStep)
|
||||||
|
@ -149,6 +149,7 @@ public:
|
|||||||
|
|
||||||
virtual qint64 elapsed() const;
|
virtual qint64 elapsed() const;
|
||||||
|
|
||||||
|
// ### Qt6: Remove these two functions
|
||||||
void setStartTime(qint64 startTime);
|
void setStartTime(qint64 startTime);
|
||||||
qint64 startTime() const;
|
qint64 startTime() const;
|
||||||
|
|
||||||
@ -157,6 +158,7 @@ Q_SIGNALS:
|
|||||||
void stopped();
|
void stopped();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
// ### Qt6: Remove timestep argument
|
||||||
void advanceAnimation(qint64 timeStep = -1);
|
void advanceAnimation(qint64 timeStep = -1);
|
||||||
virtual void start();
|
virtual void start();
|
||||||
virtual void stop();
|
virtual void stop();
|
||||||
|
@ -132,9 +132,8 @@ private:
|
|||||||
class Q_CORE_EXPORT QAnimationDriverPrivate : public QObjectPrivate
|
class Q_CORE_EXPORT QAnimationDriverPrivate : public QObjectPrivate
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
QAnimationDriverPrivate() : running(false), startTime(0) {}
|
QAnimationDriverPrivate() : running(false) {}
|
||||||
bool running;
|
bool running;
|
||||||
qint64 startTime;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class Q_CORE_EXPORT QAbstractAnimationTimer : public QObject
|
class Q_CORE_EXPORT QAbstractAnimationTimer : public QObject
|
||||||
@ -193,6 +192,10 @@ public:
|
|||||||
int runningAnimationCount();
|
int runningAnimationCount();
|
||||||
void registerProfilerCallback(void (*cb)(qint64));
|
void registerProfilerCallback(void (*cb)(qint64));
|
||||||
|
|
||||||
|
void startAnimationDriver();
|
||||||
|
void stopAnimationDriver();
|
||||||
|
qint64 elapsed() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void timerEvent(QTimerEvent *);
|
void timerEvent(QTimerEvent *);
|
||||||
|
|
||||||
@ -233,6 +236,9 @@ private:
|
|||||||
int closestPausedAnimationTimerTimeToFinish();
|
int closestPausedAnimationTimerTimeToFinish();
|
||||||
|
|
||||||
void (*profilerCallback)(qint64);
|
void (*profilerCallback)(qint64);
|
||||||
|
|
||||||
|
qint64 driverStartTime; // The time the animation driver was started
|
||||||
|
qint64 temporalDrift; // The delta between animation driver time and wall time.
|
||||||
};
|
};
|
||||||
|
|
||||||
class QAnimationTimer : public QAbstractAnimationTimer
|
class QAnimationTimer : public QAbstractAnimationTimer
|
||||||
|
Loading…
Reference in New Issue
Block a user