blob: 3e9047cc5a7b949a8ee29b2a23ca45f25d59a7c8 [file] [log] [blame]
/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of the test suite of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
** 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 https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3 as published by the Free Software
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
** included in the packaging of this file. Please review the following
** information to ensure the GNU General Public License requirements will
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include <qtest.h>
#include <QQmlEngine>
#include <QQmlComponent>
#include <QDebug>
#include <QJSValueIterator>
#include <private/qquickvaluetypes_p.h>
#include <private/qqmlglobal_p.h>
#include <private/qv4engine_p.h>
#include <private/qv4variantobject_p.h>
#include "../../shared/util.h"
#include "testtypes.h"
QT_BEGIN_NAMESPACE
extern int qt_defaultDpi(void);
QT_END_NAMESPACE
class tst_qqmlvaluetypes : public QQmlDataTest
{
Q_OBJECT
public:
tst_qqmlvaluetypes() {}
private slots:
void initTestCase();
void point();
void pointf();
void size();
void sizef();
void sizereadonly();
void rect();
void rectf();
void vector2d();
void vector3d();
void vector4d();
void quaternion();
void matrix4x4();
void font();
void color();
void variant();
void locale();
void bindingAssignment();
void bindingRead();
void staticAssignment();
void scriptAccess();
void autoBindingRemoval();
void valueSources();
void valueInterceptors();
void bindingConflict();
void deletedObject();
void bindingVariantCopy();
void scriptVariantCopy();
void enums();
void conflictingBindings();
void returnValues();
void varAssignment();
void bindingsSpliceCorrectly();
void nonValueTypeComparison();
void initializeByWrite();
void groupedInterceptors();
void groupedInterceptors_data();
void customValueType();
void customValueTypeInQml();
void gadgetInheritance();
void gadgetTemplateInheritance();
void sequences();
void toStringConversion();
void enumerableProperties();
void enumProperties();
void scarceTypes();
void nonValueTypes();
private:
QQmlEngine engine;
};
void tst_qqmlvaluetypes::initTestCase()
{
QQmlDataTest::initTestCase();
registerTypes();
}
void tst_qqmlvaluetypes::point()
{
{
QQmlComponent component(&engine, testFileUrl("point_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->property("p_x").toInt(), 10);
QCOMPARE(object->property("p_y").toInt(), 4);
QCOMPARE(object->property("copy"), QVariant(QPoint(10, 4)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("point_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->point(), QPoint(11, 12));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("point_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QPoint(10, 4)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), true);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
QCOMPARE(object->property("equalsOther").toBool(), false);
QCOMPARE(object->property("pointEqualsPointf").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::pointf()
{
{
QQmlComponent component(&engine, testFileUrl("pointf_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(float(object->property("p_x").toDouble()), float(11.3));
QCOMPARE(float(object->property("p_y").toDouble()), float(-10.9));
QCOMPARE(object->property("copy"), QVariant(QPointF(11.3, -10.9)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("pointf_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->pointf(), QPointF(6.8, 9.3));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("pointf_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QPointF(11.3, -10.9)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), true);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
QCOMPARE(object->property("equalsOther").toBool(), false);
QCOMPARE(object->property("pointfEqualsPoint").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::size()
{
{
QQmlComponent component(&engine, testFileUrl("size_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->property("s_width").toInt(), 1912);
QCOMPARE(object->property("s_height").toInt(), 1913);
QCOMPARE(object->property("copy"), QVariant(QSize(1912, 1913)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("size_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->size(), QSize(13, 88));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("size_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QSize(1912, 1913)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), true);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
QCOMPARE(object->property("equalsOther").toBool(), false);
QCOMPARE(object->property("sizeEqualsSizef").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::sizef()
{
{
QQmlComponent component(&engine, testFileUrl("sizef_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(float(object->property("s_width").toDouble()), float(0.1));
QCOMPARE(float(object->property("s_height").toDouble()), float(100923.2));
QCOMPARE(object->property("copy"), QVariant(QSizeF(0.1, 100923.2)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("sizef_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->sizef(), QSizeF(44.3, 92.8));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("sizef_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QSizeF(0.1, 100923)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), true);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
QCOMPARE(object->property("equalsOther").toBool(), false);
QCOMPARE(object->property("sizefEqualsSize").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::variant()
{
{
QQmlComponent component(&engine, testFileUrl("variant_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(float(object->property("s_width").toDouble()), float(0.1));
QCOMPARE(float(object->property("s_height").toDouble()), float(100923.2));
QCOMPARE(object->property("copy"), QVariant(QSizeF(0.1, 100923.2)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("variant_write.1.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QVERIFY(object->property("complete").toBool());
QVERIFY(object->property("success").toBool());
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("variant_write.2.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QVERIFY(object->property("complete").toBool());
QVERIFY(object->property("success").toBool());
delete object;
}
}
void tst_qqmlvaluetypes::locale()
{
{
QQmlComponent component(&engine, testFileUrl("locale_read.qml"));
QScopedPointer<QObject> object(component.create());
QVERIFY(!object.isNull());
#if QT_CONFIG(im)
QVERIFY(QQml_guiProvider()->inputMethod());
QInputMethod *inputMethod = qobject_cast<QInputMethod*>(QQml_guiProvider()->inputMethod());
QLocale locale = inputMethod->locale();
QCOMPARE(object->property("amText").toString(), locale.amText());
QCOMPARE(object->property("decimalPoint").toString().at(0), locale.decimalPoint());
QCOMPARE(object->property("exponential").toString().at(0), locale.exponential());
// Sunday is 0 in JavaScript.
QCOMPARE(object->property("firstDayOfWeek").toInt(), int(locale.firstDayOfWeek() == Qt::Sunday ? 0 : locale.firstDayOfWeek()));
QCOMPARE(object->property("groupSeparator").toString().at(0), locale.groupSeparator());
QCOMPARE(object->property("measurementSystem").toInt(), int(locale.measurementSystem()));
QCOMPARE(object->property("name").toString(), locale.name());
QCOMPARE(object->property("nativeCountryName").toString(), locale.nativeCountryName());
QCOMPARE(object->property("nativeLanguageName").toString(), locale.nativeLanguageName());
QCOMPARE(object->property("negativeSign").toString().at(0), locale.negativeSign());
QCOMPARE(object->property("percent").toString().at(0), locale.percent());
QCOMPARE(object->property("pmText").toString(), locale.pmText());
QCOMPARE(object->property("positiveSign").toString().at(0), locale.positiveSign());
QCOMPARE(object->property("textDirection").toInt(), int(locale.textDirection()));
QCOMPARE(object->property("uiLanguages").toStringList(), locale.uiLanguages());
QList<Qt::DayOfWeek> weekDays;
foreach (const QVariant &weekDay, object->property("weekDays").toList()) {
weekDays.append(Qt::DayOfWeek(weekDay.toInt()));
}
QCOMPARE(weekDays, locale.weekdays());
QCOMPARE(object->property("zeroDigit").toString().at(0), locale.zeroDigit());
#endif // im
}
}
void tst_qqmlvaluetypes::sizereadonly()
{
{
QQmlComponent component(&engine, testFileUrl("sizereadonly_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->property("s_width").toInt(), 1912);
QCOMPARE(object->property("s_height").toInt(), 1913);
QCOMPARE(object->property("copy"), QVariant(QSize(1912, 1913)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("sizereadonly_writeerror.qml"));
QVERIFY(component.isError());
QCOMPARE(component.errors().at(0).description(), QLatin1String("Invalid property assignment: \"sizereadonly\" is a read-only property"));
}
{
QQmlComponent component(&engine, testFileUrl("sizereadonly_writeerror2.qml"));
QVERIFY(component.isError());
QCOMPARE(component.errors().at(0).description(), QLatin1String("Invalid property assignment: \"sizereadonly\" is a read-only property"));
}
{
QQmlComponent component(&engine, testFileUrl("sizereadonly_writeerror3.qml"));
QVERIFY(component.isError());
QCOMPARE(component.errors().at(0).description(), QLatin1String("Invalid property assignment: \"sizereadonly\" is a read-only property"));
}
{
QQmlComponent component(&engine, testFileUrl("sizereadonly_writeerror4.qml"));
QObject *object = component.create();
QVERIFY(object);
QCOMPARE(object->property("sizereadonly").toSize(), QSize(1912, 1913));
delete object;
}
}
void tst_qqmlvaluetypes::rect()
{
{
QQmlComponent component(&engine, testFileUrl("rect_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->property("r_x").toInt(), 2);
QCOMPARE(object->property("r_y").toInt(), 3);
QCOMPARE(object->property("r_width").toInt(), 109);
QCOMPARE(object->property("r_height").toInt(), 102);
QCOMPARE(object->property("r_left").toInt(), 2);
QCOMPARE(object->property("r_right").toInt(), 110);
QCOMPARE(object->property("r_top").toInt(), 3);
QCOMPARE(object->property("r_bottom").toInt(), 104);
QCOMPARE(object->property("copy"), QVariant(QRect(2, 3, 109, 102)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("rect_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect(), QRect(1234, 7, 56, 63));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("rect_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QRect(2, 3, 109, 102)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), true);
QCOMPARE(object->property("equalsSelf").toBool(), true);
QCOMPARE(object->property("equalsOther").toBool(), false);
QCOMPARE(object->property("rectEqualsRectf").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::rectf()
{
{
QQmlComponent component(&engine, testFileUrl("rectf_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(float(object->property("r_x").toDouble()), float(103.8));
QCOMPARE(float(object->property("r_y").toDouble()), float(99.2));
QCOMPARE(float(object->property("r_width").toDouble()), float(88.1));
QCOMPARE(float(object->property("r_height").toDouble()), float(77.6));
QCOMPARE(float(object->property("r_left").toDouble()), float(103.8));
QCOMPARE(float(object->property("r_right").toDouble()), float(191.9));
QCOMPARE(float(object->property("r_top").toDouble()), float(99.2));
QCOMPARE(float(object->property("r_bottom").toDouble()), float(176.8));
QCOMPARE(object->property("copy"), QVariant(QRectF(103.8, 99.2, 88.1, 77.6)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("rectf_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rectf(), QRectF(70.1, -113.2, 80924.8, 99.2));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("rectf_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QRectF(103.8, 99.2, 88.1, 77.6)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), true);
QCOMPARE(object->property("equalsSelf").toBool(), true);
QCOMPARE(object->property("equalsOther").toBool(), false);
QCOMPARE(object->property("rectfEqualsRect").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::vector2d()
{
{
QQmlComponent component(&engine, testFileUrl("vector2d_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE((float)object->property("v_x").toDouble(), (float)32.88);
QCOMPARE((float)object->property("v_y").toDouble(), (float)1.3);
QCOMPARE(object->property("copy"), QVariant(QVector2D(32.88f, 1.3f)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector2d_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->vector2(), QVector2D(-0.3f, -12.9f));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector2d_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QVector2D(32.88, 1.3)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector2d_invokables.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QVERIFY(object->property("success").toBool());
delete object;
}
}
void tst_qqmlvaluetypes::vector3d()
{
{
QQmlComponent component(&engine, testFileUrl("vector3d_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE((float)object->property("v_x").toDouble(), (float)23.88);
QCOMPARE((float)object->property("v_y").toDouble(), (float)3.1);
QCOMPARE((float)object->property("v_z").toDouble(), (float)4.3);
QCOMPARE(object->property("copy"), QVariant(QVector3D(23.88f, 3.1f, 4.3f)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector3d_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->vector(), QVector3D(-0.3f, -12.9f, 907.4f));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector3d_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QVector3D(23.88, 3.1, 4.3)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), true);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
QCOMPARE(object->property("equalsOther").toBool(), false);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector3d_invokables.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QVERIFY(object->property("success").toBool());
delete object;
}
}
void tst_qqmlvaluetypes::vector4d()
{
{
QQmlComponent component(&engine, testFileUrl("vector4d_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE((float)object->property("v_x").toDouble(), (float)54.2);
QCOMPARE((float)object->property("v_y").toDouble(), (float)23.88);
QCOMPARE((float)object->property("v_z").toDouble(), (float)3.1);
QCOMPARE((float)object->property("v_w").toDouble(), (float)4.3);
QCOMPARE(object->property("copy"), QVariant(QVector4D(54.2f, 23.88f, 3.1f, 4.3f)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector4d_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->vector4(), QVector4D(-0.3f, -12.9f, 907.4f, 88.5f));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector4d_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QVector4D(54.2, 23.88, 3.1, 4.3)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("vector4d_invokables.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QVERIFY(object->property("success").toBool());
delete object;
}
}
void tst_qqmlvaluetypes::quaternion()
{
{
QQmlComponent component(&engine, testFileUrl("quaternion_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE((float)object->property("v_scalar").toDouble(), (float)4.3);
QCOMPARE((float)object->property("v_x").toDouble(), (float)54.2);
QCOMPARE((float)object->property("v_y").toDouble(), (float)23.88);
QCOMPARE((float)object->property("v_z").toDouble(), (float)3.1);
QCOMPARE(object->property("copy"), QVariant(QQuaternion(4.3f, 54.2f, 23.88f, 3.1f)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("quaternion_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->quaternion(), QQuaternion(88.5f, -0.3f, -12.9f, 907.4f));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("quaternion_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QQuaternion(4.3, 54.2, 23.88, 3.1)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::matrix4x4()
{
{
QQmlComponent component(&engine, testFileUrl("matrix4x4_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE((float)object->property("v_m11").toDouble(), (float)1);
QCOMPARE((float)object->property("v_m12").toDouble(), (float)2);
QCOMPARE((float)object->property("v_m13").toDouble(), (float)3);
QCOMPARE((float)object->property("v_m14").toDouble(), (float)4);
QCOMPARE((float)object->property("v_m21").toDouble(), (float)5);
QCOMPARE((float)object->property("v_m22").toDouble(), (float)6);
QCOMPARE((float)object->property("v_m23").toDouble(), (float)7);
QCOMPARE((float)object->property("v_m24").toDouble(), (float)8);
QCOMPARE((float)object->property("v_m31").toDouble(), (float)9);
QCOMPARE((float)object->property("v_m32").toDouble(), (float)10);
QCOMPARE((float)object->property("v_m33").toDouble(), (float)11);
QCOMPARE((float)object->property("v_m34").toDouble(), (float)12);
QCOMPARE((float)object->property("v_m41").toDouble(), (float)13);
QCOMPARE((float)object->property("v_m42").toDouble(), (float)14);
QCOMPARE((float)object->property("v_m43").toDouble(), (float)15);
QCOMPARE((float)object->property("v_m44").toDouble(), (float)16);
QCOMPARE(object->property("copy"),
QVariant(QMatrix4x4(1, 2, 3, 4,
5, 6, 7, 8,
9, 10, 11, 12,
13, 14, 15, 16)));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("matrix4x4_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->matrix(), QMatrix4x4(11, 12, 13, 14,
21, 22, 23, 24,
31, 32, 33, 34,
41, 42, 43, 44));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("matrix4x4_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QMatrix4x4(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16)");
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("matrix4x4_invokables.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("success").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::font()
{
{
QQmlComponent component(&engine, testFileUrl("font_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
QVERIFY(object != nullptr);
QCOMPARE(object->property("f_family").toString(), object->font().family());
QCOMPARE(object->property("f_bold").toBool(), object->font().bold());
QCOMPARE(object->property("f_weight").toInt(), object->font().weight());
QCOMPARE(object->property("f_italic").toBool(), object->font().italic());
QCOMPARE(object->property("f_underline").toBool(), object->font().underline());
QCOMPARE(object->property("f_overline").toBool(), object->font().overline());
QCOMPARE(object->property("f_strikeout").toBool(), object->font().strikeOut());
// If QFont::pixelSize() was set, QFont::pointSizeF() would return -1.
// If QFont::pointSizeF() was set, QFont::pixelSize() would return -1.
// QQuickFontValueType doesn't follow this semantic (if its -1 it calculates the value of
// the property from the other one)
double expectedPointSizeF = object->font().pointSizeF();
if (expectedPointSizeF == -1) expectedPointSizeF = object->font().pixelSize() * qreal(72.) / qreal(qt_defaultDpi());
int expectedPixelSize = object->font().pixelSize();
if (expectedPixelSize == -1) expectedPixelSize = int((object->font().pointSizeF() * qt_defaultDpi()) / qreal(72.));
QCOMPARE(object->property("f_pointSize").toDouble(), expectedPointSizeF);
QCOMPARE(object->property("f_pixelSize").toInt(), expectedPixelSize);
QCOMPARE(object->property("f_capitalization").toInt(), (int)object->font().capitalization());
QCOMPARE(object->property("f_letterSpacing").toDouble(), object->font().letterSpacing());
QCOMPARE(object->property("f_wordSpacing").toDouble(), object->font().wordSpacing());
QCOMPARE(object->property("copy"), QVariant(object->font()));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("font_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QFont font;
font.setFamily("Helvetica");
font.setBold(false);
font.setWeight(QFont::Normal);
font.setItalic(false);
font.setUnderline(false);
font.setStrikeOut(false);
font.setPointSize(15);
font.setCapitalization(QFont::AllLowercase);
font.setLetterSpacing(QFont::AbsoluteSpacing, 9.7);
font.setWordSpacing(11.2);
QFont f = object->font();
QCOMPARE(f.family(), font.family());
QCOMPARE(f.bold(), font.bold());
QCOMPARE(f.weight(), font.weight());
QCOMPARE(f.italic(), font.italic());
QCOMPARE(f.underline(), font.underline());
QCOMPARE(f.strikeOut(), font.strikeOut());
QCOMPARE(f.pointSize(), font.pointSize());
QCOMPARE(f.capitalization(), font.capitalization());
QCOMPARE(f.letterSpacing(), font.letterSpacing());
QCOMPARE(f.wordSpacing(), font.wordSpacing());
delete object;
}
// Test pixelSize
{
QQmlComponent component(&engine, testFileUrl("font_write.2.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().pixelSize(), 10);
delete object;
}
// Test pixelSize and pointSize
{
QQmlComponent component(&engine, testFileUrl("font_write.3.qml"));
QTest::ignoreMessage(QtWarningMsg, "Both point size and pixel size set. Using pixel size.");
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().pixelSize(), 10);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("font_write.4.qml"));
QTest::ignoreMessage(QtWarningMsg, "Both point size and pixel size set. Using pixel size.");
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().pixelSize(), 10);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("font_write.5.qml"));
QObject *object = qobject_cast<QObject *>(component.create());
QVERIFY(object != nullptr);
MyTypeObject *object1 = object->findChild<MyTypeObject *>("object1");
QVERIFY(object1 != nullptr);
MyTypeObject *object2 = object->findChild<MyTypeObject *>("object2");
QVERIFY(object2 != nullptr);
QCOMPARE(object1->font().pixelSize(), 19);
QCOMPARE(object2->font().pointSize(), 14);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("font_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString tostring = QLatin1String("QFont(") + object->font().toString() + QLatin1Char(')');
QCOMPARE(object->property("tostring").toString(), tostring);
QCOMPARE(object->property("equalsString").toBool(), true);
QCOMPARE(object->property("equalsColor").toBool(), false);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
QCOMPARE(object->property("equalsSelf").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::color()
{
{
QQmlComponent component(&engine, testFileUrl("color_read.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE((float)object->property("v_r").toDouble(), (float)0.2);
QCOMPARE((float)object->property("v_g").toDouble(), (float)0.88);
QCOMPARE((float)object->property("v_b").toDouble(), (float)0.6);
QCOMPARE((float)object->property("v_a").toDouble(), (float)0.34);
QCOMPARE(qRound(object->property("hsv_h").toDouble() * 100), 43);
QCOMPARE(qRound(object->property("hsv_s").toDouble() * 100), 77);
QCOMPARE(qRound(object->property("hsv_v").toDouble() * 100), 88);
QCOMPARE(qRound(object->property("hsl_h").toDouble() * 100), 43);
QCOMPARE(qRound(object->property("hsl_s").toDouble() * 100), 74);
QCOMPARE(qRound(object->property("hsl_l").toDouble() * 100), 54);
QCOMPARE(object->property("valid").userType(), QMetaType::Bool);
QVERIFY(object->property("valid").toBool());
QCOMPARE(object->property("invalid").userType(), QMetaType::Bool);
QVERIFY(!object->property("invalid").toBool());
QColor comparison;
comparison.setRedF(0.2);
comparison.setGreenF(0.88);
comparison.setBlueF(0.6);
comparison.setAlphaF(0.34);
QCOMPARE(object->property("copy"), QVariant(comparison));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("color_write.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QColor newColor;
newColor.setRedF(0.5);
newColor.setGreenF(0.38);
newColor.setBlueF(0.3);
newColor.setAlphaF(0.7);
QCOMPARE(object->color(), newColor);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("color_write_HSV.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QColor newColor;
newColor.setHsvF(0.43, 0.77, 0.88, 0.7);
QCOMPARE(object->color(), newColor);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("color_write_HSL.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QColor newColor;
newColor.setHslF(0.43, 0.74, 0.54, 0.7);
QCOMPARE(object->color(), newColor);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("color_compare.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QString colorString("#5733e199");
QCOMPARE(object->property("colorToString").toString(), colorString);
QCOMPARE(object->property("colorEqualsIdenticalRgba").toBool(), true);
QCOMPARE(object->property("colorEqualsDifferentAlpha").toBool(), false);
QCOMPARE(object->property("colorEqualsDifferentRgba").toBool(), false);
QCOMPARE(object->property("colorToStringEqualsColorString").toBool(), true);
QCOMPARE(object->property("colorToStringEqualsDifferentAlphaString").toBool(), true);
QCOMPARE(object->property("colorToStringEqualsDifferentRgbaString").toBool(), false);
QCOMPARE(object->property("colorEqualsColorString").toBool(), true); // maintaining behaviour with QtQuick 1.0
QCOMPARE(object->property("colorEqualsDifferentAlphaString").toBool(), true); // maintaining behaviour with QtQuick 1.0
QCOMPARE(object->property("colorEqualsDifferentRgbaString").toBool(), false);
QCOMPARE(object->property("equalsColor").toBool(), true);
QCOMPARE(object->property("equalsVector3d").toBool(), false);
QCOMPARE(object->property("equalsSize").toBool(), false);
QCOMPARE(object->property("equalsPoint").toBool(), false);
QCOMPARE(object->property("equalsRect").toBool(), false);
// Color == Property and Property == Color should return the same result.
QCOMPARE(object->property("equalsColorRHS").toBool(), object->property("equalsColor").toBool());
QCOMPARE(object->property("colorEqualsCopy").toBool(), true);
QCOMPARE(object->property("copyEqualsColor").toBool(), object->property("colorEqualsCopy").toBool());
delete object;
}
}
// Test bindings can write to value types
void tst_qqmlvaluetypes::bindingAssignment()
{
// binding declaration
{
QQmlComponent component(&engine, testFileUrl("bindingAssignment.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect().x(), 10);
QCOMPARE(object->rect().y(), 15);
object->setProperty("value", QVariant(92));
QCOMPARE(object->rect().x(), 92);
QCOMPARE(object->rect().y(), 97);
delete object;
}
// function assignment should fail without crashing
{
QString warning1 = testFileUrl("bindingAssignment.2.qml").toString() + QLatin1String(":6:5: Invalid use of Qt.binding() in a binding declaration.");
QString warning2 = testFileUrl("bindingAssignment.2.qml").toString() + QLatin1String(":10: Cannot assign JavaScript function to value-type property");
QTest::ignoreMessage(QtWarningMsg, qPrintable(warning1));
QTest::ignoreMessage(QtWarningMsg, qPrintable(warning2));
QQmlComponent component(&engine, testFileUrl("bindingAssignment.2.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect().x(), 5);
object->setProperty("value", QVariant(92));
QCOMPARE(object->rect().x(), 5);
delete object;
}
}
// Test bindings can read from value types
void tst_qqmlvaluetypes::bindingRead()
{
QQmlComponent component(&engine, testFileUrl("bindingRead.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->property("value").toInt(), 2);
object->setRect(QRect(19, 3, 88, 2));
QCOMPARE(object->property("value").toInt(), 19);
delete object;
}
// Test static values can assign to value types
void tst_qqmlvaluetypes::staticAssignment()
{
QQmlComponent component(&engine, testFileUrl("staticAssignment.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect().x(), 9);
delete object;
}
// Test scripts can read/write value types
void tst_qqmlvaluetypes::scriptAccess()
{
QQmlComponent component(&engine, testFileUrl("scriptAccess.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->property("valuePre").toInt(), 2);
QCOMPARE(object->rect().x(), 19);
QCOMPARE(object->property("valuePost").toInt(), 19);
delete object;
}
// Test that assigning a constant from script removes any binding
void tst_qqmlvaluetypes::autoBindingRemoval()
{
{
QQmlComponent component(&engine, testFileUrl("autoBindingRemoval.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect().x(), 10);
object->setProperty("value", QVariant(13));
QCOMPARE(object->rect().x(), 13);
object->emitRunScript();
QCOMPARE(object->rect().x(), 42);
object->setProperty("value", QVariant(92));
QCOMPARE(object->rect().x(), 42);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("autoBindingRemoval.2.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect().x(), 10);
object->setProperty("value", QVariant(13));
QCOMPARE(object->rect().x(), 13);
object->emitRunScript();
QCOMPARE(object->rect(), QRect(10, 10, 10, 10));
object->setProperty("value", QVariant(92));
QCOMPARE(object->rect(), QRect(10, 10, 10, 10));
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("autoBindingRemoval.3.qml"));
QString warning = component.url().toString() + ":6:5: Unable to assign [undefined] to QRect";
QTest::ignoreMessage(QtWarningMsg, qPrintable(warning));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
object->setProperty("value", QVariant(QRect(9, 22, 33, 44)));
QCOMPARE(object->rect(), QRect(9, 22, 33, 44));
object->emitRunScript();
QCOMPARE(object->rect(), QRect(44, 22, 33, 44));
object->setProperty("value", QVariant(QRect(19, 3, 4, 8)));
QCOMPARE(object->rect(), QRect(44, 22, 33, 44));
delete object;
}
}
// Test that property value sources assign to value types
void tst_qqmlvaluetypes::valueSources()
{
QQmlComponent component(&engine, testFileUrl("valueSources.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect().x(), 3345);
delete object;
}
static void checkNoErrors(QQmlComponent& component)
{
QList<QQmlError> errors = component.errors();
if (errors.isEmpty())
return;
for (int ii = 0; ii < errors.count(); ++ii) {
const QQmlError &error = errors.at(ii);
qWarning("%d:%d:%s",error.line(),error.column(),error.description().toUtf8().constData());
}
}
// Test that property value interceptors can be applied to value types
void tst_qqmlvaluetypes::valueInterceptors()
{
QQmlComponent component(&engine, testFileUrl("valueInterceptors.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
checkNoErrors(component);
QVERIFY(object != nullptr);
QCOMPARE(object->rect().x(), 13);
object->setProperty("value", 99);
QCOMPARE(object->rect().x(), 112);
delete object;
}
// Test that you can't assign a binding to the "root" value type, and a sub-property
void tst_qqmlvaluetypes::bindingConflict()
{
QQmlComponent component(&engine, testFileUrl("bindingConflict.qml"));
QCOMPARE(component.isError(), true);
}
#define CPP_TEST(type, v) \
{ \
type *t = new type; \
QVariant value(v); \
t->setValue(value); \
QCOMPARE(t->value(), value); \
delete t; \
}
// Test that accessing a reference to a valuetype after the owning object is deleted
// doesn't crash
void tst_qqmlvaluetypes::deletedObject()
{
QQmlComponent component(&engine, testFileUrl("deletedObject.qml"));
QTest::ignoreMessage(QtDebugMsg, "Test: 2");
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QObject *dObject = qvariant_cast<QObject *>(object->property("object"));
QVERIFY(dObject != nullptr);
delete dObject;
QTest::ignoreMessage(QtDebugMsg, "Test: undefined");
object->emitRunScript();
delete object;
}
// Test that value types can be assigned to another value type property in a binding
void tst_qqmlvaluetypes::bindingVariantCopy()
{
QQmlComponent component(&engine, testFileUrl("bindingVariantCopy.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect(), QRect(19, 33, 5, 99));
delete object;
}
// Test that value types can be assigned to another value type property in script
void tst_qqmlvaluetypes::scriptVariantCopy()
{
QQmlComponent component(&engine, testFileUrl("scriptVariantCopy.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->rect(), QRect(2, 3, 109, 102));
object->emitRunScript();
QCOMPARE(object->rect(), QRect(19, 33, 5, 99));
delete object;
}
void tst_qqmlvaluetypes::enums()
{
{
QQmlComponent component(&engine, testFileUrl("enums.1.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().capitalization(), QFont::AllUppercase);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("enums.2.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().capitalization(), QFont::AllUppercase);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("enums.3.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().capitalization(), QFont::AllUppercase);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("enums.4.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().capitalization(), QFont::AllUppercase);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("enums.5.qml"));
MyTypeObject *object = qobject_cast<MyTypeObject *>(component.create());
QVERIFY(object != nullptr);
QCOMPARE(object->font().capitalization(), QFont::AllUppercase);
delete object;
}
}
// Tests switching between "conflicting" bindings (eg. a binding on the core
// property, to a binding on the value-type sub-property)
void tst_qqmlvaluetypes::conflictingBindings()
{
{
QQmlComponent component(&engine, testFileUrl("conflicting.1.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 12);
QMetaObject::invokeMethod(object, "toggle");
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 6);
QMetaObject::invokeMethod(object, "toggle");
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 12);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("conflicting.2.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 6);
QMetaObject::invokeMethod(object, "toggle");
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 12);
QMetaObject::invokeMethod(object, "toggle");
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 6);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("conflicting.3.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 12);
QMetaObject::invokeMethod(object, "toggle");
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 24);
QMetaObject::invokeMethod(object, "toggle");
QCOMPARE(qvariant_cast<QFont>(object->property("font")).pixelSize(), 12);
delete object;
}
}
void tst_qqmlvaluetypes::returnValues()
{
QQmlComponent component(&engine, testFileUrl("returnValues.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test1").toBool(), true);
QCOMPARE(object->property("test2").toBool(), true);
QCOMPARE(object->property("size").toSize(), QSize(13, 14));
delete object;
}
void tst_qqmlvaluetypes::varAssignment()
{
QQmlComponent component(&engine, testFileUrl("varAssignment.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("x").toInt(), 1);
QCOMPARE(object->property("y").toInt(), 2);
QCOMPARE(object->property("z").toInt(), 3);
delete object;
}
// Test bindings splice together correctly
void tst_qqmlvaluetypes::bindingsSpliceCorrectly()
{
{
QQmlComponent component(&engine, testFileUrl("bindingsSpliceCorrectly.1.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test").toBool(), true);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("bindingsSpliceCorrectly.2.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test").toBool(), true);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("bindingsSpliceCorrectly.3.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test").toBool(), true);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("bindingsSpliceCorrectly.4.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test").toBool(), true);
delete object;
}
{
QQmlComponent component(&engine, testFileUrl("bindingsSpliceCorrectly.5.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test").toBool(), true);
delete object;
}
}
void tst_qqmlvaluetypes::nonValueTypeComparison()
{
QQmlComponent component(&engine, testFileUrl("nonValueTypeComparison.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test1").toBool(), true);
QCOMPARE(object->property("test2").toBool(), true);
delete object;
}
void tst_qqmlvaluetypes::initializeByWrite()
{
QQmlComponent component(&engine, testFileUrl("initializeByWrite.qml"));
QObject *object = component.create();
QVERIFY(object != nullptr);
QCOMPARE(object->property("test").toBool(), true);
delete object;
}
void tst_qqmlvaluetypes::groupedInterceptors_data()
{
QTest::addColumn<QString>("qmlfile");
QTest::addColumn<QColor>("expectedInitialColor");
QTest::addColumn<QColor>("setColor");
QTest::addColumn<QColor>("expectedFinalColor");
QColor c0, c1, c2;
c0.setRgbF(0.1f, 0.2f, 0.3f, 0.4f);
c1.setRgbF(0.2f, 0.4f, 0.6f, 0.8f);
c2.setRgbF(0.8f, 0.6f, 0.4f, 0.2f);
QTest::newRow("value-interceptor") << QString::fromLatin1("grouped_interceptors_value.qml") << c0 << c1 << c2;
QTest::newRow("component-interceptor") << QString::fromLatin1("grouped_interceptors_component.qml") << QColor(128, 0, 255) << QColor(50, 100, 200) << QColor(0, 100, 200);
QTest::newRow("ignore-interceptor") << QString::fromLatin1("grouped_interceptors_ignore.qml") << QColor(128, 0, 255) << QColor(50, 100, 200) << QColor(128, 100, 200);
}
static bool fuzzyCompare(qreal a, qreal b)
{
const qreal EPSILON = 0.0001;
return (a + EPSILON > b) && (a - EPSILON < b);
}
void tst_qqmlvaluetypes::groupedInterceptors()
{
QFETCH(QString, qmlfile);
QFETCH(QColor, expectedInitialColor);
QFETCH(QColor, setColor);
QFETCH(QColor, expectedFinalColor);
QQmlComponent component(&engine, testFileUrl(qmlfile));
QObject *object = component.create();
QVERIFY2(object != nullptr, qPrintable(component.errorString()));
QColor initialColor = object->property("color").value<QColor>();
QVERIFY(fuzzyCompare(initialColor.redF(), expectedInitialColor.redF()));
QVERIFY(fuzzyCompare(initialColor.greenF(), expectedInitialColor.greenF()));
QVERIFY(fuzzyCompare(initialColor.blueF(), expectedInitialColor.blueF()));
QVERIFY(fuzzyCompare(initialColor.alphaF(), expectedInitialColor.alphaF()));
object->setProperty("color", setColor);
QColor finalColor = object->property("color").value<QColor>();
QVERIFY(fuzzyCompare(finalColor.redF(), expectedFinalColor.redF()));
QVERIFY(fuzzyCompare(finalColor.greenF(), expectedFinalColor.greenF()));
QVERIFY(fuzzyCompare(finalColor.blueF(), expectedFinalColor.blueF()));
QVERIFY(fuzzyCompare(finalColor.alphaF(), expectedFinalColor.alphaF()));
delete object;
}
struct MyDesk
{
Q_PROPERTY(int monitorCount MEMBER monitorCount)
Q_GADGET
public:
MyDesk() : monitorCount(1) {}
int monitorCount;
};
bool operator==(const MyDesk &lhs, const MyDesk &rhs)
{ return lhs.monitorCount == rhs.monitorCount; }
bool operator!=(const MyDesk &lhs, const MyDesk &rhs)
{ return lhs.monitorCount != rhs.monitorCount; }
Q_DECLARE_METATYPE(MyDesk)
struct MyOffice
{
Q_PROPERTY(int chairs MEMBER m_chairs)
Q_PROPERTY(MyDesk desk READ desk WRITE setDesk)
Q_PROPERTY(QVariant myThing READ myThing WRITE setMyThing)
Q_GADGET
public:
MyOffice() : m_chairs(0) {}
MyDesk desk() const { return m_desk; }
void setDesk(const MyDesk &d) { m_desk = d; }
QVariant myThing() const { return m_myThing; }
void setMyThing(const QVariant &thingy) { m_myThing = thingy; }
int m_chairs;
MyDesk m_desk;
QVariant m_myThing;
};
Q_DECLARE_METATYPE(MyOffice)
void tst_qqmlvaluetypes::customValueType()
{
QJSEngine engine;
MyOffice cppOffice;
cppOffice.m_chairs = 2;
QVariantMap m;
m.insert(QStringLiteral("hasChair"), false);
m.insert(QStringLiteral("textOnWhiteboard"), QStringLiteral("Blah blah"));
cppOffice.m_myThing = m;
QJSValue office = engine.toScriptValue(cppOffice);
QCOMPARE(office.property("chairs").toInt(), 2);
office.setProperty("chairs", 1);
QCOMPARE(office.property("chairs").toInt(), 1);
QCOMPARE(cppOffice.m_chairs, 2);
QJSValue jsDesk = office.property("desk");
QCOMPARE(jsDesk.property("monitorCount").toInt(), 1);
jsDesk.setProperty("monitorCount", 2);
QCOMPARE(jsDesk.property("monitorCount").toInt(), 2);
QCOMPARE(cppOffice.desk().monitorCount, 1);
office.setProperty("desk", jsDesk);
cppOffice = engine.fromScriptValue<MyOffice>(office);
QCOMPARE(cppOffice.m_chairs, 1);
QCOMPARE(cppOffice.desk().monitorCount, 2);
QJSValue thingy = office.property("myThing");
QVERIFY(thingy.hasProperty("hasChair"));
QVERIFY(thingy.property("hasChair").isBool());
QCOMPARE(thingy.property("hasChair").toBool(), false);
QVERIFY(thingy.property("textOnWhiteboard").isString());
QVERIFY(thingy.hasProperty("textOnWhiteboard"));
QCOMPARE(thingy.property("textOnWhiteboard").toString(), QStringLiteral("Blah blah"));
}
struct BaseGadget
{
Q_GADGET
Q_PROPERTY(int baseProperty READ baseProperty WRITE setBaseProperty)
public:
BaseGadget() : m_baseProperty(0) {}
BaseGadget(int initValue) : m_baseProperty(initValue) {}
int baseProperty() const { return m_baseProperty; }
void setBaseProperty(int value) { m_baseProperty = value; }
int m_baseProperty;
Q_INVOKABLE void functionInBaseGadget(int value) { m_baseProperty = value; }
};
Q_DECLARE_METATYPE(BaseGadget)
struct DerivedGadget : public BaseGadget
{
Q_GADGET
Q_PROPERTY(int derivedProperty READ derivedProperty WRITE setDerivedProperty)
public:
DerivedGadget() : m_derivedProperty(0) {}
int derivedProperty() const { return m_derivedProperty; }
void setDerivedProperty(int value) { m_derivedProperty = value; }
int m_derivedProperty;
Q_INVOKABLE void functionInDerivedGadget(int value) { m_derivedProperty = value; }
};
// QTBUG-66744: we want a Q_GADGET giving us generic type safety in C++ and property access in Qml
template <typename T>
struct DerivedTypedGadget : public BaseGadget
{
// cannot use Q_GADGET here
public:
DerivedTypedGadget() {}
};
class DerivedTypedGadgetDummyType {};
Q_DECLARE_METATYPE(DerivedTypedGadget<DerivedTypedGadgetDummyType>)
class TypeWithCustomValueType : public QObject
{
Q_OBJECT
Q_PROPERTY(MyDesk desk MEMBER m_desk)
Q_PROPERTY(DerivedGadget derivedGadget READ derivedGadget WRITE setDerivedGadget)
public:
MyDesk m_desk;
DerivedGadget derivedGadget() const { return m_derivedGadget; }
void setDerivedGadget(const DerivedGadget &value) { m_derivedGadget = value; }
DerivedGadget m_derivedGadget;
};
void tst_qqmlvaluetypes::customValueTypeInQml()
{
qmlRegisterType<TypeWithCustomValueType>("Test", 1, 0, "TypeWithCustomValueType");
QQmlComponent component(&engine, testFileUrl("customvaluetype.qml"));
QScopedPointer<QObject> object(component.create());
QVERIFY(!object.isNull());
TypeWithCustomValueType *t = qobject_cast<TypeWithCustomValueType*>(object.data());
Q_ASSERT(t);
QCOMPARE(t->m_desk.monitorCount, 3);
QCOMPARE(t->m_derivedGadget.baseProperty(), 42);
}
Q_DECLARE_METATYPE(DerivedGadget)
void tst_qqmlvaluetypes::gadgetInheritance()
{
QJSEngine engine;
QJSValue value = engine.toScriptValue(DerivedGadget());
QCOMPARE(value.property("baseProperty").toInt(), 0);
value.setProperty("baseProperty", 10);
QCOMPARE(value.property("baseProperty").toInt(), 10);
QJSValue method = value.property("functionInBaseGadget");
method.call(QJSValueList() << QJSValue(42));
QCOMPARE(value.property("baseProperty").toInt(), 42);
}
void tst_qqmlvaluetypes::gadgetTemplateInheritance()
{
QJSEngine engine;
QJSValue value = engine.toScriptValue(DerivedTypedGadget<DerivedTypedGadgetDummyType>());
QCOMPARE(value.property("baseProperty").toInt(), 0);
value.setProperty("baseProperty", 10);
QCOMPARE(value.property("baseProperty").toInt(), 10);
QJSValue method = value.property("functionInBaseGadget");
method.call(QJSValueList() << QJSValue(42));
QCOMPARE(value.property("baseProperty").toInt(), 42);
}
void tst_qqmlvaluetypes::sequences()
{
QJSEngine engine;
{
QList<BaseGadget> gadgetList{1, 4, 7, 8, 15};
QJSValue value = engine.toScriptValue(gadgetList);
QCOMPARE(value.property("length").toInt(), gadgetList.length());
for (int i = 0; i < gadgetList.length(); ++i)
QCOMPARE(value.property(i).property("baseProperty").toInt(), gadgetList.at(i).baseProperty());
}
{
std::vector<BaseGadget> container{1, 4, 7, 8, 15};
QJSValue value = engine.toScriptValue(container);
QCOMPARE(value.property("length").toInt(), int(container.size()));
for (size_t i = 0; i < container.size(); ++i)
QCOMPARE(value.property(i).property("baseProperty").toInt(), container.at(i).baseProperty());
}
{
QVector<QChar> qcharVector{1, 4, 42, 8, 15};
QJSValue value = engine.toScriptValue(qcharVector);
QCOMPARE(value.property("length").toInt(), qcharVector.length());
for (int i = 0; i < qcharVector.length(); ++i)
QCOMPARE(value.property(i).toString(), qcharVector.at(i));
}
{
MyTypeObject a, b, c;
QSet<QObject*> objSet{&a, &b, &c};
QJSValue value = engine.toScriptValue(objSet);
QCOMPARE(value.property("length").toInt(), objSet.size());
for (int i = 0; i < objSet.size(); ++i)
QCOMPARE(value.property(i).property("point").property("x").toInt(), a.point().x());
}
{
MyTypeObject a, b, c;
QSet<MyTypeObject*> container{&a, &b, &c};
QJSValue value = engine.toScriptValue(container);
QCOMPARE(value.property("length").toInt(), container.size());
for (int i = 0; i < container.size(); ++i)
QCOMPARE(value.property(i).property("point").property("x").toInt(), a.point().x());
}
}
struct StringLessGadget {
Q_GADGET
};
Q_DECLARE_METATYPE(StringLessGadget)
static QString StringLessGadget_to_QString(const StringLessGadget &)
{
return QLatin1String("Surprise!");
}
void tst_qqmlvaluetypes::toStringConversion()
{
QJSEngine engine;
StringLessGadget g;
QJSValue value = engine.toScriptValue(g);
QJSValue method = value.property("toString");
QJSValue stringConversion = method.callWithInstance(value);
QCOMPARE(stringConversion.toString(), QString("StringLessGadget()"));
QMetaType::registerConverter<StringLessGadget, QString>(StringLessGadget_to_QString);
stringConversion = method.callWithInstance(value);
QCOMPARE(stringConversion.toString(), StringLessGadget_to_QString(g));
}
void tst_qqmlvaluetypes::enumerableProperties()
{
QJSEngine engine;
DerivedGadget g;
QJSValue value = engine.toScriptValue(g);
QSet<QString> names;
QJSValueIterator it(value);
while (it.hasNext()) {
it.next();
const QString name = it.name();
QVERIFY(!names.contains(name));
names.insert(name);
}
QCOMPARE(names.count(), 2);
QVERIFY(names.contains(QStringLiteral("baseProperty")));
QVERIFY(names.contains(QStringLiteral("derivedProperty")));
}
struct GadgetWithEnum
{
Q_GADGET
public:
enum MyEnum { FirstValue, SecondValue };
Q_ENUM(MyEnum)
Q_PROPERTY(MyEnum enumProperty READ enumProperty)
MyEnum enumProperty() const { return SecondValue; }
};
void tst_qqmlvaluetypes::enumProperties()
{
QJSEngine engine;
// When creating the property cache for the gadget when MyEnum is _not_ a registered
// meta-type, then QMetaProperty::type() will return QMetaType::Int and consequently
// property-read meta-calls will return an int (as expected in this test). However if we
// explicitly register the gadget, then QMetaProperty::type() will return the user-type
// and QQmlValueTypeWrapper should still handle that and return an integer/number for the
// enum property when it is read.
qRegisterMetaType<GadgetWithEnum::MyEnum>();
GadgetWithEnum g;
QJSValue value = engine.toScriptValue(g);
QJSValue enumValue = value.property("enumProperty");
QVERIFY(enumValue.isNumber());
QCOMPARE(enumValue.toInt(), int(g.enumProperty()));
}
void tst_qqmlvaluetypes::scarceTypes()
{
// These should not be treated as value types because we want the scarce resource
// mechanism to clear them when going out of scope. The scarce resource mechanism
// only works on QV4::VariantObject as that has an additional level of redirection.
QVERIFY(!QQmlValueTypeFactory::isValueType(qMetaTypeId<QImage>()));
QVERIFY(!QQmlValueTypeFactory::isValueType(qMetaTypeId<QPixmap>()));
QV4::ExecutionEngine engine;
QV4::Scope scope(&engine);
QImage img(20, 20, QImage::Format_ARGB32);
QV4::ScopedObject imgValue(scope, engine.fromVariant(QVariant::fromValue(img)));
QCOMPARE(QByteArray(imgValue->vtable()->className), QByteArray("VariantObject"));
QPixmap pixmap;
QV4::ScopedObject pixmapValue(scope, engine.fromVariant(QVariant::fromValue(img)));
QCOMPARE(QByteArray(pixmapValue->vtable()->className), QByteArray("VariantObject"));
}
#define CHECK_TYPE_IS_NOT_VALUETYPE(Type, typeId, cppType) \
QVERIFY(!QQmlValueTypeFactory::isValueType(QMetaType::Type));
void tst_qqmlvaluetypes::nonValueTypes()
{
CHECK_TYPE_IS_NOT_VALUETYPE(UnknownType, 0, void)
QT_FOR_EACH_STATIC_PRIMITIVE_TYPE(CHECK_TYPE_IS_NOT_VALUETYPE);
}
#undef CHECK_TYPE_IS_NOT_VALUETYPE
QTEST_MAIN(tst_qqmlvaluetypes)
#include "tst_qqmlvaluetypes.moc"