Alessandro Bonazzi 8c43d5cf2f Patch level : 12.00
Files correlati     : cg0.exe cg0700a.msk cg0700b.msk cg3.exe cg4.exe

Bug                 :

Commento:
Merge 1.0 libraries
2025-04-06 00:42:21 +02:00

433 lines
14 KiB
C++

/****************************************************************************
**
** Copyright (C) 2015 The Qt Company Ltd.
** Contact: http://www.qt.io/licensing/
**
** This file is part of the test suite of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL$
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms
** and conditions see http://www.qt.io/terms-conditions. For further
** information use the contact form at http://www.qt.io/contact-us.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 or version 3 as published by the Free
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
** following information to ensure the GNU Lesser General Public License
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** As a special exception, The Qt Company gives you certain additional
** rights. These rights are described in The Qt Company LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3.0 as published by the Free Software
** Foundation and appearing in the file LICENSE.GPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU General Public License version 3.0 requirements will be
** met: http://www.gnu.org/copyleft/gpl.html.
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include <qtest.h>
#include <QtTest/QtTest>
#include <QtDeclarative/qdeclarativeengine.h>
#include <QtDeclarative/qdeclarativeimageprovider.h>
#include <private/qdeclarativeimage_p.h>
#include <QImageReader>
#include <QWaitCondition>
#include "../../../shared/util.h"
#ifdef Q_OS_SYMBIAN
// In Symbian OS test data is located in applications private dir
#define SRCDIR "."
#endif
Q_DECLARE_METATYPE(QDeclarativeImageProvider*);
class tst_qdeclarativeimageprovider : public QObject
{
Q_OBJECT
public:
tst_qdeclarativeimageprovider()
{
}
private slots:
void requestImage_sync_data();
void requestImage_sync();
void requestImage_async_data();
void requestImage_async();
void requestPixmap_sync_data();
void requestPixmap_sync();
void requestPixmap_async();
void removeProvider_data();
void removeProvider();
void threadTest();
private:
QString newImageFileName() const;
void fillRequestTestsData(const QString &id);
void runTest(bool async, QDeclarativeImageProvider *provider);
};
class TestQImageProvider : public QDeclarativeImageProvider
{
public:
TestQImageProvider(bool *deleteWatch = 0)
: QDeclarativeImageProvider(Image), deleteWatch(deleteWatch)
{
}
~TestQImageProvider()
{
if (deleteWatch)
*deleteWatch = true;
}
QImage requestImage(const QString &id, QSize *size, const QSize& requestedSize)
{
lastImageId = id;
if (id == QLatin1String("no-such-file.png"))
return QImage();
int width = 100;
int height = 100;
QImage image(width, height, QImage::Format_RGB32);
if (size)
*size = QSize(width, height);
if (requestedSize.isValid())
image = image.scaled(requestedSize);
return image;
}
bool *deleteWatch;
QString lastImageId;
};
Q_DECLARE_METATYPE(TestQImageProvider*);
class TestQPixmapProvider : public QDeclarativeImageProvider
{
public:
TestQPixmapProvider(bool *deleteWatch = 0)
: QDeclarativeImageProvider(Pixmap), deleteWatch(deleteWatch)
{
}
~TestQPixmapProvider()
{
if (deleteWatch)
*deleteWatch = true;
}
QPixmap requestPixmap(const QString &id, QSize *size, const QSize& requestedSize)
{
lastImageId = id;
if (id == QLatin1String("no-such-file.png"))
return QPixmap();
int width = 100;
int height = 100;
QPixmap image(width, height);
if (size)
*size = QSize(width, height);
if (requestedSize.isValid())
image = image.scaled(requestedSize);
return image;
}
bool *deleteWatch;
QString lastImageId;
};
Q_DECLARE_METATYPE(TestQPixmapProvider*);
QString tst_qdeclarativeimageprovider::newImageFileName() const
{
// need to generate new filenames each time or else images are loaded
// from cache and we won't get loading status changes when testing
// async loading
static int count = 0;
return QString("image://test/image-%1.png").arg(count++);
}
void tst_qdeclarativeimageprovider::fillRequestTestsData(const QString &id)
{
QTest::addColumn<QString>("source");
QTest::addColumn<QString>("imageId");
QTest::addColumn<QString>("properties");
QTest::addColumn<QSize>("size");
QTest::addColumn<QString>("error");
QString fileName = newImageFileName();
QTest::newRow(QTest::toString(id + " simple test"))
<< "image://test/" + fileName << fileName << "" << QSize(100,100) << "";
fileName = newImageFileName();
QTest::newRow(QTest::toString(id + " simple test with capitalization"))//As it's a URL, should make no difference
<< "image://Test/" + fileName << fileName << "" << QSize(100,100) << "";
fileName = newImageFileName();
QTest::newRow(QTest::toString(id + " url with no id"))
<< "image://test/" + fileName << "" + fileName << "" << QSize(100,100) << "";
fileName = newImageFileName();
QTest::newRow(QTest::toString(id + " url with path"))
<< "image://test/test/path" + fileName << "test/path" + fileName << "" << QSize(100,100) << "";
fileName = newImageFileName();
QTest::newRow(QTest::toString(id + " url with fragment"))
<< "image://test/faq.html?#question13" + fileName << "faq.html?#question13" + fileName << "" << QSize(100,100) << "";
fileName = newImageFileName();
QTest::newRow(QTest::toString(id + " url with query"))
<< "image://test/cgi-bin/drawgraph.cgi?type=pie&color=green" + fileName << "cgi-bin/drawgraph.cgi?type=pie&color=green" + fileName
<< "" << QSize(100,100) << "";
fileName = newImageFileName();
QTest::newRow(QTest::toString(id + " scaled image"))
<< "image://test/" + fileName << fileName << "sourceSize: \"80x30\"" << QSize(80,30) << "";
QTest::newRow(QTest::toString(id + " missing"))
<< "image://test/no-such-file.png" << "no-such-file.png" << "" << QSize(100,100)
<< "file::2:1: QML Image: Failed to get image from provider: image://test/no-such-file.png";
QTest::newRow(QTest::toString(id + " unknown provider"))
<< "image://bogus/exists.png" << "" << "" << QSize()
<< "file::2:1: QML Image: Failed to get image from provider: image://bogus/exists.png";
}
void tst_qdeclarativeimageprovider::runTest(bool async, QDeclarativeImageProvider *provider)
{
QFETCH(QString, source);
QFETCH(QString, imageId);
QFETCH(QString, properties);
QFETCH(QSize, size);
QFETCH(QString, error);
if (!error.isEmpty())
QTest::ignoreMessage(QtWarningMsg, error.toUtf8());
QDeclarativeEngine engine;
engine.addImageProvider("test", provider);
QVERIFY(engine.imageProvider("test") != 0);
QString componentStr = "import QtQuick 1.0\nImage { source: \"" + source + "\"; "
+ (async ? "asynchronous: true; " : "")
+ properties + " }";
QDeclarativeComponent component(&engine);
component.setData(componentStr.toLatin1(), QUrl::fromLocalFile(""));
QDeclarativeImage *obj = qobject_cast<QDeclarativeImage*>(component.create());
QVERIFY(obj != 0);
if (async)
QTRY_VERIFY(obj->status() == QDeclarativeImage::Loading);
QCOMPARE(obj->source(), QUrl(source));
if (error.isEmpty()) {
if (async)
QTRY_VERIFY(obj->status() == QDeclarativeImage::Ready);
else
QVERIFY(obj->status() == QDeclarativeImage::Ready);
if (QByteArray(QTest::currentDataTag()).startsWith("qimage"))
QCOMPARE(static_cast<TestQImageProvider*>(provider)->lastImageId, imageId);
else
QCOMPARE(static_cast<TestQPixmapProvider*>(provider)->lastImageId, imageId);
QCOMPARE(obj->width(), qreal(size.width()));
QCOMPARE(obj->height(), qreal(size.height()));
QCOMPARE(obj->pixmap().width(), size.width());
QCOMPARE(obj->pixmap().height(), size.height());
QCOMPARE(obj->fillMode(), QDeclarativeImage::Stretch);
QCOMPARE(obj->progress(), 1.0);
} else {
if (async)
QTRY_VERIFY(obj->status() == QDeclarativeImage::Error);
else
QVERIFY(obj->status() == QDeclarativeImage::Error);
}
delete obj;
}
void tst_qdeclarativeimageprovider::requestImage_sync_data()
{
fillRequestTestsData("qimage|sync");
}
void tst_qdeclarativeimageprovider::requestImage_sync()
{
bool deleteWatch = false;
runTest(false, new TestQImageProvider(&deleteWatch));
QVERIFY(deleteWatch);
}
void tst_qdeclarativeimageprovider::requestImage_async_data()
{
fillRequestTestsData("qimage|async");
}
void tst_qdeclarativeimageprovider::requestImage_async()
{
bool deleteWatch = false;
runTest(true, new TestQImageProvider(&deleteWatch));
QVERIFY(deleteWatch);
}
void tst_qdeclarativeimageprovider::requestPixmap_sync_data()
{
fillRequestTestsData("qpixmap");
}
void tst_qdeclarativeimageprovider::requestPixmap_sync()
{
bool deleteWatch = false;
runTest(false, new TestQPixmapProvider(&deleteWatch));
QVERIFY(deleteWatch);
}
void tst_qdeclarativeimageprovider::requestPixmap_async()
{
QDeclarativeEngine engine;
QDeclarativeImageProvider *provider = new TestQPixmapProvider();
engine.addImageProvider("test", provider);
QVERIFY(engine.imageProvider("test") != 0);
// pixmaps are loaded synchronously regardless of 'asynchronous' value
QString componentStr = "import QtQuick 1.0\nImage { asynchronous: true; source: \"image://test/pixmap-async-test.png\" }";
QDeclarativeComponent component(&engine);
component.setData(componentStr.toLatin1(), QUrl::fromLocalFile(""));
QDeclarativeImage *obj = qobject_cast<QDeclarativeImage*>(component.create());
QVERIFY(obj != 0);
delete obj;
}
void tst_qdeclarativeimageprovider::removeProvider_data()
{
QTest::addColumn<QDeclarativeImageProvider*>("provider");
QTest::newRow("qimage") << static_cast<QDeclarativeImageProvider*>(new TestQImageProvider);
QTest::newRow("qpixmap") << static_cast<QDeclarativeImageProvider*>(new TestQPixmapProvider);
}
void tst_qdeclarativeimageprovider::removeProvider()
{
QFETCH(QDeclarativeImageProvider*, provider);
QDeclarativeEngine engine;
engine.addImageProvider("test", provider);
QVERIFY(engine.imageProvider("test") != 0);
// add provider, confirm it works
QString componentStr = "import QtQuick 1.0\nImage { source: \"" + newImageFileName() + "\" }";
QDeclarativeComponent component(&engine);
component.setData(componentStr.toLatin1(), QUrl::fromLocalFile(""));
QDeclarativeImage *obj = qobject_cast<QDeclarativeImage*>(component.create());
QVERIFY(obj != 0);
QCOMPARE(obj->status(), QDeclarativeImage::Ready);
// remove the provider and confirm
QString fileName = newImageFileName();
QString error("file::2:1: QML Image: Failed to get image from provider: " + fileName);
QTest::ignoreMessage(QtWarningMsg, error.toUtf8());
engine.removeImageProvider("test");
obj->setSource(QUrl(fileName));
QCOMPARE(obj->status(), QDeclarativeImage::Error);
delete obj;
}
class TestThreadProvider : public QDeclarativeImageProvider
{
public:
TestThreadProvider() : QDeclarativeImageProvider(Image), ok(false) {}
~TestThreadProvider() {}
QImage requestImage(const QString &id, QSize *size, const QSize& requestedSize)
{
mutex.lock();
if (!ok)
cond.wait(&mutex);
mutex.unlock();
QVector<int> v;
for (int i = 0; i < 10000; i++)
v.prepend(i); //do some computation
QImage image(50,50, QImage::Format_RGB32);
image.fill(QColor(id).rgb());
if (size)
*size = image.size();
if (requestedSize.isValid())
image = image.scaled(requestedSize);
return image;
}
QWaitCondition cond;
QMutex mutex;
bool ok;
};
void tst_qdeclarativeimageprovider::threadTest()
{
QDeclarativeEngine engine;
TestThreadProvider *provider = new TestThreadProvider;
engine.addImageProvider("test_thread", provider);
QVERIFY(engine.imageProvider("test_thread") != 0);
QString componentStr = "import QtQuick 1.0\nItem { \n"
"Image { source: \"image://test_thread/blue\"; asynchronous: true; }\n"
"Image { source: \"image://test_thread/red\"; asynchronous: true; }\n"
"Image { source: \"image://test_thread/green\"; asynchronous: true; }\n"
"Image { source: \"image://test_thread/yellow\"; asynchronous: true; }\n"
" }";
QDeclarativeComponent component(&engine);
component.setData(componentStr.toLatin1(), QUrl::fromLocalFile(""));
QObject *obj = component.create();
//MUST not deadlock
QVERIFY(obj != 0);
QList<QDeclarativeImage *> images = obj->findChildren<QDeclarativeImage *>();
QCOMPARE(images.count(), 4);
QTest::qWait(100);
foreach(QDeclarativeImage *img, images) {
QCOMPARE(img->status(), QDeclarativeImage::Loading);
}
provider->ok = true;
provider->cond.wakeAll();
QTest::qWait(250);
foreach(QDeclarativeImage *img, images) {
QTRY_VERIFY(img->status() == QDeclarativeImage::Ready);
}
}
QTEST_MAIN(tst_qdeclarativeimageprovider)
#include "tst_qdeclarativeimageprovider.moc"