qt 6.5.1 original

This commit is contained in:
kleuter
2023-10-29 23:33:08 +01:00
parent 71d22ab6b0
commit 85d238dfda
21202 changed files with 5499099 additions and 0 deletions

View File

@ -0,0 +1,6 @@
[interpolation]
windows
osx
[frameRate]
macos

View File

@ -0,0 +1,13 @@
# Copyright (C) 2022 The Qt Company Ltd.
# SPDX-License-Identifier: BSD-3-Clause
#####################################################################
## tst_qtimeline Test:
#####################################################################
qt_internal_add_test(tst_qtimeline
SOURCES
tst_qtimeline.cpp
LIBRARIES
Qt::TestPrivate
)

View File

@ -0,0 +1,914 @@
// Copyright (C) 2020 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include <QTest>
#include <QtTest/private/qpropertytesthelper_p.h>
#include <QSignalSpy>
#include <qtimeline.h>
class tst_QTimeLine : public QObject
{
Q_OBJECT
private slots:
void range();
void currentTime();
void bindableCurrentTime();
void duration();
void bindableDuration();
void frameRate();
void bindableUpdateInterval();
void value();
void currentFrame();
void loopCount();
void bindableLoopCount();
void interpolation();
void reverse_data();
void reverse();
void toggleDirection();
void bindableDirection();
void frameChanged();
void stopped();
void finished();
void isRunning();
void multipleTimeLines();
void sineCurve();
void cosineCurve();
void outOfRange();
void stateInFinishedSignal();
void resume();
void restart();
void setPaused();
void automatedBindableTests();
protected slots:
void finishedSlot();
protected:
QTimeLine::State state;
QTimeLine * view;
};
void tst_QTimeLine::range()
{
QTimeLine timeLine(200);
QCOMPARE(timeLine.startFrame(), 0);
QCOMPARE(timeLine.endFrame(), 0);
timeLine.setFrameRange(0, 1);
QCOMPARE(timeLine.startFrame(), 0);
QCOMPARE(timeLine.endFrame(), 1);
timeLine.setFrameRange(10, 20);
QCOMPARE(timeLine.startFrame(), 10);
QCOMPARE(timeLine.endFrame(), 20);
timeLine.setStartFrame(6);
QCOMPARE(timeLine.startFrame(), 6);
timeLine.setEndFrame(16);
QCOMPARE(timeLine.endFrame(), 16);
// Verify that you can change the range in the timeLine
timeLine.setFrameRange(1000, 2000);
QSignalSpy spy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(spy.isValid());
timeLine.start(); // make sure that the logic works for a running timeline
QTRY_COMPARE(timeLine.state(), QTimeLine::Running);
timeLine.setCurrentTime(timeLine.duration()/2);
int oldValue = timeLine.currentFrame();
timeLine.setFrameRange(0, 500);
QVERIFY(timeLine.currentFrame() < oldValue);
timeLine.setEndFrame(10000);
timeLine.setStartFrame(5000);
QVERIFY(timeLine.currentFrame() > oldValue);
timeLine.setFrameRange(0, 500);
QTRY_VERIFY(spy.size() > 1);
QVERIFY(timeLine.currentFrame() < oldValue);
}
void tst_QTimeLine::currentTime()
{
QTimeLine timeLine(2000);
timeLine.setUpdateInterval((timeLine.duration()/2) / 33);
timeLine.setFrameRange(10, 20);
QCOMPARE(timeLine.currentTime(), 0);
timeLine.start();
QTRY_COMPARE(timeLine.state(), QTimeLine::Running);
QTRY_VERIFY(timeLine.currentTime() > timeLine.duration()/2 - timeLine.duration()/4);
QVERIFY(timeLine.currentTime() < timeLine.duration()/2 + timeLine.duration()/4);
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(timeLine.currentTime(), timeLine.duration());
QSignalSpy spy(&timeLine, &QTimeLine::valueChanged);
QVERIFY(spy.isValid());
spy.clear();
timeLine.setCurrentTime(timeLine.duration()/2);
timeLine.setCurrentTime(timeLine.duration()/2);
QCOMPARE(spy.size(), 1);
spy.clear();
QCOMPARE(timeLine.currentTime(), timeLine.duration()/2);
timeLine.resume();
// Let it update on its own
QCOMPARE(timeLine.state(), QTimeLine::Running);
QTRY_VERIFY(timeLine.currentTime() > timeLine.duration()/2);
QVERIFY(timeLine.currentTime() < timeLine.duration());
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(timeLine.currentTime(), timeLine.duration());
// Reverse should decrease the currentTime
timeLine.setCurrentTime(timeLine.duration()/2);
timeLine.start();
// Let it update on its own
int currentTime = timeLine.currentTime();
QTRY_VERIFY(timeLine.currentTime() > currentTime);
QCOMPARE(timeLine.state(), QTimeLine::Running);
currentTime = timeLine.currentTime();
timeLine.setDirection(QTimeLine::Backward);
QTRY_VERIFY(timeLine.currentTime() < currentTime);
timeLine.stop();
}
void tst_QTimeLine::bindableCurrentTime()
{
QTimeLine timeLine(2000);
QProperty<int> currentTimeObserver([&]() { return timeLine.currentTime(); });
timeLine.setUpdateInterval((timeLine.duration() / 2) / 33);
timeLine.setFrameRange(10, 20);
QCOMPARE(timeLine.currentTime(), 0);
QCOMPARE(currentTimeObserver.value(), 0);
QCOMPARE(currentTimeObserver.value(), timeLine.currentTime());
timeLine.start();
QTRY_COMPARE(timeLine.state(), QTimeLine::Running);
QCOMPARE(currentTimeObserver.value(), timeLine.currentTime());
QTRY_VERIFY(timeLine.currentTime() > timeLine.duration() / 2 - timeLine.duration() / 4);
QVERIFY(timeLine.currentTime() < timeLine.duration() / 2 + timeLine.duration() / 4);
QCOMPARE(currentTimeObserver.value(), timeLine.currentTime());
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(timeLine.currentTime(), timeLine.duration());
QCOMPARE(currentTimeObserver.value(), timeLine.currentTime());
QSignalSpy spy(&timeLine, &QTimeLine::valueChanged);
QVERIFY(spy.isValid());
spy.clear();
QProperty<int> referenceCurrentTime(timeLine.duration() / 2);
timeLine.bindableCurrentTime().setBinding([&]() { return referenceCurrentTime.value(); });
QCOMPARE(spy.size(), 1);
// setting it a second time to check that valueChanged() is emitted only once
referenceCurrentTime = timeLine.duration() / 2;
QCOMPARE(spy.size(), 1);
spy.clear();
QCOMPARE(timeLine.currentTime(), timeLine.duration() / 2);
QCOMPARE(currentTimeObserver.value(), timeLine.duration() / 2);
timeLine.resume();
// Let it update on its own
QCOMPARE(timeLine.state(), QTimeLine::Running);
QTRY_VERIFY(currentTimeObserver.value() > timeLine.duration() / 2);
QVERIFY(currentTimeObserver.value() < timeLine.duration());
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(currentTimeObserver.value(), timeLine.duration());
// the resume above should have broken the connection to referenceCurrentTime, check that:
spy.clear();
referenceCurrentTime = 0;
QCOMPARE(currentTimeObserver.value(), timeLine.duration());
QCOMPARE(spy.size(), 0);
}
void tst_QTimeLine::duration()
{
QTimeLine timeLine(200);
timeLine.setFrameRange(10, 20);
QCOMPARE(timeLine.duration(), 200);
timeLine.setDuration(1000);
QCOMPARE(timeLine.duration(), 1000);
timeLine.start();
QTRY_COMPARE(timeLine.state(), QTimeLine::Running);
QTRY_VERIFY(timeLine.currentTime() > 0);
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(timeLine.currentTime(), 1000);
// The duration shouldn't change
QCOMPARE(timeLine.duration(), 1000);
}
void tst_QTimeLine::bindableDuration()
{
QTimeLine timeLine(200);
QProperty<int> durationObserver;
durationObserver.setBinding([&]() { return timeLine.duration(); });
QCOMPARE(durationObserver.value(), timeLine.duration());
timeLine.setFrameRange(10, 20);
QCOMPARE(timeLine.duration(), 200);
QProperty<int> referenceDuration(500);
timeLine.bindableDuration().setBinding([&]() { return referenceDuration.value(); });
QCOMPARE(durationObserver.value(), referenceDuration.value());
QCOMPARE(timeLine.duration(), 500);
timeLine.start();
QTRY_COMPARE(timeLine.state(), QTimeLine::Running);
QTRY_VERIFY(timeLine.currentTime() > 0);
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(timeLine.currentTime(), 500);
// The duration shouldn't change
QCOMPARE(timeLine.duration(), 500);
referenceDuration = 30;
QCOMPARE(timeLine.duration(), 30);
QCOMPARE(durationObserver.value(), 30);
}
void tst_QTimeLine::frameRate()
{
QTimeLine timeLine;
timeLine.setFrameRange(100, 2000);
QCOMPARE(timeLine.updateInterval(), 1000 / 25);
timeLine.setUpdateInterval(1000 / 60);
QCOMPARE(timeLine.updateInterval(), 1000 / 60);
// Default speed
timeLine.setUpdateInterval(1000 / 33);
QSignalSpy spy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(spy.isValid());
timeLine.start();
QTest::qWait(timeLine.duration()*2);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
int slowCount = spy.size();
// Faster!!
timeLine.setUpdateInterval(1000 / 100);
spy.clear();
timeLine.setCurrentTime(0);
timeLine.start();
QTest::qWait(timeLine.duration()*2);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
QVERIFY2(slowCount < spy.size(), QByteArray::number(spy.size()));
}
void tst_QTimeLine::bindableUpdateInterval()
{
QTimeLine timeLine;
timeLine.setFrameRange(100, 2000);
QProperty<int> updateIntervalObserver;
updateIntervalObserver.setBinding([&]() { return timeLine.updateInterval(); });
QCOMPARE(updateIntervalObserver.value(), 1000 / 25);
QProperty<int> updateIntervalReference(1000 / 60);
timeLine.bindableUpdateInterval().setBinding([&]() { return updateIntervalReference.value(); });
updateIntervalReference = 1000 / 60;
QCOMPARE(updateIntervalObserver.value(), 1000 / 60);
// Default speed
updateIntervalReference = 1000 / 33;
QSignalSpy spy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(spy.isValid());
timeLine.start();
QTest::qWait(timeLine.duration() * 2);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
int slowCount = spy.size();
// Faster!!
updateIntervalReference = 1000 / 100;
spy.clear();
timeLine.setCurrentTime(0);
timeLine.start();
QTest::qWait(timeLine.duration() * 2);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
QVERIFY2(slowCount < spy.size(), QByteArray::number(spy.size()));
}
void tst_QTimeLine::value()
{
QTimeLine timeLine(4500); // Should be at least 5% under 5000ms
QCOMPARE(timeLine.currentValue(), 0.0);
// Default speed
QSignalSpy spy(&timeLine, &QTimeLine::valueChanged);
QVERIFY(spy.isValid());
timeLine.start();
QTRY_VERIFY(timeLine.currentValue() > 0);
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(timeLine.currentValue(), 1.0);
QVERIFY(spy.size() > 0);
// Reverse should decrease the value
timeLine.setCurrentTime(100);
timeLine.start();
// Let it update on its own
QCOMPARE(timeLine.state(), QTimeLine::Running);
QTRY_VERIFY(timeLine.currentValue());
qreal value = timeLine.currentValue();
timeLine.setDirection(QTimeLine::Backward);
QTRY_VERIFY(timeLine.currentValue() < value);
timeLine.stop();
}
void tst_QTimeLine::currentFrame()
{
QTimeLine timeLine(2000);
timeLine.setFrameRange(10, 20);
QCOMPARE(timeLine.currentFrame(), 10);
// Default speed
QSignalSpy spy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(spy.isValid());
timeLine.start();
QTRY_VERIFY(timeLine.currentFrame() > 10);
QTRY_COMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(timeLine.currentFrame(), 20);
// Reverse should decrease the value
timeLine.setCurrentTime(timeLine.duration()/2);
timeLine.start();
// Let it update on its own
QCOMPARE(timeLine.state(), QTimeLine::Running);
QTRY_VERIFY(timeLine.currentTime() > timeLine.duration()/2); // wait for continuation
int value = timeLine.currentFrame();
timeLine.setDirection(QTimeLine::Backward);
QTRY_VERIFY(timeLine.currentFrame() < value);
timeLine.stop();
}
void tst_QTimeLine::loopCount()
{
QTimeLine timeLine(200);
QCOMPARE(timeLine.loopCount(), 1);
timeLine.setFrameRange(10, 20);
QCOMPARE(timeLine.loopCount(), 1);
timeLine.setLoopCount(0);
QCOMPARE(timeLine.loopCount(), 0);
// Default speed endless looping
QSignalSpy spy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(spy.isValid());
timeLine.start();
QTest::qWait(timeLine.duration());
QCOMPARE(timeLine.state(), QTimeLine::Running);
//QCOMPARE(timeLine.currentFrame(), 20);
QTest::qWait(timeLine.duration()*6);
QCOMPARE(timeLine.state(), QTimeLine::Running);
QVERIFY(timeLine.currentTime() >= 0);
QVERIFY(timeLine.currentFrame() >= 10);
QVERIFY(timeLine.currentFrame() <= 20);
QCOMPARE(timeLine.state(), QTimeLine::Running);
timeLine.stop();
timeLine.setDuration(2500); // some platforms have a very low resolution timer
timeLine.setFrameRange(0, 2);
timeLine.setLoopCount(4);
QSignalSpy finishedSpy(&timeLine, &QTimeLine::finished);
QSignalSpy frameChangedSpy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(finishedSpy.isValid());
QVERIFY(frameChangedSpy.isValid());
QEventLoop loop;
connect(&timeLine, SIGNAL(finished()), &loop, SLOT(quit()));
for(int i=0;i<2;i++) {
timeLine.start();
// we clear the list after the start so we don't catch
// a frameChanged signal for the frame 0 at the beginning
finishedSpy.clear();
frameChangedSpy.clear();
loop.exec();
QCOMPARE(finishedSpy.size(), 1);
QCOMPARE(frameChangedSpy.size(), 11);
for (int i = 0; i < 11; ++i)
QCOMPARE(frameChangedSpy.at(i).at(0).toInt(), (i+1) % 3);
}
timeLine.setDirection(QTimeLine::Backward);
timeLine.start();
loop.exec();
QCOMPARE(finishedSpy.size(), 2);
QCOMPARE(frameChangedSpy.size(), 22);
for (int i = 11; i < 22; ++i) {
QCOMPARE(frameChangedSpy.at(i).at(0).toInt(), 2 - (i+2) % 3);
}
}
void tst_QTimeLine::bindableLoopCount()
{
QTimeLine timeLine(200);
QProperty<int> referenceLoopCount(1);
timeLine.bindableLoopCount().setBinding([&]() { return referenceLoopCount.value(); });
QProperty<int> loopCountObserver([&]() { return timeLine.loopCount(); });
QCOMPARE(referenceLoopCount.value(), 1);
QCOMPARE(timeLine.loopCount(), 1);
QCOMPARE(loopCountObserver.value(), 1);
timeLine.setFrameRange(10, 20);
QCOMPARE(referenceLoopCount.value(), 1);
QCOMPARE(timeLine.loopCount(), 1);
QCOMPARE(loopCountObserver.value(), 1);
referenceLoopCount = 0;
QCOMPARE(referenceLoopCount.value(), 0);
QCOMPARE(timeLine.loopCount(), 0);
QCOMPARE(loopCountObserver.value(), 0);
// Default speed endless looping
QSignalSpy spy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(spy.isValid());
timeLine.start();
QTest::qWait(timeLine.duration());
QCOMPARE(timeLine.state(), QTimeLine::Running);
// QCOMPARE(timeLine.currentFrame(), 20);
QTest::qWait(timeLine.duration() * 6);
QCOMPARE(timeLine.state(), QTimeLine::Running);
QVERIFY(timeLine.currentTime() >= 0);
QVERIFY(timeLine.currentFrame() >= 10);
QVERIFY(timeLine.currentFrame() <= 20);
QCOMPARE(timeLine.state(), QTimeLine::Running);
timeLine.stop();
timeLine.setDuration(2500); // some platforms have a very low resolution timer
timeLine.setFrameRange(0, 2);
referenceLoopCount = 4;
QSignalSpy finishedSpy(&timeLine, &QTimeLine::finished);
QSignalSpy frameChangedSpy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(finishedSpy.isValid());
QVERIFY(frameChangedSpy.isValid());
QEventLoop loop;
connect(&timeLine, SIGNAL(finished()), &loop, SLOT(quit()));
for (int i = 0; i < 2; i++) {
timeLine.start();
// we clear the list after the start so we don't catch
// a frameChanged signal for the frame 0 at the beginning
finishedSpy.clear();
frameChangedSpy.clear();
loop.exec();
QCOMPARE(finishedSpy.size(), 1);
QCOMPARE(frameChangedSpy.size(), 11);
for (int i = 0; i < 11; ++i)
QCOMPARE(frameChangedSpy.at(i).at(0).toInt(), (i + 1) % 3);
}
timeLine.setDirection(QTimeLine::Backward);
timeLine.start();
loop.exec();
QCOMPARE(finishedSpy.size(), 2);
QCOMPARE(frameChangedSpy.size(), 22);
for (int i = 11; i < 22; ++i)
QCOMPARE(frameChangedSpy.at(i).at(0).toInt(), 2 - (i + 2) % 3);
}
void tst_QTimeLine::interpolation()
{
// also tests bindableEasingCurve
QTimeLine timeLine(400);
QProperty<QEasingCurve> easingCurveObserver([&]() { return timeLine.easingCurve(); });
QCOMPARE(timeLine.easingCurve(), QEasingCurve::InOutSine);
QCOMPARE(easingCurveObserver.value(), QEasingCurve::InOutSine);
timeLine.setFrameRange(100, 200);
QProperty<QEasingCurve> referenceEasingCurve(QEasingCurve::Linear);
timeLine.bindableEasingCurve().setBinding([&]() { return referenceEasingCurve.value(); });
QCOMPARE(timeLine.easingCurve(), QEasingCurve::Linear);
QCOMPARE(easingCurveObserver.value(), QEasingCurve::Linear);
// smooth
referenceEasingCurve = QEasingCurve::InOutSine;
QCOMPARE(timeLine.easingCurve(), QEasingCurve::InOutSine);
QCOMPARE(easingCurveObserver.value(), QEasingCurve::InOutSine);
timeLine.start();
QTest::qWait(100);
QCOMPARE(timeLine.state(), QTimeLine::Running);
int firstValue = timeLine.currentFrame();
QTest::qWait(200);
int endValue = timeLine.currentFrame();
timeLine.stop();
timeLine.setCurrentTime(0);
// linear
referenceEasingCurve = QEasingCurve::Linear;
QCOMPARE(timeLine.easingCurve(), QEasingCurve::Linear);
QCOMPARE(easingCurveObserver.value(), QEasingCurve::Linear);
timeLine.start();
QTest::qWait(100);
QCOMPARE(timeLine.state(), QTimeLine::Running);
// Smooth accellerates slowly so in the beginning so it is farther behind
if (firstValue >= timeLine.currentFrame())
QEXPECT_FAIL("", "QTBUG-24796: QTimeLine exhibits inconsistent timing behaviour", Abort);
QVERIFY(firstValue < timeLine.currentFrame());
QTest::qWait(200);
QVERIFY(endValue > timeLine.currentFrame());
timeLine.stop();
}
void tst_QTimeLine::reverse_data()
{
QTest::addColumn<int>("duration");
QTest::addColumn<int>("start");
QTest::addColumn<int>("end");
QTest::addColumn<int>("direction");
QTest::addColumn<int>("direction2");
QTest::addColumn<int>("direction3");
QTest::addColumn<int>("startTime");
QTest::addColumn<int>("currentFrame");
QTest::addColumn<qreal>("currentValue");
QTest::addColumn<int>("wait");
QTest::addColumn<int>("state");
QTest::addColumn<int>("wait2");
QTest::newRow("start at end") << 200 << 1000 << 2000 << (int)QTimeLine::Backward << (int)QTimeLine::Forward << (int)QTimeLine::Backward << 200 << 2000 << qreal(1.0) << 40 << (int)QTimeLine::Running << 140;
QTest::newRow("start at half") << 200 << 1000 << 2000 << (int)QTimeLine::Backward << (int)QTimeLine::Forward << (int)QTimeLine::Backward << 100 << 1500 << qreal(0.5) << 40 << (int)QTimeLine::Running << 140;
QTest::newRow("start at quarter") << 200 << 1000 << 2000 << (int)QTimeLine::Backward << (int)QTimeLine::Forward << (int)QTimeLine::Backward << 50 << 1250 << qreal(0.25) << 40 << (int)QTimeLine::Running << 140;
}
void tst_QTimeLine::reverse()
{
QFETCH(int, duration);
QFETCH(int, start);
QFETCH(int, end);
QFETCH(int, direction);
QFETCH(int, direction2);
QFETCH(int, direction3);
QFETCH(int, startTime);
QFETCH(int, currentFrame);
QFETCH(qreal, currentValue);
QFETCH(int, wait);
QFETCH(int, state);
QFETCH(int, wait2);
QTimeLine timeLine(duration);
timeLine.setEasingCurve(QEasingCurve::Linear);
timeLine.setFrameRange(start, end);
timeLine.setDirection((QTimeLine::Direction)direction);
timeLine.setDirection((QTimeLine::Direction)direction2);
timeLine.setDirection((QTimeLine::Direction)direction3);
QCOMPARE(timeLine.direction(), ((QTimeLine::Direction)direction));
timeLine.setCurrentTime(startTime);
timeLine.setDirection((QTimeLine::Direction)direction);
timeLine.setDirection((QTimeLine::Direction)direction2);
timeLine.setDirection((QTimeLine::Direction)direction3);
QCOMPARE(timeLine.currentFrame(), currentFrame);
QCOMPARE(timeLine.currentValue(), currentValue);
timeLine.start();
QTest::qWait(wait);
QCOMPARE(timeLine.state(), (QTimeLine::State)state);
int firstValue = timeLine.currentFrame();
timeLine.setDirection((QTimeLine::Direction)direction2);
timeLine.setDirection((QTimeLine::Direction)direction3);
timeLine.setDirection((QTimeLine::Direction)direction2);
timeLine.setDirection((QTimeLine::Direction)direction3);
QTest::qWait(wait2);
int endValue = timeLine.currentFrame();
QVERIFY(endValue < firstValue);
}
void tst_QTimeLine::toggleDirection()
{
QTimeLine timeLine;
QCOMPARE(timeLine.direction(), QTimeLine::Forward);
timeLine.toggleDirection();
QCOMPARE(timeLine.direction(), QTimeLine::Backward);
timeLine.toggleDirection();
QCOMPARE(timeLine.direction(), QTimeLine::Forward);
}
void tst_QTimeLine::bindableDirection()
{
// Note: enum values are cast to int so that QCOMPARE will show
// the values if they don't match.
QTimeLine timeLine;
QProperty<QTimeLine::Direction> directionObserver([&]() { return timeLine.direction(); });
QProperty<QTimeLine::Direction> referenceDirection(QTimeLine::Forward);
timeLine.bindableDirection().setBinding([&]() { return referenceDirection.value(); });
QCOMPARE(referenceDirection.value(), QTimeLine::Forward);
QCOMPARE(timeLine.direction(), QTimeLine::Forward);
QCOMPARE(directionObserver.value(), QTimeLine::Forward);
referenceDirection = QTimeLine::Backward;
QCOMPARE(referenceDirection.value(), QTimeLine::Backward);
QCOMPARE(timeLine.direction(), QTimeLine::Backward);
QCOMPARE(directionObserver.value(), QTimeLine::Backward);
referenceDirection = QTimeLine::Forward;
QCOMPARE(referenceDirection.value(), QTimeLine::Forward);
QCOMPARE(timeLine.direction(), QTimeLine::Forward);
QCOMPARE(directionObserver.value(), QTimeLine::Forward);
}
void tst_QTimeLine::frameChanged()
{
QTimeLine timeLine;
timeLine.setEasingCurve(QEasingCurve::Linear);
timeLine.setFrameRange(0,9);
timeLine.setUpdateInterval(800);
QSignalSpy spy(&timeLine, &QTimeLine::frameChanged);
QVERIFY(spy.isValid());
// Test what happens when duration expires before all frames are emitted.
timeLine.start();
QTest::qWait(timeLine.duration()/2);
QCOMPARE(timeLine.state(), QTimeLine::Running);
QCOMPARE(spy.size(), 0);
QTest::qWait(timeLine.duration());
if (timeLine.state() != QTimeLine::NotRunning)
QEXPECT_FAIL("", "QTBUG-24796: QTimeLine runs slower than it should", Abort);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
if (spy.size() != 1)
QEXPECT_FAIL("", "QTBUG-24796: QTimeLine runs slower than it should", Abort);
QCOMPARE(spy.size(), 1);
// Test what happens when the frames are all emitted well before duration expires.
timeLine.setUpdateInterval(5);
spy.clear();
timeLine.setCurrentTime(0);
timeLine.start();
QTest::qWait(timeLine.duration()*2);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(spy.size(), 10);
}
void tst_QTimeLine::stopped()
{
QTimeLine timeLine;
timeLine.setFrameRange(0, 9);
qRegisterMetaType<QTimeLine::State>("QTimeLine::State");
QSignalSpy spy(&timeLine, &QTimeLine::stateChanged);
QVERIFY(spy.isValid());
timeLine.start();
QTest::qWait(timeLine.duration()*2);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
QCOMPARE(spy.size(), 2);
spy.clear();
timeLine.start();
timeLine.stop();
QCOMPARE(spy.size(), 2);
timeLine.setDirection(QTimeLine::Backward);
QCOMPARE(timeLine.loopCount(), 1);
}
void tst_QTimeLine::finished()
{
QTimeLine timeLine;
timeLine.setFrameRange(0,9);
QSignalSpy spy(&timeLine, &QTimeLine::finished);
QVERIFY(spy.isValid());
timeLine.start();
QTRY_COMPARE(spy.size(), 1);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
spy.clear();
timeLine.start();
timeLine.stop();
QCOMPARE(spy.size(), 0);
}
void tst_QTimeLine::isRunning()
{
QTimeLine timeLine;
timeLine.setFrameRange(0,9);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
timeLine.start();
QCOMPARE(timeLine.state(), QTimeLine::Running);
timeLine.stop();
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
timeLine.start();
QTest::qWait(timeLine.duration()*2);
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
}
void tst_QTimeLine::multipleTimeLines()
{
// Stopping a timer shouldn't affect the other timers
QTimeLine timeLine(200);
timeLine.setFrameRange(0,99);
QSignalSpy spy(&timeLine, &QTimeLine::finished);
QVERIFY(spy.isValid());
QTimeLine timeLineKiller;
timeLineKiller.setFrameRange(0,99);
timeLineKiller.start();
timeLine.start();
timeLineKiller.stop();
QTest::qWait(timeLine.duration()*2);
QCOMPARE(spy.size(), 1);
}
void tst_QTimeLine::sineCurve()
{
QTimeLine timeLine(1000);
timeLine.setEasingCurve(QEasingCurve::SineCurve);
QCOMPARE(timeLine.valueForTime(0), qreal(0));
QCOMPARE(timeLine.valueForTime(250), qreal(0.5));
QCOMPARE(timeLine.valueForTime(500), qreal(1));
QCOMPARE(timeLine.valueForTime(750), qreal(0.5));
QCOMPARE(timeLine.valueForTime(1000), qreal(0));
}
void tst_QTimeLine::cosineCurve()
{
QTimeLine timeLine(1000);
timeLine.setEasingCurve(QEasingCurve::CosineCurve);
QCOMPARE(timeLine.valueForTime(0), qreal(0.5));
QCOMPARE(timeLine.valueForTime(250), qreal(1));
QCOMPARE(timeLine.valueForTime(500), qreal(0.5));
QCOMPARE(timeLine.valueForTime(750), qreal(0));
QCOMPARE(timeLine.valueForTime(1000), qreal(0.5));
}
void tst_QTimeLine::outOfRange()
{
QTimeLine timeLine(1000);
QCOMPARE(timeLine.valueForTime(-100), qreal(0));
QCOMPARE(timeLine.valueForTime(2000), qreal(1));
timeLine.setEasingCurve(QEasingCurve::SineCurve);
QCOMPARE(timeLine.valueForTime(2000), qreal(0));
}
void tst_QTimeLine::stateInFinishedSignal()
{
QTimeLine timeLine(50);
connect(&timeLine, SIGNAL(finished()), this, SLOT(finishedSlot()));
state = QTimeLine::State(-1);
timeLine.start();
QTest::qWait(250);
QCOMPARE(state, QTimeLine::NotRunning);
}
void tst_QTimeLine::finishedSlot()
{
QTimeLine *timeLine = qobject_cast<QTimeLine *>(sender());
if (timeLine)
state = timeLine->state();
}
void tst_QTimeLine::resume()
{
QTimeLine timeLine(1000);
{
QCOMPARE(timeLine.currentTime(), 0);
timeLine.start();
QTRY_VERIFY(timeLine.currentTime() > 0);
timeLine.stop();
int oldCurrentTime = timeLine.currentTime();
QVERIFY(oldCurrentTime > 0);
QVERIFY(oldCurrentTime < 1000);
timeLine.resume();
QTRY_VERIFY(timeLine.currentTime() > oldCurrentTime);
timeLine.stop();
int currentTime = timeLine.currentTime();
QVERIFY(currentTime < 1000);
}
timeLine.setDirection(QTimeLine::Backward);
{
timeLine.setCurrentTime(1000);
QCOMPARE(timeLine.currentTime(), 1000);
timeLine.start();
QTRY_VERIFY(timeLine.currentTime() < 1000);
timeLine.stop();
int oldCurrentTime = timeLine.currentTime();
QVERIFY(oldCurrentTime < 1000);
QVERIFY(oldCurrentTime > 0);
timeLine.resume();
QTRY_VERIFY(timeLine.currentTime() < oldCurrentTime);
timeLine.stop();
int currentTime = timeLine.currentTime();
QVERIFY(currentTime < oldCurrentTime);
QVERIFY(currentTime > 0);
}
}
void tst_QTimeLine::restart()
{
QTimeLine timeLine(100);
timeLine.setFrameRange(0,9);
timeLine.start();
QTRY_COMPARE(timeLine.currentFrame(), timeLine.endFrame());
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
// A restart with the same duration
timeLine.start();
QCOMPARE(timeLine.state(), QTimeLine::Running);
QCOMPARE(timeLine.currentFrame(), timeLine.startFrame());
QCOMPARE(timeLine.currentTime(), 0);
QTRY_COMPARE(timeLine.currentFrame(), timeLine.endFrame());
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
// Set a smaller duration and restart
timeLine.setDuration(50);
timeLine.start();
QCOMPARE(timeLine.state(), QTimeLine::Running);
QCOMPARE(timeLine.currentFrame(), timeLine.startFrame());
QCOMPARE(timeLine.currentTime(), 0);
QTRY_COMPARE(timeLine.currentFrame(), timeLine.endFrame());
QCOMPARE(timeLine.state(), QTimeLine::NotRunning);
// Set a longer duration and restart
timeLine.setDuration(150);
timeLine.start();
QCOMPARE(timeLine.state(), QTimeLine::Running);
QCOMPARE(timeLine.currentFrame(), timeLine.startFrame());
QCOMPARE(timeLine.currentTime(), 0);
}
void tst_QTimeLine::setPaused()
{
const int EndTime = 10000;
QTimeLine timeLine(EndTime);
{
QCOMPARE(timeLine.currentTime(), 0);
timeLine.start();
QTRY_VERIFY(timeLine.currentTime() != 0); // wait for start
timeLine.setPaused(true);
int oldCurrentTime = timeLine.currentTime();
QVERIFY(oldCurrentTime > 0);
QVERIFY(oldCurrentTime < EndTime);
QTest::qWait(1000);
timeLine.setPaused(false);
QTRY_VERIFY(timeLine.currentTime() > oldCurrentTime);
QVERIFY(timeLine.currentTime() > 0);
QVERIFY(timeLine.currentTime() < EndTime);
timeLine.stop();
}
}
void tst_QTimeLine::automatedBindableTests()
{
QTimeLine timeLine(200);
QTestPrivate::testReadWritePropertyBasics(timeLine, 1000, 2000, "duration");
if (QTest::currentTestFailed()) {
qDebug() << "Failed property test for duration";
return;
}
QTestPrivate::testReadWritePropertyBasics(timeLine, 10, 20, "updateInterval");
if (QTest::currentTestFailed()) {
qDebug() << "Failed property test for updateInterval";
return;
}
QTestPrivate::testReadWritePropertyBasics(timeLine, 10, 20, "currentTime");
if (QTest::currentTestFailed()) {
qDebug() << "Failed property test for currentTime";
return;
}
QTestPrivate::testReadWritePropertyBasics(timeLine, QTimeLine::Forward, QTimeLine::Backward,
"direction");
if (QTest::currentTestFailed()) {
qDebug() << "Failed property test for direction";
return;
}
QTestPrivate::testReadWritePropertyBasics(timeLine, 4, 5, "loopCount");
if (QTest::currentTestFailed()) {
qDebug() << "Failed property test for loopCount";
return;
}
QTestPrivate::testReadWritePropertyBasics<QTimeLine, QEasingCurve>(
timeLine, QEasingCurve::InQuad, QEasingCurve::OutQuad, "easingCurve");
if (QTest::currentTestFailed()) {
qDebug() << "Failed property test for easingCurve";
return;
}
}
QTEST_MAIN(tst_QTimeLine)
#include "tst_qtimeline.moc"