7929 lines
307 KiB
C++
7929 lines
307 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
#include <qtest.h>
|
|
#include <QtQml/qqmlengine.h>
|
|
#include <QtQml/qqmlcomponent.h>
|
|
#include <QtQml/qqmlincubator.h>
|
|
#include <QtCore/qiterable.h>
|
|
#include <QtCore/qcoreapplication.h>
|
|
#include <QtCore/qfile.h>
|
|
#include <QtCore/qdebug.h>
|
|
#include <QtCore/qfileinfo.h>
|
|
#include <QtCore/qdir.h>
|
|
#include <QtCore/qscopeguard.h>
|
|
#include <QtCore/qrandom.h>
|
|
#include <QtGui/qevent.h>
|
|
#include <QSignalSpy>
|
|
#include <QFont>
|
|
#include <QQmlFileSelector>
|
|
#include <QFileSelector>
|
|
#include <QEasingCurve>
|
|
#include <QScopeGuard>
|
|
|
|
#include <private/qqmlproperty_p.h>
|
|
#include <private/qqmlmetatype_p.h>
|
|
#include <private/qqmlglobal_p.h>
|
|
#include <private/qqmlscriptstring_p.h>
|
|
#include <private/qqmlvmemetaobject_p.h>
|
|
#include <private/qqmlcomponent_p.h>
|
|
#include <private/qqmltype_p_p.h>
|
|
#include <private/qv4debugging_p.h>
|
|
#include <private/qqmlcomponentattached_p.h>
|
|
#include <QtQml/private/qqmlexpression_p.h>
|
|
|
|
#include "testtypes.h"
|
|
#include <QtQuickTestUtils/private/qmlutils_p.h>
|
|
#include <QtQuickTestUtils/private/testhttpserver_p.h>
|
|
|
|
#include <deque>
|
|
|
|
#if defined(Q_OS_MAC)
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
using namespace Qt::StringLiterals;
|
|
|
|
DEFINE_BOOL_CONFIG_OPTION(qmlCheckTypes, QML_CHECK_TYPES)
|
|
|
|
static inline bool isCaseSensitiveFileSystem(const QString &path) {
|
|
Q_UNUSED(path);
|
|
#if defined(Q_OS_MAC)
|
|
return pathconf(path.toLatin1().constData(), _PC_CASE_SENSITIVE);
|
|
#elif defined(Q_OS_WIN)
|
|
return false;
|
|
#else
|
|
return true;
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
This test case covers QML language issues. This covers everything that does not
|
|
involve evaluating ECMAScript expressions and bindings.
|
|
|
|
Evaluation of expressions and bindings is covered in qmlecmascript
|
|
*/
|
|
class tst_qqmllanguage : public QQmlDataTest
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
tst_qqmllanguage();
|
|
|
|
private slots:
|
|
void initTestCase() override;
|
|
void cleanupTestCase();
|
|
|
|
void errors_data();
|
|
void errors();
|
|
|
|
void insertedSemicolon_data();
|
|
void insertedSemicolon();
|
|
|
|
void simpleObject();
|
|
void simpleContainer();
|
|
void interfaceProperty();
|
|
void interfaceQList();
|
|
void assignObjectToSignal();
|
|
void assignObjectToVariant();
|
|
void assignLiteralSignalProperty();
|
|
void assignQmlComponent();
|
|
void assignValueTypes();
|
|
void assignTypeExtremes();
|
|
void assignCompositeToType();
|
|
void assignLiteralToVar();
|
|
void assignLiteralToJSValue();
|
|
void assignNullStrings();
|
|
void bindJSValueToVar();
|
|
void bindJSValueToVariant();
|
|
void bindJSValueToType();
|
|
void bindTypeToJSValue();
|
|
void customParserTypes();
|
|
void customParserTypeInInlineComponent();
|
|
void rootAsQmlComponent();
|
|
void rootItemIsComponent();
|
|
void inlineQmlComponents();
|
|
void idProperty();
|
|
void autoNotifyConnection();
|
|
void assignSignal();
|
|
void assignSignalFunctionExpression();
|
|
void overrideSignal_data();
|
|
void overrideSignal();
|
|
void dynamicProperties();
|
|
void dynamicPropertiesNested();
|
|
void listProperties();
|
|
void listPropertiesInheritanceNoCrash();
|
|
void badListItemType();
|
|
void dynamicObjectProperties();
|
|
void dynamicSignalsAndSlots();
|
|
void simpleBindings();
|
|
void noDoubleEvaluationForFlushedBindings_data();
|
|
void noDoubleEvaluationForFlushedBindings();
|
|
void autoComponentCreation();
|
|
void autoComponentCreationInGroupProperty();
|
|
void propertyValueSource();
|
|
void requiredProperty();
|
|
void requiredPropertyFromCpp_data();
|
|
void requiredPropertyFromCpp();
|
|
void attachedProperties();
|
|
void dynamicObjects();
|
|
void valueTypes();
|
|
void cppnamespace();
|
|
void aliasProperties();
|
|
void aliasPropertiesAndSignals();
|
|
void aliasPropertyChangeSignals();
|
|
void qtbug_89822();
|
|
void componentCompositeType();
|
|
void i18n();
|
|
void i18n_data();
|
|
void onCompleted();
|
|
void onDestruction();
|
|
void scriptString();
|
|
void scriptStringJs();
|
|
void scriptStringWithoutSourceCode();
|
|
void scriptStringComparison();
|
|
void defaultPropertyListOrder();
|
|
void defaultPropertyWithInitializer_data();
|
|
void defaultPropertyWithInitializer();
|
|
void declaredPropertyValues();
|
|
void dontDoubleCallClassBegin();
|
|
void reservedWords_data();
|
|
void reservedWords();
|
|
void inlineAssignmentsOverrideBindings();
|
|
void nestedComponentRoots();
|
|
void registrationOrder();
|
|
void readonly();
|
|
void readonlyObjectProperties();
|
|
void receivers();
|
|
void registeredCompositeType();
|
|
void registeredCompositeTypeWithEnum();
|
|
void registeredCompositeTypeWithAttachedProperty();
|
|
void implicitImportsLast();
|
|
|
|
void basicRemote_data();
|
|
void basicRemote();
|
|
void importsBuiltin_data();
|
|
void importsBuiltin();
|
|
void importsLocal_data();
|
|
void importsLocal();
|
|
void importsRemote_data();
|
|
void importsRemote();
|
|
void importsInstalled_data();
|
|
void importsInstalled();
|
|
void importsInstalledRemote_data();
|
|
void importsInstalledRemote();
|
|
void importsPath_data();
|
|
void importsPath();
|
|
void importsOrder_data();
|
|
void importsOrder();
|
|
void importIncorrectCase();
|
|
void importJs_data();
|
|
void importJs();
|
|
void importJsModule_data();
|
|
void importJsModule();
|
|
void explicitSelfImport();
|
|
void importInternalType();
|
|
|
|
void qmlAttachedPropertiesObjectMethod();
|
|
void customOnProperty();
|
|
void variantNotify();
|
|
|
|
void revisions();
|
|
void revisionOverloads();
|
|
|
|
void subclassedUncreateableRevision_data();
|
|
void subclassedUncreateableRevision();
|
|
|
|
void subclassedExtendedUncreateableRevision_data();
|
|
void subclassedExtendedUncreateableRevision();
|
|
|
|
void uncreatableTypesAsProperties();
|
|
|
|
void propertyInit();
|
|
void remoteLoadCrash();
|
|
void signalWithDefaultArg();
|
|
void signalParameterTypes();
|
|
void functionParameterTypes();
|
|
|
|
// regression tests for crashes
|
|
void crash1();
|
|
void crash2();
|
|
|
|
void globalEnums();
|
|
void lowercaseEnumRuntime_data();
|
|
void lowercaseEnumRuntime();
|
|
void lowercaseEnumCompileTime_data();
|
|
void lowercaseEnumCompileTime();
|
|
void scopedEnum();
|
|
void scopedEnumsWithNameClash();
|
|
void scopedEnumsWithResolvedNameClash();
|
|
void enumNoScopeLeak();
|
|
void qmlEnums();
|
|
void literals_data();
|
|
void literals();
|
|
|
|
void objectDeletionNotify_data();
|
|
void objectDeletionNotify();
|
|
|
|
void scopedProperties();
|
|
|
|
void deepProperty();
|
|
|
|
void groupAssignmentFailure();
|
|
|
|
void compositeSingletonProperties();
|
|
void compositeSingletonSameEngine();
|
|
void compositeSingletonDifferentEngine();
|
|
void compositeSingletonNonTypeError();
|
|
void compositeSingletonQualifiedNamespace();
|
|
void compositeSingletonModule();
|
|
void compositeSingletonModuleVersioned();
|
|
void compositeSingletonModuleQualified();
|
|
void compositeSingletonInstantiateError();
|
|
void compositeSingletonDynamicPropertyError();
|
|
void compositeSingletonDynamicSignalAndJavaScriptPragma();
|
|
void compositeSingletonQmlRegisterTypeError();
|
|
void compositeSingletonQmldirNoPragmaError();
|
|
void compositeSingletonQmlDirError();
|
|
void compositeSingletonRemote();
|
|
void compositeSingletonSelectors();
|
|
void compositeSingletonRegistered();
|
|
void compositeSingletonCircular();
|
|
|
|
void singletonsHaveContextAndEngine();
|
|
|
|
void customParserBindingScopes();
|
|
void customParserEvaluateEnum();
|
|
void customParserProperties();
|
|
void customParserWithExtendedObject();
|
|
void nestedCustomParsers();
|
|
|
|
void preservePropertyCacheOnGroupObjects();
|
|
void propertyCacheInSync();
|
|
|
|
void rootObjectInCreationNotForSubObjects();
|
|
void lazyDeferredSubObject();
|
|
void deferredProperties();
|
|
void executeDeferredPropertiesOnce();
|
|
void deferredProperties_extra();
|
|
|
|
void noChildEvents();
|
|
|
|
void earlyIdObjectAccess();
|
|
|
|
void deleteSingletons();
|
|
|
|
void arrayBuffer_data();
|
|
void arrayBuffer();
|
|
|
|
void defaultListProperty();
|
|
void namespacedPropertyTypes();
|
|
|
|
void qmlTypeCanBeResolvedByName_data();
|
|
void qmlTypeCanBeResolvedByName();
|
|
|
|
void instanceof_data();
|
|
void instanceof();
|
|
|
|
void concurrentLoadQmlDir();
|
|
|
|
void accessDeletedObject();
|
|
|
|
void lowercaseTypeNames();
|
|
|
|
void thisInQmlScope();
|
|
|
|
void valueTypeGroupPropertiesInBehavior();
|
|
|
|
void retrieveQmlTypeId();
|
|
|
|
void polymorphicFunctionLookup();
|
|
void anchorsToParentInPropertyChanges();
|
|
|
|
void typeWrapperToVariant();
|
|
|
|
void extendedForeignTypes();
|
|
void foreignTypeSingletons();
|
|
|
|
void inlineComponent();
|
|
void inlineComponent_data();
|
|
void inlineComponentReferenceCycle_data();
|
|
void inlineComponentReferenceCycle();
|
|
void nestedInlineComponentNotAllowed();
|
|
void inlineComponentStaticTypeResolution();
|
|
void inlineComponentInSingleton();
|
|
void nonExistingInlineComponent_data();
|
|
void nonExistingInlineComponent();
|
|
void inlineComponentFoundBeforeOtherImports();
|
|
void inlineComponentDuplicateNameError();
|
|
void inlineComponentWithAliasInstantiatedWithNewProperties();
|
|
|
|
void selfReference();
|
|
void selfReferencingSingleton();
|
|
|
|
void listContainingDeletedObject();
|
|
void overrideSingleton();
|
|
void revisionedPropertyOfAttachedObjectProperty();
|
|
|
|
void arrayToContainer();
|
|
void qualifiedScopeInCustomParser();
|
|
void accessNullPointerPropertyCache();
|
|
void bareInlineComponent();
|
|
|
|
void checkUncreatableNoReason();
|
|
|
|
void checkURLtoURLObject();
|
|
void registerValueTypes();
|
|
void extendedNamespace();
|
|
void extendedNamespaceByObject();
|
|
void extendedByAttachedType();
|
|
void factorySingleton();
|
|
void extendedSingleton();
|
|
void qtbug_85932();
|
|
void qtbug_86482();
|
|
|
|
void multiExtension();
|
|
void multiExtensionExtra();
|
|
void multiExtensionIndirect();
|
|
void multiExtensionQmlTypes();
|
|
void extensionSpecial();
|
|
void extensionRevision();
|
|
void extendedGroupProperty();
|
|
void invalidInlineComponent();
|
|
void warnOnInjectedParameters();
|
|
#if QT_CONFIG(wheelevent)
|
|
void warnOnInjectedParametersFromCppSignal();
|
|
#endif
|
|
|
|
void qtbug_85615();
|
|
|
|
void hangOnWarning();
|
|
|
|
void listEnumConversion();
|
|
void deepInlineComponentScriptBinding();
|
|
|
|
void propertyObserverOnReadonly();
|
|
void valueTypeWithEnum();
|
|
void enumsFromRelatedTypes();
|
|
|
|
void propertyAndAliasMustHaveDistinctNames_data();
|
|
void propertyAndAliasMustHaveDistinctNames();
|
|
|
|
void variantListConversion();
|
|
void thisInArrowFunction();
|
|
|
|
void jittedAsCast();
|
|
void propertyNecromancy();
|
|
void generalizedGroupedProperty();
|
|
|
|
void groupedAttachedProperty_data();
|
|
void groupedAttachedProperty();
|
|
|
|
void ambiguousContainingType();
|
|
void objectAsBroken();
|
|
void customValueTypes();
|
|
void valueTypeList();
|
|
void componentMix();
|
|
void uncreatableAttached();
|
|
void resetGadgetProperty();
|
|
void leakingAttributesQmlAttached();
|
|
void leakingAttributesQmlSingleton();
|
|
void leakingAttributesQmlForeign();
|
|
void attachedOwnProperties();
|
|
void bindableOnly();
|
|
void v4SequenceMethods();
|
|
void v4SequenceMethodsWithParams_data();
|
|
void v4SequenceMethodsWithParams();
|
|
void jsFunctionOverridesImport();
|
|
void bindingAliasToComponentUrl();
|
|
void badGroupedProperty();
|
|
void functionInGroupedProperty();
|
|
void signalInlineComponentArg();
|
|
void functionSignatureEnforcement();
|
|
void importPrecedence();
|
|
void nullIsNull();
|
|
void multiRequired();
|
|
void isNullOrUndefined();
|
|
|
|
void objectAndGadgetMethodCallsRejectThisObject();
|
|
void objectAndGadgetMethodCallsAcceptThisObject();
|
|
|
|
void longConversion();
|
|
|
|
void objectMethodClone();
|
|
void unregisteredValueTypeConversion();
|
|
|
|
private:
|
|
QQmlEngine engine;
|
|
QStringList defaultImportPathList;
|
|
|
|
void testType(const QString& qml, const QString& type, const QString& error, bool partialMatch = false);
|
|
|
|
// When calling into JavaScript, the specific type of the return value can differ if that return
|
|
// value is a number. This is not only the case for non-integral numbers, or numbers that do not
|
|
// fit into the (signed) integer range, but it also depends on which optimizations are run. So,
|
|
// to check if the return value is of a number type, use this method instead of checking against
|
|
// a specific userType.
|
|
static bool isJSNumberType(int userType)
|
|
{
|
|
return userType == QMetaType::Int || userType == QMetaType::UInt
|
|
|| userType == QMetaType::Double;
|
|
}
|
|
|
|
void getSingletonInstance(QQmlEngine& engine, const char* fileName, const char* propertyName, QObject** result /* out */);
|
|
void getSingletonInstance(QObject* o, const char* propertyName, QObject** result /* out */);
|
|
};
|
|
|
|
#define DETERMINE_ERRORS(errorfile,expected,actual)\
|
|
QList<QByteArray> expected; \
|
|
QList<QByteArray> actual; \
|
|
do { \
|
|
QFile file(testFile(errorfile)); \
|
|
QVERIFY(file.open(QIODevice::ReadOnly | QIODevice::Text)); \
|
|
QByteArray data = file.readAll(); \
|
|
file.close(); \
|
|
expected = data.split('\n'); \
|
|
expected.removeAll(QByteArray("")); \
|
|
QList<QQmlError> errors = component.errors(); \
|
|
for (int ii = 0; ii < errors.count(); ++ii) { \
|
|
const QQmlError &error = errors.at(ii); \
|
|
QByteArray errorStr = QByteArray::number(error.line()) + ':' + \
|
|
QByteArray::number(error.column()) + ':' + \
|
|
error.description().toUtf8(); \
|
|
actual << errorStr; \
|
|
} \
|
|
} while (false);
|
|
|
|
#define VERIFY_ERRORS(errorfile) \
|
|
if (!errorfile) { \
|
|
if (qgetenv("DEBUG") != "" && !component.errors().isEmpty()) \
|
|
qWarning() << "Unexpected Errors:" << component.errors(); \
|
|
QVERIFY2(!component.isError(), qPrintable(component.errorString())); \
|
|
QVERIFY(component.errors().isEmpty()); \
|
|
} else { \
|
|
DETERMINE_ERRORS(errorfile,expected,actual);\
|
|
if (qgetenv("DEBUG") != "" && expected != actual) \
|
|
qWarning() << "Expected:" << expected << "Actual:" << actual; \
|
|
if (qgetenv("QDECLARATIVELANGUAGE_UPDATEERRORS") != "" && expected != actual) {\
|
|
QFile file(testFile(errorfile)); \
|
|
QVERIFY(file.open(QIODevice::WriteOnly)); \
|
|
for (int ii = 0; ii < actual.count(); ++ii) { \
|
|
file.write(actual.at(ii)); file.write("\n"); \
|
|
} \
|
|
file.close(); \
|
|
} else { \
|
|
QCOMPARE(actual, expected); \
|
|
} \
|
|
}
|
|
|
|
void tst_qqmllanguage::cleanupTestCase()
|
|
{
|
|
if (dataDirectoryUrl().scheme() != QLatin1String("qrc"))
|
|
QVERIFY(QFile::remove(testFile(QString::fromUtf8("I18nType\303\201\303\242\303\243\303\244\303\245.qml"))));
|
|
}
|
|
|
|
void tst_qqmllanguage::insertedSemicolon_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::addColumn<QString>("errorFile");
|
|
QTest::addColumn<bool>("create");
|
|
|
|
QTest::newRow("insertedSemicolon.1") << "insertedSemicolon.1.qml" << "insertedSemicolon.1.errors.txt" << false;
|
|
}
|
|
|
|
void tst_qqmllanguage::insertedSemicolon()
|
|
{
|
|
QFETCH(QString, file);
|
|
QFETCH(QString, errorFile);
|
|
QFETCH(bool, create);
|
|
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
|
|
QScopedPointer<QObject> object;
|
|
|
|
if(create) {
|
|
object.reset(component.create());
|
|
QVERIFY(object.isNull());
|
|
}
|
|
|
|
VERIFY_ERRORS(errorFile.toLatin1().constData());
|
|
}
|
|
|
|
void tst_qqmllanguage::errors_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::addColumn<QString>("errorFile");
|
|
QTest::addColumn<bool>("create");
|
|
|
|
QTest::newRow("nonexistantProperty.1") << "nonexistantProperty.1.qml" << "nonexistantProperty.1.errors.txt" << false;
|
|
QTest::newRow("nonexistantProperty.2") << "nonexistantProperty.2.qml" << "nonexistantProperty.2.errors.txt" << false;
|
|
QTest::newRow("nonexistantProperty.3") << "nonexistantProperty.3.qml" << "nonexistantProperty.3.errors.txt" << false;
|
|
QTest::newRow("nonexistantProperty.4") << "nonexistantProperty.4.qml" << "nonexistantProperty.4.errors.txt" << false;
|
|
QTest::newRow("nonexistantProperty.5") << "nonexistantProperty.5.qml" << "nonexistantProperty.5.errors.txt" << false;
|
|
QTest::newRow("nonexistantProperty.6") << "nonexistantProperty.6.qml" << "nonexistantProperty.6.errors.txt" << false;
|
|
QTest::newRow("nonexistantProperty.7") << "nonexistantProperty.7.qml" << "nonexistantProperty.7.errors.txt" << false;
|
|
QTest::newRow("nonexistantProperty.8") << "nonexistantProperty.8.qml" << "nonexistantProperty.8.errors.txt" << false;
|
|
|
|
QTest::newRow("wrongType (string for int)") << "wrongType.1.qml" << "wrongType.1.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for bool)") << "wrongType.2.qml" << "wrongType.2.errors.txt" << false;
|
|
QTest::newRow("wrongType (bad rect)") << "wrongType.3.qml" << "wrongType.3.errors.txt" << false;
|
|
|
|
QTest::newRow("wrongType (invalid enum)") << "wrongType.4.qml" << "wrongType.4.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for uint)") << "wrongType.5.qml" << "wrongType.5.errors.txt" << false;
|
|
QTest::newRow("wrongType (string for real)") << "wrongType.6.qml" << "wrongType.6.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for color)") << "wrongType.7.qml" << "wrongType.7.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for date)") << "wrongType.8.qml" << "wrongType.8.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for time)") << "wrongType.9.qml" << "wrongType.9.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for datetime)") << "wrongType.10.qml" << "wrongType.10.errors.txt" << false;
|
|
QTest::newRow("wrongType (string for point)") << "wrongType.11.qml" << "wrongType.11.errors.txt" << false;
|
|
QTest::newRow("wrongType (color for size)") << "wrongType.12.qml" << "wrongType.12.errors.txt" << false;
|
|
QTest::newRow("wrongType (number string for int)") << "wrongType.13.qml" << "wrongType.13.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for string)") << "wrongType.14.qml" << "wrongType.14.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for url)") << "wrongType.15.qml" << "wrongType.15.errors.txt" << false;
|
|
QTest::newRow("wrongType (invalid object)") << "wrongType.16.qml" << "wrongType.16.errors.txt" << false;
|
|
QTest::newRow("wrongType (int for enum)") << "wrongType.17.qml" << "wrongType.17.errors.txt" << false;
|
|
|
|
QTest::newRow("readOnly.1") << "readOnly.1.qml" << "readOnly.1.errors.txt" << false;
|
|
QTest::newRow("readOnly.2") << "readOnly.2.qml" << "readOnly.2.errors.txt" << false;
|
|
QTest::newRow("readOnly.3") << "readOnly.3.qml" << "readOnly.3.errors.txt" << false;
|
|
QTest::newRow("readOnly.4") << "readOnly.4.qml" << "readOnly.4.errors.txt" << false;
|
|
QTest::newRow("readOnly.5") << "readOnly.5.qml" << "readOnly.5.errors.txt" << false;
|
|
|
|
QTest::newRow("listAssignment.1") << "listAssignment.1.qml" << "listAssignment.1.errors.txt" << false;
|
|
QTest::newRow("listAssignment.2") << "listAssignment.2.qml" << "listAssignment.2.errors.txt" << false;
|
|
QTest::newRow("listAssignment.3") << "listAssignment.3.qml" << "listAssignment.3.errors.txt" << false;
|
|
|
|
QTest::newRow("invalidID.1") << "invalidID.qml" << "invalidID.errors.txt" << false;
|
|
QTest::newRow("invalidID.2") << "invalidID.2.qml" << "invalidID.2.errors.txt" << false;
|
|
QTest::newRow("invalidID.3") << "invalidID.3.qml" << "invalidID.3.errors.txt" << false;
|
|
QTest::newRow("invalidID.4") << "invalidID.4.qml" << "invalidID.4.errors.txt" << false;
|
|
QTest::newRow("invalidID.5") << "invalidID.5.qml" << "invalidID.5.errors.txt" << false;
|
|
QTest::newRow("invalidID.6") << "invalidID.6.qml" << "invalidID.6.errors.txt" << false;
|
|
QTest::newRow("invalidID.7") << "invalidID.7.qml" << "invalidID.7.errors.txt" << false;
|
|
QTest::newRow("invalidID.8") << "invalidID.8.qml" << "invalidID.8.errors.txt" << false;
|
|
QTest::newRow("invalidID.9") << "invalidID.9.qml" << "invalidID.9.errors.txt" << false;
|
|
QTest::newRow("invalidID.10") << "invalidID.10.qml" << "invalidID.10.errors.txt" << false;
|
|
|
|
QTest::newRow("scriptString.1") << "scriptString.1.qml" << "scriptString.1.errors.txt" << false;
|
|
QTest::newRow("scriptString.2") << "scriptString.2.qml" << "scriptString.2.errors.txt" << false;
|
|
|
|
QTest::newRow("unsupportedProperty") << "unsupportedProperty.qml" << "unsupportedProperty.errors.txt" << false;
|
|
QTest::newRow("nullDotProperty") << "nullDotProperty.qml" << "nullDotProperty.errors.txt" << true;
|
|
QTest::newRow("fakeDotProperty") << "fakeDotProperty.qml" << "fakeDotProperty.errors.txt" << false;
|
|
QTest::newRow("duplicateIDs") << "duplicateIDs.qml" << "duplicateIDs.errors.txt" << false;
|
|
QTest::newRow("unregisteredObject") << "unregisteredObject.qml" << "unregisteredObject.errors.txt" << false;
|
|
QTest::newRow("empty") << "empty.qml" << "empty.errors.txt" << false;
|
|
QTest::newRow("missingObject") << "missingObject.qml" << "missingObject.errors.txt" << false;
|
|
QTest::newRow("failingComponent") << "failingComponentTest.qml" << "failingComponent.errors.txt" << false;
|
|
QTest::newRow("missingSignal") << "missingSignal.qml" << "missingSignal.errors.txt" << false;
|
|
QTest::newRow("missingSignal2") << "missingSignal.2.qml" << "missingSignal.2.errors.txt" << false;
|
|
QTest::newRow("finalOverride") << "finalOverride.qml" << "finalOverride.errors.txt" << false;
|
|
QTest::newRow("customParserIdNotAllowed") << "customParserIdNotAllowed.qml" << "customParserIdNotAllowed.errors.txt" << false;
|
|
|
|
QTest::newRow("nullishCoalescing_LHS_Or") << "nullishCoalescing_LHS_Or.qml" << "nullishCoalescing_LHS_Or.errors.txt" << false;
|
|
QTest::newRow("nullishCoalescing_LHS_And") << "nullishCoalescing_LHS_And.qml" << "nullishCoalescing_LHS_And.errors.txt" << false;
|
|
QTest::newRow("nullishCoalescing_RHS_Or") << "nullishCoalescing_RHS_Or.qml" << "nullishCoalescing_RHS_Or.errors.txt" << false;
|
|
QTest::newRow("nullishCoalescing_RHS_And") << "nullishCoalescing_RHS_And.qml" << "nullishCoalescing_RHS_And.errors.txt" << false;
|
|
|
|
QTest::newRow("questionDotEOF") << "questionDotEOF.qml" << "questionDotEOF.errors.txt" << false;
|
|
QTest::newRow("optionalChaining.LHS") << "optionalChaining.LHS.qml" << "optionalChaining.LHS.errors.txt" << false;
|
|
|
|
|
|
QTest::newRow("invalidGroupedProperty.1") << "invalidGroupedProperty.1.qml" << "invalidGroupedProperty.1.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.2") << "invalidGroupedProperty.2.qml" << "invalidGroupedProperty.2.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.3") << "invalidGroupedProperty.3.qml" << "invalidGroupedProperty.3.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.4") << "invalidGroupedProperty.4.qml" << "invalidGroupedProperty.4.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.5") << "invalidGroupedProperty.5.qml" << "invalidGroupedProperty.5.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.6") << "invalidGroupedProperty.6.qml" << "invalidGroupedProperty.6.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.7") << "invalidGroupedProperty.7.qml" << "invalidGroupedProperty.7.errors.txt" << true;
|
|
QTest::newRow("invalidGroupedProperty.8") << "invalidGroupedProperty.8.qml" << "invalidGroupedProperty.8.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.9") << "invalidGroupedProperty.9.qml" << "invalidGroupedProperty.9.errors.txt" << false;
|
|
QTest::newRow("invalidGroupedProperty.10") << "invalidGroupedProperty.10.qml" << "invalidGroupedProperty.10.errors.txt" << false;
|
|
|
|
QTest::newRow("importNamespaceConflict") << "importNamespaceConflict.qml" << "importNamespaceConflict.errors.txt" << false;
|
|
QTest::newRow("importVersionMissing (builtin)") << "importVersionMissingBuiltIn.qml" << "importVersionMissingBuiltIn.errors.txt" << false;
|
|
QTest::newRow("importVersionMissing (installed)") << "importVersionMissingInstalled.qml" << "importVersionMissingInstalled.errors.txt" << false;
|
|
QTest::newRow("importNonExist (installed)") << "importNonExist.qml" << "importNonExist.errors.txt" << false;
|
|
QTest::newRow("importNonExistOlder (installed)") << "importNonExistOlder.qml" << "importNonExistOlder.errors.txt" << false;
|
|
QTest::newRow("importNewerVersion (installed)") << "importNewerVersion.qml" << "importNewerVersion.errors.txt" << false;
|
|
QTest::newRow("invalidImportID") << "invalidImportID.qml" << "invalidImportID.errors.txt" << false;
|
|
QTest::newRow("importFile") << "importFile.qml" << "importFile.errors.txt" << false;
|
|
|
|
QTest::newRow("signal.1") << "signal.1.qml" << "signal.1.errors.txt" << false;
|
|
QTest::newRow("signal.2") << "signal.2.qml" << "signal.2.errors.txt" << false;
|
|
QTest::newRow("signal.3") << "signal.3.qml" << "signal.3.errors.txt" << false;
|
|
QTest::newRow("signal.4") << "signal.4.qml" << "signal.4.errors.txt" << false;
|
|
QTest::newRow("signal.5") << "signal.5.qml" << "signal.5.errors.txt" << false;
|
|
QTest::newRow("signal.6") << "signal.6.qml" << "signal.6.errors.txt" << false;
|
|
|
|
QTest::newRow("method.1") << "method.1.qml" << "method.1.errors.txt" << false;
|
|
|
|
QTest::newRow("property.1") << "property.1.qml" << "property.1.errors.txt" << false;
|
|
QTest::newRow("property.2") << "property.2.qml" << "property.2.errors.txt" << false;
|
|
QTest::newRow("property.3") << "property.3.qml" << "property.3.errors.txt" << false;
|
|
QTest::newRow("property.4") << "property.4.qml" << "property.4.errors.txt" << false;
|
|
QTest::newRow("property.6") << "property.6.qml" << "property.6.errors.txt" << false;
|
|
QTest::newRow("property.7") << "property.7.qml" << "property.7.errors.txt" << false;
|
|
|
|
QTest::newRow("importScript.1") << "importscript.1.qml" << "importscript.1.errors.txt" << false;
|
|
|
|
QTest::newRow("Component.1") << "component.1.qml" << "component.1.errors.txt" << false;
|
|
QTest::newRow("Component.2") << "component.2.qml" << "component.2.errors.txt" << false;
|
|
QTest::newRow("Component.3") << "component.3.qml" << "component.3.errors.txt" << false;
|
|
QTest::newRow("Component.4") << "component.4.qml" << "component.4.errors.txt" << false;
|
|
QTest::newRow("Component.5") << "component.5.qml" << "component.5.errors.txt" << false;
|
|
QTest::newRow("Component.6") << "component.6.qml" << "component.6.errors.txt" << false;
|
|
QTest::newRow("Component.7") << "component.7.qml" << "component.7.errors.txt" << false;
|
|
QTest::newRow("Component.8") << "component.8.qml" << "component.8.errors.txt" << false;
|
|
QTest::newRow("Component.9") << "component.9.qml" << "component.9.errors.txt" << false;
|
|
|
|
QTest::newRow("MultiSet.1") << "multiSet.1.qml" << "multiSet.1.errors.txt" << false;
|
|
QTest::newRow("MultiSet.2") << "multiSet.2.qml" << "multiSet.2.errors.txt" << false;
|
|
QTest::newRow("MultiSet.3") << "multiSet.3.qml" << "multiSet.3.errors.txt" << false;
|
|
QTest::newRow("MultiSet.4") << "multiSet.4.qml" << "multiSet.4.errors.txt" << false;
|
|
QTest::newRow("MultiSet.5") << "multiSet.5.qml" << "multiSet.5.errors.txt" << false;
|
|
QTest::newRow("MultiSet.6") << "multiSet.6.qml" << "multiSet.6.errors.txt" << false;
|
|
QTest::newRow("MultiSet.7") << "multiSet.7.qml" << "multiSet.7.errors.txt" << false;
|
|
QTest::newRow("MultiSet.8") << "multiSet.8.qml" << "multiSet.8.errors.txt" << false;
|
|
QTest::newRow("MultiSet.9") << "multiSet.9.qml" << "multiSet.9.errors.txt" << false;
|
|
QTest::newRow("MultiSet.10") << "multiSet.10.qml" << "multiSet.10.errors.txt" << false;
|
|
QTest::newRow("MultiSet.11") << "multiSet.11.qml" << "multiSet.11.errors.txt" << false;
|
|
|
|
QTest::newRow("dynamicMeta.1") << "dynamicMeta.1.qml" << "dynamicMeta.1.errors.txt" << false;
|
|
QTest::newRow("dynamicMeta.2") << "dynamicMeta.2.qml" << "dynamicMeta.2.errors.txt" << false;
|
|
QTest::newRow("dynamicMeta.3") << "dynamicMeta.3.qml" << "dynamicMeta.3.errors.txt" << false;
|
|
QTest::newRow("dynamicMeta.4") << "dynamicMeta.4.qml" << "dynamicMeta.4.errors.txt" << false;
|
|
QTest::newRow("dynamicMeta.5") << "dynamicMeta.5.qml" << "dynamicMeta.5.errors.txt" << false;
|
|
|
|
QTest::newRow("invalidAlias.1") << "invalidAlias.1.qml" << "invalidAlias.1.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.2") << "invalidAlias.2.qml" << "invalidAlias.2.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.3") << "invalidAlias.3.qml" << "invalidAlias.3.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.4") << "invalidAlias.4.qml" << "invalidAlias.4.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.5") << "invalidAlias.5.qml" << "invalidAlias.5.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.6") << "invalidAlias.6.qml" << "invalidAlias.6.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.7") << "invalidAlias.7.qml" << "invalidAlias.7.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.8") << "invalidAlias.8.qml" << "invalidAlias.8.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.9") << "invalidAlias.9.qml" << "invalidAlias.9.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.10") << "invalidAlias.10.qml" << "invalidAlias.10.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.11") << "invalidAlias.11.qml" << "invalidAlias.11.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.12") << "invalidAlias.12.qml" << "invalidAlias.12.errors.txt" << false;
|
|
QTest::newRow("invalidAlias.13") << "invalidAlias.13.qml" << "invalidAlias.13.errors.txt" << false;
|
|
|
|
QTest::newRow("invalidAttachedProperty.1") << "invalidAttachedProperty.1.qml" << "invalidAttachedProperty.1.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.2") << "invalidAttachedProperty.2.qml" << "invalidAttachedProperty.2.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.3") << "invalidAttachedProperty.3.qml" << "invalidAttachedProperty.3.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.4") << "invalidAttachedProperty.4.qml" << "invalidAttachedProperty.4.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.5") << "invalidAttachedProperty.5.qml" << "invalidAttachedProperty.5.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.6") << "invalidAttachedProperty.6.qml" << "invalidAttachedProperty.6.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.7") << "invalidAttachedProperty.7.qml" << "invalidAttachedProperty.7.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.8") << "invalidAttachedProperty.8.qml" << "invalidAttachedProperty.8.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.9") << "invalidAttachedProperty.9.qml" << "invalidAttachedProperty.9.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.10") << "invalidAttachedProperty.10.qml" << "invalidAttachedProperty.10.errors.txt" << false;
|
|
QTest::newRow("invalidAttachedProperty.11") << "invalidAttachedProperty.11.qml" << "invalidAttachedProperty.11.errors.txt" << false;
|
|
|
|
QTest::newRow("assignValueToSignal") << "assignValueToSignal.qml" << "assignValueToSignal.errors.txt" << false;
|
|
QTest::newRow("emptySignal") << "emptySignal.qml" << "emptySignal.errors.txt" << false;
|
|
|
|
QTest::newRow("nestedErrors") << "nestedErrors.qml" << "nestedErrors.errors.txt" << false;
|
|
QTest::newRow("defaultGrouped") << "defaultGrouped.qml" << "defaultGrouped.errors.txt" << false;
|
|
QTest::newRow("doubleSignal") << "doubleSignal.qml" << "doubleSignal.errors.txt" << false;
|
|
QTest::newRow("missingValueTypeProperty") << "missingValueTypeProperty.qml" << "missingValueTypeProperty.errors.txt" << false;
|
|
QTest::newRow("objectValueTypeProperty") << "objectValueTypeProperty.qml" << "objectValueTypeProperty.errors.txt" << false;
|
|
QTest::newRow("enumTypes") << "enumTypes.qml" << "enumTypes.errors.txt" << false;
|
|
QTest::newRow("noCreation") << "noCreation.qml" << "noCreation.errors.txt" << false;
|
|
QTest::newRow("destroyedSignal") << "destroyedSignal.qml" << "destroyedSignal.errors.txt" << false;
|
|
QTest::newRow("assignToNamespace") << "assignToNamespace.qml" << "assignToNamespace.errors.txt" << false;
|
|
QTest::newRow("invalidOn") << "invalidOn.qml" << "invalidOn.errors.txt" << false;
|
|
QTest::newRow("invalidProperty") << "invalidProperty.qml" << "invalidProperty.errors.txt" << false;
|
|
QTest::newRow("nonScriptableProperty") << "nonScriptableProperty.qml" << "nonScriptableProperty.errors.txt" << false;
|
|
QTest::newRow("notAvailable") << "notAvailable.qml" << "notAvailable.errors.txt" << false;
|
|
QTest::newRow("singularProperty") << "singularProperty.qml" << "singularProperty.errors.txt" << false;
|
|
QTest::newRow("singularProperty.2") << "singularProperty.2.qml" << "singularProperty.2.errors.txt" << false;
|
|
|
|
QTest::newRow("scopedEnumList") << "scopedEnumList.qml" << "scopedEnumList.errors.txt" << false;
|
|
QTest::newRow("lowercase enum value") << "lowercaseQmlEnum.1.qml" << "lowercaseQmlEnum.1.errors.txt" << false;
|
|
QTest::newRow("lowercase enum type") << "lowercaseQmlEnum.2.qml" << "lowercaseQmlEnum.2.errors.txt" << false;
|
|
QTest::newRow("string enum value") << "invalidQmlEnumValue.1.qml" << "invalidQmlEnumValue.1.errors.txt" << false;
|
|
QTest::newRow("identifier enum type") << "invalidQmlEnumValue.2.qml" << "invalidQmlEnumValue.2.errors.txt" << false;
|
|
QTest::newRow("enum value too large") << "invalidQmlEnumValue.3.qml" << "invalidQmlEnumValue.3.errors.txt" << false;
|
|
QTest::newRow("non-integer enum value") << "invalidQmlEnumValue.4.qml" << "invalidQmlEnumValue.4.errors.txt" << false;
|
|
|
|
const QString expectedError = isCaseSensitiveFileSystem(dataDirectory()) ?
|
|
QStringLiteral("incorrectCase.errors.sensitive.txt") :
|
|
QStringLiteral("incorrectCase.errors.insensitive.txt");
|
|
QTest::newRow("incorrectCase") << "incorrectCase.qml" << expectedError << false;
|
|
|
|
QTest::newRow("metaobjectRevision.1") << "metaobjectRevision.1.qml" << "metaobjectRevision.1.errors.txt" << false;
|
|
QTest::newRow("metaobjectRevision.2") << "metaobjectRevision.2.qml" << "metaobjectRevision.2.errors.txt" << false;
|
|
QTest::newRow("metaobjectRevision.3") << "metaobjectRevision.3.qml" << "metaobjectRevision.3.errors.txt" << false;
|
|
|
|
QTest::newRow("invalidRoot.1") << "invalidRoot.1.qml" << "invalidRoot.1.errors.txt" << false;
|
|
QTest::newRow("invalidRoot.2") << "invalidRoot.2.qml" << "invalidRoot.2.errors.txt" << false;
|
|
QTest::newRow("invalidRoot.3") << "invalidRoot.3.qml" << "invalidRoot.3.errors.txt" << false;
|
|
QTest::newRow("invalidRoot.4") << "invalidRoot.4.qml" << "invalidRoot.4.errors.txt" << false;
|
|
|
|
QTest::newRow("invalidTypeName.1") << "invalidTypeName.1.qml" << "invalidTypeName.1.errors.txt" << false;
|
|
QTest::newRow("invalidTypeName.2") << "invalidTypeName.2.qml" << "invalidTypeName.2.errors.txt" << false;
|
|
QTest::newRow("invalidTypeName.3") << "invalidTypeName.3.qml" << "invalidTypeName.3.errors.txt" << false;
|
|
QTest::newRow("invalidTypeName.4") << "invalidTypeName.4.qml" << "invalidTypeName.4.errors.txt" << false;
|
|
|
|
QTest::newRow("Major version isolation") << "majorVersionIsolation.qml" << "majorVersionIsolation.errors.txt" << false;
|
|
|
|
QTest::newRow("badCompositeRegistration.1") << "badCompositeRegistration.1.qml" << "badCompositeRegistration.1.errors.txt" << false;
|
|
QTest::newRow("badCompositeRegistration.2") << "badCompositeRegistration.2.qml" << "badCompositeRegistration.2.errors.txt" << false;
|
|
|
|
QTest::newRow("assignComponentToWrongType") << "assignComponentToWrongType.qml" << "assignComponentToWrongType.errors.txt" << false;
|
|
QTest::newRow("cyclicAlias") << "cyclicAlias.qml" << "cyclicAlias.errors.txt" << false;
|
|
|
|
QTest::newRow("fuzzed.1") << "fuzzed.1.qml" << "fuzzed.1.errors.txt" << false;
|
|
QTest::newRow("fuzzed.2") << "fuzzed.2.qml" << "fuzzed.2.errors.txt" << false;
|
|
QTest::newRow("fuzzed.3") << "fuzzed.3.qml" << "fuzzed.3.errors.txt" << false;
|
|
|
|
QTest::newRow("bareQmlImport") << "bareQmlImport.qml" << "bareQmlImport.errors.txt" << false;
|
|
|
|
QTest::newRow("typeAnnotations.2") << "typeAnnotations.2.qml" << "typeAnnotations.2.errors.txt" << false;
|
|
|
|
QTest::newRow("propertyUnknownType") << "propertyUnknownType.qml" << "propertyUnknownType.errors.txt" << false;
|
|
|
|
QTest::newRow("selfInstantiation") << "SelfInstantiation.qml" << "SelfInstantiation.errors.txt" << false;
|
|
}
|
|
|
|
void tst_qqmllanguage::errors()
|
|
{
|
|
#ifdef Q_OS_ANDROID
|
|
if (qstrcmp(QTest::currentDataTag(), "fuzzed.2") == 0) {
|
|
QSKIP("Gives different errors on Android");
|
|
/* Only gives one error on Android:
|
|
|
|
qrc:/data/fuzzed.2.qml:1:1: "
|
|
import"
|
|
^
|
|
So, it seems to complain about the first import (which is understandable)
|
|
*/
|
|
}
|
|
#endif
|
|
QFETCH(QString, file);
|
|
QFETCH(QString, errorFile);
|
|
QFETCH(bool, create);
|
|
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
QTRY_VERIFY(!component.isLoading());
|
|
|
|
QScopedPointer<QObject> object;
|
|
|
|
if (create) {
|
|
object.reset(component.create());
|
|
QVERIFY(object.isNull());
|
|
}
|
|
|
|
VERIFY_ERRORS(errorFile.toLatin1().constData());
|
|
}
|
|
|
|
void tst_qqmllanguage::simpleObject()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("simpleObject.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::simpleContainer()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("simpleContainer.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyContainer> container(qobject_cast<MyContainer*>(component.create()));
|
|
QVERIFY(container != nullptr);
|
|
QCOMPARE(container->getChildren()->size(),2);
|
|
}
|
|
|
|
void tst_qqmllanguage::interfaceProperty()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("interfaceProperty.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(object->interface());
|
|
QCOMPARE(object->interface()->id, 913);
|
|
}
|
|
|
|
void tst_qqmllanguage::interfaceQList()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("interfaceQList.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyContainer> container(qobject_cast<MyContainer*>(component.create()));
|
|
QVERIFY(container != nullptr);
|
|
QCOMPARE(container->getQListInterfaces()->size(), 2);
|
|
for(int ii = 0; ii < 2; ++ii)
|
|
QCOMPARE(container->getQListInterfaces()->at(ii)->id, 913);
|
|
}
|
|
|
|
void tst_qqmllanguage::assignObjectToSignal()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignObjectToSignal.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QTest::ignoreMessage(QtWarningMsg, "MyQmlObject::basicSlot");
|
|
emit object->basicSignal();
|
|
}
|
|
|
|
void tst_qqmllanguage::assignObjectToVariant()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignObjectToVariant.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QVariant v = object->property("a");
|
|
QVERIFY(v.typeId() == qMetaTypeId<QObject *>());
|
|
}
|
|
|
|
void tst_qqmllanguage::assignLiteralSignalProperty()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignLiteralSignalProperty.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->onLiteralSignal(), 10);
|
|
}
|
|
|
|
// Test is an external component can be loaded and assigned (to a qlist)
|
|
void tst_qqmllanguage::assignQmlComponent()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignQmlComponent.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyContainer> object(qobject_cast<MyContainer *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->getChildren()->size(), 1);
|
|
QObject *child = object->getChildren()->at(0);
|
|
QCOMPARE(child->property("x"), QVariant(10));
|
|
QCOMPARE(child->property("y"), QVariant(11));
|
|
}
|
|
|
|
// Test literal assignment to all the value types
|
|
void tst_qqmllanguage::assignValueTypes()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignValueTypes.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->flagProperty(), MyTypeObject::FlagVal1 | MyTypeObject::FlagVal3);
|
|
QCOMPARE(object->enumProperty(), MyTypeObject::EnumVal2);
|
|
QCOMPARE(object->qtEnumProperty(), Qt::RichText);
|
|
QCOMPARE(object->mirroredEnumProperty(), MyTypeObject::MirroredEnumVal3);
|
|
QCOMPARE(object->relatedEnumProperty(), MyEnumContainer::RelatedValue);
|
|
QCOMPARE(object->stringProperty(), QString("Hello World!"));
|
|
QCOMPARE(object->uintProperty(), uint(10));
|
|
QCOMPARE(object->intProperty(), -19);
|
|
QCOMPARE((float)object->realProperty(), float(23.2));
|
|
QCOMPARE((float)object->doubleProperty(), float(-19.7));
|
|
QCOMPARE((float)object->floatProperty(), float(8.5));
|
|
QCOMPARE(object->colorProperty(), QColor("red"));
|
|
QCOMPARE(object->dateProperty(), QDate(1982, 11, 25));
|
|
QCOMPARE(object->timeProperty(), QTime(11, 11, 32));
|
|
QCOMPARE(object->dateTimeProperty(), QDateTime(QDate(2009, 5, 12), QTime(13, 22, 1)));
|
|
QCOMPARE(object->pointProperty(), QPoint(99,13));
|
|
QCOMPARE(object->pointFProperty(), QPointF(-10.1, 12.3));
|
|
QCOMPARE(object->sizeProperty(), QSize(99, 13));
|
|
QCOMPARE(object->sizeFProperty(), QSizeF(0.1, 0.2));
|
|
QCOMPARE(object->rectProperty(), QRect(9, 7, 100, 200));
|
|
QCOMPARE(object->rectFProperty(), QRectF(1000.1, -10.9, 400, 90.99));
|
|
QCOMPARE(object->boolProperty(), true);
|
|
QCOMPARE(object->variantProperty(), QVariant("Hello World!"));
|
|
QCOMPARE(object->vectorProperty(), QVector3D(10, 1, 2.2f));
|
|
QCOMPARE(object->vector2Property(), QVector2D(2, 3));
|
|
QCOMPARE(object->vector4Property(), QVector4D(10, 1, 2.2f, 2.3f));
|
|
const QUrl encoded = QUrl::fromEncoded("main.qml?with%3cencoded%3edata", QUrl::TolerantMode);
|
|
QCOMPARE(object->urlProperty(), encoded);
|
|
QVERIFY(object->objectProperty() != nullptr);
|
|
MyTypeObject *child = qobject_cast<MyTypeObject *>(object->objectProperty());
|
|
QVERIFY(child != nullptr);
|
|
QCOMPARE(child->intProperty(), 8);
|
|
|
|
//these used to go via script. Ensure they no longer do
|
|
QCOMPARE(object->property("qtEnumTriggeredChange").toBool(), false);
|
|
QCOMPARE(object->property("mirroredEnumTriggeredChange").toBool(), false);
|
|
}
|
|
|
|
// Test edge case type assignments
|
|
void tst_qqmllanguage::assignTypeExtremes()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignTypeExtremes.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->uintProperty(), 0xEE6B2800);
|
|
QCOMPARE(object->intProperty(), -0x77359400);
|
|
}
|
|
|
|
// Test that a composite type can assign to a property of its base type
|
|
void tst_qqmllanguage::assignCompositeToType()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignCompositeToType.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
}
|
|
|
|
// Test that literals are stored correctly in "var" properties
|
|
// Note that behaviour differs from "variant" properties in that
|
|
// no conversion from "special strings" to QVariants is performed.
|
|
void tst_qqmllanguage::assignLiteralToVar()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignLiteralToVar.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QVERIFY(isJSNumberType(object->property("test1").typeId()));
|
|
QCOMPARE(object->property("test2").typeId(), (int)QMetaType::Double);
|
|
QCOMPARE(object->property("test3").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test4").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test5").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test6").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test7").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test8").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test9").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test10").typeId(), QMetaType::Bool);
|
|
QCOMPARE(object->property("test11").typeId(), QMetaType::Bool);
|
|
QCOMPARE(object->property("test12").typeId(), QMetaType::QColor);
|
|
QCOMPARE(object->property("test13").typeId(), QMetaType::QRectF);
|
|
QCOMPARE(object->property("test14").typeId(), QMetaType::QPointF);
|
|
QCOMPARE(object->property("test15").typeId(), QMetaType::QSizeF);
|
|
QCOMPARE(object->property("test16").typeId(), QMetaType::QVector3D);
|
|
QVERIFY(isJSNumberType(object->property("variantTest1Bound").typeId()));
|
|
QVERIFY(isJSNumberType(object->property("test1Bound").typeId()));
|
|
|
|
QCOMPARE(object->property("test1"), QVariant(5));
|
|
QCOMPARE(object->property("test2"), QVariant((double)1.7));
|
|
QCOMPARE(object->property("test3"), QVariant(QString(QLatin1String("Hello world!"))));
|
|
QCOMPARE(object->property("test4"), QVariant(QString(QLatin1String("#FF008800"))));
|
|
QCOMPARE(object->property("test5"), QVariant(QString(QLatin1String("10,10,10x10"))));
|
|
QCOMPARE(object->property("test6"), QVariant(QString(QLatin1String("10,10"))));
|
|
QCOMPARE(object->property("test7"), QVariant(QString(QLatin1String("10x10"))));
|
|
QCOMPARE(object->property("test8"), QVariant(QString(QLatin1String("100,100,100"))));
|
|
QCOMPARE(object->property("test9"), QVariant(QString(QLatin1String("#FF008800"))));
|
|
QCOMPARE(object->property("test10"), QVariant(bool(true)));
|
|
QCOMPARE(object->property("test11"), QVariant(bool(false)));
|
|
QCOMPARE(object->property("test12"), QVariant(QColor::fromRgbF(0.2f, 0.3f, 0.4f, 0.5f)));
|
|
QCOMPARE(object->property("test13"), QVariant(QRectF(10, 10, 10, 10)));
|
|
QCOMPARE(object->property("test14"), QVariant(QPointF(10, 10)));
|
|
QCOMPARE(object->property("test15"), QVariant(QSizeF(10, 10)));
|
|
QCOMPARE(object->property("test16"), QVariant(QVector3D(100, 100, 100)));
|
|
QCOMPARE(object->property("variantTest1Bound"), QVariant(9));
|
|
QCOMPARE(object->property("test1Bound"), QVariant(11));
|
|
}
|
|
|
|
void tst_qqmllanguage::assignLiteralToJSValue()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignLiteralToJSValue.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(root != nullptr);
|
|
|
|
{
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test1");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(5));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test2");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(1.7));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test3");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("Hello world!")));
|
|
}{
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test4");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("#FF008800")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test5");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("10,10,10x10")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test6");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("10,10")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test7");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("10x10")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test8");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("100,100,100")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test9");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("#FF008800")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test10");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isBool());
|
|
QCOMPARE(value.toBool(), true);
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test11");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isBool());
|
|
QCOMPARE(value.toBool(), false);
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test20");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isCallable());
|
|
QCOMPARE(value.call(QList<QJSValue> () << QJSValue(4)).toInt(), 12);
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test21");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isUndefined());
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test22");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNull());
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test1Bound");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(9));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test20Bound");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(27));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("test23");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isQObject());
|
|
QCOMPARE(value.toQObject()->objectName(), "blah");
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::assignNullStrings()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignNullStrings.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(object->stringProperty().isNull());
|
|
QVERIFY(object->byteArrayProperty().isNull());
|
|
QMetaObject::invokeMethod(object.data(), "assignNullStringsFromJs", Qt::DirectConnection);
|
|
QVERIFY(object->stringProperty().isNull());
|
|
QVERIFY(object->byteArrayProperty().isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::bindJSValueToVar()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignLiteralToJSValue.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(root != nullptr);
|
|
|
|
QObject *object = root->findChild<QObject *>("varProperties");
|
|
|
|
QVERIFY(isJSNumberType(object->property("test1").typeId()));
|
|
QVERIFY(isJSNumberType(object->property("test2").typeId()));
|
|
QCOMPARE(object->property("test3").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test4").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test5").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test6").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test7").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test8").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test9").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test10").typeId(), QMetaType::Bool);
|
|
QCOMPARE(object->property("test11").typeId(), QMetaType::Bool);
|
|
QCOMPARE(object->property("test12").typeId(), QMetaType::QColor);
|
|
QCOMPARE(object->property("test13").typeId(), QMetaType::QRectF);
|
|
QCOMPARE(object->property("test14").typeId(), QMetaType::QPointF);
|
|
QCOMPARE(object->property("test15").typeId(), QMetaType::QSizeF);
|
|
QCOMPARE(object->property("test16").typeId(), QMetaType::QVector3D);
|
|
QVERIFY(isJSNumberType(object->property("test1Bound").typeId()));
|
|
QVERIFY(isJSNumberType(object->property("test20Bound").typeId()));
|
|
|
|
QCOMPARE(object->property("test1"), QVariant(5));
|
|
QCOMPARE(object->property("test2"), QVariant((double)1.7));
|
|
QCOMPARE(object->property("test3"), QVariant(QString(QLatin1String("Hello world!"))));
|
|
QCOMPARE(object->property("test4"), QVariant(QString(QLatin1String("#FF008800"))));
|
|
QCOMPARE(object->property("test5"), QVariant(QString(QLatin1String("10,10,10x10"))));
|
|
QCOMPARE(object->property("test6"), QVariant(QString(QLatin1String("10,10"))));
|
|
QCOMPARE(object->property("test7"), QVariant(QString(QLatin1String("10x10"))));
|
|
QCOMPARE(object->property("test8"), QVariant(QString(QLatin1String("100,100,100"))));
|
|
QCOMPARE(object->property("test9"), QVariant(QString(QLatin1String("#FF008800"))));
|
|
QCOMPARE(object->property("test10"), QVariant(bool(true)));
|
|
QCOMPARE(object->property("test11"), QVariant(bool(false)));
|
|
QCOMPARE(object->property("test12"), QVariant(QColor::fromRgbF(0.2f, 0.3f, 0.4f, 0.5f)));
|
|
QCOMPARE(object->property("test13"), QVariant(QRectF(10, 10, 10, 10)));
|
|
QCOMPARE(object->property("test14"), QVariant(QPointF(10, 10)));
|
|
QCOMPARE(object->property("test15"), QVariant(QSizeF(10, 10)));
|
|
QCOMPARE(object->property("test16"), QVariant(QVector3D(100, 100, 100)));
|
|
QCOMPARE(object->property("test1Bound"), QVariant(9));
|
|
QCOMPARE(object->property("test20Bound"), QVariant(27));
|
|
}
|
|
|
|
void tst_qqmllanguage::bindJSValueToVariant()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignLiteralToJSValue.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(root != nullptr);
|
|
|
|
QObject *object = root->findChild<QObject *>("variantProperties");
|
|
|
|
QVERIFY(isJSNumberType(object->property("test1").typeId()));
|
|
QVERIFY(isJSNumberType(object->property("test2").typeId()));
|
|
QCOMPARE(object->property("test3").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test4").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test5").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test6").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test7").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test8").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test9").typeId(), QMetaType::QString);
|
|
QCOMPARE(object->property("test10").typeId(), QMetaType::Bool);
|
|
QCOMPARE(object->property("test11").typeId(), QMetaType::Bool);
|
|
QCOMPARE(object->property("test12").typeId(), QMetaType::QColor);
|
|
QCOMPARE(object->property("test13").typeId(), QMetaType::QRectF);
|
|
QCOMPARE(object->property("test14").typeId(), QMetaType::QPointF);
|
|
QCOMPARE(object->property("test15").typeId(), QMetaType::QSizeF);
|
|
QCOMPARE(object->property("test16").typeId(), QMetaType::QVector3D);
|
|
QVERIFY(isJSNumberType(object->property("test1Bound").typeId()));
|
|
QVERIFY(isJSNumberType(object->property("test20Bound").typeId()));
|
|
|
|
QCOMPARE(object->property("test1"), QVariant(5));
|
|
QCOMPARE(object->property("test2"), QVariant((double)1.7));
|
|
QCOMPARE(object->property("test3"), QVariant(QString(QLatin1String("Hello world!"))));
|
|
QCOMPARE(object->property("test4"), QVariant(QString(QLatin1String("#FF008800"))));
|
|
QCOMPARE(object->property("test5"), QVariant(QString(QLatin1String("10,10,10x10"))));
|
|
QCOMPARE(object->property("test6"), QVariant(QString(QLatin1String("10,10"))));
|
|
QCOMPARE(object->property("test7"), QVariant(QString(QLatin1String("10x10"))));
|
|
QCOMPARE(object->property("test8"), QVariant(QString(QLatin1String("100,100,100"))));
|
|
QCOMPARE(object->property("test9"), QVariant(QString(QLatin1String("#FF008800"))));
|
|
QCOMPARE(object->property("test10"), QVariant(bool(true)));
|
|
QCOMPARE(object->property("test11"), QVariant(bool(false)));
|
|
QCOMPARE(object->property("test12"), QVariant(QColor::fromRgbF(0.2f, 0.3f, 0.4f, 0.5f)));
|
|
QCOMPARE(object->property("test13"), QVariant(QRectF(10, 10, 10, 10)));
|
|
QCOMPARE(object->property("test14"), QVariant(QPointF(10, 10)));
|
|
QCOMPARE(object->property("test15"), QVariant(QSizeF(10, 10)));
|
|
QCOMPARE(object->property("test16"), QVariant(QVector3D(100, 100, 100)));
|
|
QCOMPARE(object->property("test1Bound"), QVariant(9));
|
|
QCOMPARE(object->property("test20Bound"), QVariant(27));
|
|
}
|
|
|
|
void tst_qqmllanguage::bindJSValueToType()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignLiteralToJSValue.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(root != nullptr);
|
|
|
|
{
|
|
MyTypeObject *object = root->findChild<MyTypeObject *>("typedProperties");
|
|
|
|
QCOMPARE(object->intProperty(), 5);
|
|
QCOMPARE(object->doubleProperty(), double(1.7));
|
|
QCOMPARE(object->stringProperty(), QString(QLatin1String("Hello world!")));
|
|
QCOMPARE(object->boolProperty(), true);
|
|
QCOMPARE(object->colorProperty(), QColor::fromRgbF(0.2f, 0.3f, 0.4f, 0.5f));
|
|
QCOMPARE(object->rectFProperty(), QRectF(10, 10, 10, 10));
|
|
QCOMPARE(object->pointFProperty(), QPointF(10, 10));
|
|
QCOMPARE(object->sizeFProperty(), QSizeF(10, 10));
|
|
QCOMPARE(object->vectorProperty(), QVector3D(100, 100, 100));
|
|
} {
|
|
MyTypeObject *object = root->findChild<MyTypeObject *>("stringProperties");
|
|
|
|
QCOMPARE(object->intProperty(), 1);
|
|
QCOMPARE(object->doubleProperty(), double(1.7));
|
|
QCOMPARE(object->stringProperty(), QString(QLatin1String("Hello world!")));
|
|
QCOMPARE(object->boolProperty(), true);
|
|
QCOMPARE(object->colorProperty(), QColor::fromRgb(0x00, 0x88, 0x00, 0xFF));
|
|
QCOMPARE(object->rectFProperty(), QRectF(10, 10, 10, 10));
|
|
QCOMPARE(object->pointFProperty(), QPointF(10, 10));
|
|
QCOMPARE(object->sizeFProperty(), QSizeF(10, 10));
|
|
QCOMPARE(object->vectorProperty(), QVector3D(100, 100, 100));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::bindTypeToJSValue()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("bindTypeToJSValue.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(root != nullptr);
|
|
|
|
{
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("flagProperty");
|
|
QVERIFY(object);
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(MyTypeObject::FlagVal1 | MyTypeObject::FlagVal3));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("enumProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(MyTypeObject::EnumVal2));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("stringProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("Hello World!")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("uintProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(10));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("intProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(-19));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("realProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(23.2));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("doubleProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(-19.7));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("floatProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isNumber());
|
|
QCOMPARE(value.toNumber(), qreal(8.5));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("colorProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isObject());
|
|
QCOMPARE(value.property(QLatin1String("r")).toNumber(), qreal(1.0));
|
|
QCOMPARE(value.property(QLatin1String("g")).toNumber(), qreal(0.0));
|
|
QCOMPARE(value.property(QLatin1String("b")).toNumber(), qreal(0.0));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("dateProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QCOMPARE(value.toDateTime().isValid(), true);
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("timeProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QCOMPARE(value.toDateTime().isValid(), true);
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("dateTimeProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QCOMPARE(value.toDateTime().isValid(), true);
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("pointProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isObject());
|
|
QCOMPARE(value.property(QLatin1String("x")).toNumber(), qreal(99));
|
|
QCOMPARE(value.property(QLatin1String("y")).toNumber(), qreal(13));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("pointFProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isObject());
|
|
QCOMPARE(value.property(QLatin1String("x")).toNumber(), qreal(-10.1));
|
|
QCOMPARE(value.property(QLatin1String("y")).toNumber(), qreal(12.3));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("rectProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isObject());
|
|
QCOMPARE(value.property(QLatin1String("x")).toNumber(), qreal(9));
|
|
QCOMPARE(value.property(QLatin1String("y")).toNumber(), qreal(7));
|
|
QCOMPARE(value.property(QLatin1String("width")).toNumber(), qreal(100));
|
|
QCOMPARE(value.property(QLatin1String("height")).toNumber(), qreal(200));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("rectFProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isObject());
|
|
QCOMPARE(value.property(QLatin1String("x")).toNumber(), qreal(1000.1));
|
|
QCOMPARE(value.property(QLatin1String("y")).toNumber(), qreal(-10.9));
|
|
QCOMPARE(value.property(QLatin1String("width")).toNumber(), qreal(400));
|
|
QCOMPARE(value.property(QLatin1String("height")).toNumber(), qreal(90.99));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("boolProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isBool());
|
|
QCOMPARE(value.toBool(), true);
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("variantProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("Hello World!")));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("vectorProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isObject());
|
|
QCOMPARE(value.property(QLatin1String("x")).toNumber(), qreal(10.0f));
|
|
QCOMPARE(value.property(QLatin1String("y")).toNumber(), qreal(1.0f));
|
|
QCOMPARE(value.property(QLatin1String("z")).toNumber(), qreal(2.2f));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("vector4Property");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isObject());
|
|
QCOMPARE(value.property(QLatin1String("x")).toNumber(), qreal(10.0f));
|
|
QCOMPARE(value.property(QLatin1String("y")).toNumber(), qreal(1.0f));
|
|
QCOMPARE(value.property(QLatin1String("z")).toNumber(), qreal(2.2f));
|
|
QCOMPARE(value.property(QLatin1String("w")).toNumber(), qreal(2.3f));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("urlProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
const QUrl encoded = QUrl::fromEncoded("main.qml?with%3cencoded%3edata", QUrl::TolerantMode);
|
|
QCOMPARE(value.toString(), encoded.toString());
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("objectProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isQObject());
|
|
QVERIFY(qobject_cast<MyTypeObject *>(value.toQObject()));
|
|
} {
|
|
MyQmlObject *object = root->findChild<MyQmlObject *>("varProperty");
|
|
QJSValue value = object->qjsvalue();
|
|
QVERIFY(value.isString());
|
|
QCOMPARE(value.toString(), QString(QLatin1String("Hello World!")));
|
|
}
|
|
}
|
|
|
|
// Tests that custom parser types can be instantiated
|
|
void tst_qqmllanguage::customParserTypes()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("customParserTypes.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("count"), QVariant(2));
|
|
}
|
|
|
|
// Tests that custom pursor types can be instantiated in ICs
|
|
void tst_qqmllanguage::customParserTypeInInlineComponent()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("customParserTypeInIC.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("count"), 2);
|
|
}
|
|
|
|
// Tests that the root item can be a custom component
|
|
void tst_qqmllanguage::rootAsQmlComponent()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("rootAsQmlComponent.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyContainer> object(qobject_cast<MyContainer *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("x"), QVariant(11));
|
|
QCOMPARE(object->getChildren()->size(), 2);
|
|
}
|
|
|
|
void tst_qqmllanguage::rootItemIsComponent()
|
|
{
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg,
|
|
QRegularExpression(
|
|
".*/rootItemIsComponent\\.qml:3:1: Using a Component as the root of "
|
|
"a QML document is deprecated: types defined in qml documents are "
|
|
"automatically wrapped into Components when needed\\."));
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg,
|
|
QRegularExpression(
|
|
".*/EvilComponentType\\.qml:3:1: Using a Component as the root of a "
|
|
"QML document is deprecated: types defined in qml documents are automatically "
|
|
"wrapped into Components when needed\\."));
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg,
|
|
QRegularExpression(".*/rootItemIsComponent\\.qml:7:36: Using a Component as the root "
|
|
"of an inline component is deprecated: inline components are "
|
|
"automatically wrapped into Components when needed\\."));
|
|
QQmlComponent component(&engine, testFileUrl("rootItemIsComponent.qml"));
|
|
}
|
|
|
|
// Tests that components can be specified inline
|
|
void tst_qqmllanguage::inlineQmlComponents()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("inlineQmlComponents.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyContainer> object(qobject_cast<MyContainer *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->getChildren()->size(), 1);
|
|
QQmlComponent *comp = qobject_cast<QQmlComponent *>(object->getChildren()->at(0));
|
|
QVERIFY(comp != nullptr);
|
|
QScopedPointer<MyQmlObject> compObject(qobject_cast<MyQmlObject *>(comp->create()));
|
|
QVERIFY(compObject != nullptr);
|
|
QCOMPARE(compObject->value(), 11);
|
|
}
|
|
|
|
// Tests that types that have an id property have it set
|
|
void tst_qqmllanguage::idProperty()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("idProperty.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyContainer> object(qobject_cast<MyContainer *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->getChildren()->size(), 2);
|
|
MyTypeObject *child =
|
|
qobject_cast<MyTypeObject *>(object->getChildren()->at(0));
|
|
QVERIFY(child != nullptr);
|
|
QCOMPARE(child->id(), QString("myObjectId"));
|
|
QCOMPARE(object->property("object"), QVariant::fromValue((QObject *)child));
|
|
|
|
child =
|
|
qobject_cast<MyTypeObject *>(object->getChildren()->at(1));
|
|
QVERIFY(child != nullptr);
|
|
QCOMPARE(child->id(), QString("name.with.dots"));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("idPropertyMismatch.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(!root.isNull());
|
|
QQmlContext *ctx = qmlContext(root.data());
|
|
QVERIFY(ctx);
|
|
QCOMPARE(ctx->nameForObject(root.data()), QStringLiteral("root"));
|
|
QCOMPARE(ctx->objectForName(QStringLiteral("root")), root.data());
|
|
}
|
|
}
|
|
|
|
// Tests automatic connection to notify signals if "onBlahChanged" syntax is used
|
|
// even if the notify signal for "blah" is not called "blahChanged"
|
|
void tst_qqmllanguage::autoNotifyConnection()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("autoNotifyConnection.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QMetaProperty prop = object->metaObject()->property(object->metaObject()->indexOfProperty("receivedNotify"));
|
|
QVERIFY(prop.isValid());
|
|
|
|
QCOMPARE(prop.read(object.data()), QVariant::fromValue(false));
|
|
object->setPropertyWithNotify(1);
|
|
QCOMPARE(prop.read(object.data()), QVariant::fromValue(true));
|
|
}
|
|
|
|
// Tests that signals can be assigned to
|
|
void tst_qqmllanguage::assignSignal()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignSignal.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QTest::ignoreMessage(QtWarningMsg, "MyQmlObject::basicSlot");
|
|
emit object->basicSignal();
|
|
QTest::ignoreMessage(QtWarningMsg, "MyQmlObject::basicSlotWithArgs(9)");
|
|
emit object->basicParameterizedSignal(9);
|
|
}
|
|
|
|
void tst_qqmllanguage::assignSignalFunctionExpression()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("assignSignalFunctionExpression.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QTest::ignoreMessage(QtWarningMsg, "MyQmlObject::basicSlot");
|
|
emit object->basicSignal();
|
|
QTest::ignoreMessage(QtWarningMsg, "MyQmlObject::basicSlotWithArgs(9)");
|
|
emit object->basicParameterizedSignal(9);
|
|
}
|
|
|
|
void tst_qqmllanguage::overrideSignal_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::addColumn<QString>("errorFile");
|
|
|
|
QTest::newRow("override signal with signal") << "overrideSignal.1.qml" << "overrideSignal.1.errors.txt";
|
|
QTest::newRow("override signal with method") << "overrideSignal.2.qml" << "overrideSignal.2.errors.txt";
|
|
QTest::newRow("override signal with property") << "overrideSignal.3.qml" << "";
|
|
QTest::newRow("override signal of alias property with signal") << "overrideSignal.4.qml" << "overrideSignal.4.errors.txt";
|
|
QTest::newRow("override signal of superclass with signal") << "overrideSignal.5.qml" << "overrideSignal.5.errors.txt";
|
|
QTest::newRow("override builtin signal with signal") << "overrideSignal.6.qml" << "overrideSignal.6.errors.txt";
|
|
}
|
|
|
|
void tst_qqmllanguage::overrideSignal()
|
|
{
|
|
QFETCH(QString, file);
|
|
QFETCH(QString, errorFile);
|
|
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
if (errorFile.isEmpty()) {
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(object->property("success").toBool());
|
|
} else {
|
|
VERIFY_ERRORS(errorFile.toLatin1().constData());
|
|
}
|
|
}
|
|
|
|
// Tests the creation and assignment of dynamic properties
|
|
void tst_qqmllanguage::dynamicProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("dynamicProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QCOMPARE(object->property("intProperty"), QVariant(10));
|
|
QCOMPARE(object->property("boolProperty"), QVariant(false));
|
|
QCOMPARE(object->property("doubleProperty"), QVariant(-10.1));
|
|
QCOMPARE(object->property("realProperty"), QVariant((qreal)-19.9));
|
|
QCOMPARE(object->property("stringProperty"), QVariant("Hello World!"));
|
|
QCOMPARE(object->property("urlProperty"), QVariant(QUrl("main.qml")));
|
|
QCOMPARE(object->property("colorProperty"), QVariant(QColor("red")));
|
|
QVariant date = object->property("dateProperty");
|
|
if (!date.convert(QMetaType(QMetaType::QDate)))
|
|
QFAIL("could not convert to date");
|
|
QCOMPARE(date, QVariant(QDate(1945, 9, 2)));
|
|
QCOMPARE(object->property("varProperty"), QVariant("Hello World!"));
|
|
}
|
|
|
|
// Test that nested types can use dynamic properties
|
|
void tst_qqmllanguage::dynamicPropertiesNested()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("dynamicPropertiesNested.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("super_a").toInt(), 11); // Overridden
|
|
QCOMPARE(object->property("super_c").toInt(), 14); // Inherited
|
|
QCOMPARE(object->property("a").toInt(), 13); // New
|
|
QCOMPARE(object->property("b").toInt(), 12); // New
|
|
}
|
|
|
|
// Tests the creation and assignment to dynamic list properties
|
|
void tst_qqmllanguage::listProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("listProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("test").toInt(), 2);
|
|
}
|
|
|
|
// Tests that initializing list properties of a base class does not crash
|
|
// (QTBUG-82171)
|
|
void tst_qqmllanguage::listPropertiesInheritanceNoCrash()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("listPropertiesChild.qml"));
|
|
QScopedPointer<QObject> object(component.create()); // should not crash
|
|
QVERIFY(object != nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::badListItemType()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("badListItemType.qml"));
|
|
QVERIFY(component.isError());
|
|
VERIFY_ERRORS("badListItemType.errors.txt");
|
|
}
|
|
|
|
// Tests the creation and assignment of dynamic object properties
|
|
// ### Not complete
|
|
void tst_qqmllanguage::dynamicObjectProperties()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("dynamicObjectProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("objectProperty"), QVariant::fromValue((QObject*)nullptr));
|
|
QVERIFY(object->property("objectProperty2") != QVariant::fromValue((QObject*)nullptr));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("dynamicObjectProperties.2.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QVERIFY(object->property("objectProperty") != QVariant::fromValue((QObject*)nullptr));
|
|
}
|
|
}
|
|
|
|
// Tests the declaration of dynamic signals and slots
|
|
void tst_qqmllanguage::dynamicSignalsAndSlots()
|
|
{
|
|
QTest::ignoreMessage(QtDebugMsg, "1921");
|
|
|
|
QQmlComponent component(&engine, testFileUrl("dynamicSignalsAndSlots.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(object->metaObject()->indexOfMethod("signal1()") != -1);
|
|
QVERIFY(object->metaObject()->indexOfMethod("signal2()") != -1);
|
|
QVERIFY(object->metaObject()->indexOfMethod("slot1()") != -1);
|
|
QVERIFY(object->metaObject()->indexOfMethod("slot2()") != -1);
|
|
|
|
QCOMPARE(object->property("test").toInt(), 0);
|
|
QMetaObject::invokeMethod(object.data(), "slot3", Qt::DirectConnection, Q_ARG(QVariant, QVariant(10)));
|
|
QCOMPARE(object->property("test").toInt(), 10);
|
|
}
|
|
|
|
void tst_qqmllanguage::simpleBindings()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("simpleBindings.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("value1"), QVariant(10));
|
|
QCOMPARE(object->property("value2"), QVariant(10));
|
|
QCOMPARE(object->property("value3"), QVariant(21));
|
|
QCOMPARE(object->property("value4"), QVariant(10));
|
|
QCOMPARE(object->property("objectProperty"), QVariant::fromValue(object.data()));
|
|
}
|
|
|
|
class EvaluationCounter : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
int counter = 0;
|
|
Q_INVOKABLE void increaseEvaluationCounter() { ++counter; }
|
|
};
|
|
|
|
void tst_qqmllanguage::noDoubleEvaluationForFlushedBindings_data()
|
|
{
|
|
QTest::addColumn<QString>("fileName");
|
|
QTest::newRow("order1") << QString("noDoubleEvaluationForFlushedBindings.qml");
|
|
QTest::newRow("order2") << QString("noDoubleEvaluationForFlushedBindings.2.qml");
|
|
}
|
|
|
|
void tst_qqmllanguage::noDoubleEvaluationForFlushedBindings()
|
|
{
|
|
QFETCH(QString, fileName);
|
|
QQmlEngine engine;
|
|
|
|
EvaluationCounter stats;
|
|
engine.rootContext()->setContextProperty("stats", &stats);
|
|
|
|
QQmlComponent component(&engine, testFileUrl(fileName));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(!obj.isNull());
|
|
|
|
QCOMPARE(stats.counter, 1);
|
|
}
|
|
|
|
void tst_qqmllanguage::autoComponentCreation()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("autoComponentCreation.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(object->componentProperty() != nullptr);
|
|
QScopedPointer<MyTypeObject> child(qobject_cast<MyTypeObject *>(object->componentProperty()->create()));
|
|
QVERIFY(child != nullptr);
|
|
QCOMPARE(child->realProperty(), qreal(9));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("autoComponentCreation.2.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(object->componentProperty() != nullptr);
|
|
QScopedPointer<MyTypeObject> child(qobject_cast<MyTypeObject *>(object->componentProperty()->create()));
|
|
QVERIFY(child != nullptr);
|
|
QCOMPARE(child->realProperty(), qreal(9));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::autoComponentCreationInGroupProperty()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("autoComponentCreationInGroupProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(object->componentProperty() != nullptr);
|
|
QScopedPointer<MyTypeObject> child(qobject_cast<MyTypeObject *>(object->componentProperty()->create()));
|
|
QVERIFY(child != nullptr);
|
|
QCOMPARE(child->realProperty(), qreal(9));
|
|
}
|
|
|
|
void tst_qqmllanguage::propertyValueSource()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("propertyValueSource.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
|
|
QList<QObject *> valueSources;
|
|
QObjectList allChildren = object->findChildren<QObject*>();
|
|
foreach (QObject *child, allChildren) {
|
|
if (qobject_cast<QQmlPropertyValueSource *>(child))
|
|
valueSources.append(child);
|
|
}
|
|
|
|
QCOMPARE(valueSources.size(), 1);
|
|
MyPropertyValueSource *valueSource =
|
|
qobject_cast<MyPropertyValueSource *>(valueSources.at(0));
|
|
QVERIFY(valueSource != nullptr);
|
|
QCOMPARE(valueSource->prop.object(), qobject_cast<QObject*>(object.data()));
|
|
QCOMPARE(valueSource->prop.name(), QString(QLatin1String("intProperty")));
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("propertyValueSource.2.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
|
|
QList<QObject *> valueSources;
|
|
QObjectList allChildren = object->findChildren<QObject*>();
|
|
foreach (QObject *child, allChildren) {
|
|
if (qobject_cast<QQmlPropertyValueSource *>(child))
|
|
valueSources.append(child);
|
|
}
|
|
|
|
QCOMPARE(valueSources.size(), 1);
|
|
MyPropertyValueSource *valueSource =
|
|
qobject_cast<MyPropertyValueSource *>(valueSources.at(0));
|
|
QVERIFY(valueSource != nullptr);
|
|
QCOMPARE(valueSource->prop.object(), qobject_cast<QObject*>(object.data()));
|
|
QCOMPARE(valueSource->prop.name(), QString(QLatin1String("intProperty")));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::requiredProperty()
|
|
{
|
|
QQmlEngine engine;
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("requiredProperties.1.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object);
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("requiredProperties.2.qml"));
|
|
QVERIFY(!component.errors().empty());
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("requiredProperties.4.qml"));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!component.errors().empty());
|
|
QVERIFY(component.errorString().contains("Required property objectName was not initialized"));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("requiredProperties.3.qml"));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!component.errors().empty());
|
|
QVERIFY(component.errorString().contains("Required property i was not initialized"));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("requiredProperties.5.qml"));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!component.errors().empty());
|
|
QVERIFY(component.errorString().contains("Required property i was not initialized"));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("requiredProperties.6.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object);
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("requiredProperties.7.qml"));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!component.errors().empty());
|
|
QVERIFY(component.errorString().contains("Property blub was marked as required but does not exist"));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("RequiredListPropertiesUser.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object);
|
|
}
|
|
}
|
|
|
|
class MyClassWithRequiredProperty : public QObject
|
|
{
|
|
public:
|
|
Q_OBJECT
|
|
Q_PROPERTY(int test MEMBER m_test REQUIRED NOTIFY testChanged)
|
|
Q_SIGNAL void testChanged();
|
|
private:
|
|
int m_test;
|
|
};
|
|
|
|
class ChildClassWithoutOwnRequired : public MyClassWithRequiredProperty
|
|
{
|
|
public:
|
|
Q_OBJECT
|
|
Q_PROPERTY(int test2 MEMBER m_test2 NOTIFY test2Changed)
|
|
Q_SIGNAL void test2Changed();
|
|
private:
|
|
int m_test2;
|
|
};
|
|
|
|
class ChildClassWithOwnRequired : public MyClassWithRequiredProperty
|
|
{
|
|
public:
|
|
Q_OBJECT
|
|
Q_PROPERTY(int test2 MEMBER m_test2 REQUIRED NOTIFY test2Changed)
|
|
Q_SIGNAL void test2Changed();
|
|
private:
|
|
int m_test2;
|
|
};
|
|
|
|
void tst_qqmllanguage::requiredPropertyFromCpp_data()
|
|
{
|
|
qmlRegisterType<MyClassWithRequiredProperty>("example.org", 1, 0, "MyClass");
|
|
qmlRegisterType<ChildClassWithoutOwnRequired>("example.org", 1, 0, "Child");
|
|
qmlRegisterType<ChildClassWithOwnRequired>("example.org", 1, 0, "Child2");
|
|
|
|
|
|
QTest::addColumn<QUrl>("setFile");
|
|
QTest::addColumn<QUrl>("notSetFile");
|
|
QTest::addColumn<QString>("errorMessage");
|
|
QTest::addColumn<int>("expectedValue");
|
|
|
|
QTest::addRow("direct") << testFileUrl("cppRequiredProperty.qml") << testFileUrl("cppRequiredPropertyNotSet.qml") << QString(":4 Required property test was not initialized\n") << 42;
|
|
QTest::addRow("in parent") << testFileUrl("cppRequiredPropertyInParent.qml") << testFileUrl("cppRequiredPropertyInParentNotSet.qml") << QString(":4 Required property test was not initialized\n") << 42;
|
|
QTest::addRow("in child and parent") << testFileUrl("cppRequiredPropertyInChildAndParent.qml") << testFileUrl("cppRequiredPropertyInChildAndParentNotSet.qml") << QString(":4 Required property test2 was not initialized\n") << 18;
|
|
}
|
|
|
|
void tst_qqmllanguage::requiredPropertyFromCpp()
|
|
{
|
|
QQmlEngine engine;
|
|
QFETCH(QUrl, setFile);
|
|
QFETCH(QUrl, notSetFile);
|
|
QFETCH(QString, errorMessage);
|
|
QFETCH(int, expectedValue);
|
|
{
|
|
QQmlComponent comp(&engine, notSetFile);
|
|
QScopedPointer<QObject> o { comp.create() };
|
|
QVERIFY(o.isNull());
|
|
QVERIFY(comp.isError());
|
|
QCOMPARE(comp.errorString(), notSetFile.toString() + errorMessage);
|
|
}
|
|
{
|
|
QQmlComponent comp(&engine, setFile);
|
|
QScopedPointer<QObject> o { comp.create() };
|
|
QVERIFY(!o.isNull());
|
|
QCOMPARE(o->property("test").toInt(), expectedValue);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::attachedProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("attachedProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QObject *attached = qmlAttachedPropertiesObject<MyQmlObject>(object.data());
|
|
QVERIFY(attached != nullptr);
|
|
QCOMPARE(attached->property("value"), QVariant(10));
|
|
QCOMPARE(attached->property("value2"), QVariant(13));
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("attachedPropertyDerived.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(MyQmlObjectWithAttachedCounter::attachedCount, 1);
|
|
}
|
|
}
|
|
|
|
// Tests non-static object properties
|
|
void tst_qqmllanguage::dynamicObjects()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("dynamicObject.1.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::valueTypes()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("valueTypes.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QString message = component.url().toString() + ":2:1: QML MyTypeObject: Binding loop detected for property \"rectProperty.width\"";
|
|
QTest::ignoreMessage(QtWarningMsg, qPrintable(message));
|
|
QTest::ignoreMessage(QtWarningMsg, qPrintable(message));
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
|
|
|
|
QCOMPARE(object->rectProperty(), QRect(10, 11, 12, 13));
|
|
QCOMPARE(object->rectProperty2(), QRect(10, 11, 12, 13));
|
|
QCOMPARE(object->intProperty(), 10);
|
|
object->doAction();
|
|
QCOMPARE(object->rectProperty(), QRect(12, 11, 14, 13));
|
|
QCOMPARE(object->rectProperty2(), QRect(12, 11, 14, 13));
|
|
QCOMPARE(object->intProperty(), 12);
|
|
|
|
// ###
|
|
#if 0
|
|
QQmlProperty p(object, "rectProperty.x");
|
|
QCOMPARE(p.read(), QVariant(12));
|
|
p.write(13);
|
|
QCOMPARE(p.read(), QVariant(13));
|
|
|
|
quint32 r = QQmlPropertyPrivate::saveValueType(p.coreIndex(), p.valueTypeCoreIndex());
|
|
QQmlProperty p2;
|
|
QQmlPropertyPrivate::restore(p2, r, object);
|
|
QCOMPARE(p2.read(), QVariant(13));
|
|
#endif
|
|
}
|
|
|
|
void tst_qqmllanguage::cppnamespace()
|
|
{
|
|
QScopedPointer<QObject> object;
|
|
|
|
auto create = [&](const char *file) {
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
VERIFY_ERRORS(0);
|
|
object.reset(component.create());
|
|
QVERIFY(object != nullptr);
|
|
};
|
|
|
|
auto createAndCheck = [&](const char *file) {
|
|
create(file);
|
|
return !QTest::currentTestFailed();
|
|
};
|
|
|
|
QVERIFY(createAndCheck("cppnamespace.qml"));
|
|
QCOMPARE(object->property("intProperty").toInt(),
|
|
(int)MyNamespace::MyOtherNSEnum::OtherKey2);
|
|
|
|
QVERIFY(createAndCheck("cppstaticnamespace.qml"));
|
|
QCOMPARE(object->property("intProperty").toInt(),
|
|
(int)MyStaticNamespace::MyOtherNSEnum::OtherKey2);
|
|
|
|
QVERIFY(createAndCheck("cppnamespace.2.qml"));
|
|
QVERIFY(createAndCheck("cppstaticnamespace.2.qml"));
|
|
}
|
|
|
|
void tst_qqmllanguage::aliasProperties()
|
|
{
|
|
// Simple "int" alias
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.1.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
// Read through alias
|
|
QCOMPARE(object->property("valueAlias").toInt(), 10);
|
|
object->setProperty("value", QVariant(13));
|
|
QCOMPARE(object->property("valueAlias").toInt(), 13);
|
|
|
|
// Write through alias
|
|
object->setProperty("valueAlias", QVariant(19));
|
|
QCOMPARE(object->property("valueAlias").toInt(), 19);
|
|
QCOMPARE(object->property("value").toInt(), 19);
|
|
}
|
|
|
|
// Complex object alias
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.2.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
// Read through alias
|
|
MyQmlObject *v =
|
|
qvariant_cast<MyQmlObject *>(object->property("aliasObject"));
|
|
QVERIFY(v != nullptr);
|
|
QCOMPARE(v->value(), 10);
|
|
|
|
// Write through alias
|
|
MyQmlObject *v2 = new MyQmlObject();
|
|
v2->setParent(object.data());
|
|
object->setProperty("aliasObject", QVariant::fromValue(v2));
|
|
MyQmlObject *v3 =
|
|
qvariant_cast<MyQmlObject *>(object->property("aliasObject"));
|
|
QVERIFY(v3 != nullptr);
|
|
QCOMPARE(v3, v2);
|
|
}
|
|
|
|
// Nested aliases
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.3.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("value").toInt(), 1892);
|
|
QCOMPARE(object->property("value2").toInt(), 1892);
|
|
|
|
object->setProperty("value", QVariant(1313));
|
|
QCOMPARE(object->property("value").toInt(), 1313);
|
|
QCOMPARE(object->property("value2").toInt(), 1313);
|
|
|
|
object->setProperty("value2", QVariant(8080));
|
|
QCOMPARE(object->property("value").toInt(), 8080);
|
|
QCOMPARE(object->property("value2").toInt(), 8080);
|
|
}
|
|
|
|
// Enum aliases
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.4.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("enumAlias").toInt(), 1);
|
|
}
|
|
|
|
// Id aliases
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.5.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QVariant v = object->property("otherAlias");
|
|
QCOMPARE(v.typeId(), qMetaTypeId<MyQmlObject *>());
|
|
MyQmlObject *o = qvariant_cast<MyQmlObject*>(v);
|
|
QCOMPARE(o->value(), 10);
|
|
|
|
delete o;
|
|
|
|
v = object->property("otherAlias");
|
|
QCOMPARE(v.typeId(), qMetaTypeId<MyQmlObject *>());
|
|
o = qvariant_cast<MyQmlObject*>(v);
|
|
QVERIFY(!o);
|
|
}
|
|
|
|
// Nested aliases - this used to cause a crash
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.6.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("a").toInt(), 1923);
|
|
}
|
|
|
|
// Ptr Alias Cleanup - check that aliases to ptr types return 0
|
|
// if the object aliased to is removed
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.7.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QObject *object1 = qvariant_cast<QObject *>(object->property("object"));
|
|
QVERIFY(object1 != nullptr);
|
|
QObject *object2 = qvariant_cast<QObject *>(object1->property("object"));
|
|
QVERIFY(object2 != nullptr);
|
|
|
|
QObject *alias = qvariant_cast<QObject *>(object->property("aliasedObject"));
|
|
QCOMPARE(alias, object2);
|
|
|
|
delete object1;
|
|
|
|
QObject *alias2 = object.data(); // "Random" start value
|
|
int status = -1;
|
|
void *a[] = { &alias2, nullptr, &status };
|
|
QMetaObject::metacall(object.data(), QMetaObject::ReadProperty,
|
|
object->metaObject()->indexOfProperty("aliasedObject"), a);
|
|
QVERIFY(!alias2);
|
|
}
|
|
|
|
// Simple composite type
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.8.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("value").toInt(), 10);
|
|
}
|
|
|
|
// Complex composite type
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.9.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("value").toInt(), 10);
|
|
}
|
|
|
|
// Valuetype alias
|
|
// Simple "int" alias
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.10.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
// Read through alias
|
|
QCOMPARE(object->property("valueAlias").toRect(), QRect(10, 11, 9, 8));
|
|
object->setProperty("rectProperty", QVariant(QRect(33, 12, 99, 100)));
|
|
QCOMPARE(object->property("valueAlias").toRect(), QRect(33, 12, 99, 100));
|
|
|
|
// Write through alias
|
|
object->setProperty("valueAlias", QVariant(QRect(3, 3, 4, 9)));
|
|
QCOMPARE(object->property("valueAlias").toRect(), QRect(3, 3, 4, 9));
|
|
QCOMPARE(object->property("rectProperty").toRect(), QRect(3, 3, 4, 9));
|
|
}
|
|
|
|
// Valuetype sub-alias
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.11.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
// Read through alias
|
|
QCOMPARE(object->property("aliasProperty").toInt(), 19);
|
|
object->setProperty("rectProperty", QVariant(QRect(33, 8, 102, 111)));
|
|
QCOMPARE(object->property("aliasProperty").toInt(), 33);
|
|
|
|
// Write through alias
|
|
object->setProperty("aliasProperty", QVariant(4));
|
|
QCOMPARE(object->property("aliasProperty").toInt(), 4);
|
|
QCOMPARE(object->property("rectProperty").toRect(), QRect(4, 8, 102, 111));
|
|
}
|
|
|
|
// Nested aliases with a qml file
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.12.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QPointer<QObject> subObject = qvariant_cast<QObject*>(object->property("referencingSubObject"));
|
|
QVERIFY(!subObject.isNull());
|
|
|
|
QVERIFY(subObject->property("success").toBool());
|
|
}
|
|
|
|
// Nested aliases with a qml file with reverse ordering
|
|
{
|
|
// This is known to fail at the moment.
|
|
QQmlComponent component(&engine, testFileUrl("alias.13.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QPointer<QObject> subObject = qvariant_cast<QObject*>(object->property("referencingSubObject"));
|
|
QVERIFY(!subObject.isNull());
|
|
|
|
QVERIFY(subObject->property("success").toBool());
|
|
}
|
|
|
|
// "Nested" aliases within an object that require iterative resolution
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.14.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QPointer<QObject> subObject = qvariant_cast<QObject*>(object->property("referencingSubObject"));
|
|
QVERIFY(!subObject.isNull());
|
|
|
|
QVERIFY(subObject->property("success").toBool());
|
|
}
|
|
|
|
// Property bindings on group properties that are actually aliases (QTBUG-51043)
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.15.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QPointer<QObject> subItem = qvariant_cast<QObject*>(object->property("symbol"));
|
|
QVERIFY(!subItem.isNull());
|
|
|
|
QCOMPARE(subItem->property("y").toInt(), 1);
|
|
}
|
|
|
|
// Nested property bindings on group properties that are actually aliases (QTBUG-94983)
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.15a.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QPointer<QObject> subItem = qvariant_cast<QObject*>(object->property("symbol"));
|
|
QVERIFY(!subItem.isNull());
|
|
|
|
QPointer<QObject> subSubItem = qvariant_cast<QObject*>(subItem->property("layer"));
|
|
|
|
QCOMPARE(subSubItem->property("enabled").value<bool>(), true);
|
|
}
|
|
|
|
// Alias to sub-object with binding (QTBUG-57041)
|
|
{
|
|
// This is shold *not* crash.
|
|
QQmlComponent component(&engine, testFileUrl("alias.16.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
}
|
|
|
|
// Alias to grouped property
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.17.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
QVERIFY(object->property("success").toBool());
|
|
}
|
|
|
|
// Alias to grouped property updates
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.17.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
QObject *aliasUser = object->findChild<QObject*>(QLatin1String("aliasUser"));
|
|
QVERIFY(aliasUser);
|
|
QQmlProperty checkValueProp(object.get(), "checkValue");
|
|
QVERIFY(checkValueProp.isValid());
|
|
checkValueProp.write(777);
|
|
QCOMPARE(object->property("checkValue").toInt(), 777);
|
|
QCOMPARE(aliasUser->property("topMargin").toInt(), 777);
|
|
}
|
|
|
|
// Write to alias to grouped property
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.17.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
QObject *aliasUser = object->findChild<QObject*>(QLatin1String("aliasUser"));
|
|
QVERIFY(aliasUser);
|
|
QQmlProperty topMarginProp {aliasUser, "topMargin"};
|
|
QVERIFY(topMarginProp.isValid());
|
|
topMarginProp.write(777);
|
|
QObject *myItem = object->findChild<QObject*>(QLatin1String("myItem"));
|
|
QVERIFY(myItem);
|
|
auto anchors = myItem->property("anchors").value<QObject*>();
|
|
QVERIFY(anchors);
|
|
QCOMPARE(anchors->property("topMargin").toInt(), 777);
|
|
}
|
|
|
|
// Binding to alias to grouped property gets updated
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.17.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
QObject *aliasUser = object->findChild<QObject*>(QLatin1String("aliasUser"));
|
|
QVERIFY(aliasUser);
|
|
QQmlProperty topMarginProp {aliasUser, "topMargin"};
|
|
QVERIFY(topMarginProp.isValid());
|
|
topMarginProp.write(20);
|
|
QObject *myText = object->findChild<QObject*>(QLatin1String("myText"));
|
|
QVERIFY(myText);
|
|
auto text = myText->property("text").toString();
|
|
QCOMPARE(text, "alias:\n20");
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.18.qml"));
|
|
VERIFY_ERRORS("alias.18.errors.txt");
|
|
}
|
|
|
|
// Binding on deep alias
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("alias.19.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
QCOMPARE(object->property("height").toInt(), 960);
|
|
}
|
|
}
|
|
|
|
// QTBUG-13374 Test that alias properties and signals can coexist
|
|
void tst_qqmllanguage::aliasPropertiesAndSignals()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("aliasPropertiesAndSignals.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o);
|
|
QCOMPARE(o->property("test").toBool(), true);
|
|
}
|
|
|
|
void tst_qqmllanguage::qtbug_89822()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("qtbug_89822.qml"));
|
|
VERIFY_ERRORS("qtbug_89822.errors.txt");
|
|
}
|
|
|
|
// Test that the root element in a composite type can be a Component
|
|
void tst_qqmllanguage::componentCompositeType()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("componentCompositeType.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
}
|
|
|
|
class TestType : public QObject {
|
|
Q_OBJECT
|
|
public:
|
|
TestType(QObject *p=nullptr) : QObject(p) {}
|
|
};
|
|
|
|
class TestType2 : public QObject {
|
|
Q_OBJECT
|
|
public:
|
|
TestType2(QObject *p=nullptr) : QObject(p) {}
|
|
};
|
|
|
|
void tst_qqmllanguage::i18n_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::addColumn<QString>("stringProperty");
|
|
QTest::newRow("i18nStrings") << "i18nStrings.qml" << QString::fromUtf8("Test \303\241\303\242\303\243\303\244\303\245 (5 accented 'a' letters)");
|
|
QTest::newRow("i18nDeclaredPropertyNames") << "i18nDeclaredPropertyNames.qml" << QString::fromUtf8("Test \303\241\303\242\303\243\303\244\303\245: 10");
|
|
QTest::newRow("i18nDeclaredPropertyUse") << "i18nDeclaredPropertyUse.qml" << QString::fromUtf8("Test \303\241\303\242\303\243\303\244\303\245: 15");
|
|
QTest::newRow("i18nScript") << "i18nScript.qml" << QString::fromUtf8("Test \303\241\303\242\303\243\303\244\303\245: 20");
|
|
QTest::newRow("i18nType") << "i18nType.qml" << QString::fromUtf8("Test \303\241\303\242\303\243\303\244\303\245: 30");
|
|
QTest::newRow("i18nNameSpace") << "i18nNameSpace.qml" << QString::fromUtf8("Test \303\241\303\242\303\243\303\244\303\245: 40");
|
|
}
|
|
|
|
void tst_qqmllanguage::i18n()
|
|
{
|
|
QFETCH(QString, file);
|
|
QFETCH(QString, stringProperty);
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->stringProperty(), stringProperty);
|
|
}
|
|
|
|
// Check that the Component::onCompleted attached property works
|
|
void tst_qqmllanguage::onCompleted()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("onCompleted.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QTest::ignoreMessage(QtDebugMsg, "Completed 6 10");
|
|
QTest::ignoreMessage(QtDebugMsg, "Completed 6 10");
|
|
QTest::ignoreMessage(QtDebugMsg, "Completed 10 11");
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
}
|
|
|
|
// Check that the Component::onDestruction attached property works
|
|
void tst_qqmllanguage::onDestruction()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("onDestruction.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QTest::ignoreMessage(QtDebugMsg, "Destruction 6 10");
|
|
QTest::ignoreMessage(QtDebugMsg, "Destruction 6 10");
|
|
QTest::ignoreMessage(QtDebugMsg, "Destruction 10 11");
|
|
}
|
|
|
|
// Check that assignments to QQmlScriptString properties work
|
|
void tst_qqmllanguage::scriptString()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptString.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QCOMPARE(object->scriptProperty().stringLiteral(), QString());
|
|
bool ok;
|
|
QCOMPARE(object->scriptProperty().numberLiteral(&ok), qreal(0.));
|
|
QCOMPARE(ok, false);
|
|
|
|
const QQmlScriptStringPrivate *scriptPrivate = QQmlScriptStringPrivate::get(object->scriptProperty());
|
|
QVERIFY(scriptPrivate != nullptr);
|
|
QCOMPARE(scriptPrivate->script, QString("foo + bar"));
|
|
QCOMPARE(scriptPrivate->scope, qobject_cast<QObject*>(object.data()));
|
|
QCOMPARE(scriptPrivate->context, qmlContext(object.data()));
|
|
|
|
QVERIFY(object->grouped() != nullptr);
|
|
const QQmlScriptStringPrivate *groupedPrivate = QQmlScriptStringPrivate::get(object->grouped()->script());
|
|
QCOMPARE(groupedPrivate->script, QString("console.log(1921)"));
|
|
QCOMPARE(groupedPrivate->scope, qobject_cast<QObject*>(object.data()));
|
|
QCOMPARE(groupedPrivate->context, qmlContext(object.data()));
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptString2.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->scriptProperty().stringLiteral(), QString("hello\\n\\\"world\\\""));
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptString3.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
bool ok;
|
|
QCOMPARE(object->scriptProperty().numberLiteral(&ok), qreal(12.345));
|
|
QCOMPARE(ok, true);
|
|
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptString4.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
bool ok;
|
|
QCOMPARE(object->scriptProperty().booleanLiteral(&ok), true);
|
|
QCOMPARE(ok, true);
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptString5.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->scriptProperty().isNullLiteral(), true);
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptString6.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->scriptProperty().isUndefinedLiteral(), true);
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptString7.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QQmlScriptString ss = object->scriptProperty();
|
|
|
|
{
|
|
QQmlExpression expr(ss, /*context*/nullptr, object.data());
|
|
QCOMPARE(expr.evaluate().toInt(), int(100));
|
|
}
|
|
|
|
{
|
|
SimpleObjectWithCustomParser testScope;
|
|
QVERIFY(testScope.metaObject()->indexOfProperty("intProperty") != object->metaObject()->indexOfProperty("intProperty"));
|
|
|
|
testScope.setIntProperty(42);
|
|
QQmlExpression expr(ss, /*context*/nullptr, &testScope);
|
|
QCOMPARE(expr.evaluate().toInt(), int(42));
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check that assignments to QQmlScriptString properties works also from within Javascript
|
|
void tst_qqmllanguage::scriptStringJs()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scriptStringJs.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QQmlContext *context = QQmlEngine::contextForObject(object.data());
|
|
QVERIFY(context != nullptr);
|
|
bool ok;
|
|
|
|
QCOMPARE(QQmlScriptStringPrivate::get(object->scriptProperty())->script, QString("\" hello \\\" world \""));
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QVERIFY(!object->scriptProperty().isUndefinedLiteral());
|
|
QVERIFY(!object->scriptProperty().isNullLiteral());
|
|
QCOMPARE(object->scriptProperty().stringLiteral(), QString(" hello \\\" world "));
|
|
QVERIFY(object->scriptProperty().numberLiteral(&ok) == 0.0 && !ok);
|
|
QVERIFY(!object->scriptProperty().booleanLiteral(&ok) && !ok);
|
|
|
|
QJSValue inst = engine.newQObject(object.data());
|
|
QJSValue func = engine.evaluate("(function(value) { this.scriptProperty = value })");
|
|
|
|
func.callWithInstance(inst, QJSValueList() << "test a \"string ");
|
|
QCOMPARE(QQmlScriptStringPrivate::get(object->scriptProperty())->script, QString("\"test a \\\"string \""));
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QVERIFY(!object->scriptProperty().isUndefinedLiteral());
|
|
QVERIFY(!object->scriptProperty().isNullLiteral());
|
|
QCOMPARE(object->scriptProperty().stringLiteral(), QString("test a \\\"string "));
|
|
QVERIFY(object->scriptProperty().numberLiteral(&ok) == 0.0 && !ok);
|
|
QVERIFY(!object->scriptProperty().booleanLiteral(&ok) && !ok);
|
|
|
|
func.callWithInstance(inst, QJSValueList() << QJSValue::UndefinedValue);
|
|
QCOMPARE(QQmlScriptStringPrivate::get(object->scriptProperty())->script, QString("undefined"));
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QVERIFY(object->scriptProperty().isUndefinedLiteral());
|
|
QVERIFY(!object->scriptProperty().isNullLiteral());
|
|
QVERIFY(object->scriptProperty().stringLiteral().isEmpty());
|
|
QVERIFY(object->scriptProperty().numberLiteral(&ok) == 0.0 && !ok);
|
|
QVERIFY(!object->scriptProperty().booleanLiteral(&ok) && !ok);
|
|
|
|
func.callWithInstance(inst, QJSValueList() << true);
|
|
QCOMPARE(QQmlScriptStringPrivate::get(object->scriptProperty())->script, QString("true"));
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QVERIFY(!object->scriptProperty().isUndefinedLiteral());
|
|
QVERIFY(!object->scriptProperty().isNullLiteral());
|
|
QVERIFY(object->scriptProperty().stringLiteral().isEmpty());
|
|
QVERIFY(object->scriptProperty().numberLiteral(&ok) == 0.0 && !ok);
|
|
QVERIFY(object->scriptProperty().booleanLiteral(&ok) && ok);
|
|
|
|
func.callWithInstance(inst, QJSValueList() << false);
|
|
QCOMPARE(QQmlScriptStringPrivate::get(object->scriptProperty())->script, QString("false"));
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QVERIFY(!object->scriptProperty().isUndefinedLiteral());
|
|
QVERIFY(!object->scriptProperty().isNullLiteral());
|
|
QVERIFY(object->scriptProperty().stringLiteral().isEmpty());
|
|
QVERIFY(object->scriptProperty().numberLiteral(&ok) == 0.0 && !ok);
|
|
QVERIFY(!object->scriptProperty().booleanLiteral(&ok) && ok);
|
|
|
|
func.callWithInstance(inst, QJSValueList() << QJSValue::NullValue);
|
|
QCOMPARE(QQmlScriptStringPrivate::get(object->scriptProperty())->script, QString("null"));
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QVERIFY(!object->scriptProperty().isUndefinedLiteral());
|
|
QVERIFY(object->scriptProperty().isNullLiteral());
|
|
QVERIFY(object->scriptProperty().stringLiteral().isEmpty());
|
|
QVERIFY(object->scriptProperty().numberLiteral(&ok) == 0.0 && !ok);
|
|
QVERIFY(!object->scriptProperty().booleanLiteral(&ok) && !ok);
|
|
|
|
func.callWithInstance(inst, QJSValueList() << 12.34);
|
|
QCOMPARE(QQmlScriptStringPrivate::get(object->scriptProperty())->script, QString("12.34"));
|
|
QVERIFY(!object->scriptProperty().isEmpty());
|
|
QVERIFY(!object->scriptProperty().isUndefinedLiteral());
|
|
QVERIFY(!object->scriptProperty().isNullLiteral());
|
|
QVERIFY(object->scriptProperty().stringLiteral().isEmpty());
|
|
QVERIFY(object->scriptProperty().numberLiteral(&ok) == 12.34 && ok);
|
|
QVERIFY(!object->scriptProperty().booleanLiteral(&ok) && !ok);
|
|
}
|
|
|
|
struct FreeUnitData
|
|
{
|
|
static void cleanup(const QV4::CompiledData::Unit *readOnlyQmlUnit)
|
|
{
|
|
if (readOnlyQmlUnit && !(readOnlyQmlUnit->flags & QV4::CompiledData::Unit::StaticData))
|
|
free(const_cast<QV4::CompiledData::Unit *>(readOnlyQmlUnit));
|
|
}
|
|
};
|
|
|
|
void tst_qqmllanguage::scriptStringWithoutSourceCode()
|
|
{
|
|
QUrl url = testFileUrl("scriptString7.qml");
|
|
QScopedPointer<const QV4::CompiledData::Unit, FreeUnitData> readOnlyQmlUnit;
|
|
{
|
|
QQmlEnginePrivate *eng = QQmlEnginePrivate::get(&engine);
|
|
QQmlRefPointer<QQmlTypeData> td = eng->typeLoader.getType(url);
|
|
Q_ASSERT(td);
|
|
QVERIFY(!td->backupSourceCode().isValid());
|
|
|
|
QQmlRefPointer<QV4::ExecutableCompilationUnit> compilationUnit = td->compilationUnit();
|
|
readOnlyQmlUnit.reset(compilationUnit->unitData());
|
|
Q_ASSERT(readOnlyQmlUnit);
|
|
QV4::CompiledData::Unit *qmlUnit = reinterpret_cast<QV4::CompiledData::Unit *>(malloc(readOnlyQmlUnit->unitSize));
|
|
memcpy(qmlUnit, readOnlyQmlUnit.data(), readOnlyQmlUnit->unitSize);
|
|
|
|
qmlUnit->flags &= ~QV4::CompiledData::Unit::StaticData;
|
|
compilationUnit->setUnitData(qmlUnit);
|
|
|
|
const QV4::CompiledData::Object *rootObject = compilationUnit->objectAt(/*root object*/0);
|
|
QCOMPARE(compilationUnit->stringAt(rootObject->inheritedTypeNameIndex), QString("MyTypeObject"));
|
|
quint32 i;
|
|
for (i = 0; i < rootObject->nBindings; ++i) {
|
|
const QV4::CompiledData::Binding *binding = rootObject->bindingTable() + i;
|
|
if (compilationUnit->stringAt(binding->propertyNameIndex) != QString("scriptProperty"))
|
|
continue;
|
|
QCOMPARE(compilationUnit->bindingValueAsScriptString(binding), QString("intProperty"));
|
|
const_cast<QV4::CompiledData::Binding*>(binding)->stringIndex = 0; // empty string index
|
|
QVERIFY(compilationUnit->bindingValueAsScriptString(binding).isEmpty());
|
|
break;
|
|
}
|
|
QVERIFY(i < rootObject->nBindings);
|
|
}
|
|
QQmlComponent component(&engine, url);
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyTypeObject> object(qobject_cast<MyTypeObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QQmlScriptString ss = object->scriptProperty();
|
|
QVERIFY(!ss.isEmpty());
|
|
QCOMPARE(ss.stringLiteral(), QString());
|
|
bool ok;
|
|
QCOMPARE(ss.numberLiteral(&ok), qreal(0.));
|
|
QCOMPARE(ok, false);
|
|
|
|
const QQmlScriptStringPrivate *scriptPrivate = QQmlScriptStringPrivate::get(ss);
|
|
QVERIFY(scriptPrivate != nullptr);
|
|
QVERIFY(scriptPrivate->script.isEmpty());
|
|
QCOMPARE(scriptPrivate->scope, qobject_cast<QObject*>(object.data()));
|
|
QCOMPARE(scriptPrivate->context, qmlContext(object.data()));
|
|
|
|
{
|
|
QQmlExpression expr(ss, /*context*/nullptr, object.data());
|
|
QCOMPARE(expr.evaluate().toInt(), int(100));
|
|
}
|
|
}
|
|
|
|
// Test the QQmlScriptString comparison operators. The script strings are considered
|
|
// equal if there evaluation would produce the same result.
|
|
void tst_qqmllanguage::scriptStringComparison()
|
|
{
|
|
QQmlComponent component1(&engine, testFileUrl("scriptString.qml"));
|
|
QVERIFY(!component1.isError() && component1.errors().isEmpty());
|
|
QScopedPointer<MyTypeObject> object1(qobject_cast<MyTypeObject*>(component1.create()));
|
|
QVERIFY(object1 != nullptr);
|
|
|
|
QQmlComponent component2(&engine, testFileUrl("scriptString2.qml"));
|
|
QVERIFY(!component2.isError() && component2.errors().isEmpty());
|
|
QScopedPointer<MyTypeObject> object2(qobject_cast<MyTypeObject*>(component2.create()));
|
|
QVERIFY(object2 != nullptr);
|
|
|
|
QQmlComponent component3(&engine, testFileUrl("scriptString3.qml"));
|
|
QVERIFY(!component3.isError() && component3.errors().isEmpty());
|
|
QScopedPointer<MyTypeObject> object3(qobject_cast<MyTypeObject*>(component3.create()));
|
|
QVERIFY(object3 != nullptr);
|
|
|
|
//QJSValue inst1 = engine.newQObject(object1);
|
|
QJSValue inst2 = engine.newQObject(object2.data());
|
|
QJSValue inst3 = engine.newQObject(object3.data());
|
|
QJSValue func = engine.evaluate("(function(value) { this.scriptProperty = value })");
|
|
|
|
const QString s = "hello\\n\\\"world\\\"";
|
|
const qreal n = 12.345;
|
|
bool ok;
|
|
|
|
QCOMPARE(object2->scriptProperty().stringLiteral(), s);
|
|
QVERIFY(object3->scriptProperty().numberLiteral(&ok) == n && ok);
|
|
QCOMPARE(object1->scriptProperty(), object1->scriptProperty());
|
|
QCOMPARE(object2->scriptProperty(), object2->scriptProperty());
|
|
QCOMPARE(object3->scriptProperty(), object3->scriptProperty());
|
|
QVERIFY(object2->scriptProperty() != object3->scriptProperty());
|
|
QVERIFY(object1->scriptProperty() != object2->scriptProperty());
|
|
QVERIFY(object1->scriptProperty() != object3->scriptProperty());
|
|
|
|
func.callWithInstance(inst2, QJSValueList() << n);
|
|
QCOMPARE(object2->scriptProperty(), object3->scriptProperty());
|
|
|
|
func.callWithInstance(inst2, QJSValueList() << s);
|
|
QVERIFY(object2->scriptProperty() != object3->scriptProperty());
|
|
func.callWithInstance(inst3, QJSValueList() << s);
|
|
QCOMPARE(object2->scriptProperty(), object3->scriptProperty());
|
|
|
|
func.callWithInstance(inst2, QJSValueList() << QJSValue::UndefinedValue);
|
|
QVERIFY(object2->scriptProperty() != object3->scriptProperty());
|
|
func.callWithInstance(inst3, QJSValueList() << QJSValue::UndefinedValue);
|
|
QCOMPARE(object2->scriptProperty(), object3->scriptProperty());
|
|
|
|
func.callWithInstance(inst2, QJSValueList() << QJSValue::NullValue);
|
|
QVERIFY(object2->scriptProperty() != object3->scriptProperty());
|
|
func.callWithInstance(inst3, QJSValueList() << QJSValue::NullValue);
|
|
QCOMPARE(object2->scriptProperty(), object3->scriptProperty());
|
|
|
|
func.callWithInstance(inst2, QJSValueList() << false);
|
|
QVERIFY(object2->scriptProperty() != object3->scriptProperty());
|
|
func.callWithInstance(inst3, QJSValueList() << false);
|
|
QCOMPARE(object2->scriptProperty(), object3->scriptProperty());
|
|
|
|
func.callWithInstance(inst2, QJSValueList() << true);
|
|
QVERIFY(object2->scriptProperty() != object3->scriptProperty());
|
|
func.callWithInstance(inst3, QJSValueList() << true);
|
|
QCOMPARE(object2->scriptProperty(), object3->scriptProperty());
|
|
|
|
QVERIFY(object1->scriptProperty() != object2->scriptProperty());
|
|
object2->setScriptProperty(object1->scriptProperty());
|
|
QCOMPARE(object1->scriptProperty(), object2->scriptProperty());
|
|
|
|
QVERIFY(object1->scriptProperty() != object3->scriptProperty());
|
|
func.callWithInstance(inst3, QJSValueList() << engine.toScriptValue(object1->scriptProperty()));
|
|
QCOMPARE(object1->scriptProperty(), object3->scriptProperty());
|
|
|
|
// While this are two instances of the same object they are still considered different
|
|
// because the (none literal) script string may access variables which have different
|
|
// values in both instances and hence evaluated to different results.
|
|
QScopedPointer<MyTypeObject> object1_2(qobject_cast<MyTypeObject*>(component1.create()));
|
|
QVERIFY(object1_2 != nullptr);
|
|
QVERIFY(object1->scriptProperty() != object1_2->scriptProperty());
|
|
}
|
|
|
|
// Check that default property assignments are correctly spliced into explicit
|
|
// property assignments
|
|
void tst_qqmllanguage::defaultPropertyListOrder()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("defaultPropertyListOrder.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<MyContainer> container(qobject_cast<MyContainer *>(component.create()));
|
|
QVERIFY(container != nullptr);
|
|
|
|
QCOMPARE(container->getChildren()->size(), 6);
|
|
QCOMPARE(container->getChildren()->at(0)->property("index"), QVariant(0));
|
|
QCOMPARE(container->getChildren()->at(1)->property("index"), QVariant(1));
|
|
QCOMPARE(container->getChildren()->at(2)->property("index"), QVariant(2));
|
|
QCOMPARE(container->getChildren()->at(3)->property("index"), QVariant(3));
|
|
QCOMPARE(container->getChildren()->at(4)->property("index"), QVariant(4));
|
|
QCOMPARE(container->getChildren()->at(5)->property("index"), QVariant(5));
|
|
}
|
|
|
|
void tst_qqmllanguage::defaultPropertyWithInitializer_data()
|
|
{
|
|
QTest::addColumn<QUrl>("file");
|
|
QTest::addColumn<QString>("objectName");
|
|
|
|
QTest::newRow("base") << testFileUrl("DefaultPropertyWithInitializer.qml") << u"default"_s;
|
|
QTest::newRow("user") << testFileUrl("DefaultPropertyWithInitializerUser.qml") << u"changed"_s;
|
|
QTest::newRow("list base") << testFileUrl("DefaultPropertyWithListInitializer.qml") << u"1"_s;
|
|
QTest::newRow("list user") << testFileUrl("DefaultPropertyWithListInitializerUser.qml") << u"2"_s;
|
|
}
|
|
|
|
void tst_qqmllanguage::defaultPropertyWithInitializer()
|
|
{
|
|
QFETCH(QUrl, file);
|
|
QFETCH(QString, objectName);
|
|
|
|
QQmlComponent component(&engine, file);
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(root);
|
|
auto entry = root->property("entry").value<QObject *>();
|
|
QVERIFY(entry);
|
|
QCOMPARE(entry->objectName(), objectName);
|
|
}
|
|
|
|
void tst_qqmllanguage::declaredPropertyValues()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("declaredPropertyValues.qml"));
|
|
VERIFY_ERRORS(0);
|
|
}
|
|
|
|
void tst_qqmllanguage::dontDoubleCallClassBegin()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("dontDoubleCallClassBegin.qml"));
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o);
|
|
|
|
MyParserStatus *o2 = qobject_cast<MyParserStatus *>(qvariant_cast<QObject *>(o->property("object")));
|
|
QVERIFY(o2);
|
|
QCOMPARE(o2->classBeginCount(), 1);
|
|
QCOMPARE(o2->componentCompleteCount(), 1);
|
|
}
|
|
|
|
void tst_qqmllanguage::reservedWords_data()
|
|
{
|
|
QTest::addColumn<QByteArray>("word");
|
|
|
|
QTest::newRow("abstract") << QByteArray("abstract");
|
|
QTest::newRow("as") << QByteArray("as");
|
|
QTest::newRow("boolean") << QByteArray("boolean");
|
|
QTest::newRow("break") << QByteArray("break");
|
|
QTest::newRow("byte") << QByteArray("byte");
|
|
QTest::newRow("case") << QByteArray("case");
|
|
QTest::newRow("catch") << QByteArray("catch");
|
|
QTest::newRow("char") << QByteArray("char");
|
|
QTest::newRow("class") << QByteArray("class");
|
|
QTest::newRow("continue") << QByteArray("continue");
|
|
QTest::newRow("const") << QByteArray("const");
|
|
QTest::newRow("debugger") << QByteArray("debugger");
|
|
QTest::newRow("default") << QByteArray("default");
|
|
QTest::newRow("delete") << QByteArray("delete");
|
|
QTest::newRow("do") << QByteArray("do");
|
|
QTest::newRow("double") << QByteArray("double");
|
|
QTest::newRow("else") << QByteArray("else");
|
|
QTest::newRow("enum") << QByteArray("enum");
|
|
QTest::newRow("export") << QByteArray("export");
|
|
QTest::newRow("extends") << QByteArray("extends");
|
|
QTest::newRow("false") << QByteArray("false");
|
|
QTest::newRow("final") << QByteArray("final");
|
|
QTest::newRow("finally") << QByteArray("finally");
|
|
QTest::newRow("float") << QByteArray("float");
|
|
QTest::newRow("for") << QByteArray("for");
|
|
QTest::newRow("function") << QByteArray("function");
|
|
QTest::newRow("goto") << QByteArray("goto");
|
|
QTest::newRow("if") << QByteArray("if");
|
|
QTest::newRow("implements") << QByteArray("implements");
|
|
QTest::newRow("import") << QByteArray("import");
|
|
QTest::newRow("pragma") << QByteArray("pragma");
|
|
QTest::newRow("in") << QByteArray("in");
|
|
QTest::newRow("instanceof") << QByteArray("instanceof");
|
|
QTest::newRow("int") << QByteArray("int");
|
|
QTest::newRow("interface") << QByteArray("interface");
|
|
QTest::newRow("long") << QByteArray("long");
|
|
QTest::newRow("native") << QByteArray("native");
|
|
QTest::newRow("new") << QByteArray("new");
|
|
QTest::newRow("null") << QByteArray("null");
|
|
QTest::newRow("package") << QByteArray("package");
|
|
QTest::newRow("private") << QByteArray("private");
|
|
QTest::newRow("protected") << QByteArray("protected");
|
|
QTest::newRow("public") << QByteArray("public");
|
|
QTest::newRow("return") << QByteArray("return");
|
|
QTest::newRow("short") << QByteArray("short");
|
|
QTest::newRow("static") << QByteArray("static");
|
|
QTest::newRow("super") << QByteArray("super");
|
|
QTest::newRow("switch") << QByteArray("switch");
|
|
QTest::newRow("synchronized") << QByteArray("synchronized");
|
|
QTest::newRow("this") << QByteArray("this");
|
|
QTest::newRow("throw") << QByteArray("throw");
|
|
QTest::newRow("throws") << QByteArray("throws");
|
|
QTest::newRow("transient") << QByteArray("transient");
|
|
QTest::newRow("true") << QByteArray("true");
|
|
QTest::newRow("try") << QByteArray("try");
|
|
QTest::newRow("typeof") << QByteArray("typeof");
|
|
QTest::newRow("var") << QByteArray("var");
|
|
QTest::newRow("void") << QByteArray("void");
|
|
QTest::newRow("volatile") << QByteArray("volatile");
|
|
QTest::newRow("while") << QByteArray("while");
|
|
QTest::newRow("with") << QByteArray("with");
|
|
}
|
|
|
|
void tst_qqmllanguage::reservedWords()
|
|
{
|
|
QFETCH(QByteArray, word);
|
|
QQmlComponent component(&engine);
|
|
component.setData("import QtQuick 2.0\nQtObject { property string " + word + " }", QUrl());
|
|
QCOMPARE(component.errorString(), QLatin1String(":2 Expected token `identifier'\n"));
|
|
}
|
|
|
|
// Check that first child of qml is of given type. Empty type insists on error.
|
|
void tst_qqmllanguage::testType(const QString& qml, const QString& type, const QString& expectederror, bool partialMatch)
|
|
{
|
|
if (engine.importPathList() == defaultImportPathList)
|
|
engine.addImportPath(testFile("lib"));
|
|
|
|
QQmlComponent component(&engine);
|
|
component.setData(qml.toUtf8(), testFileUrl("empty.qml")); // just a file for relative local imports
|
|
|
|
QTRY_VERIFY(!component.isLoading());
|
|
|
|
if (type.isEmpty()) {
|
|
QVERIFY(component.isError());
|
|
QString actualerror;
|
|
foreach (const QQmlError e, component.errors()) {
|
|
if (!actualerror.isEmpty())
|
|
actualerror.append("; ");
|
|
actualerror.append(e.description());
|
|
}
|
|
QCOMPARE(actualerror.left(partialMatch ? expectederror.size(): -1),expectederror);
|
|
} else {
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
const QMetaObject *meta = object->metaObject();
|
|
for (; meta; meta = meta->superClass()) {
|
|
const QString className(meta->className());
|
|
if (!className.contains("_QMLTYPE_") && !className.contains("_QML_")) {
|
|
QCOMPARE(className, type);
|
|
break;
|
|
}
|
|
}
|
|
QVERIFY(meta != nullptr);
|
|
}
|
|
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
// QTBUG-17276
|
|
void tst_qqmllanguage::inlineAssignmentsOverrideBindings()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("inlineAssignmentsOverrideBindings.qml"));
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
QCOMPARE(o->property("test").toInt(), 11);
|
|
}
|
|
|
|
// QTBUG-19354
|
|
void tst_qqmllanguage::nestedComponentRoots()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("nestedComponentRoots.qml"));
|
|
}
|
|
|
|
// Import tests (QT-558)
|
|
void tst_qqmllanguage::importsBuiltin_data()
|
|
{
|
|
// QT-610
|
|
|
|
QTest::addColumn<QString>("qml");
|
|
QTest::addColumn<QString>("type");
|
|
QTest::addColumn<QString>("error");
|
|
|
|
// import built-ins
|
|
QTest::newRow("missing import")
|
|
<< "Test {}"
|
|
<< ""
|
|
<< "Test is not a type";
|
|
QTest::newRow("not in version 0.0")
|
|
<< "import org.qtproject.Test 0.0\n"
|
|
"Test {}"
|
|
<< ""
|
|
<< "Test is not a type";
|
|
QTest::newRow("version not installed")
|
|
<< "import org.qtproject.Test 99.0\n"
|
|
"Test {}"
|
|
<< ""
|
|
<< "module \"org.qtproject.Test\" version 99.0 is not installed";
|
|
QTest::newRow("in version 0.0")
|
|
<< "import org.qtproject.Test 0.0\n"
|
|
"TestTP {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("qualified in version 0.0")
|
|
<< "import org.qtproject.Test 0.0 as T\n"
|
|
"T.TestTP {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("in version 1.0")
|
|
<< "import org.qtproject.Test 1.0\n"
|
|
"Test {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("qualified wrong")
|
|
<< "import org.qtproject.Test 1.0 as T\n" // QT-610
|
|
"Test {}"
|
|
<< ""
|
|
<< "Test is not a type";
|
|
QTest::newRow("qualified right")
|
|
<< "import org.qtproject.Test 1.0 as T\n"
|
|
"T.Test {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("qualified right but not in version 0.0")
|
|
<< "import org.qtproject.Test 0.0 as T\n"
|
|
"T.Test {}"
|
|
<< ""
|
|
<< "T.Test is not a type";
|
|
QTest::newRow("in version 1.1")
|
|
<< "import org.qtproject.Test 1.1\n"
|
|
"Test {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("in version 1.3")
|
|
<< "import org.qtproject.Test 1.3\n"
|
|
"Test {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("in version 1.5")
|
|
<< "import org.qtproject.Test 1.5\n"
|
|
"Test {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("changed in version 1.8")
|
|
<< "import org.qtproject.Test 1.8\n"
|
|
"Test {}"
|
|
<< "TestType2"
|
|
<< "";
|
|
QTest::newRow("in version 1.12")
|
|
<< "import org.qtproject.Test 1.12\n"
|
|
"Test {}"
|
|
<< "TestType2"
|
|
<< "";
|
|
QTest::newRow("old in version 1.9")
|
|
<< "import org.qtproject.Test 1.9\n"
|
|
"OldTest {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("old in version 1.11")
|
|
<< "import org.qtproject.Test 1.11\n"
|
|
"OldTest {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("multiversion 1")
|
|
<< "import org.qtproject.Test 1.11\n"
|
|
"import org.qtproject.Test 1.12\n"
|
|
"Test {}"
|
|
<< (!qmlCheckTypes()?"TestType2":"")
|
|
<< (!qmlCheckTypes()?"":"Test is ambiguous. Found in org/qtproject/Test/ in version 1.12 and 1.11");
|
|
QTest::newRow("multiversion 2")
|
|
<< "import org.qtproject.Test 1.11\n"
|
|
"import org.qtproject.Test 1.12\n"
|
|
"OldTest {}"
|
|
<< (!qmlCheckTypes()?"TestType":"")
|
|
<< (!qmlCheckTypes()?"":"OldTest is ambiguous. Found in org/qtproject/Test/ in version 1.12 and 1.11");
|
|
QTest::newRow("qualified multiversion 3")
|
|
<< "import org.qtproject.Test 1.0 as T0\n"
|
|
"import org.qtproject.Test 1.8 as T8\n"
|
|
"T0.Test {}"
|
|
<< "TestType"
|
|
<< "";
|
|
QTest::newRow("qualified multiversion 4")
|
|
<< "import org.qtproject.Test 1.0 as T0\n"
|
|
"import org.qtproject.Test 1.8 as T8\n"
|
|
"T8.Test {}"
|
|
<< "TestType2"
|
|
<< "";
|
|
}
|
|
|
|
void tst_qqmllanguage::importsBuiltin()
|
|
{
|
|
QFETCH(QString, qml);
|
|
QFETCH(QString, type);
|
|
QFETCH(QString, error);
|
|
testType(qml,type,error);
|
|
}
|
|
|
|
void tst_qqmllanguage::importsLocal_data()
|
|
{
|
|
QTest::addColumn<QString>("qml");
|
|
QTest::addColumn<QString>("type");
|
|
QTest::addColumn<QString>("error");
|
|
|
|
// import locals
|
|
QTest::newRow("local import")
|
|
<< "import \"subdir\"\n" // QT-613
|
|
"Test {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("local import second")
|
|
<< "import QtQuick 2.0\nimport \"subdir\"\n"
|
|
"Test {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("local import subsubdir")
|
|
<< "import QtQuick 2.0\nimport \"subdir/subsubdir\"\n"
|
|
"SubTest {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("local import QTBUG-7721 A")
|
|
<< "subdir.Test {}" // no longer allowed (QTBUG-7721)
|
|
<< ""
|
|
<< "subdir.Test - subdir is neither a type nor a namespace";
|
|
QTest::newRow("local import QTBUG-7721 B")
|
|
<< "import \"subdir\" as X\n"
|
|
"X.subsubdir.SubTest {}" // no longer allowed (QTBUG-7721)
|
|
<< ""
|
|
<< "X.subsubdir.SubTest - subsubdir is not a type";
|
|
QTest::newRow("local import as")
|
|
<< "import \"subdir\" as T\n"
|
|
"T.Test {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("wrong local import as")
|
|
<< "import \"subdir\" as T\n"
|
|
"Test {}"
|
|
<< ""
|
|
<< "Test is not a type";
|
|
QTest::newRow("library precedence over local import")
|
|
<< "import \"subdir\"\n"
|
|
"import org.qtproject.Test 1.0\n"
|
|
"Test {}"
|
|
<< (!qmlCheckTypes()?"TestType":"")
|
|
<< (!qmlCheckTypes()?"":"Test is ambiguous. Found in org/qtproject/Test/ and in subdir/");
|
|
|
|
if (dataDirectoryUrl().scheme() != QLatin1String("qrc")) {
|
|
// file URL doesn't work with qrc scheme
|
|
QTest::newRow("file URL survives percent-encoding")
|
|
<< "import \"" + QUrl::fromLocalFile(QDir::currentPath() + "/{subdir}").toString() + "\"\n"
|
|
"Test {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::importsLocal()
|
|
{
|
|
QFETCH(QString, qml);
|
|
QFETCH(QString, type);
|
|
QFETCH(QString, error);
|
|
testType(qml,type,error);
|
|
}
|
|
|
|
void tst_qqmllanguage::basicRemote_data()
|
|
{
|
|
QTest::addColumn<QUrl>("url");
|
|
QTest::addColumn<QString>("type");
|
|
QTest::addColumn<QString>("error");
|
|
|
|
QString serverdir = "/qtest/qml/qqmllanguage/";
|
|
|
|
QTest::newRow("no need for qmldir") << QUrl(serverdir+"Test.qml") << "" << "";
|
|
QTest::newRow("absent qmldir") << QUrl(serverdir+"/noqmldir/Test.qml") << "" << "";
|
|
QTest::newRow("need qmldir") << QUrl(serverdir+"TestNamed.qml") << "" << "";
|
|
}
|
|
|
|
void tst_qqmllanguage::basicRemote()
|
|
{
|
|
QFETCH(QUrl, url);
|
|
QFETCH(QString, type);
|
|
QFETCH(QString, error);
|
|
|
|
ThreadedTestHTTPServer server(dataDirectory());
|
|
|
|
url = server.baseUrl().resolved(url);
|
|
|
|
QQmlComponent component(&engine, url);
|
|
|
|
QTRY_VERIFY(!component.isLoading());
|
|
|
|
if (error.isEmpty()) {
|
|
if (component.isError())
|
|
qDebug() << component.errors();
|
|
QVERIFY(!component.isError());
|
|
} else {
|
|
QVERIFY(component.isError());
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::importsRemote_data()
|
|
{
|
|
QTest::addColumn<QString>("qml");
|
|
QTest::addColumn<QString>("type");
|
|
QTest::addColumn<QString>("error");
|
|
|
|
QString serverdir = "{{ServerBaseUrl}}/qtest/qml/qqmllanguage";
|
|
|
|
QTest::newRow("remote import") << "import \""+serverdir+"\"\nTest {}" << "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("remote import with subdir") << "import \""+serverdir+"\"\nTestSubDir {}" << "QQuickText"
|
|
<< "";
|
|
QTest::newRow("remote import with local") << "import \""+serverdir+"\"\nTestLocal {}" << "QQuickImage"
|
|
<< "";
|
|
QTest::newRow("remote import with qualifier") << "import \""+serverdir+"\" as NS\nNS.NamedLocal {}" << "QQuickImage"
|
|
<< "";
|
|
QTest::newRow("wrong remote import with undeclared local") << "import \""+serverdir+"\"\nWrongTestLocal {}" << ""
|
|
<< "WrongTestLocal is not a type";
|
|
QTest::newRow("wrong remote import of internal local") << "import \""+serverdir+"\"\nLocalInternal {}" << ""
|
|
<< "LocalInternal is not a type";
|
|
QTest::newRow("wrong remote import of undeclared local") << "import \""+serverdir+"\"\nUndeclaredLocal {}" << ""
|
|
<< "UndeclaredLocal is not a type";
|
|
}
|
|
|
|
void tst_qqmllanguage::importsRemote()
|
|
{
|
|
QFETCH(QString, qml);
|
|
QFETCH(QString, type);
|
|
QFETCH(QString, error);
|
|
|
|
ThreadedTestHTTPServer server(dataDirectory());
|
|
|
|
qml.replace(QStringLiteral("{{ServerBaseUrl}}"), server.baseUrl().toString());
|
|
|
|
testType(qml,type,error);
|
|
}
|
|
|
|
void tst_qqmllanguage::importsInstalled_data()
|
|
{
|
|
// QT-610
|
|
|
|
QTest::addColumn<QString>("qml");
|
|
QTest::addColumn<QString>("type");
|
|
QTest::addColumn<QString>("error");
|
|
|
|
// import installed
|
|
QTest::newRow("installed import 0")
|
|
<< "import org.qtproject.installedtest0 0.0\n"
|
|
"InstalledTestTP {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("installed import 0 as TP")
|
|
<< "import org.qtproject.installedtest0 0.0 as TP\n"
|
|
"TP.InstalledTestTP {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("installed import 1")
|
|
<< "import org.qtproject.installedtest 1.0\n"
|
|
"InstalledTest {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("installed import 2")
|
|
<< "import org.qtproject.installedtest 1.3\n"
|
|
"InstalledTest {}"
|
|
<< "QQuickRectangle"
|
|
<< "";
|
|
QTest::newRow("installed import 3")
|
|
<< "import org.qtproject.installedtest 1.4\n"
|
|
"InstalledTest {}"
|
|
<< "QQuickText"
|
|
<< "";
|
|
QTest::newRow("installed import minor version not available") // QTBUG-11936
|
|
<< "import org.qtproject.installedtest 0.1\n"
|
|
"InstalledTest {}"
|
|
<< ""
|
|
<< "module \"org.qtproject.installedtest\" version 0.1 is not installed";
|
|
QTest::newRow("installed import minor version not available") // QTBUG-9627
|
|
<< "import org.qtproject.installedtest 1.10\n"
|
|
"InstalledTest {}"
|
|
<< ""
|
|
<< "module \"org.qtproject.installedtest\" version 1.10 is not installed";
|
|
QTest::newRow("installed import major version not available") // QTBUG-9627
|
|
<< "import org.qtproject.installedtest 9.0\n"
|
|
"InstalledTest {}"
|
|
<< ""
|
|
<< "module \"org.qtproject.installedtest\" version 9.0 is not installed";
|
|
QTest::newRow("installed import visibility") // QT-614
|
|
<< "import org.qtproject.installedtest 1.4\n"
|
|
"PrivateType {}"
|
|
<< ""
|
|
<< "PrivateType is not a type";
|
|
QTest::newRow("installed import version QML clash")
|
|
<< "import org.qtproject.installedtest1 1.0\n"
|
|
"Test {}"
|
|
<< ""
|
|
<< "\"Test\" version 1.0 is defined more than once in module \"org.qtproject.installedtest1\"";
|
|
QTest::newRow("installed import version JS clash")
|
|
<< "import org.qtproject.installedtest2 1.0\n"
|
|
"Test {}"
|
|
<< ""
|
|
<< "\"Test\" version 1.0 is defined more than once in module \"org.qtproject.installedtest2\"";
|
|
}
|
|
|
|
void tst_qqmllanguage::importsInstalled()
|
|
{
|
|
QFETCH(QString, qml);
|
|
QFETCH(QString, type);
|
|
QFETCH(QString, error);
|
|
testType(qml,type,error);
|
|
}
|
|
|
|
void tst_qqmllanguage::importsInstalledRemote_data()
|
|
{
|
|
// Repeat the tests for local installed data
|
|
importsInstalled_data();
|
|
}
|
|
|
|
void tst_qqmllanguage::importsInstalledRemote()
|
|
{
|
|
QFETCH(QString, qml);
|
|
QFETCH(QString, type);
|
|
QFETCH(QString, error);
|
|
|
|
ThreadedTestHTTPServer server(dataDirectory());
|
|
|
|
QString serverdir = server.urlString("/lib/");
|
|
engine.setImportPathList(QStringList(defaultImportPathList) << serverdir);
|
|
|
|
testType(qml,type,error);
|
|
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
void tst_qqmllanguage::importsPath_data()
|
|
{
|
|
QTest::addColumn<QStringList>("importPath");
|
|
QTest::addColumn<QString>("qml");
|
|
QTest::addColumn<QString>("value");
|
|
|
|
QTest::newRow("local takes priority normal")
|
|
<< (QStringList() << testFile("lib") << "{{ServerBaseUrl}}/lib2/")
|
|
<< "import testModule 1.0\n"
|
|
"Test {}"
|
|
<< "foo";
|
|
|
|
QTest::newRow("local takes priority reversed")
|
|
<< (QStringList() << "{{ServerBaseUrl}}/lib/" << testFile("lib2"))
|
|
<< "import testModule 1.0\n"
|
|
"Test {}"
|
|
<< "bar";
|
|
|
|
QTest::newRow("earlier takes priority 1")
|
|
<< (QStringList() << "{{ServerBaseUrl}}/lib/" << "{{ServerBaseUrl}}/lib2/")
|
|
<< "import testModule 1.0\n"
|
|
"Test {}"
|
|
<< "foo";
|
|
|
|
QTest::newRow("earlier takes priority 2")
|
|
<< (QStringList() << "{{ServerBaseUrl}}/lib2/" << "{{ServerBaseUrl}}/lib/")
|
|
<< "import testModule 1.0\n"
|
|
"Test {}"
|
|
<< "bar";
|
|
|
|
QTest::newRow("major version takes priority over unversioned")
|
|
<< (QStringList() << "{{ServerBaseUrl}}/lib/" << "{{ServerBaseUrl}}/lib3/")
|
|
<< "import testModule 1.0\n"
|
|
"Test {}"
|
|
<< "baz";
|
|
|
|
QTest::newRow("major version takes priority over minor")
|
|
<< (QStringList() << "{{ServerBaseUrl}}/lib4/" << "{{ServerBaseUrl}}/lib3/")
|
|
<< "import testModule 1.0\n"
|
|
"Test {}"
|
|
<< "baz";
|
|
|
|
QTest::newRow("minor version takes priority over unversioned")
|
|
<< (QStringList() << "{{ServerBaseUrl}}/lib/" << "{{ServerBaseUrl}}/lib4/")
|
|
<< "import testModule 1.0\n"
|
|
"Test {}"
|
|
<< "qux";
|
|
}
|
|
|
|
void tst_qqmllanguage::importsPath()
|
|
{
|
|
QFETCH(QStringList, importPath);
|
|
QFETCH(QString, qml);
|
|
QFETCH(QString, value);
|
|
|
|
ThreadedTestHTTPServer server(dataDirectory());
|
|
|
|
for (int i = 0; i < importPath.size(); ++i)
|
|
importPath[i].replace(QStringLiteral("{{ServerBaseUrl}}"), server.baseUrl().toString());
|
|
|
|
engine.setImportPathList(QStringList(defaultImportPathList) << importPath);
|
|
|
|
QQmlComponent component(&engine);
|
|
component.setData(qml.toUtf8(), testFileUrl("empty.qml"));
|
|
|
|
QTRY_VERIFY(component.isReady());
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("test").toString(), value);
|
|
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
void tst_qqmllanguage::importsOrder_data()
|
|
{
|
|
QTest::addColumn<QString>("qml");
|
|
QTest::addColumn<QString>("type");
|
|
QTest::addColumn<QString>("error");
|
|
QTest::addColumn<bool>("partialMatch");
|
|
|
|
QTest::newRow("double import") <<
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"InstalledTest {}"
|
|
<< (!qmlCheckTypes()?"QQuickText":"")
|
|
<< (!qmlCheckTypes()?"":"InstalledTest is ambiguous. Found in lib/org/qtproject/installedtest/ in version 1.4 and 1.4")
|
|
<< false;
|
|
QTest::newRow("installed import overrides 1") <<
|
|
"import org.qtproject.installedtest 1.0\n"
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"InstalledTest {}"
|
|
<< (!qmlCheckTypes()?"QQuickText":"")
|
|
<< (!qmlCheckTypes()?"":"InstalledTest is ambiguous. Found in lib/org/qtproject/installedtest/ in version 1.4 and 1.0")
|
|
<< false;
|
|
QTest::newRow("installed import overrides 2") <<
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"import org.qtproject.installedtest 1.0\n"
|
|
"InstalledTest {}"
|
|
<< (!qmlCheckTypes()?"QQuickRectangle":"")
|
|
<< (!qmlCheckTypes()?"":"InstalledTest is ambiguous. Found in lib/org/qtproject/installedtest/ in version 1.0 and 1.4")
|
|
<< false;
|
|
QTest::newRow("installed import re-overrides 1") <<
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"import org.qtproject.installedtest 1.0\n"
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"InstalledTest {}"
|
|
<< (!qmlCheckTypes()?"QQuickText":"")
|
|
<< (!qmlCheckTypes()?"":"InstalledTest is ambiguous. Found in lib/org/qtproject/installedtest/ in version 1.4 and 1.0")
|
|
<< false;
|
|
QTest::newRow("installed import re-overrides 2") <<
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"import org.qtproject.installedtest 1.0\n"
|
|
"import org.qtproject.installedtest 1.4\n"
|
|
"import org.qtproject.installedtest 1.0\n"
|
|
"InstalledTest {}"
|
|
<< (!qmlCheckTypes()?"QQuickRectangle":"")
|
|
<< (!qmlCheckTypes()?"":"InstalledTest is ambiguous. Found in lib/org/qtproject/installedtest/ in version 1.0 and 1.4")
|
|
<< false;
|
|
QTest::newRow("installed import versus builtin 1") <<
|
|
"import org.qtproject.installedtest 1.5\n"
|
|
"import QtQuick 2.0\n"
|
|
"Rectangle {}"
|
|
<< (!qmlCheckTypes()?"QQuickRectangle":"")
|
|
<< (!qmlCheckTypes()?"":"Rectangle is ambiguous. Found in file://")
|
|
<< true;
|
|
QTest::newRow("installed import versus builtin 2") <<
|
|
"import QtQuick 2.0\n"
|
|
"import org.qtproject.installedtest 1.5\n"
|
|
"Rectangle {}"
|
|
<< (!qmlCheckTypes()?"QQuickText":"")
|
|
<< (!qmlCheckTypes()?"":"Rectangle is ambiguous. Found in lib/org/qtproject/installedtest/ and in file://")
|
|
<< true;
|
|
QTest::newRow("namespaces cannot be overridden by types 1") <<
|
|
"import QtQuick 2.0 as Rectangle\n"
|
|
"import org.qtproject.installedtest 1.5\n"
|
|
"Rectangle {}"
|
|
<< ""
|
|
<< "Namespace Rectangle cannot be used as a type"
|
|
<< false;
|
|
QTest::newRow("namespaces cannot be overridden by types 2") <<
|
|
"import QtQuick 2.0 as Rectangle\n"
|
|
"import org.qtproject.installedtest 1.5\n"
|
|
"Rectangle.Image {}"
|
|
<< "QQuickImage"
|
|
<< ""
|
|
<< false;
|
|
QTest::newRow("local last 1") <<
|
|
"LocalLast {}"
|
|
<< "QQuickText"
|
|
<< ""
|
|
<< false;
|
|
QTest::newRow("local last 2") <<
|
|
"import org.qtproject.installedtest 1.0\n"
|
|
"LocalLast {}"
|
|
<< (!qmlCheckTypes()?"QQuickRectangle":"")// i.e. from org.qtproject.installedtest, not data/LocalLast.qml
|
|
<< (!qmlCheckTypes()?"":"LocalLast is ambiguous. Found in lib/org/qtproject/installedtest/ and in ")
|
|
<< false;
|
|
QTest::newRow("local last 3") << //Forces it to load the local qmldir to resolve types, but they shouldn't override anything
|
|
"import org.qtproject.installedtest 1.0\n"
|
|
"LocalLast {LocalLast2{}}"
|
|
<< (!qmlCheckTypes()?"QQuickRectangle":"")// i.e. from org.qtproject.installedtest, not data/LocalLast.qml
|
|
<< (!qmlCheckTypes()?"":"LocalLast is ambiguous. Found in lib/org/qtproject/installedtest/ and in ")
|
|
<< false;
|
|
}
|
|
|
|
void tst_qqmllanguage::importsOrder()
|
|
{
|
|
QFETCH(QString, qml);
|
|
QFETCH(QString, type);
|
|
QFETCH(QString, error);
|
|
QFETCH(bool, partialMatch);
|
|
testType(qml,type,error,partialMatch);
|
|
}
|
|
|
|
void tst_qqmllanguage::importIncorrectCase()
|
|
{
|
|
if (engine.importPathList() == defaultImportPathList)
|
|
engine.addImportPath(testFile("lib"));
|
|
|
|
// Load "importIncorrectCase.qml" using wrong case
|
|
QQmlComponent component(&engine, testFileUrl("ImportIncorrectCase.qml"));
|
|
|
|
QList<QQmlError> errors = component.errors();
|
|
QCOMPARE(errors.size(), 1);
|
|
|
|
const QString expectedError = isCaseSensitiveFileSystem(dataDirectory()) ?
|
|
QStringLiteral("No such file or directory") :
|
|
QStringLiteral("File name case mismatch");
|
|
QCOMPARE(errors.at(0).description(), expectedError);
|
|
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
void tst_qqmllanguage::importJs_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::addColumn<QString>("errorFile");
|
|
QTest::addColumn<bool>("performTest");
|
|
|
|
QTest::newRow("defaultVersion")
|
|
<< "importJs.1.qml"
|
|
<< "importJs.1.errors.txt"
|
|
<< true;
|
|
|
|
QTest::newRow("specifiedVersion")
|
|
<< "importJs.2.qml"
|
|
<< "importJs.2.errors.txt"
|
|
<< true;
|
|
|
|
QTest::newRow("excludeExcessiveVersion")
|
|
<< "importJs.3.qml"
|
|
<< "importJs.3.errors.txt"
|
|
<< false;
|
|
|
|
QTest::newRow("includeAppropriateVersion")
|
|
<< "importJs.4.qml"
|
|
<< "importJs.4.errors.txt"
|
|
<< true;
|
|
|
|
QTest::newRow("noDefaultVersion")
|
|
<< "importJs.5.qml"
|
|
<< "importJs.5.errors.txt"
|
|
<< false;
|
|
|
|
QTest::newRow("repeatImportFails")
|
|
<< "importJs.6.qml"
|
|
<< "importJs.6.errors.txt"
|
|
<< false;
|
|
|
|
QTest::newRow("multipleVersionImportFails")
|
|
<< "importJs.7.qml"
|
|
<< "importJs.7.errors.txt"
|
|
<< false;
|
|
|
|
QTest::newRow("namespacedImport")
|
|
<< "importJs.8.qml"
|
|
<< "importJs.8.errors.txt"
|
|
<< true;
|
|
|
|
QTest::newRow("namespacedVersionedImport")
|
|
<< "importJs.9.qml"
|
|
<< "importJs.9.errors.txt"
|
|
<< true;
|
|
|
|
QTest::newRow("namespacedRepeatImport")
|
|
<< "importJs.10.qml"
|
|
<< "importJs.10.errors.txt"
|
|
<< true;
|
|
|
|
QTest::newRow("emptyScript")
|
|
<< "importJs.11.qml"
|
|
<< "importJs.11.errors.txt"
|
|
<< true;
|
|
}
|
|
|
|
void tst_qqmllanguage::importJs()
|
|
{
|
|
QFETCH(QString, file);
|
|
QFETCH(QString, errorFile);
|
|
QFETCH(bool, performTest);
|
|
|
|
engine.setImportPathList(QStringList(defaultImportPathList) << testFile("lib"));
|
|
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
|
|
{
|
|
DETERMINE_ERRORS(errorFile,expected,actual);
|
|
QCOMPARE(actual.size(), expected.size());
|
|
for (int i = 0; i < expected.size(); ++i)
|
|
{
|
|
const int compareLen = qMin(expected.at(i).size(), actual.at(i).size());
|
|
QCOMPARE(actual.at(i).left(compareLen), expected.at(i).left(compareLen));
|
|
}
|
|
}
|
|
|
|
if (performTest) {
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("test").toBool(),true);
|
|
}
|
|
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
void tst_qqmllanguage::importJsModule_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
|
|
QTest::newRow("plainImport")
|
|
<< "importJsModule.1.qml";
|
|
|
|
QTest::newRow("ImportQmlStyle")
|
|
<< "importJsModule.2.qml";
|
|
|
|
QTest::newRow("plainImportWithCycle")
|
|
<< "importJsModule.3.qml";
|
|
}
|
|
|
|
void tst_qqmllanguage::importJsModule()
|
|
{
|
|
QFETCH(QString, file);
|
|
|
|
engine.setImportPathList(QStringList(defaultImportPathList) << testFile("lib"));
|
|
auto importPathGuard = qScopeGuard([this]{
|
|
engine.setImportPathList(defaultImportPathList);
|
|
});
|
|
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
QVERIFY2(!component.isError(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("test").toBool(),true);
|
|
}
|
|
|
|
void tst_qqmllanguage::explicitSelfImport()
|
|
{
|
|
engine.setImportPathList(QStringList(defaultImportPathList) << testFile("lib"));
|
|
|
|
QQmlComponent component(&engine, testFileUrl("mixedModuleWithSelfImport.qml"));
|
|
QVERIFY(component.errors().size() == 0);
|
|
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
void tst_qqmllanguage::importInternalType()
|
|
{
|
|
QQmlEngine engine;
|
|
engine.addImportPath(dataDirectory());
|
|
|
|
{
|
|
QQmlComponent component(&engine);
|
|
component.setData("import modulewithinternaltypes 1.0\nPublicType{}", QUrl());
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(!obj.isNull());
|
|
QVERIFY(obj->property("myInternalType").value<QObject*>() != 0);
|
|
}
|
|
{
|
|
QQmlComponent component(&engine);
|
|
component.setData("import modulewithinternaltypes 1.0\nPublicTypeWithExplicitImport{}", QUrl());
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(!obj.isNull());
|
|
QVERIFY(obj->property("myInternalType").value<QObject*>() != 0);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::qmlAttachedPropertiesObjectMethod()
|
|
{
|
|
QObject object;
|
|
|
|
QCOMPARE(qmlAttachedPropertiesObject<MyQmlObject>(&object, false), (QObject *)nullptr);
|
|
QVERIFY(qmlAttachedPropertiesObject<MyQmlObject>(&object, true));
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("qmlAttachedPropertiesObjectMethod.1.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(qmlAttachedPropertiesObject<MyQmlObject>(object.data(), false), (QObject *)nullptr);
|
|
QVERIFY(qmlAttachedPropertiesObject<MyQmlObject>(object.data(), true) != nullptr);
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("qmlAttachedPropertiesObjectMethod.2.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QVERIFY(qmlAttachedPropertiesObject<MyQmlObject>(object.data(), false) != nullptr);
|
|
QVERIFY(qmlAttachedPropertiesObject<MyQmlObject>(object.data(), true) != nullptr);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::crash1()
|
|
{
|
|
QQmlComponent component(&engine);
|
|
component.setData("import QtQuick 2.0\nComponent {}", QUrl());
|
|
}
|
|
|
|
void tst_qqmllanguage::crash2()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("crash2.qml"));
|
|
}
|
|
|
|
// QTBUG-8676
|
|
void tst_qqmllanguage::customOnProperty()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("customOnProperty.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("on").toInt(), 10);
|
|
}
|
|
|
|
// QTBUG-12601
|
|
void tst_qqmllanguage::variantNotify()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("variantNotify.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
QEXPECT_FAIL("", "var properties always trigger notify", Continue);
|
|
QCOMPARE(object->property("notifyCount").toInt(), 1);
|
|
}
|
|
|
|
void tst_qqmllanguage::revisions()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("revisions11.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyRevisionedClass> object(qobject_cast<MyRevisionedClass*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->prop2(), 10.0);
|
|
}
|
|
{
|
|
QQmlEngine myEngine;
|
|
QQmlComponent component(&myEngine, testFileUrl("revisionssub11.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyRevisionedSubclass> object(qobject_cast<MyRevisionedSubclass*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->prop1(), 10.0);
|
|
QCOMPARE(object->prop2(), 10.0);
|
|
QCOMPARE(object->prop3(), 10.0);
|
|
QCOMPARE(object->prop4(), 10.0);
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("versionedbase.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MySubclass> object(qobject_cast<MySubclass*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->prop1(), 10.0);
|
|
QCOMPARE(object->prop2(), 10.0);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::revisionOverloads()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("allowedRevisionOverloads.qml"));
|
|
VERIFY_ERRORS(0);
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("disallowedRevisionOverloads.qml"));
|
|
QEXPECT_FAIL("", "QTBUG-13849", Abort);
|
|
QVERIFY(0);
|
|
VERIFY_ERRORS("disallowedRevisionOverloads.errors.txt");
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::subclassedUncreateableRevision_data()
|
|
{
|
|
QTest::addColumn<QString>("version");
|
|
QTest::addColumn<QString>("prop");
|
|
QTest::addColumn<bool>("shouldWork");
|
|
|
|
QTest::newRow("prop1 exists in 1.0") << "1.0" << "prop1" << true;
|
|
QTest::newRow("prop2 does not exist in 1.0") << "1.0" << "prop2" << false;
|
|
QTest::newRow("prop3 does not exist in 1.0") << "1.0" << "prop3" << false;
|
|
|
|
QTest::newRow("prop1 exists in 1.1") << "1.1" << "prop1" << true;
|
|
QTest::newRow("prop2 works because it's re-declared in Derived") << "1.1" << "prop2" << true;
|
|
QTest::newRow("prop3 only works if the Base REVISION 1 is picked up") << "1.1" << "prop3" << true;
|
|
|
|
}
|
|
|
|
void tst_qqmllanguage::subclassedUncreateableRevision()
|
|
{
|
|
QFETCH(QString, version);
|
|
QFETCH(QString, prop);
|
|
QFETCH(bool, shouldWork);
|
|
|
|
{
|
|
QQmlEngine engine;
|
|
QString qml = QString("import QtQuick 2.0\nimport Test %1\nMyUncreateableBaseClass {}").arg(version);
|
|
QQmlComponent c(&engine);
|
|
QTest::ignoreMessage(QtWarningMsg, "QQmlComponent: Component is not ready");
|
|
c.setData(qml.toUtf8(), QUrl::fromLocalFile(QDir::currentPath()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QCOMPARE(obj.data(), static_cast<QObject*>(nullptr));
|
|
QCOMPARE(c.errors().size(), 1);
|
|
QCOMPARE(c.errors().first().description(), QString("Cannot create MyUncreateableBaseClass"));
|
|
}
|
|
|
|
QQmlEngine engine;
|
|
QString qml = QString("import QtQuick 2.0\nimport Test %1\nMyCreateableDerivedClass {\n%3: true\n}").arg(version).arg(prop);
|
|
QQmlComponent c(&engine);
|
|
if (!shouldWork)
|
|
QTest::ignoreMessage(QtWarningMsg, "QQmlComponent: Component is not ready");
|
|
c.setData(qml.toUtf8(), QUrl::fromLocalFile(QDir::currentPath()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
if (!shouldWork) {
|
|
QCOMPARE(obj.data(), static_cast<QObject*>(nullptr));
|
|
return;
|
|
}
|
|
|
|
QVERIFY(obj);
|
|
MyUncreateableBaseClass *base = qobject_cast<MyUncreateableBaseClass*>(obj.data());
|
|
QVERIFY(base);
|
|
QCOMPARE(base->property(prop.toLatin1()).toBool(), true);
|
|
}
|
|
|
|
void tst_qqmllanguage::subclassedExtendedUncreateableRevision_data()
|
|
{
|
|
QTest::addColumn<QString>("version");
|
|
QTest::addColumn<QString>("prop");
|
|
QTest::addColumn<bool>("shouldWork");
|
|
|
|
QTest::newRow("prop1 exists in 1.0") << "1.0" << "prop1" << true;
|
|
QTest::newRow("prop2 does not exist in 1.0") << "1.0" << "prop2" << false;
|
|
QTest::newRow("prop3 does not exist in 1.0") << "1.0" << "prop3" << false;
|
|
QTest::newRow("prop4 exists in 1.0") << "1.0" << "prop4" << true;
|
|
QTest::newRow("prop5 exists in 1.0") << "1.0" << "prop5" << true;
|
|
|
|
QTest::newRow("prop1 exists in 1.1") << "1.1" << "prop1" << true;
|
|
QTest::newRow("prop2 exists in 1.1") << "1.1" << "prop2" << true;
|
|
QTest::newRow("prop3 exists in 1.1") << "1.1" << "prop3" << true;
|
|
QTest::newRow("prop4 exists in 1.1") << "1.1" << "prop4" << true;
|
|
QTest::newRow("prop5 exists in 1.1") << "1.1" << "prop5" << true;
|
|
}
|
|
|
|
void tst_qqmllanguage::subclassedExtendedUncreateableRevision()
|
|
{
|
|
QFETCH(QString, version);
|
|
QFETCH(QString, prop);
|
|
QFETCH(bool, shouldWork);
|
|
|
|
{
|
|
QQmlEngine engine;
|
|
QString qml = QString("import QtQuick 2.0\nimport Test %1\nMyExtendedUncreateableBaseClass {}").arg(version);
|
|
QQmlComponent c(&engine);
|
|
QTest::ignoreMessage(QtWarningMsg, "QQmlComponent: Component is not ready");
|
|
c.setData(qml.toUtf8(), QUrl::fromLocalFile(QDir::currentPath()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QCOMPARE(obj.data(), static_cast<QObject*>(nullptr));
|
|
QCOMPARE(c.errors().size(), 1);
|
|
QCOMPARE(c.errors().first().description(), QString("Cannot create MyExtendedUncreateableBaseClass"));
|
|
}
|
|
|
|
QQmlEngine engine;
|
|
QString qml = QString("import QtQuick 2.0\nimport Test %1\nMyExtendedCreateableDerivedClass {\n%3: true\n}").arg(version).arg(prop);
|
|
QQmlComponent c(&engine);
|
|
if (!shouldWork)
|
|
QTest::ignoreMessage(QtWarningMsg, "QQmlComponent: Component is not ready");
|
|
c.setData(qml.toUtf8(), QUrl::fromLocalFile(QDir::currentPath()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
if (!shouldWork) {
|
|
QCOMPARE(obj.data(), static_cast<QObject*>(nullptr));
|
|
return;
|
|
}
|
|
|
|
QVERIFY(obj);
|
|
MyExtendedUncreateableBaseClass *base = qobject_cast<MyExtendedUncreateableBaseClass*>(obj.data());
|
|
QVERIFY(base);
|
|
QCOMPARE(base->property(prop.toLatin1()).toBool(), true);
|
|
}
|
|
|
|
void tst_qqmllanguage::uncreatableTypesAsProperties()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("uncreatableTypeAsProperty.qml"));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
}
|
|
|
|
tst_qqmllanguage::tst_qqmllanguage()
|
|
: QQmlDataTest(QT_QMLTEST_DATADIR)
|
|
{
|
|
}
|
|
|
|
void tst_qqmllanguage::initTestCase()
|
|
{
|
|
QQmlDataTest::initTestCase();
|
|
if (dataDirectoryUrl().scheme() == QLatin1String("qrc"))
|
|
engine.addImportPath(dataDirectory());
|
|
else
|
|
QVERIFY2(QDir::setCurrent(dataDirectory()), qPrintable("Could not chdir to " + dataDirectory()));
|
|
|
|
|
|
defaultImportPathList = engine.importPathList();
|
|
|
|
registerTypes();
|
|
// Registered here because it uses testFileUrl
|
|
qmlRegisterType(testFileUrl("CompositeType.qml"), "Test", 1, 0, "RegisteredCompositeType");
|
|
qmlRegisterType(testFileUrl("CompositeType.DoesNotExist.qml"), "Test", 1, 0, "RegisteredCompositeType2");
|
|
qmlRegisterType(testFileUrl("invalidRoot.1.qml"), "Test", 1, 0, "RegisteredCompositeType3");
|
|
qmlRegisterType(testFileUrl("CompositeTypeWithEnum.qml"), "Test", 1, 0, "RegisteredCompositeTypeWithEnum");
|
|
qmlRegisterType(testFileUrl("CompositeTypeWithAttachedProperty.qml"), "Test", 1, 0, "RegisteredCompositeTypeWithAttachedProperty");
|
|
|
|
// Registering the TestType class in other modules should have no adverse effects
|
|
qmlRegisterType<TestType>("org.qtproject.TestPre", 1, 0, "Test");
|
|
|
|
qmlRegisterType<TestType>("org.qtproject.Test", 0, 0, "TestTP");
|
|
qmlRegisterType<TestType>("org.qtproject.Test", 1, 0, "Test");
|
|
qmlRegisterType<TestType>("org.qtproject.Test", 1, 5, "Test");
|
|
qmlRegisterType<TestType2>("org.qtproject.Test", 1, 8, "Test");
|
|
qmlRegisterType<TestType>("org.qtproject.Test", 1, 9, "OldTest");
|
|
qmlRegisterType<TestType2>("org.qtproject.Test", 1, 12, "Test");
|
|
|
|
// Registering the TestType class in other modules should have no adverse effects
|
|
qmlRegisterType<TestType>("org.qtproject.TestPost", 1, 0, "Test");
|
|
|
|
// Create locale-specific file
|
|
// For POSIX, this will just be data/I18nType.qml, since POSIX is 7-bit
|
|
// For iso8859-1 locale, this will just be data/I18nType?????.qml where ????? is 5 8-bit characters
|
|
// For utf-8 locale, this will be data/I18nType??????????.qml where ?????????? is 5 8-bit characters, UTF-8 encoded
|
|
if (dataDirectoryUrl().scheme() != QLatin1String("qrc")) {
|
|
QFile in(testFileUrl(QLatin1String("I18nType30.qml")).toLocalFile());
|
|
QVERIFY2(in.open(QIODevice::ReadOnly), qPrintable(QString::fromLatin1("Cannot open '%1': %2").arg(in.fileName(), in.errorString())));
|
|
QFile out(testFileUrl(QString::fromUtf8("I18nType\303\201\303\242\303\243\303\244\303\245.qml")).toLocalFile());
|
|
QVERIFY2(out.open(QIODevice::WriteOnly), qPrintable(QString::fromLatin1("Cannot open '%1': %2").arg(out.fileName(), out.errorString())));
|
|
out.write(in.readAll());
|
|
}
|
|
|
|
// Register a Composite Singleton.
|
|
qmlRegisterSingletonType(testFileUrl("singleton/RegisteredCompositeSingletonType.qml"), "org.qtproject.Test", 1, 0, "RegisteredSingleton");
|
|
}
|
|
|
|
void tst_qqmllanguage::aliasPropertyChangeSignals()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("aliasPropertyChangeSignals.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->property("test").toBool(), true);
|
|
}
|
|
|
|
// QTCREATORBUG-2769
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("aliasPropertyChangeSignals.2.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->property("test").toBool(), true);
|
|
}
|
|
}
|
|
|
|
// Tests property initializers
|
|
void tst_qqmllanguage::propertyInit()
|
|
{
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("propertyInit.1.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->property("test").toInt(), 1);
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("propertyInit.2.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->property("test").toInt(), 123);
|
|
}
|
|
}
|
|
|
|
// Test that registration order doesn't break type availability
|
|
// QTBUG-16878
|
|
void tst_qqmllanguage::registrationOrder()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("registrationOrder.qml"));
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
QCOMPARE(o->metaObject(), &MyVersion2Class::staticMetaObject);
|
|
}
|
|
|
|
void tst_qqmllanguage::readonly()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("readonly.qml"));
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->property("test1").toInt(), 10);
|
|
QCOMPARE(o->property("test2").toInt(), 18);
|
|
QCOMPARE(o->property("test3").toInt(), 13);
|
|
|
|
o->setProperty("testData", 13);
|
|
|
|
QCOMPARE(o->property("test1").toInt(), 10);
|
|
QCOMPARE(o->property("test2").toInt(), 22);
|
|
QCOMPARE(o->property("test3").toInt(), 13);
|
|
|
|
o->setProperty("testData2", 2);
|
|
|
|
QCOMPARE(o->property("test1").toInt(), 10);
|
|
QCOMPARE(o->property("test2").toInt(), 22);
|
|
QCOMPARE(o->property("test3").toInt(), 2);
|
|
|
|
o->setProperty("test1", 11);
|
|
o->setProperty("test2", 11);
|
|
o->setProperty("test3", 11);
|
|
|
|
QCOMPARE(o->property("test1").toInt(), 10);
|
|
QCOMPARE(o->property("test2").toInt(), 22);
|
|
QCOMPARE(o->property("test3").toInt(), 2);
|
|
}
|
|
|
|
void tst_qqmllanguage::readonlyObjectProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("readonlyObjectProperty.qml"));
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QQmlProperty prop(o.data(), QStringLiteral("subObject"), &engine);
|
|
QVERIFY(!prop.isWritable());
|
|
QVERIFY(!prop.write(QVariant::fromValue(o.data())));
|
|
|
|
QObject *subObject = qvariant_cast<QObject*>(prop.read());
|
|
QVERIFY(subObject);
|
|
QCOMPARE(subObject->property("readWrite").toInt(), int(42));
|
|
subObject->setProperty("readWrite", QVariant::fromValue(int(100)));
|
|
QCOMPARE(subObject->property("readWrite").toInt(), int(100));
|
|
}
|
|
|
|
void tst_qqmllanguage::receivers()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("receivers.qml"));
|
|
|
|
QScopedPointer<MyReceiversTestObject> o(qobject_cast<MyReceiversTestObject*>(component.create()));
|
|
QVERIFY(o != nullptr);
|
|
QCOMPARE(o->mySignalCount(), 1);
|
|
QCOMPARE(o->propChangedCount(), 2);
|
|
QCOMPARE(o->myUnconnectedSignalCount(), 0);
|
|
|
|
QVERIFY(o->isSignalConnected(QMetaMethod::fromSignal(&MyReceiversTestObject::mySignal)));
|
|
QVERIFY(o->isSignalConnected(QMetaMethod::fromSignal(&MyReceiversTestObject::propChanged)));
|
|
QVERIFY(!o->isSignalConnected(QMetaMethod::fromSignal(&MyReceiversTestObject::myUnconnectedSignal)));
|
|
}
|
|
|
|
void tst_qqmllanguage::registeredCompositeType()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("registeredCompositeType.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
}
|
|
|
|
// QTBUG-43582
|
|
void tst_qqmllanguage::registeredCompositeTypeWithEnum()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("registeredCompositeTypeWithEnum.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->property("enumValue0").toInt(), static_cast<int>(MyCompositeBaseType::EnumValue0));
|
|
QCOMPARE(o->property("enumValue42").toInt(), static_cast<int>(MyCompositeBaseType::EnumValue42));
|
|
QCOMPARE(o->property("enumValue15").toInt(), static_cast<int>(MyCompositeBaseType::ScopedCompositeEnum::EnumValue15));
|
|
}
|
|
|
|
// QTBUG-43581
|
|
void tst_qqmllanguage::registeredCompositeTypeWithAttachedProperty()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("registeredCompositeTypeWithAttachedProperty.qml"));
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->property("attachedProperty").toString(), QStringLiteral("test"));
|
|
}
|
|
|
|
// QTBUG-18268
|
|
void tst_qqmllanguage::remoteLoadCrash()
|
|
{
|
|
ThreadedTestHTTPServer server(dataDirectory());
|
|
|
|
QQmlComponent component(&engine);
|
|
component.setData("import QtQuick 2.0; Text {}", server.url("/remoteLoadCrash.qml"));
|
|
while (component.isLoading())
|
|
QCoreApplication::processEvents( QEventLoop::ExcludeUserInputEvents | QEventLoop::WaitForMoreEvents, 50);
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
}
|
|
|
|
void tst_qqmllanguage::signalWithDefaultArg()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("signalWithDefaultArg.qml"));
|
|
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject *>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
|
|
QCOMPARE(object->property("signalCount").toInt(), 0);
|
|
QCOMPARE(object->property("signalArg").toInt(), 0);
|
|
|
|
emit object->signalWithDefaultArg();
|
|
QCOMPARE(object-> property("signalCount").toInt(), 1);
|
|
QCOMPARE(object->property("signalArg").toInt(), 5);
|
|
|
|
emit object->signalWithDefaultArg(15);
|
|
QCOMPARE(object->property("signalCount").toInt(), 2);
|
|
QCOMPARE(object->property("signalArg").toInt(), 15);
|
|
|
|
|
|
QMetaObject::invokeMethod(object.data(), "emitNoArgSignal");
|
|
QCOMPARE(object->property("signalCount").toInt(), 3);
|
|
QCOMPARE(object->property("signalArg").toInt(), 5);
|
|
|
|
QMetaObject::invokeMethod(object.data(), "emitArgSignal");
|
|
QCOMPARE(object->property("signalCount").toInt(), 4);
|
|
QCOMPARE(object->property("signalArg").toInt(), 22);
|
|
}
|
|
|
|
void tst_qqmllanguage::signalParameterTypes()
|
|
{
|
|
// bound signal handlers
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("signalParameterTypes.1.qml"));
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(obj != nullptr);
|
|
QVERIFY(obj->property("success").toBool());
|
|
}
|
|
|
|
// dynamic signal connections
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("signalParameterTypes.2.qml"));
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(obj != nullptr);
|
|
QVERIFY(obj->property("success").toBool());
|
|
}
|
|
|
|
// dynamic signal connections
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("signalParameterTypes.3.qml"));
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(obj != nullptr);
|
|
QVERIFY(obj->property("success").toBool());
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::functionParameterTypes()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("functionParameterTypes.qml"));
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY2(!obj.isNull(), qPrintable(component.errorString()));
|
|
const QMetaObject *metaObject = obj->metaObject();
|
|
|
|
{
|
|
QMetaMethod slot = metaObject->method(metaObject->indexOfSlot("returnItem()"));
|
|
QVERIFY(slot.isValid());
|
|
QCOMPARE(slot.returnType(), QMetaType::QObjectStar);
|
|
QObject *returnedPtr = nullptr;
|
|
slot.invoke(obj.data(), Qt::DirectConnection, Q_RETURN_ARG(QObject*, returnedPtr));
|
|
QCOMPARE(returnedPtr, obj.data());
|
|
}
|
|
|
|
{
|
|
QMetaMethod slot = metaObject->method(metaObject->indexOfSlot("takeString(QString)"));
|
|
QVERIFY(slot.isValid());
|
|
QCOMPARE(slot.parameterCount(), 1);
|
|
QCOMPARE(slot.parameterType(0), int(QMetaType::QString));
|
|
}
|
|
}
|
|
|
|
// QTBUG-20639
|
|
void tst_qqmllanguage::globalEnums()
|
|
{
|
|
qRegisterMetaType<MyEnum1Class::EnumA>();
|
|
qRegisterMetaType<MyEnum2Class::EnumB>();
|
|
qRegisterMetaType<Qt::TextFormat>();
|
|
|
|
QQmlComponent component(&engine, testFileUrl("globalEnums.qml"));
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
MyEnum1Class *enum1Class = o->findChild<MyEnum1Class *>(QString::fromLatin1("enum1Class"));
|
|
QVERIFY(enum1Class != nullptr);
|
|
QVERIFY(enum1Class->getValue() == -1);
|
|
|
|
MyEnumDerivedClass *enum2Class = o->findChild<MyEnumDerivedClass *>(QString::fromLatin1("enumDerivedClass"));
|
|
QVERIFY(enum2Class != nullptr);
|
|
QVERIFY(enum2Class->getValueA() == -1);
|
|
QVERIFY(enum2Class->getValueB() == -1);
|
|
QVERIFY(enum2Class->getValueC() == 0);
|
|
QVERIFY(enum2Class->getValueD() == 0);
|
|
QVERIFY(enum2Class->getValueE() == -1);
|
|
QVERIFY(enum2Class->getValueE2() == -1);
|
|
|
|
QVERIFY(enum2Class->property("aValue") == 0);
|
|
QVERIFY(enum2Class->property("bValue") == 0);
|
|
QVERIFY(enum2Class->property("cValue") == 0);
|
|
QVERIFY(enum2Class->property("dValue") == 0);
|
|
QVERIFY(enum2Class->property("eValue") == 0);
|
|
QVERIFY(enum2Class->property("e2Value") == 0);
|
|
|
|
QSignalSpy signalA(enum2Class, SIGNAL(valueAChanged(MyEnum1Class::EnumA)));
|
|
QSignalSpy signalB(enum2Class, SIGNAL(valueBChanged(MyEnum2Class::EnumB)));
|
|
|
|
QMetaObject::invokeMethod(o.data(), "setEnumValues");
|
|
|
|
QVERIFY(enum1Class->getValue() == MyEnum1Class::A_13);
|
|
QVERIFY(enum2Class->getValueA() == MyEnum1Class::A_11);
|
|
QVERIFY(enum2Class->getValueB() == MyEnum2Class::B_37);
|
|
QVERIFY(enum2Class->getValueC() == Qt::RichText);
|
|
QVERIFY(enum2Class->getValueD() == Qt::ElideMiddle);
|
|
QVERIFY(enum2Class->getValueE() == MyEnum2Class::E_14);
|
|
QVERIFY(enum2Class->getValueE2() == MyEnum2Class::E_76);
|
|
|
|
QVERIFY(signalA.size() == 1);
|
|
QVERIFY(signalB.size() == 1);
|
|
|
|
QVERIFY(enum2Class->property("aValue") == MyEnum1Class::A_11);
|
|
QVERIFY(enum2Class->property("bValue") == 37);
|
|
QVERIFY(enum2Class->property("cValue") == 1);
|
|
QVERIFY(enum2Class->property("dValue") == 2);
|
|
QVERIFY(enum2Class->property("eValue") == 14);
|
|
QVERIFY(enum2Class->property("e2Value") == 76);
|
|
}
|
|
|
|
void tst_qqmllanguage::lowercaseEnumRuntime_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
|
|
QTest::newRow("enum from normal type") << "lowercaseEnumRuntime.1.qml";
|
|
QTest::newRow("enum from singleton type") << "lowercaseEnumRuntime.2.qml";
|
|
}
|
|
|
|
void tst_qqmllanguage::lowercaseEnumRuntime()
|
|
{
|
|
QFETCH(QString, file);
|
|
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
VERIFY_ERRORS(0);
|
|
delete component.create();
|
|
}
|
|
|
|
void tst_qqmllanguage::lowercaseEnumCompileTime_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
|
|
QTest::newRow("assignment to int property") << "lowercaseEnumCompileTime.1.qml";
|
|
QTest::newRow("assignment to enum property") << "lowercaseEnumCompileTime.2.qml";
|
|
}
|
|
|
|
void tst_qqmllanguage::lowercaseEnumCompileTime()
|
|
{
|
|
QFETCH(QString, file);
|
|
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
VERIFY_ERRORS(0);
|
|
delete component.create();
|
|
}
|
|
|
|
void tst_qqmllanguage::scopedEnum()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scopedEnum.qml"));
|
|
|
|
QScopedPointer<MyTypeObject> o(qobject_cast<MyTypeObject *>(component.create()));
|
|
QVERIFY(o != nullptr);
|
|
|
|
QCOMPARE(o->scopedEnum(), MyTypeObject::MyScopedEnum::ScopedVal1);
|
|
QCOMPARE(o->intProperty(), (int)MyTypeObject::MyScopedEnum::ScopedVal2);
|
|
QCOMPARE(o->property("listValue").toInt(), (int)MyTypeObject::MyScopedEnum::ScopedVal3);
|
|
QCOMPARE(o->property("noScope").toInt(), (int)MyTypeObject::MyScopedEnum::ScopedVal1);
|
|
|
|
QMetaObject::invokeMethod(o.data(), "assignNewValue");
|
|
QCOMPARE(o->scopedEnum(), MyTypeObject::MyScopedEnum::ScopedVal2);
|
|
QCOMPARE(o->property("noScope").toInt(), (int)MyTypeObject::MyScopedEnum::ScopedVal2);
|
|
}
|
|
|
|
void tst_qqmllanguage::scopedEnumsWithNameClash()
|
|
{
|
|
auto typeId = qmlRegisterUncreatableMetaObject(
|
|
ScopedEnumsWithNameClash::staticMetaObject,
|
|
"ScopedEnumsWithNameClashTest", 1, 0, "ScopedEnum", "Dummy reason");
|
|
auto registryGuard = qScopeGuard([typeId]() {
|
|
QQmlMetaType::unregisterType(typeId);
|
|
});
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("scopedEnumsWithNameClash.qml"));
|
|
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, "Previously registered enum will be overwritten due to name clash: ScopedEnumsWithNameClash.ScopedVal1");
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, "Previously registered enum will be overwritten due to name clash: ScopedEnumsWithNameClash.ScopedVal2");
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, "Previously registered enum will be overwritten due to name clash: ScopedEnumsWithNameClash.ScopedVal3");
|
|
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(obj != nullptr);
|
|
QVERIFY(obj->property("success").toBool());
|
|
}
|
|
|
|
void tst_qqmllanguage::scopedEnumsWithResolvedNameClash()
|
|
{
|
|
auto typeId = qmlRegisterUncreatableMetaObject(
|
|
ScopedEnumsWithResolvedNameClash::staticMetaObject,
|
|
"ScopedEnumsWithResolvedNameClashTest", 1, 0, "ScopedEnum", "Dummy reason");
|
|
auto registryGuard = qScopeGuard([typeId]() {
|
|
QQmlMetaType::unregisterType(typeId);
|
|
});
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("scopedEnumsWithResolvedNameClash.qml"));
|
|
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(obj != nullptr);
|
|
QVERIFY(obj->property("success").toBool());
|
|
}
|
|
|
|
class ObjectA : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
enum TestEnum {
|
|
Default = 42,
|
|
TestValue1,
|
|
TestValue2
|
|
};
|
|
Q_ENUM(TestEnum)
|
|
|
|
ObjectA() = default;
|
|
};
|
|
|
|
class ObjectB : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY(ObjectA::TestEnum testEnum READ testEnum WRITE setTestEnum NOTIFY testEnumChanged)
|
|
|
|
public:
|
|
ObjectB() = default;
|
|
ObjectA::TestEnum testEnum() const {return m_testEnum;}
|
|
|
|
public slots:
|
|
void setTestEnum(ObjectA::TestEnum testEnum) {auto old = m_testEnum; m_testEnum = testEnum; if (old != m_testEnum) testEnumChanged(m_testEnum);}
|
|
signals:
|
|
void testEnumChanged(ObjectA::TestEnum testEnum);
|
|
private:
|
|
ObjectA::TestEnum m_testEnum = ObjectA::Default;
|
|
};
|
|
|
|
class ObjectC : public ObjectB
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
ObjectC() = default;
|
|
};
|
|
|
|
void tst_qqmllanguage::enumNoScopeLeak()
|
|
{
|
|
qmlRegisterType<ObjectA>("test", 1, 0, "ObjectA");
|
|
qmlRegisterType<ObjectB>("test", 1, 0, "ObjectB");
|
|
qmlRegisterType<ObjectC>("test", 1, 0, "ObjectC");
|
|
QQmlEngine engine;
|
|
auto url = testFileUrl("enumScopeLeak.qml");
|
|
QQmlComponent component(&engine, url);
|
|
const auto msg = url.toString() + ":5:5: Unable to assign [undefined] to ObjectA::TestEnum";
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, msg.toUtf8().constData());
|
|
QScopedPointer<QObject> root(component.create());
|
|
QVERIFY(root);
|
|
QCOMPARE(root->property("testEnum").toInt(), ObjectA::Default);
|
|
}
|
|
|
|
void tst_qqmllanguage::qmlEnums()
|
|
{
|
|
QQmlEngine engine;
|
|
engine.setImportPathList(QStringList(defaultImportPathList) << testFile("lib"));
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("TypeWithEnum.qml"));
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o);
|
|
QCOMPARE(o->property("enumValue").toInt(), 1);
|
|
QCOMPARE(o->property("enumValue2").toInt(), 2);
|
|
QCOMPARE(o->property("scopedEnumValue").toInt(), 1);
|
|
|
|
QCOMPARE(o->property("otherEnumValue1").toInt(), 24);
|
|
QCOMPARE(o->property("otherEnumValue2").toInt(), 25);
|
|
QCOMPARE(o->property("otherEnumValue3").toInt(), 24);
|
|
QCOMPARE(o->property("otherEnumValue4").toInt(), 25);
|
|
QCOMPARE(o->property("otherEnumValue5").toInt(), 1);
|
|
QCOMPARE(o->property("otherEnumValue6").toInt(), -42);
|
|
}
|
|
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("usingTypeWithEnum.qml"));
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o);
|
|
QCOMPARE(o->property("enumValue").toInt(), 1);
|
|
QCOMPARE(o->property("enumValue2").toInt(), 0);
|
|
QCOMPARE(o->property("scopedEnumValue").toInt(), 2);
|
|
QCOMPARE(o->property("enumValueFromSingleton").toInt(), 42);
|
|
// while this next test verifies current duplication behavior, I'm not sure it should be codified
|
|
QCOMPARE(o->property("duplicatedEnumValueFromSingleton").toInt(), 2);
|
|
QCOMPARE(o->property("scopedEnumValueFromSingleton1").toInt(), 43);
|
|
QCOMPARE(o->property("scopedEnumValueFromSingleton2").toInt(), 2);
|
|
QCOMPARE(o->property("scopedEnumValueFromSingleton3").toInt(), 2);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::literals_data()
|
|
{
|
|
QTest::addColumn<QString>("property");
|
|
QTest::addColumn<QVariant>("value");
|
|
|
|
QTest::newRow("hex") << "n1" << QVariant(0xfe32);
|
|
// Octal integer literals are deprecated
|
|
// QTest::newRow("octal") << "n2" << QVariant(015);
|
|
QTest::newRow("fp1") << "n3" << QVariant(-4.2E11);
|
|
QTest::newRow("fp2") << "n4" << QVariant(.1e9);
|
|
QTest::newRow("fp3") << "n5" << QVariant(3e-12);
|
|
QTest::newRow("fp4") << "n6" << QVariant(3e+12);
|
|
QTest::newRow("fp5") << "n7" << QVariant(0.1e9);
|
|
QTest::newRow("large-int1") << "n8" << QVariant((double) 1152921504606846976);
|
|
QTest::newRow("large-int2") << "n9" << QVariant(100000000000000000000.);
|
|
|
|
QTest::newRow("special1") << "c1" << QVariant(QString("\b"));
|
|
QTest::newRow("special2") << "c2" << QVariant(QString("\f"));
|
|
QTest::newRow("special3") << "c3" << QVariant(QString("\n"));
|
|
QTest::newRow("special4") << "c4" << QVariant(QString("\r"));
|
|
QTest::newRow("special5") << "c5" << QVariant(QString("\t"));
|
|
QTest::newRow("special6") << "c6" << QVariant(QString("\v"));
|
|
QTest::newRow("special7") << "c7" << QVariant(QString("\'"));
|
|
QTest::newRow("special8") << "c8" << QVariant(QString("\""));
|
|
QTest::newRow("special9") << "c9" << QVariant(QString("\\"));
|
|
// We don't handle octal escape sequences
|
|
QTest::newRow("special10") << "c10" << QVariant(QString(1, QChar(0xa9)));
|
|
QTest::newRow("special11") << "c11" << QVariant(QString(1, QChar(0x00A9)));
|
|
}
|
|
|
|
void tst_qqmllanguage::literals()
|
|
{
|
|
QFETCH(QString, property);
|
|
QFETCH(QVariant, value);
|
|
|
|
QQmlComponent component(&engine, testFile("literals.qml"));
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property(property.toLatin1()), value);
|
|
}
|
|
|
|
void tst_qqmllanguage::objectDeletionNotify_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
|
|
QTest::newRow("property QtObject") << "objectDeletionNotify.1.qml";
|
|
QTest::newRow("property variant") << "objectDeletionNotify.2.qml";
|
|
QTest::newRow("property var") << "objectDeletionNotify.3.qml";
|
|
QTest::newRow("property var guard removed") << "objectDeletionNotify.4.qml";
|
|
}
|
|
|
|
void tst_qqmllanguage::objectDeletionNotify()
|
|
{
|
|
QFETCH(QString, file);
|
|
|
|
QQmlComponent component(&engine, testFile(file));
|
|
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("success").toBool(), true);
|
|
|
|
QMetaObject::invokeMethod(object.data(), "destroyObject");
|
|
|
|
// Process the deletion event
|
|
QCoreApplication::sendPostedEvents(nullptr, QEvent::DeferredDelete);
|
|
QCoreApplication::processEvents();
|
|
|
|
QCOMPARE(object->property("success").toBool(), true);
|
|
}
|
|
|
|
void tst_qqmllanguage::scopedProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("scopedProperties.qml"));
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
QVERIFY(o->property("success").toBool());
|
|
}
|
|
|
|
void tst_qqmllanguage::deepProperty()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("deepProperty.qml"));
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
QFont font = qvariant_cast<QFont>(qvariant_cast<QObject*>(o->property("someObject"))->property("font"));
|
|
QCOMPARE(font.family(), QStringLiteral("test"));
|
|
}
|
|
|
|
void tst_qqmllanguage::groupAssignmentFailure()
|
|
{
|
|
auto ep = std::make_unique<QQmlEngine>();
|
|
QTest::failOnWarning("QQmlComponent: Component destroyed while completion pending");
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, QRegularExpression(".*Invalid property assignment: url expected - Assigning null to incompatible properties in QML is deprecated. This will become a compile error in future versions of Qt..*"));
|
|
QQmlComponent component(ep.get(), testFileUrl("groupFailure.qml"));
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o);
|
|
ep.reset();
|
|
// ~QQmlComponent should not crash here
|
|
}
|
|
|
|
// Tests that the implicit import has lowest precedence, in the case where
|
|
// there are conflicting types and types only found in the local import.
|
|
// Tests that just check one (or the root) type are in ::importsOrder
|
|
void tst_qqmllanguage::implicitImportsLast()
|
|
{
|
|
if (qmlCheckTypes())
|
|
QSKIP("This test is about maintaining the same choice when type is ambiguous.");
|
|
|
|
if (engine.importPathList() == defaultImportPathList)
|
|
engine.addImportPath(testFile("lib"));
|
|
|
|
QQmlComponent component(&engine, testFileUrl("localOrderTest.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QVERIFY(QString(object->metaObject()->superClass()->superClass()->className())
|
|
.startsWith(QLatin1String("QQuickMouseArea")));
|
|
QObject* object2 = object->property("item").value<QObject*>();
|
|
QVERIFY(object2 != nullptr);
|
|
QCOMPARE(QString(object2->metaObject()->superClass()->className()),
|
|
QLatin1String("QQuickRectangle"));
|
|
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
void tst_qqmllanguage::getSingletonInstance(QQmlEngine& engine, const char* fileName, const char* propertyName, QObject** result /* out */)
|
|
{
|
|
QVERIFY(fileName != nullptr);
|
|
QVERIFY(propertyName != nullptr);
|
|
|
|
if (!fileName || !propertyName)
|
|
return;
|
|
|
|
QQmlComponent component(&engine, testFileUrl(fileName));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
|
|
getSingletonInstance(object.data(), propertyName, result);
|
|
}
|
|
|
|
void tst_qqmllanguage::getSingletonInstance(QObject* o, const char* propertyName, QObject** result /* out */)
|
|
{
|
|
QVERIFY(o != nullptr);
|
|
QVERIFY(propertyName != nullptr);
|
|
|
|
if (!o || !propertyName)
|
|
return;
|
|
|
|
QVariant variant = o->property(propertyName);
|
|
QVERIFY(variant.isValid());
|
|
|
|
QObject *singleton = nullptr;
|
|
if (variant.typeId() == qMetaTypeId<QObject *>())
|
|
singleton = variant.value<QObject*>();
|
|
else if (variant.typeId() == qMetaTypeId<QJSValue>())
|
|
singleton = variant.value<QJSValue>().toQObject();
|
|
|
|
QVERIFY(singleton != nullptr);
|
|
*result = singleton;
|
|
}
|
|
|
|
void verifyCompositeSingletonPropertyValues(QObject* o, const char* n1, int v1, const char* n2, int v2)
|
|
{
|
|
QCOMPARE(o->property(n1).typeId(), (int)QMetaType::Int);
|
|
QCOMPARE(o->property(n1), QVariant(v1));
|
|
|
|
QCOMPARE(o->property(n2).typeId(), QMetaType::QString);
|
|
QString numStr;
|
|
QCOMPARE(o->property(n2), QVariant(QString(QLatin1String("Test value: ")).append(numStr.setNum(v2))));
|
|
}
|
|
|
|
// Reads values from a composite singleton type
|
|
void tst_qqmllanguage::compositeSingletonProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest1.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 125, "value2", -55);
|
|
}
|
|
|
|
// Checks that the addresses of the composite singletons used in the same
|
|
// engine are the same.
|
|
void tst_qqmllanguage::compositeSingletonSameEngine()
|
|
{
|
|
QObject* s1 = nullptr;
|
|
getSingletonInstance(engine, "singletonTest2.qml", "singleton1", &s1);
|
|
QVERIFY(s1 != nullptr);
|
|
s1->setProperty("testProp2", QVariant(13));
|
|
|
|
QObject* s2 = nullptr;
|
|
getSingletonInstance(engine, "singletonTest3.qml", "singleton2", &s2);
|
|
QVERIFY(s2 != nullptr);
|
|
QCOMPARE(s2->property("testProp2"), QVariant(13));
|
|
|
|
QCOMPARE(s1, s2);
|
|
}
|
|
|
|
// Checks that the addresses of the composite singletons used in different
|
|
// engines are different.
|
|
void tst_qqmllanguage::compositeSingletonDifferentEngine()
|
|
{
|
|
QQmlEngine e2;
|
|
|
|
QObject* s1 = nullptr;
|
|
getSingletonInstance(engine, "singletonTest2.qml", "singleton1", &s1);
|
|
QVERIFY(s1 != nullptr);
|
|
s1->setProperty("testProp2", QVariant(13));
|
|
|
|
QObject* s2 = nullptr;
|
|
getSingletonInstance(e2, "singletonTest3.qml", "singleton2", &s2);
|
|
QVERIFY(s2 != nullptr);
|
|
QCOMPARE(s2->property("testProp2"), QVariant(25));
|
|
|
|
QVERIFY(s1 != s2);
|
|
}
|
|
|
|
// pragma Singleton in a non-type qml file fails
|
|
void tst_qqmllanguage::compositeSingletonNonTypeError()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest4.qml"));
|
|
VERIFY_ERRORS("singletonTest4.error.txt");
|
|
}
|
|
|
|
// Loads the singleton using a namespace qualifier
|
|
void tst_qqmllanguage::compositeSingletonQualifiedNamespace()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest5.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 125, "value2", -55);
|
|
|
|
// lets verify that the singleton instance we are using is the same
|
|
// when loaded through another file (without namespace!)
|
|
QObject *s1 = nullptr;
|
|
getSingletonInstance(o.data(), "singletonInstance", &s1);
|
|
QVERIFY(s1 != nullptr);
|
|
|
|
QObject* s2 = nullptr;
|
|
getSingletonInstance(engine, "singletonTest5a.qml", "singletonInstance", &s2);
|
|
QVERIFY(s2 != nullptr);
|
|
|
|
QCOMPARE(s1, s2);
|
|
}
|
|
|
|
// Loads a singleton from a module
|
|
void tst_qqmllanguage::compositeSingletonModule()
|
|
{
|
|
engine.addImportPath(testFile("singleton/module"));
|
|
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest6.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 125, "value2", -55);
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value3", 125, "value4", -55);
|
|
|
|
// lets verify that the singleton instance we are using is the same
|
|
// when loaded through another file
|
|
QObject *s1 = nullptr;
|
|
getSingletonInstance(o.data(), "singletonInstance", &s1);
|
|
QVERIFY(s1 != nullptr);
|
|
|
|
QObject* s2 = nullptr;
|
|
getSingletonInstance(engine, "singletonTest6a.qml", "singletonInstance", &s2);
|
|
QVERIFY(s2 != nullptr);
|
|
|
|
QCOMPARE(s1, s2);
|
|
}
|
|
|
|
// Loads a singleton from a module with a higher version
|
|
void tst_qqmllanguage::compositeSingletonModuleVersioned()
|
|
{
|
|
engine.addImportPath(testFile("singleton/module"));
|
|
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest7.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 225, "value2", 55);
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value3", 225, "value4", 55);
|
|
|
|
// lets verify that the singleton instance we are using is the same
|
|
// when loaded through another file
|
|
QObject *s1 = nullptr;
|
|
getSingletonInstance(o.data(), "singletonInstance", &s1);
|
|
QVERIFY(s1 != nullptr);
|
|
|
|
QObject* s2 = nullptr;
|
|
getSingletonInstance(engine, "singletonTest7a.qml", "singletonInstance", &s2);
|
|
QVERIFY(s2 != nullptr);
|
|
|
|
QCOMPARE(s1, s2);
|
|
}
|
|
|
|
// Loads a singleton from a module with a qualified namespace
|
|
void tst_qqmllanguage::compositeSingletonModuleQualified()
|
|
{
|
|
engine.addImportPath(testFile("singleton/module"));
|
|
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest8.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 225, "value2", 55);
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value3", 225, "value4", 55);
|
|
|
|
// lets verify that the singleton instance we are using is the same
|
|
// when loaded through another file
|
|
QObject *s1 = nullptr;
|
|
getSingletonInstance(o.data(), "singletonInstance", &s1);
|
|
QVERIFY(s1 != nullptr);
|
|
|
|
QObject* s2 = nullptr;
|
|
getSingletonInstance(engine, "singletonTest8a.qml", "singletonInstance", &s2);
|
|
QVERIFY(s2 != nullptr);
|
|
|
|
QCOMPARE(s1, s2);
|
|
}
|
|
|
|
// Tries to instantiate a type with a pragma Singleton and fails
|
|
void tst_qqmllanguage::compositeSingletonInstantiateError()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest9.qml"));
|
|
VERIFY_ERRORS("singletonTest9.error.txt");
|
|
}
|
|
|
|
// Having a composite singleton type as dynamic property type is allowed
|
|
void tst_qqmllanguage::compositeSingletonDynamicPropertyError()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest10.qml"));
|
|
VERIFY_ERRORS(0);
|
|
}
|
|
|
|
void tst_qqmllanguage::compositeSingletonDynamicSignalAndJavaScriptPragma()
|
|
{
|
|
{
|
|
// Having a composite singleton type as dynamic signal parameter succeeds
|
|
// (like C++ singleton)
|
|
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest11.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 99, "value2", -55);
|
|
}
|
|
{
|
|
// Load a composite singleton type and a javascript file that has .pragma library
|
|
// in it. This will make sure that the javascript .pragma does not get mixed with
|
|
// the pragma Singleton changes.
|
|
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest16.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
// The value1 that is read from the SingletonType was changed from 125 to 99
|
|
// above. As the type is a singleton and
|
|
// the engine has not been destroyed, we just retrieve the old instance and
|
|
// the value is still 99.
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 99, "value2", 333);
|
|
}
|
|
}
|
|
|
|
// Use qmlRegisterType to register a qml composite type with pragma Singleton defined in it.
|
|
// This will fail as qmlRegisterType will only instantiate CompositeTypes.
|
|
void tst_qqmllanguage::compositeSingletonQmlRegisterTypeError()
|
|
{
|
|
qmlRegisterType(testFileUrl("singleton/registeredComposite/CompositeType.qml"),
|
|
"CompositeSingletonTest", 1, 0, "RegisteredCompositeType");
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest12.qml"));
|
|
VERIFY_ERRORS("singletonTest12.error.txt");
|
|
}
|
|
|
|
// Qmldir defines a type as a singleton, but the qml file does not have a pragma Singleton.
|
|
void tst_qqmllanguage::compositeSingletonQmldirNoPragmaError()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest13.qml"));
|
|
VERIFY_ERRORS("singletonTest13.error.txt");
|
|
}
|
|
|
|
// Invalid singleton definition in the qmldir file results in an error
|
|
void tst_qqmllanguage::compositeSingletonQmlDirError()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest14.qml"));
|
|
VERIFY_ERRORS("singletonTest14.error.txt");
|
|
}
|
|
|
|
// Load a remote composite singleton type via qmldir that defines the type as a singleton
|
|
void tst_qqmllanguage::compositeSingletonRemote()
|
|
{
|
|
ThreadedTestHTTPServer server(dataDirectory());
|
|
|
|
QFile f(testFile("singletonTest15.qml"));
|
|
QVERIFY(f.open(QIODevice::ReadOnly));
|
|
QByteArray contents = f.readAll();
|
|
f.close();
|
|
|
|
contents.replace(QByteArrayLiteral("{{ServerBaseUrl}}"), server.baseUrl().toString().toUtf8());
|
|
|
|
QQmlComponent component(&engine);
|
|
component.setData(contents, testFileUrl("singletonTest15.qml"));
|
|
|
|
while (component.isLoading())
|
|
QCoreApplication::processEvents( QEventLoop::ExcludeUserInputEvents | QEventLoop::WaitForMoreEvents, 50);
|
|
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 525, "value2", 355);
|
|
}
|
|
|
|
// Reads values from a Singleton accessed through selectors.
|
|
void tst_qqmllanguage::compositeSingletonSelectors()
|
|
{
|
|
QQmlEngine e2;
|
|
QQmlFileSelector qmlSelector(&e2);
|
|
qmlSelector.setExtraSelectors(QStringList() << "basicSelector");
|
|
QQmlComponent component(&e2, testFileUrl("singletonTest1.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 625, "value2", 455);
|
|
}
|
|
|
|
// Reads values from a Singleton that was registered through the C++ API:
|
|
// qmlRegisterSingletonType.
|
|
void tst_qqmllanguage::compositeSingletonRegistered()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("singletonTest17.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
verifyCompositeSingletonPropertyValues(o.data(), "value1", 925, "value2", 755);
|
|
}
|
|
|
|
void tst_qqmllanguage::compositeSingletonCircular()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("circularSingleton.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QQmlTestMessageHandler messageHandler;
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
// ensure we aren't hitting the recursion warning
|
|
QVERIFY2(messageHandler.messages().isEmpty(), qPrintable(messageHandler.messageString()));
|
|
|
|
QCOMPARE(o->property("value").toInt(), 2);
|
|
}
|
|
|
|
void tst_qqmllanguage::singletonsHaveContextAndEngine()
|
|
{
|
|
QObject *qmlSingleton = nullptr;
|
|
getSingletonInstance(engine, "singletonTest18.qml", "qmlSingleton", &qmlSingleton);
|
|
QVERIFY(qmlContext(qmlSingleton));
|
|
QCOMPARE(qmlEngine(qmlSingleton), &engine);
|
|
|
|
QObject *jsSingleton = nullptr;
|
|
getSingletonInstance(engine, "singletonTest18.qml", "jsSingleton", &jsSingleton);
|
|
QVERIFY(qmlContext(jsSingleton));
|
|
QCOMPARE(qmlEngine(jsSingleton), &engine);
|
|
|
|
QObject *cppSingleton = nullptr;
|
|
getSingletonInstance(engine, "singletonTest18.qml", "cppSingleton", &cppSingleton);
|
|
QVERIFY(qmlContext(cppSingleton));
|
|
QCOMPARE(qmlEngine(cppSingleton), &engine);
|
|
}
|
|
|
|
void tst_qqmllanguage::customParserBindingScopes()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("customParserBindingScopes.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
QPointer<QObject> child = qvariant_cast<QObject*>(o->property("child"));
|
|
QVERIFY(!child.isNull());
|
|
QCOMPARE(child->property("testProperty").toInt(), 42);
|
|
}
|
|
|
|
void tst_qqmllanguage::customParserEvaluateEnum()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("customParserEvaluateEnum.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::customParserProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("customParserProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
SimpleObjectWithCustomParser *testObject = qobject_cast<SimpleObjectWithCustomParser*>(o.data());
|
|
QVERIFY(testObject);
|
|
QCOMPARE(testObject->customBindingsCount(), 0);
|
|
QCOMPARE(testObject->intProperty(), 42);
|
|
QCOMPARE(testObject->property("qmlString").toString(), QStringLiteral("Hello"));
|
|
QVERIFY(!testObject->property("someObject").isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::customParserWithExtendedObject()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("customExtendedParserProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
SimpleObjectWithCustomParser *testObject = qobject_cast<SimpleObjectWithCustomParser*>(o.data());
|
|
QVERIFY(testObject);
|
|
QCOMPARE(testObject->customBindingsCount(), 0);
|
|
QCOMPARE(testObject->intProperty(), 42);
|
|
QCOMPARE(testObject->property("qmlString").toString(), QStringLiteral("Hello"));
|
|
QVERIFY(!testObject->property("someObject").isNull());
|
|
|
|
QVariant returnValue;
|
|
QVERIFY(QMetaObject::invokeMethod(o.data(), "getExtendedProperty", Q_RETURN_ARG(QVariant, returnValue)));
|
|
QCOMPARE(returnValue.toInt(), 1584);
|
|
}
|
|
|
|
void tst_qqmllanguage::nestedCustomParsers()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("nestedCustomParsers.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
SimpleObjectWithCustomParser *testObject = qobject_cast<SimpleObjectWithCustomParser*>(o.data());
|
|
QVERIFY(testObject);
|
|
QCOMPARE(testObject->customBindingsCount(), 1);
|
|
SimpleObjectWithCustomParser *nestedObject = qobject_cast<SimpleObjectWithCustomParser*>(testObject->property("nested").value<QObject*>());
|
|
QVERIFY(nestedObject);
|
|
QCOMPARE(nestedObject->customBindingsCount(), 1);
|
|
}
|
|
|
|
void tst_qqmllanguage::preservePropertyCacheOnGroupObjects()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("preservePropertyCacheOnGroupObjects.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
QObject *subObject = qvariant_cast<QObject*>(o->property("subObject"));
|
|
QVERIFY(subObject);
|
|
QCOMPARE(subObject->property("value").toInt(), 42);
|
|
|
|
QQmlData *ddata = QQmlData::get(subObject);
|
|
QVERIFY(ddata);
|
|
const QQmlPropertyCache *subCache = ddata->propertyCache.data();
|
|
QVERIFY(subCache);
|
|
const QQmlPropertyData *pd = subCache->property(QStringLiteral("newProperty"), /*object*/nullptr, /*context*/nullptr);
|
|
QVERIFY(pd);
|
|
QCOMPARE(pd->propType(), QMetaType::fromType<int>());
|
|
}
|
|
|
|
void tst_qqmllanguage::propertyCacheInSync()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("propertyCacheInSync.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
QObject *anchors = qvariant_cast<QObject*>(o->property("anchors"));
|
|
QVERIFY(anchors);
|
|
QQmlVMEMetaObject *vmemo = QQmlVMEMetaObject::get(anchors);
|
|
QVERIFY(vmemo);
|
|
QQmlPropertyCache::ConstPtr vmemoCache = vmemo->propertyCache();
|
|
QVERIFY(vmemoCache);
|
|
QQmlData *ddata = QQmlData::get(anchors);
|
|
QVERIFY(ddata);
|
|
QVERIFY(ddata->propertyCache);
|
|
// Those always have to be in sync and correct.
|
|
QCOMPARE(ddata->propertyCache, vmemoCache);
|
|
QCOMPARE(anchors->property("margins").toInt(), 50);
|
|
}
|
|
|
|
void tst_qqmllanguage::rootObjectInCreationNotForSubObjects()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("rootObjectInCreationNotForSubObjects.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
// QQmlComponent should have set this back to false anyway
|
|
QQmlData *ddata = QQmlData::get(o.data());
|
|
QVERIFY(!ddata->rootObjectInCreation);
|
|
|
|
QObject *subObject = qvariant_cast<QObject*>(o->property("subObject"));
|
|
QVERIFY(!subObject);
|
|
|
|
qmlExecuteDeferred(o.data());
|
|
|
|
subObject = qvariant_cast<QObject*>(o->property("subObject"));
|
|
QVERIFY(subObject);
|
|
|
|
ddata = QQmlData::get(subObject);
|
|
// This should never have been set in the first place as there is no
|
|
// QQmlComponent to set it back to false.
|
|
QVERIFY(!ddata->rootObjectInCreation);
|
|
}
|
|
|
|
// QTBUG-63036
|
|
void tst_qqmllanguage::lazyDeferredSubObject()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("lazyDeferredSubObject.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QObject *subObject = qvariant_cast<QObject *>(object->property("subObject"));
|
|
QVERIFY(subObject);
|
|
|
|
QCOMPARE(object->objectName(), QStringLiteral("custom"));
|
|
QCOMPARE(subObject->objectName(), QStringLiteral("custom"));
|
|
}
|
|
|
|
// QTBUG-63200
|
|
void tst_qqmllanguage::deferredProperties()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("deferredProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QObject *innerObj = object->findChild<QObject *>(QStringLiteral("innerobj"));
|
|
QVERIFY(!innerObj);
|
|
|
|
QObject *outerObj = object->findChild<QObject *>(QStringLiteral("outerobj"));
|
|
QVERIFY(!outerObj);
|
|
|
|
QObject *groupProperty = object->property("groupProperty").value<QObject *>();
|
|
QVERIFY(!groupProperty);
|
|
|
|
QQmlListProperty<QObject> listProperty = object->property("listProperty").value<QQmlListProperty<QObject>>();
|
|
QCOMPARE(listProperty.count(&listProperty), 0);
|
|
|
|
QQmlData *qmlData = QQmlData::get(object.data());
|
|
QVERIFY(qmlData);
|
|
|
|
QCOMPARE(qmlData->deferredData.size(), 2); // MyDeferredListProperty.qml + deferredListProperty.qml
|
|
QCOMPARE(qmlData->deferredData.first()->bindings.size(), 3); // "innerobj", "innerlist1", "innerlist2"
|
|
QCOMPARE(qmlData->deferredData.last()->bindings.size(), 3); // "outerobj", "outerlist1", "outerlist2"
|
|
|
|
qmlExecuteDeferred(object.data());
|
|
|
|
QCOMPARE(qmlData->deferredData.size(), 0);
|
|
|
|
innerObj = object->findChild<QObject *>(QStringLiteral("innerobj")); // MyDeferredListProperty.qml
|
|
QVERIFY(innerObj);
|
|
QCOMPARE(innerObj->property("wasCompleted"), QVariant(true));
|
|
|
|
outerObj = object->findChild<QObject *>(QStringLiteral("outerobj")); // deferredListProperty.qml
|
|
QVERIFY(outerObj);
|
|
QCOMPARE(outerObj->property("wasCompleted"), QVariant(true));
|
|
|
|
groupProperty = object->property("groupProperty").value<QObject *>();
|
|
QCOMPARE(groupProperty, outerObj);
|
|
|
|
listProperty = object->property("listProperty").value<QQmlListProperty<QObject>>();
|
|
QCOMPARE(listProperty.count(&listProperty), 4);
|
|
|
|
QCOMPARE(listProperty.at(&listProperty, 0)->objectName(), QStringLiteral("innerlist1")); // MyDeferredListProperty.qml
|
|
QCOMPARE(listProperty.at(&listProperty, 0)->property("wasCompleted"), QVariant(true));
|
|
QCOMPARE(listProperty.at(&listProperty, 1)->objectName(), QStringLiteral("innerlist2")); // MyDeferredListProperty.qml
|
|
QCOMPARE(listProperty.at(&listProperty, 1)->property("wasCompleted"), QVariant(true));
|
|
|
|
QCOMPARE(listProperty.at(&listProperty, 2)->objectName(), QStringLiteral("outerlist1")); // deferredListProperty.qml
|
|
QCOMPARE(listProperty.at(&listProperty, 2)->property("wasCompleted"), QVariant(true));
|
|
QCOMPARE(listProperty.at(&listProperty, 3)->objectName(), QStringLiteral("outerlist2")); // deferredListProperty.qml
|
|
QCOMPARE(listProperty.at(&listProperty, 3)->property("wasCompleted"), QVariant(true));
|
|
}
|
|
|
|
static void beginDeferredOnce(QQmlEnginePrivate *enginePriv,
|
|
const QQmlProperty &property, QQmlComponentPrivate::DeferredState *deferredState)
|
|
{
|
|
QObject *object = property.object();
|
|
QQmlData *ddata = QQmlData::get(object);
|
|
Q_ASSERT(!ddata->deferredData.isEmpty());
|
|
|
|
int propertyIndex = property.index();
|
|
|
|
for (auto dit = ddata->deferredData.rbegin(); dit != ddata->deferredData.rend(); ++dit) {
|
|
QQmlData::DeferredData *deferData = *dit;
|
|
|
|
auto range = deferData->bindings.equal_range(propertyIndex);
|
|
if (range.first == deferData->bindings.end())
|
|
continue;
|
|
|
|
QQmlComponentPrivate::ConstructionState state;
|
|
state.setCompletePending(true);
|
|
|
|
state.initCreator(deferData->context->parent(), deferData->compilationUnit,
|
|
QQmlRefPointer<QQmlContextData>());
|
|
|
|
enginePriv->inProgressCreations++;
|
|
|
|
std::deque<const QV4::CompiledData::Binding *> reversedBindings;
|
|
std::copy(range.first, range.second, std::front_inserter(reversedBindings));
|
|
state.creator()->beginPopulateDeferred(deferData->context);
|
|
for (const QV4::CompiledData::Binding *binding: reversedBindings)
|
|
state.creator()->populateDeferredBinding(property, deferData->deferredIdx, binding);
|
|
state.creator()->finalizePopulateDeferred();
|
|
state.appendErrors(state.creator()->errors);
|
|
|
|
deferredState->push_back(std::move(state));
|
|
|
|
// Cleanup any remaining deferred bindings for this property, also in inner contexts,
|
|
// to avoid executing them later and overriding the property that was just populated.
|
|
while (dit != ddata->deferredData.rend()) {
|
|
(*dit)->bindings.remove(propertyIndex);
|
|
++dit;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void testExecuteDeferredOnce(const QQmlProperty &property)
|
|
{
|
|
QObject *object = property.object();
|
|
QQmlData *data = QQmlData::get(object);
|
|
if (data && !data->deferredData.isEmpty() && !data->wasDeleted(object)) {
|
|
QQmlEnginePrivate *ep = QQmlEnginePrivate::get(data->context->engine());
|
|
|
|
QQmlComponentPrivate::DeferredState state;
|
|
beginDeferredOnce(ep, property, &state);
|
|
|
|
// Release deferred data for those compilation units that no longer have deferred bindings
|
|
data->releaseDeferredData();
|
|
|
|
QQmlComponentPrivate::completeDeferred(ep, &state);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::executeDeferredPropertiesOnce()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("deferredProperties.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QObjectList innerObjsAtCreation = object->findChildren<QObject *>(QStringLiteral("innerobj"));
|
|
QVERIFY(innerObjsAtCreation.isEmpty());
|
|
|
|
QObjectList outerObjsAtCreation = object->findChildren<QObject *>(QStringLiteral("outerobj"));
|
|
QVERIFY(outerObjsAtCreation.isEmpty());
|
|
|
|
QObject *groupProperty = object->property("groupProperty").value<QObject *>();
|
|
QVERIFY(!groupProperty);
|
|
|
|
QQmlListProperty<QObject> listProperty = object->property("listProperty").value<QQmlListProperty<QObject>>();
|
|
QCOMPARE(listProperty.count(&listProperty), 0);
|
|
|
|
QQmlData *qmlData = QQmlData::get(object.data());
|
|
QVERIFY(qmlData);
|
|
|
|
QCOMPARE(qmlData->deferredData.size(), 2); // MyDeferredListProperty.qml + deferredListProperty.qml
|
|
QCOMPARE(qmlData->deferredData.first()->bindings.size(), 3); // "innerobj", "innerlist1", "innerlist2"
|
|
QCOMPARE(qmlData->deferredData.last()->bindings.size(), 3); // "outerobj", "outerlist1", "outerlist2"
|
|
|
|
// first execution creates the outer object
|
|
testExecuteDeferredOnce(QQmlProperty(object.data(), "groupProperty"));
|
|
|
|
QCOMPARE(qmlData->deferredData.size(), 2); // MyDeferredListProperty.qml + deferredListProperty.qml
|
|
QCOMPARE(qmlData->deferredData.first()->bindings.size(), 2); // "innerlist1", "innerlist2"
|
|
QCOMPARE(qmlData->deferredData.last()->bindings.size(), 2); // "outerlist1", "outerlist2"
|
|
|
|
QObjectList innerObjsAfterFirstExecute = object->findChildren<QObject *>(QStringLiteral("innerobj")); // MyDeferredListProperty.qml
|
|
QVERIFY(innerObjsAfterFirstExecute.isEmpty());
|
|
|
|
QObjectList outerObjsAfterFirstExecute = object->findChildren<QObject *>(QStringLiteral("outerobj")); // deferredListProperty.qml
|
|
QCOMPARE(outerObjsAfterFirstExecute.size(), 1);
|
|
QCOMPARE(outerObjsAfterFirstExecute.first()->property("wasCompleted"), QVariant(true));
|
|
|
|
groupProperty = object->property("groupProperty").value<QObject *>();
|
|
QCOMPARE(groupProperty, outerObjsAfterFirstExecute.first());
|
|
|
|
listProperty = object->property("listProperty").value<QQmlListProperty<QObject>>();
|
|
QCOMPARE(listProperty.count(&listProperty), 0);
|
|
|
|
// re-execution does nothing (to avoid overriding the property)
|
|
testExecuteDeferredOnce(QQmlProperty(object.data(), "groupProperty"));
|
|
|
|
QCOMPARE(qmlData->deferredData.size(), 2); // MyDeferredListProperty.qml + deferredListProperty.qml
|
|
QCOMPARE(qmlData->deferredData.first()->bindings.size(), 2); // "innerlist1", "innerlist2"
|
|
QCOMPARE(qmlData->deferredData.last()->bindings.size(), 2); // "outerlist1", "outerlist2"
|
|
|
|
QObjectList innerObjsAfterSecondExecute = object->findChildren<QObject *>(QStringLiteral("innerobj")); // MyDeferredListProperty.qml
|
|
QVERIFY(innerObjsAfterSecondExecute.isEmpty());
|
|
|
|
QObjectList outerObjsAfterSecondExecute = object->findChildren<QObject *>(QStringLiteral("outerobj")); // deferredListProperty.qml
|
|
QCOMPARE(outerObjsAfterFirstExecute, outerObjsAfterSecondExecute);
|
|
|
|
groupProperty = object->property("groupProperty").value<QObject *>();
|
|
QCOMPARE(groupProperty, outerObjsAfterFirstExecute.first());
|
|
|
|
listProperty = object->property("listProperty").value<QQmlListProperty<QObject>>();
|
|
QCOMPARE(listProperty.count(&listProperty), 0);
|
|
|
|
// execution of a list property should execute all outer list bindings
|
|
testExecuteDeferredOnce(QQmlProperty(object.data(), "listProperty"));
|
|
|
|
QCOMPARE(qmlData->deferredData.size(), 0);
|
|
|
|
listProperty = object->property("listProperty").value<QQmlListProperty<QObject>>();
|
|
QCOMPARE(listProperty.count(&listProperty), 2);
|
|
|
|
QCOMPARE(listProperty.at(&listProperty, 0)->objectName(), QStringLiteral("outerlist1")); // deferredListProperty.qml
|
|
QCOMPARE(listProperty.at(&listProperty, 0)->property("wasCompleted"), QVariant(true));
|
|
QCOMPARE(listProperty.at(&listProperty, 1)->objectName(), QStringLiteral("outerlist2")); // deferredListProperty.qml
|
|
QCOMPARE(listProperty.at(&listProperty, 1)->property("wasCompleted"), QVariant(true));
|
|
}
|
|
|
|
class GroupType : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY(int foo READ getFoo WRITE setFoo BINDABLE bindableFoo)
|
|
Q_PROPERTY(int bar READ getBar WRITE setBar BINDABLE bindableBar)
|
|
Q_CLASSINFO("DeferredPropertyNames", "bar")
|
|
|
|
QProperty<int> m_foo { 0 };
|
|
QProperty<int> m_bar { 0 };
|
|
|
|
public:
|
|
int getFoo() const { return m_foo; }
|
|
void setFoo(int v) { m_foo = v; }
|
|
QBindable<int> bindableFoo() const { return QBindable<int>(&m_foo); }
|
|
|
|
int getBar() const { return m_bar; }
|
|
void setBar(int v) { m_bar = v; }
|
|
QBindable<int> bindableBar() const { return QBindable<int>(&m_bar); }
|
|
};
|
|
|
|
class ExtraDeferredProperties : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY(GroupType *group READ getGroup)
|
|
Q_CLASSINFO("DeferredPropertyNames", "group,MyQmlObject")
|
|
|
|
GroupType m_group;
|
|
|
|
public:
|
|
ExtraDeferredProperties(QObject *parent = nullptr) : QObject(parent) { }
|
|
|
|
GroupType *getGroup() { return &m_group; }
|
|
};
|
|
|
|
void tst_qqmllanguage::deferredProperties_extra()
|
|
{
|
|
// Note: because ExtraDeferredProperties defers only a `group` property, the
|
|
// deferral does not actually work.
|
|
qmlRegisterType<GroupType>("deferred.properties.extra", 1, 0, "GroupType");
|
|
qmlRegisterType<ExtraDeferredProperties>("deferred.properties.extra", 1, 0,
|
|
"ExtraDeferredProperties");
|
|
QQmlComponent component(&engine);
|
|
component.setData(R"(
|
|
import QtQuick
|
|
import Test 1.0
|
|
import deferred.properties.extra 1.0
|
|
|
|
ExtraDeferredProperties {
|
|
group.foo: 4
|
|
group.bar: 4
|
|
MyQmlObject.value: 1
|
|
}
|
|
)", QUrl());
|
|
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<ExtraDeferredProperties> object(
|
|
qobject_cast<ExtraDeferredProperties *>(component.create()));
|
|
QVERIFY(object);
|
|
|
|
QCOMPARE(object->getGroup()->getFoo(), 4); // not deferred (as group itself is not deferred)
|
|
QCOMPARE(object->getGroup()->getBar(), 0); // deferred, as per group's own deferred names
|
|
// but attached property is deferred:
|
|
QVERIFY(!qmlAttachedPropertiesObject<MyQmlObject>(object.get(), false));
|
|
|
|
qmlExecuteDeferred(object.get());
|
|
|
|
auto attached = qmlAttachedPropertiesObject<MyQmlObject>(object.get(), false);
|
|
QVERIFY(attached);
|
|
QCOMPARE(attached->property("value").toInt(), 1);
|
|
}
|
|
|
|
void tst_qqmllanguage::noChildEvents()
|
|
{
|
|
QQmlComponent component(&engine);
|
|
component.setData("import QtQml 2.0; import Test 1.0; MyQmlObject { property QtObject child: QtObject {} }", QUrl());
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<MyQmlObject> object(qobject_cast<MyQmlObject*>(component.create()));
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->childAddedEventCount(), 0);
|
|
}
|
|
|
|
void tst_qqmllanguage::earlyIdObjectAccess()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("earlyIdObjectAccess.qml"));
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
QVERIFY(o->property("success").toBool());
|
|
}
|
|
|
|
void tst_qqmllanguage::deleteSingletons()
|
|
{
|
|
QPointer<QObject> singleton;
|
|
{
|
|
QQmlEngine tmpEngine;
|
|
QQmlComponent component(&tmpEngine, testFileUrl("singletonTest5.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
QObject *s1 = nullptr;
|
|
getSingletonInstance(o.data(), "singletonInstance", &s1);
|
|
QVERIFY(s1 != nullptr);
|
|
singleton = s1;
|
|
QVERIFY(singleton.data() != nullptr);
|
|
}
|
|
QVERIFY(singleton.data() == nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::arrayBuffer_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::newRow("arraybuffer_property_get") << "arraybuffer_property_get.qml";
|
|
QTest::newRow("arraybuffer_property_set") << "arraybuffer_property_set.qml";
|
|
QTest::newRow("arraybuffer_signal_arg") << "arraybuffer_signal_arg.qml";
|
|
QTest::newRow("arraybuffer_method_arg") << "arraybuffer_method_arg.qml";
|
|
QTest::newRow("arraybuffer_method_return") << "arraybuffer_method_return.qml";
|
|
QTest::newRow("arraybuffer_method_overload") << "arraybuffer_method_overload.qml";
|
|
}
|
|
|
|
void tst_qqmllanguage::arrayBuffer()
|
|
{
|
|
QFETCH(QString, file);
|
|
QQmlComponent component(&engine, testFileUrl(file));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
QCOMPARE(object->property("ok").toBool(), true);
|
|
}
|
|
|
|
void tst_qqmllanguage::defaultListProperty()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("defaultListProperty.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
}
|
|
|
|
void tst_qqmllanguage::namespacedPropertyTypes()
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("namespacedPropertyTypes.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::qmlTypeCanBeResolvedByName_data()
|
|
{
|
|
QTest::addColumn<QUrl>("componentUrl");
|
|
|
|
// Built-in C++ types
|
|
QTest::newRow("C++ - Anonymous") << testFileUrl("quickTypeByName_anon.qml");
|
|
QTest::newRow("C++ - Named") << testFileUrl("quickTypeByName_named.qml");
|
|
|
|
// Composite types with a qmldir
|
|
QTest::newRow("QML - Anonymous - qmldir") << testFileUrl("compositeTypeByName_anon_qmldir.qml");
|
|
QTest::newRow("QML - Named - qmldir") << testFileUrl("compositeTypeByName_named_qmldir.qml");
|
|
}
|
|
|
|
void tst_qqmllanguage::qmlTypeCanBeResolvedByName()
|
|
{
|
|
QFETCH(QUrl, componentUrl);
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, componentUrl);
|
|
VERIFY_ERRORS(0);
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, "[object Object]"); // a bit crude, but it will do
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
}
|
|
|
|
// Tests for the QML-only extensions of instanceof. Tests for the regular JS
|
|
// instanceof belong in tst_qqmlecmascript!
|
|
void tst_qqmllanguage::instanceof_data()
|
|
{
|
|
QTest::addColumn<QUrl>("documentToTestIn");
|
|
QTest::addColumn<QVariant>("expectedValue");
|
|
|
|
// so the way this works is that the name of the test tag defines the test
|
|
// to run.
|
|
//
|
|
// the expectedValue is either a boolean true or false for whether the two
|
|
// operands are indeed an instanceof each other, or a string for the
|
|
// expected error message.
|
|
|
|
// assert that value types don't convert to QObject
|
|
QTest::newRow("1 instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("true instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("\"foobar\" instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
|
|
// assert that Managed don't either
|
|
QTest::newRow("new String(\"foobar\") instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("new Object() instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("new Date() instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
|
|
// test that simple QtQml comparisons work
|
|
QTest::newRow("qtobjectInstance instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("qtobjectInstance instanceof Timer")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("timerInstance instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("timerInstance instanceof Timer")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("connectionsInstance instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("connectionsInstance instanceof Timer")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("connectionsInstance instanceof Connections")
|
|
<< testFileUrl("instanceof_qtqml.qml")
|
|
<< QVariant(true);
|
|
|
|
// make sure they still work when imported with a qualifier
|
|
QTest::newRow("qtobjectInstance instanceof QmlImport.QtObject")
|
|
<< testFileUrl("instanceof_qtqml_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("qtobjectInstance instanceof QmlImport.Timer")
|
|
<< testFileUrl("instanceof_qtqml_qualified.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("timerInstance instanceof QmlImport.QtObject")
|
|
<< testFileUrl("instanceof_qtqml_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("timerInstance instanceof QmlImport.Timer")
|
|
<< testFileUrl("instanceof_qtqml_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("connectionsInstance instanceof QmlImport.QtObject")
|
|
<< testFileUrl("instanceof_qtqml_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("connectionsInstance instanceof QmlImport.Timer")
|
|
<< testFileUrl("instanceof_qtqml_qualified.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("connectionsInstance instanceof QmlImport.Connections")
|
|
<< testFileUrl("instanceof_qtqml_qualified.qml")
|
|
<< QVariant(true);
|
|
|
|
// test that Quick C++ types work ok
|
|
QTest::newRow("itemInstance instanceof QtObject")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("itemInstance instanceof Timer")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("itemInstance instanceof Rectangle")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("rectangleInstance instanceof Item")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("rectangleInstance instanceof Rectangle")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("rectangleInstance instanceof MouseArea")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("mouseAreaInstance instanceof Item")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("mouseAreaInstance instanceof Rectangle")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("mouseAreaInstance instanceof MouseArea")
|
|
<< testFileUrl("instanceof_qtquick.qml")
|
|
<< QVariant(true);
|
|
|
|
// test that unqualified quick composite types work ok
|
|
QTest::newRow("rectangleInstance instanceof CustomRectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("customRectangleInstance instanceof Rectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleInstance instanceof Item")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleWithPropInstance instanceof CustomRectangleWithProp")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleWithPropInstance instanceof CustomRectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(false); // ### XXX: QTBUG-58477
|
|
QTest::newRow("customRectangleWithPropInstance instanceof Rectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleInstance instanceof MouseArea")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("customMouseAreaInstance instanceof MouseArea")
|
|
<< testFileUrl("instanceof_qtquick_composite.qml")
|
|
<< QVariant(true);
|
|
|
|
// test that they still work when qualified
|
|
QTest::newRow("rectangleInstance instanceof CustomImport.CustomRectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("customRectangleInstance instanceof QuickImport.Rectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleInstance instanceof QuickImport.Item")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleWithPropInstance instanceof CustomImport.CustomRectangleWithProp")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleWithPropInstance instanceof CustomImport.CustomRectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(false); // ### XXX: QTBUG-58477
|
|
QTest::newRow("customRectangleWithPropInstance instanceof QuickImport.Rectangle")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(true);
|
|
QTest::newRow("customRectangleInstance instanceof QuickImport.MouseArea")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(false);
|
|
QTest::newRow("customMouseAreaInstance instanceof QuickImport.MouseArea")
|
|
<< testFileUrl("instanceof_qtquick_composite_qualified.qml")
|
|
<< QVariant(true);
|
|
}
|
|
|
|
void tst_qqmllanguage::instanceof()
|
|
{
|
|
QFETCH(QUrl, documentToTestIn);
|
|
QFETCH(QVariant, expectedValue);
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, documentToTestIn);
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o != nullptr);
|
|
|
|
QQmlExpression expr(engine.contextForObject(o.data()), nullptr, QString::fromLatin1(QTest::currentDataTag()));
|
|
QVariant ret = expr.evaluate();
|
|
|
|
if (expectedValue.typeId() == QMetaType::Bool) {
|
|
// no error expected
|
|
QVERIFY2(!expr.hasError(), qPrintable(expr.error().description()));
|
|
bool returnValue = ret.toBool();
|
|
|
|
if (QTest::currentDataTag() == QLatin1String("customRectangleWithPropInstance instanceof CustomRectangle") ||
|
|
QTest::currentDataTag() == QLatin1String("customRectangleWithPropInstance instanceof CustomImport.CustomRectangle"))
|
|
QCOMPARE(returnValue, expectedValue.toBool());
|
|
} else {
|
|
QVERIFY(expr.hasError());
|
|
QCOMPARE(expr.error().description(), expectedValue.toString());
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::concurrentLoadQmlDir()
|
|
{
|
|
ThreadedTestHTTPServer server(dataDirectory());
|
|
QString serverdir = server.urlString("/lib/");
|
|
engine.setImportPathList(QStringList(defaultImportPathList) << serverdir);
|
|
|
|
QQmlComponent component(&engine, testFileUrl("concurrentLoad_main.qml"));
|
|
QTRY_VERIFY(component.isReady());
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
engine.setImportPathList(defaultImportPathList);
|
|
}
|
|
|
|
// Test that deleting an object and then accessing it doesn't crash.
|
|
// QTBUG-44153
|
|
class ObjectCreator : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public slots:
|
|
QObject *create() { return (new ObjectCreator); }
|
|
void del() { delete this; }
|
|
};
|
|
|
|
void tst_qqmllanguage::accessDeletedObject()
|
|
{
|
|
QQmlEngine engine;
|
|
|
|
QScopedPointer<ObjectCreator> creator(new ObjectCreator);
|
|
engine.rootContext()->setContextProperty("objectCreator", creator.get());
|
|
QQmlComponent component(&engine, testFileUrl("accessDeletedObject.qml"));
|
|
VERIFY_ERRORS(0);
|
|
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::lowercaseTypeNames()
|
|
{
|
|
QCOMPARE(qmlRegisterType<QObject>("Test", 1, 0, "lowerCaseTypeName"), -1);
|
|
QCOMPARE(qmlRegisterSingletonType<QObject>("Test", 1, 0, "lowerCaseTypeName", nullptr), -1);
|
|
}
|
|
|
|
void tst_qqmllanguage::thisInQmlScope()
|
|
{
|
|
QQmlEngine engine;
|
|
|
|
QQmlComponent component(&engine, testFileUrl("thisInQmlScope.qml"));
|
|
QTRY_VERIFY(component.isReady());
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
QCOMPARE(o->property("x"), QVariant(42));
|
|
QCOMPARE(o->property("y"), QVariant(42));
|
|
QCOMPARE(o->property("a"), QVariant(42));
|
|
QCOMPARE(o->property("b"), QVariant(42));
|
|
}
|
|
|
|
void tst_qqmllanguage::valueTypeGroupPropertiesInBehavior()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("groupPropertyInPropertyValueSource.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QObject *animation = qmlContext(o.data())->contextProperty("animation").value<QObject*>();
|
|
QVERIFY(animation);
|
|
|
|
QCOMPARE(animation->property("easing").value<QEasingCurve>().type(), QEasingCurve::InOutQuad);
|
|
}
|
|
|
|
void tst_qqmllanguage::retrieveQmlTypeId()
|
|
{
|
|
// Register in reverse order to provoke wrong minor version matching.
|
|
int id2 = qmlRegisterType<QObject>("Test", 2, 3, "SomeTestType");
|
|
int id1 = qmlRegisterType<QObject>("Test", 2, 1, "SomeTestType");
|
|
QCOMPARE(qmlTypeId("Test", 2, 1, "SomeTestType"), id1);
|
|
QCOMPARE(qmlTypeId("Test", 2, 2, "SomeTestType"), id1);
|
|
QCOMPARE(qmlTypeId("Test", 2, 3, "SomeTestType"), id2);
|
|
|
|
// Error cases
|
|
QCOMPARE(qmlTypeId("Test", 2, 0, "SomeTestType"), -1);
|
|
QCOMPARE(qmlTypeId("Test", 2, 3, "DoesNotExist"), -1);
|
|
QCOMPARE(qmlTypeId("DoesNotExist", 2, 3, "SomeTestType"), -1);
|
|
|
|
// Must also work for other types (defined in testtpes.cpp)
|
|
QVERIFY(qmlTypeId("Test", 1, 0, "MyExtendedUncreateableBaseClass") >= 0);
|
|
QVERIFY(qmlTypeId("Test", 1, 0, "MyUncreateableBaseClass") >= 0);
|
|
QVERIFY(qmlTypeId("Test", 1, 0, "MyTypeObjectSingleton") >= 0);
|
|
}
|
|
|
|
void tst_qqmllanguage::polymorphicFunctionLookup()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("polymorphicFunctionLookup.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QVERIFY(o->property("ok").toBool());
|
|
}
|
|
|
|
void tst_qqmllanguage::anchorsToParentInPropertyChanges()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("anchorsToParentInPropertyChagnes.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
QTRY_COMPARE(o->property("edgeWidth").toInt(), 200);
|
|
}
|
|
|
|
void tst_qqmllanguage::typeWrapperToVariant()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("typeWrapperToVariant.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
QObject *connections = qvariant_cast<QObject *>(o->property("connections"));
|
|
QVERIFY(connections);
|
|
QObject *target = qvariant_cast<QObject *>(connections->property("target"));
|
|
QVERIFY(target);
|
|
}
|
|
|
|
void tst_qqmllanguage::extendedForeignTypes()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("foreignExtended.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QObject *extended = o->property("extended").value<QObject *>();
|
|
QVERIFY(extended);
|
|
QSignalSpy extensionChangedSpy(extended, SIGNAL(extensionChanged()));
|
|
QSignalSpy extensionChangedWithValueSpy(extended, SIGNAL(extensionChangedWithValue(int)));
|
|
QVERIFY(extensionChangedWithValueSpy.isValid());
|
|
|
|
QCOMPARE(o->property("extendedBase").toInt(), 43);
|
|
QCOMPARE(o->property("extendedExtension").toInt(), 42);
|
|
QCOMPARE(o->property("foreignExtendedExtension").toInt(), 42);
|
|
|
|
QCOMPARE(extensionChangedSpy.size(), 0);
|
|
extended->setProperty("extension", 44);
|
|
QCOMPARE(extensionChangedSpy.size(), 1);
|
|
QCOMPARE(extensionChangedWithValueSpy.size(), 1);
|
|
QCOMPARE(o->property("extendedChangeCount").toInt(), 1);
|
|
QCOMPARE(o->property("extendedExtension").toInt(), 44);
|
|
|
|
QCOMPARE(o->property("foreignObjectName").toString(), QLatin1String("foreign"));
|
|
QCOMPARE(o->property("foreignExtendedObjectName").toString(), QLatin1String("foreignExtended"));
|
|
QCOMPARE(o->property("extendedInvokable").toInt(), 123);
|
|
QCOMPARE(o->property("extendedSlot").toInt(), 456);
|
|
|
|
QObject *extension = qmlExtendedObject(extended);
|
|
QVERIFY(extension != nullptr);
|
|
QVERIFY(qobject_cast<Extension *>(extension) != nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::foreignTypeSingletons() {
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("foreignSingleton.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QCOMPARE(o->property("number").toInt(), 42);
|
|
}
|
|
|
|
void tst_qqmllanguage::selfReference()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("SelfReference.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QQmlComponentPrivate *componentPrivate = QQmlComponentPrivate::get(&component);
|
|
auto compilationUnit = componentPrivate->compilationUnit;
|
|
QVERIFY(compilationUnit);
|
|
|
|
const QMetaObject *metaObject = o->metaObject();
|
|
QMetaProperty selfProperty = metaObject->property(metaObject->indexOfProperty("self"));
|
|
QCOMPARE(selfProperty.metaType().id(), compilationUnit->typeIds.id.id());
|
|
|
|
QByteArray typeName = selfProperty.typeName();
|
|
QVERIFY(typeName.endsWith('*'));
|
|
typeName = typeName.chopped(1);
|
|
QCOMPARE(typeName, metaObject->className());
|
|
|
|
QMetaMethod selfFunction = metaObject->method(metaObject->indexOfMethod("returnSelf()"));
|
|
QVERIFY(selfFunction.isValid());
|
|
QCOMPARE(selfFunction.returnType(), compilationUnit->typeIds.id.id());
|
|
|
|
QMetaMethod selfSignal;
|
|
|
|
for (int i = metaObject->methodOffset(); i < metaObject->methodCount(); ++i) {
|
|
QMetaMethod method = metaObject->method(i);
|
|
if (method.isValid() && method.name().startsWith("blah")) {
|
|
selfSignal = method;
|
|
break;
|
|
}
|
|
}
|
|
|
|
QVERIFY(selfSignal.isValid());
|
|
QCOMPARE(selfSignal.parameterCount(), 1);
|
|
QCOMPARE(selfSignal.parameterType(0), compilationUnit->typeIds.id.id());
|
|
}
|
|
|
|
void tst_qqmllanguage::selfReferencingSingleton()
|
|
{
|
|
QQmlEngine engine;
|
|
engine.addImportPath(dataDirectory());
|
|
|
|
QPointer<QObject> singletonPointer;
|
|
{
|
|
QQmlComponent component(&engine);
|
|
component.setData(QByteArray(R"(import QtQml 2.0
|
|
import selfreferencingsingletonmodule 1.0
|
|
QtObject {
|
|
property SelfReferencingSingleton singletonPointer: SelfReferencingSingleton
|
|
})"), QUrl());
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
singletonPointer = o->property("singletonPointer").value<QObject*>();
|
|
}
|
|
|
|
QVERIFY(!singletonPointer.isNull());
|
|
QCOMPARE(singletonPointer->property("dummy").toInt(), 42);
|
|
}
|
|
|
|
void tst_qqmllanguage::listContainingDeletedObject()
|
|
{
|
|
QQmlEngine engine;
|
|
auto url = testFileUrl("listContainingDeleted.qml");
|
|
const QString message = url.toString() + ":24: TypeError: Cannot read property 'enabled' of null";
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, message.toUtf8().data());
|
|
QQmlComponent comp(&engine, url);
|
|
QScopedPointer<QObject> root(comp.create());
|
|
QVERIFY(root);
|
|
|
|
auto cmp = root->property("a").value<QQmlComponent*>();
|
|
auto o = cmp->create();
|
|
|
|
QMetaObject::invokeMethod(root.get(), "doAssign", Q_ARG(QVariant, QVariant::fromValue(o)));
|
|
delete o;
|
|
QMetaObject::invokeMethod(root.get(), "use");
|
|
|
|
}
|
|
|
|
void tst_qqmllanguage::overrideSingleton()
|
|
{
|
|
auto check = [](const QString &name, const QByteArray &singletonElement) {
|
|
const QByteArray testQml = "import Test 1.0\n"
|
|
"import QtQml 2.0\n"
|
|
"QtObject { objectName: " + singletonElement + ".objectName }";
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, nullptr);
|
|
component.setData(testQml, QUrl("singleton.qml"));
|
|
QVERIFY(component.isReady());
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QCOMPARE(obj->objectName(), name);
|
|
};
|
|
|
|
check("statically registered", "BareSingleton");
|
|
|
|
BareSingleton singleton;
|
|
singleton.setObjectName("dynamically registered");
|
|
qmlRegisterSingletonInstance("Test", 1, 0, "BareSingleton", &singleton);
|
|
|
|
check("dynamically registered", "BareSingleton");
|
|
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg,
|
|
"singleton.qml:3:12: TypeError: Cannot read property 'objectName' of undefined");
|
|
check("", "UncreatableSingleton");
|
|
|
|
qmlRegisterSingletonInstance("Test", 1, 0, "UncreatableSingleton",
|
|
UncreatableSingleton::instance());
|
|
check("uncreatable", "UncreatableSingleton");
|
|
}
|
|
|
|
class AttachedObject;
|
|
class InnerObject : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY(bool revisionedProperty READ revisionedProperty WRITE setRevisionedProperty
|
|
NOTIFY revisionedPropertyChanged REVISION 2)
|
|
|
|
public:
|
|
InnerObject(QObject *parent = nullptr) : QObject(parent) {}
|
|
|
|
bool revisionedProperty() const { return m_revisionedProperty; }
|
|
void setRevisionedProperty(bool revisionedProperty)
|
|
{
|
|
if (revisionedProperty != m_revisionedProperty) {
|
|
m_revisionedProperty = revisionedProperty;
|
|
emit revisionedPropertyChanged();
|
|
}
|
|
}
|
|
|
|
static AttachedObject *qmlAttachedProperties(QObject *object);
|
|
|
|
signals:
|
|
Q_REVISION(2) void revisionedPropertyChanged();
|
|
|
|
private:
|
|
bool m_revisionedProperty = false;
|
|
};
|
|
|
|
class AttachedObject : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY(InnerObject *attached READ attached CONSTANT)
|
|
|
|
public:
|
|
explicit AttachedObject(QObject *parent = nullptr) :
|
|
QObject(parent),
|
|
m_attached(new InnerObject(this))
|
|
{}
|
|
|
|
InnerObject *attached() const { return m_attached; }
|
|
|
|
private:
|
|
InnerObject *m_attached;
|
|
};
|
|
|
|
class OuterObject : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
explicit OuterObject(QObject *parent = nullptr) : QObject(parent) {}
|
|
};
|
|
|
|
AttachedObject *InnerObject::qmlAttachedProperties(QObject *object)
|
|
{
|
|
return new AttachedObject(object);
|
|
}
|
|
|
|
QML_DECLARE_TYPE(InnerObject)
|
|
QML_DECLARE_TYPEINFO(InnerObject, QML_HAS_ATTACHED_PROPERTIES)
|
|
|
|
void tst_qqmllanguage::revisionedPropertyOfAttachedObjectProperty()
|
|
{
|
|
qmlRegisterAnonymousType<AttachedObject>("foo", 2);
|
|
qmlRegisterType<InnerObject>("foo", 2, 0, "InnerObject");
|
|
qmlRegisterType<InnerObject, 2>("foo", 2, 2, "InnerObject");
|
|
qmlRegisterType<OuterObject>("foo", 2, 2, "OuterObject");
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine);
|
|
component.setData("import foo 2.2\n"
|
|
"OuterObject {\n"
|
|
" InnerObject.attached.revisionedProperty: true\n"
|
|
"}", QUrl());
|
|
|
|
QVERIFY(component.isReady());
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(!obj.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponent()
|
|
{
|
|
QFETCH(QUrl, componentUrl);
|
|
QFETCH(QColor, color);
|
|
QFETCH(int, width);
|
|
QFETCH(bool, checkProperties);
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, componentUrl);
|
|
QScopedPointer<QObject> o(component.create());
|
|
if (component.isError()) {
|
|
qDebug() << component.errorString();
|
|
}
|
|
QVERIFY(!o.isNull());
|
|
if (checkProperties) {
|
|
auto icInstance = o->findChild<QObject *>("icInstance");
|
|
QVERIFY(icInstance);
|
|
QCOMPARE(icInstance->property("color").value<QColor>(),color);
|
|
QCOMPARE(icInstance->property("width").value<qreal>(), width);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponent_data()
|
|
{
|
|
QTest::addColumn<QUrl>("componentUrl");
|
|
QTest::addColumn<QColor>("color");
|
|
QTest::addColumn<int>("width");
|
|
QTest::addColumn<bool>("checkProperties");
|
|
|
|
QTest::newRow("Usage from other component") << testFileUrl("inlineComponentUser1.qml") << QColorConstants::Blue << 24 << true;
|
|
QTest::newRow("Reexport") << testFileUrl("inlineComponentUser2.qml") << QColorConstants::Svg::green << 24 << true;
|
|
QTest::newRow("Usage in same component") << testFileUrl("inlineComponentUser3.qml") << QColorConstants::Blue << 24 << true;
|
|
|
|
QTest::newRow("Resolution happens at instantiation") << testFileUrl("inlineComponentUser4.qml") << QColorConstants::Blue << 24 << true;
|
|
QTest::newRow("Non-toplevel IC is found") << testFileUrl("inlineComponentUser5.qml") << QColorConstants::Svg::red << 24 << true;
|
|
|
|
QTest::newRow("Resolved in correct order") << testFileUrl("inlineComponentOrder.qml") << QColorConstants::Blue << 200 << true;
|
|
|
|
QTest::newRow("ID resolves correctly") << testFileUrl("inlineComponentWithId.qml") << QColorConstants::Svg::red << 42 << true;
|
|
QTest::newRow("Alias resolves correctly") << testFileUrl("inlineComponentWithAlias.qml") << QColorConstants::Svg::lime << 42 << true;
|
|
|
|
QTest::newRow("Two inline components in same do not crash (QTBUG-86989)") << testFileUrl("twoInlineComponents.qml") << QColor() << 0 << false;
|
|
QTest::newRow("Inline components used in same file (QTBUG-89173)") << testFileUrl("inlineComponentsSameFile.qml") << QColor() << 0 << false;
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponentReferenceCycle_data()
|
|
{
|
|
QTest::addColumn<QUrl>("componentUrl");
|
|
|
|
QTest::newRow("Simple cycle") << testFileUrl("icSimpleCycle.qml");
|
|
QTest::newRow("Via property") << testFileUrl("icCycleViaProperty.qml");
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponentReferenceCycle()
|
|
{
|
|
QFETCH(QUrl, componentUrl);
|
|
QQmlEngine engine;
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, "QQmlComponent: Component is not ready");
|
|
QQmlComponent component(&engine, componentUrl);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o.isNull());
|
|
QVERIFY(component.isError());
|
|
QCOMPARE(component.errorString(), componentUrl.toString() + QLatin1String(":-1 Inline components form a cycle!\n"));
|
|
}
|
|
|
|
void tst_qqmllanguage::nestedInlineComponentNotAllowed()
|
|
{
|
|
QQmlEngine engine;
|
|
auto url = testFileUrl("nestedIC.qml");
|
|
QQmlComponent component(&engine, url);
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, "QQmlComponent: Component is not ready");
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(component.isError());
|
|
QCOMPARE(component.errorString(), QLatin1String("%1:%2").arg(url.toString(), QLatin1String("5 Nested inline components are not supported\n")));
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponentStaticTypeResolution()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("InlineComponentChild.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(o);
|
|
QCOMPARE(o->property("i").toInt(), 42);
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponentInSingleton()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("singletonICTest.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY(!o.isNull());
|
|
auto untyped = o->property("singleton1");
|
|
QVERIFY(untyped.isValid());
|
|
auto singleton1 = untyped.value<QObject*>();
|
|
QVERIFY(singleton1);
|
|
QCOMPARE(singleton1->property("iProp").value<int>(), 42);
|
|
QCOMPARE(singleton1->property("sProp").value<QString>(), QString::fromLatin1("Hello, world"));
|
|
QVERIFY(!o.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::nonExistingInlineComponent_data()
|
|
{
|
|
QTest::addColumn<QUrl>("componentUrl");
|
|
QTest::addColumn<QString>("errorMessage");
|
|
QTest::addColumn<int>("line");
|
|
QTest::addColumn<int>("column");
|
|
|
|
QTest::newRow("Property type") << testFileUrl("nonExistingICUser1.qml") << QString("Type InlineComponentProvider has no inline component type called NonExisting") << 4 << 5;
|
|
QTest::newRow("Instantiation") << testFileUrl("nonExistingICUser2.qml") << QString("Type InlineComponentProvider has no inline component type called NotExisting") << 4 << 5;
|
|
QTest::newRow("Inheritance") << testFileUrl("nonExistingICUser3.qml") << QString("Type InlineComponentProvider has no inline component type called NotExisting") << 3 << 1;
|
|
QTest::newRow("From singleton") << testFileUrl("nonExistingICUser4.qml") << QString("Type MySingleton.SingletonTypeWithIC has no inline component type called NonExisting") << 5 << 5;
|
|
|
|
QTest::newRow("Cannot access parent inline components from child") << testFileUrl("nonExistingICUser5.qml") << QString("Type InlineComponentProviderChild has no inline component type called StyledRectangle") << 4 << 5;
|
|
}
|
|
|
|
void tst_qqmllanguage::nonExistingInlineComponent()
|
|
{
|
|
QFETCH(QUrl, componentUrl);
|
|
QFETCH(QString, errorMessage);
|
|
QFETCH(int, line);
|
|
QFETCH(int, column);
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, componentUrl);
|
|
auto errors = component.errors();
|
|
QCOMPARE(errors.size(), 1);
|
|
const auto &error = errors.first();
|
|
QCOMPARE(error.description(), errorMessage);
|
|
QCOMPARE(error.line(), line);
|
|
QCOMPARE(error.column(), column);
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponentFoundBeforeOtherImports()
|
|
{
|
|
QQmlEngine engine;
|
|
QUrl url = testFileUrl("inlineComponentFoundBeforeOtherImports.qml");
|
|
QQmlComponent component(&engine, url);
|
|
|
|
QTest::ignoreMessage(QtMsgType::QtInfoMsg, "Created");
|
|
QScopedPointer<QObject> root {component.create()};
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponentDuplicateNameError()
|
|
{
|
|
QQmlEngine engine;
|
|
QUrl url = testFileUrl("inlineComponentDuplicateName.qml");
|
|
QQmlComponent component(&engine, url);
|
|
|
|
QString message = QLatin1String("%1:5 Inline component names must be unique per file\n").arg(url.toString());
|
|
QScopedPointer<QObject> root {component.create()};
|
|
QVERIFY(root.isNull());
|
|
QVERIFY(component.isError());
|
|
QCOMPARE(component.errorString(), message);
|
|
}
|
|
|
|
void tst_qqmllanguage::inlineComponentWithAliasInstantiatedWithNewProperties()
|
|
{
|
|
// this tests that metaobjects are resolved in the correct order
|
|
// so that inline components are fully resolved before they are used
|
|
// in their parent component
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("inlineComponentWithAliasInstantiated.qml"));
|
|
QScopedPointer<QObject> root {component.create()};
|
|
QVERIFY2(root, qPrintable(component.errorString()));
|
|
QCOMPARE(root->property("result").toString(), "Bar");
|
|
}
|
|
|
|
struct QJSValueConvertible {
|
|
|
|
Q_GADGET
|
|
|
|
public:
|
|
QString msg;
|
|
};
|
|
|
|
bool operator==(const QJSValueConvertible &lhs, const QJSValueConvertible &rhs) {
|
|
return lhs.msg == rhs.msg;
|
|
}
|
|
|
|
class TestItem : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY( QVector<QPointF> positions MEMBER m_points )
|
|
Q_PROPERTY( QSet<QByteArray> barrays MEMBER m_barrays )
|
|
Q_PROPERTY( QVector<QJSValueConvertible> convertibles MEMBER m_convertibles)
|
|
|
|
public:
|
|
TestItem() = default;
|
|
QVector< QPointF > m_points;
|
|
QSet<QByteArray> m_barrays;
|
|
QVector<QJSValueConvertible> m_convertibles;
|
|
};
|
|
|
|
|
|
Q_DECLARE_METATYPE(QVector<QPointF>);
|
|
Q_DECLARE_METATYPE(QSet<QByteArray>);
|
|
Q_DECLARE_METATYPE(QJSValueConvertible);
|
|
Q_DECLARE_METATYPE(QVector<QJSValueConvertible>);
|
|
|
|
void tst_qqmllanguage::arrayToContainer()
|
|
{
|
|
QMetaType::registerConverter< QJSValue, QJSValueConvertible >(
|
|
|
|
[](const QJSValue& value)
|
|
{
|
|
return QJSValueConvertible{value.toString()};
|
|
}
|
|
);
|
|
QQmlEngine engine;
|
|
qmlRegisterType<TestItem>("qt.test", 1, 0, "TestItem");
|
|
QVector<QPointF> points { QPointF (2.0, 3.0) };
|
|
QSet<QByteArray> barrays { QByteArray("hello"), QByteArray("world") };
|
|
engine.rootContext()->setContextProperty("test", QVariant::fromValue(points));
|
|
QQmlComponent component(&engine, testFileUrl("arrayToContainer.qml"));
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<TestItem> root(qobject_cast<TestItem *>(component.createWithInitialProperties( {{"vector", QVariant::fromValue(points)}, {"myset", QVariant::fromValue(barrays)} } )));
|
|
QVERIFY(root);
|
|
QCOMPARE(root->m_points.at(0), QPointF (2.0, 3.0) );
|
|
QVERIFY(root->m_barrays.contains("hello"));
|
|
QVERIFY(root->m_barrays.contains("world"));
|
|
QCOMPARE(root->m_convertibles.size(), 2);
|
|
QCOMPARE(root->m_convertibles.at(0).msg, QLatin1String("hello"));
|
|
QCOMPARE(root->m_convertibles.at(1).msg, QLatin1String("world"));
|
|
}
|
|
|
|
class EnumTester : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
enum Types
|
|
{
|
|
FIRST = 0,
|
|
SECOND,
|
|
THIRD
|
|
};
|
|
Q_ENUM(Types)
|
|
};
|
|
|
|
void tst_qqmllanguage::qualifiedScopeInCustomParser()
|
|
{
|
|
qmlRegisterUncreatableType<EnumTester>("scoped.custom.test", 1, 0, "EnumTester",
|
|
"Object only creatable in C++");
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine);
|
|
component.setData("import QtQml.Models 2.12\n"
|
|
"import scoped.custom.test 1.0 as BACKEND\n"
|
|
"ListModel {\n"
|
|
" ListElement { text: \"a\"; type: BACKEND.EnumTester.FIRST }\n"
|
|
"}\n", QUrl());
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> obj(component.create());
|
|
QVERIFY(!obj.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::checkUncreatableNoReason()
|
|
{
|
|
qmlRegisterTypesAndRevisions<UncreatableElementNoReason>("qt.uncreatable.noreason", 1, 0);
|
|
QQmlEngine engine;
|
|
QString qml = QString("import QtQuick 2.0\nimport qt.uncreatable.noreason 1.0\nUncreatableElementNoReason {}");
|
|
QQmlComponent c(&engine);
|
|
c.setData(qml.toUtf8(), QUrl::fromLocalFile(QDir::currentPath()));
|
|
QCOMPARE(c.errors().size(), 1);
|
|
QCOMPARE(c.errors().first().description(), QString("Type cannot be created in QML."));
|
|
}
|
|
|
|
void tst_qqmllanguage::checkURLtoURLObject()
|
|
{
|
|
QQmlEngine engine;
|
|
QString qml = QString("import QtQuick 2.0\nItem { property url source: 'file:///foo/bar/'; "
|
|
"Component.onCompleted: { new URL(parent.source); } }");
|
|
QQmlComponent c(&engine);
|
|
c.setData(qml.toUtf8(), QUrl::fromLocalFile(QDir::currentPath()));
|
|
QCOMPARE(c.errors().size(), 0);
|
|
}
|
|
|
|
struct TestValueType
|
|
{
|
|
Q_GADGET
|
|
Q_PROPERTY(int foo MEMBER foo)
|
|
public:
|
|
int foo = 12;
|
|
|
|
friend bool operator==(const TestValueType &a, const TestValueType &b)
|
|
{
|
|
return a.foo == b.foo;
|
|
}
|
|
|
|
friend bool operator!=(const TestValueType &a, const TestValueType &b)
|
|
{
|
|
return a.foo != b.foo;
|
|
}
|
|
};
|
|
|
|
struct TestExtendedValueType
|
|
{
|
|
Q_GADGET
|
|
Q_PROPERTY(int bar READ bar WRITE setBar)
|
|
public:
|
|
TestValueType wrapped;
|
|
|
|
int bar() const { return wrapped.foo; }
|
|
void setBar(int bar) { wrapped.foo = bar; }
|
|
};
|
|
|
|
class TestObjectType : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY(TestValueType test MEMBER test)
|
|
public:
|
|
TestValueType test;
|
|
};
|
|
|
|
void tst_qqmllanguage::registerValueTypes()
|
|
{
|
|
QTest::ignoreMessage(QtWarningMsg, "Invalid QML element name \"UpperCase\"; value type names should begin with a lowercase letter");
|
|
QVERIFY(qmlRegisterType<TestValueType>("DoesNotWork", 1, 0, "UpperCase") >= 0);
|
|
QVERIFY(qmlRegisterType<TestObjectType>("DoesWork", 1, 0, "TestObject") >= 0);
|
|
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import QtQml\nimport DoesWork\nTestObject { Component.onCompleted: test.foo = 14 }", QUrl());
|
|
QVERIFY(c.isReady());
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QCOMPARE(obj->property("test").value<TestValueType>().foo, 14);
|
|
|
|
QQmlComponent c2(&engine);
|
|
c2.setData("import QtQml\nimport DoesWork\n TestObject { Component.onCompleted: test.bar = 14 }", QUrl());
|
|
QVERIFY(c2.isReady());
|
|
QScopedPointer<QObject> obj2(c2.create());
|
|
QCOMPARE(obj2->property("test").value<TestValueType>().foo, 12);
|
|
}
|
|
|
|
QVERIFY((qmlRegisterExtendedType<TestValueType, TestExtendedValueType>("DoesWork", 1, 0, "lowerCase")) >= 0);
|
|
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import QtQml\nimport DoesWork\nTestObject { Component.onCompleted: test.foo = 14 }", QUrl());
|
|
QVERIFY(c.isReady());
|
|
QScopedPointer<QObject> obj(c.create());
|
|
// The foo property is hidden now.
|
|
QCOMPARE(obj->property("test").value<TestValueType>().foo, 12);
|
|
|
|
QQmlComponent c2(&engine);
|
|
c2.setData("import QtQml\nimport DoesWork\n TestObject { Component.onCompleted: test.bar = 14 }", QUrl());
|
|
QVERIFY(c2.isReady());
|
|
QScopedPointer<QObject> obj2(c2.create());
|
|
QCOMPARE(obj2->property("test").value<TestValueType>().foo, 14);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::accessNullPointerPropertyCache()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("NullPointerPropertyCache.qml"));
|
|
QVERIFY(c.isReady());
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QVERIFY(!obj.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::extendedNamespace()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQml\n"
|
|
"ExtendedByNamespace {\n"
|
|
" property int mine: own\n"
|
|
" property int myEnum: ExtendedByNamespace.Moo\n"
|
|
" property int fromExtension: ExtendedByNamespace.Bar\n"
|
|
"}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QVERIFY(!obj.isNull());
|
|
|
|
QCOMPARE(obj->property("mine").toInt(), 93);
|
|
QCOMPARE(obj->property("myEnum").toInt(), 16);
|
|
QCOMPARE(obj->property("fromExtension").toInt(), 9);
|
|
}
|
|
|
|
void tst_qqmllanguage::extendedNamespaceByObject()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQml\n"
|
|
"ExtendedNamespaceByObject {\n"
|
|
" extension: 10\n"
|
|
"}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QVERIFY(!obj.isNull());
|
|
|
|
QCOMPARE(obj->property("extension").toInt(), 10);
|
|
}
|
|
|
|
void tst_qqmllanguage::extendedByAttachedType()
|
|
{
|
|
QQmlEngine engine;
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQml\n"
|
|
"QtObject {\n"
|
|
" ExtendedNamespaceByObject.attachedName: \"name for extension\"\n"
|
|
"}",
|
|
QUrl());
|
|
QVERIFY2(!c.isReady(), qPrintable(c.errorString()));
|
|
}
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQml\n"
|
|
"QtObject {\n"
|
|
" Extended.attachedName: \"name for extension\"\n"
|
|
"}",
|
|
QUrl());
|
|
QVERIFY2(!c.isReady(), qPrintable(c.errorString()));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::factorySingleton()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQml\n"
|
|
"QtObject {\n"
|
|
" property int mine: FactorySingleton.foo\n"
|
|
"}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QVERIFY(!obj.isNull());
|
|
|
|
QCOMPARE(obj->property("mine").toInt(), 314);
|
|
}
|
|
|
|
void tst_qqmllanguage::extendedSingleton()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQml\n"
|
|
"QtObject {\n"
|
|
" property int a: ExtendedSingleton.foo\n"
|
|
" property int b: NamespaceExtendedSingleton.foo\n"
|
|
" property int c: ExtendedSingleton.extension\n"
|
|
" property int d: NamespaceExtendedSingleton.Bar\n"
|
|
"}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QVERIFY(!obj.isNull());
|
|
|
|
QCOMPARE(obj->property("a").toInt(), 315);
|
|
QCOMPARE(obj->property("b").toInt(), 316);
|
|
QCOMPARE(obj->property("c").toInt(), 42);
|
|
QCOMPARE(obj->property("d").toInt(), 9);
|
|
}
|
|
|
|
void tst_qqmllanguage::qtbug_85932()
|
|
{
|
|
QString warning1 = QLatin1String("%1:10:9: id is not unique").arg(testFileUrl("SingletonTest.qml").toString());
|
|
QString warning2 = QLatin1String("%1:4: Error: Due to the preceding error(s), Singleton \"SingletonTest\" could not be loaded.").arg(testFileUrl("qtbug_85932.qml").toString());
|
|
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, qPrintable(warning1));
|
|
QTest::ignoreMessage(QtMsgType::QtWarningMsg, qPrintable(warning2));
|
|
|
|
QQmlEngine engine;
|
|
QList<QQmlError> allWarnings;
|
|
QObject::connect(&engine, &QQmlEngine::warnings, [&allWarnings](const QList<QQmlError> &warnings) {
|
|
allWarnings.append(warnings);
|
|
});
|
|
|
|
QQmlComponent c(&engine, testFileUrl("qtbug_85932.qml"));
|
|
QScopedPointer<QObject> obj(c.create());
|
|
QTRY_COMPARE(allWarnings.size(), 2);
|
|
QCOMPARE(allWarnings.at(0).toString(), warning1);
|
|
QCOMPARE(allWarnings.at(1).toString(), warning2);
|
|
}
|
|
|
|
void tst_qqmllanguage::multiExtension()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\nMultiExtension {}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QCOMPARE(o->property("a").toInt(), int('a'));
|
|
QCOMPARE(o->property("b").toInt(), int('b'));
|
|
QCOMPARE(o->property("p").toInt(), int('p'));
|
|
QCOMPARE(o->property("e").toInt(), int('e'));
|
|
|
|
// Extension properties override base object properties
|
|
QCOMPARE(o->property("c").toInt(), 12);
|
|
QCOMPARE(o->property("d").toInt(), 22);
|
|
QCOMPARE(o->property("f").toInt(), 31);
|
|
QCOMPARE(o->property("g").toInt(), 44); // NB: taken from the type, not from the extension!
|
|
|
|
QObject *extension = qmlExtendedObject(o.get());
|
|
QVERIFY(extension != nullptr);
|
|
QVERIFY(qobject_cast<ExtensionB *>(extension) != nullptr);
|
|
|
|
QCOMPARE(QQmlPrivate::qmlExtendedObject(o.get(), 0), extension);
|
|
QObject *baseTypeExtension = QQmlPrivate::qmlExtendedObject(o.get(), 1);
|
|
QVERIFY(baseTypeExtension);
|
|
QVERIFY(qobject_cast<ExtensionA *>(baseTypeExtension) != nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::multiExtensionExtra()
|
|
{
|
|
QQmlEngine engine;
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\nMultiExtensionThreeExtensions {}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
|
|
QObject *extension = qmlExtendedObject(o.get());
|
|
QVERIFY(extension != nullptr);
|
|
QVERIFY(qobject_cast<Extension *>(extension) != nullptr);
|
|
|
|
QCOMPARE(QQmlPrivate::qmlExtendedObject(o.get(), 0), extension);
|
|
QObject *baseTypeExtension = QQmlPrivate::qmlExtendedObject(o.get(), 1);
|
|
QVERIFY(baseTypeExtension);
|
|
QVERIFY(qobject_cast<ExtensionB *>(baseTypeExtension) != nullptr);
|
|
QObject *baseBaseTypeExtension = QQmlPrivate::qmlExtendedObject(o.get(), 2);
|
|
QVERIFY(baseBaseTypeExtension);
|
|
QVERIFY(qobject_cast<ExtensionA *>(baseBaseTypeExtension) != nullptr);
|
|
}
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\nMultiExtensionWithExtensionInBaseBase {}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
|
|
QObject *extension = qmlExtendedObject(o.get());
|
|
QVERIFY(extension != nullptr);
|
|
QVERIFY(qobject_cast<Extension *>(extension) != nullptr);
|
|
|
|
QCOMPARE(QQmlPrivate::qmlExtendedObject(o.get(), 0), extension);
|
|
|
|
QObject *baseBaseTypeExtension = QQmlPrivate::qmlExtendedObject(o.get(), 1);
|
|
QVERIFY(baseBaseTypeExtension);
|
|
QVERIFY(qobject_cast<ExtensionB *>(baseBaseTypeExtension) != nullptr);
|
|
|
|
QObject *baseBaseBaseTypeExtension = QQmlPrivate::qmlExtendedObject(o.get(), 2);
|
|
QVERIFY(baseBaseBaseTypeExtension);
|
|
QVERIFY(qobject_cast<ExtensionA *>(baseBaseBaseTypeExtension) != nullptr);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::multiExtensionIndirect()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\nMultiExtensionIndirect {}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QCOMPARE(o->property("a").toInt(), int('a'));
|
|
QCOMPARE(o->property("b").toInt(), 77); // indirect extension is not considered
|
|
QCOMPARE(o->property("p").toInt(), int('p'));
|
|
QCOMPARE(o->property("e").toInt(), int('e'));
|
|
|
|
QCOMPARE(o->property("c").toInt(), int('c')); // indirect extension is not considered
|
|
QCOMPARE(o->property("d").toInt(), 21); // indirect extension is not considered
|
|
QCOMPARE(o->property("f").toInt(), 31);
|
|
QCOMPARE(o->property("g").toInt(), 44); // NB: taken from the type, not from the extension!
|
|
}
|
|
|
|
void tst_qqmllanguage::multiExtensionQmlTypes()
|
|
{
|
|
QQmlType extendedType = QQmlMetaType::qmlType(&MultiExtensionParent::staticMetaObject,
|
|
QStringLiteral("StaticTest"), QTypeRevision());
|
|
QVERIFY(extendedType.isValid());
|
|
QVERIFY(extendedType.extensionFunction());
|
|
QVERIFY(extendedType.extensionMetaObject() != nullptr);
|
|
|
|
QQmlType nonExtendedType = QQmlMetaType::qmlType(&ExtendedInParent::staticMetaObject,
|
|
QStringLiteral("StaticTest"), QTypeRevision());
|
|
QVERIFY(nonExtendedType.isValid());
|
|
QVERIFY(!nonExtendedType.extensionFunction());
|
|
QCOMPARE(nonExtendedType.extensionMetaObject(), nullptr);
|
|
|
|
QQmlType namespaceExtendedType = QQmlMetaType::qmlType(
|
|
&ExtendedByNamespace::staticMetaObject, QStringLiteral("StaticTest"), QTypeRevision());
|
|
QVERIFY(namespaceExtendedType.isValid());
|
|
QVERIFY(!namespaceExtendedType.extensionFunction()); // namespaces are non-creatable
|
|
QVERIFY(namespaceExtendedType.extensionMetaObject() != nullptr);
|
|
|
|
QQmlType namespaceNonExtendedType =
|
|
QQmlMetaType::qmlType(&ExtendedByNamespaceInParent::staticMetaObject,
|
|
QStringLiteral("StaticTest"), QTypeRevision());
|
|
QVERIFY(namespaceNonExtendedType.isValid());
|
|
QVERIFY(!namespaceNonExtendedType.extensionFunction());
|
|
QCOMPARE(namespaceNonExtendedType.extensionMetaObject(), nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::extensionSpecial()
|
|
{
|
|
QQmlEngine engine;
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\nExtendedInParent {}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
|
|
// property a exists only in the extension type
|
|
QCOMPARE(o->property("a").toInt(), int('a'));
|
|
|
|
// property c exists on the leaf type but since extension's properties
|
|
// are effectively FINAL, it is not used
|
|
QCOMPARE(o->property("c").toInt(), 11);
|
|
}
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\nExtendedByIndirect {}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
|
|
// there are (visibly) no properties in this case
|
|
QCOMPARE(o->property("b"), QVariant());
|
|
QCOMPARE(o->property("c"), QVariant());
|
|
QCOMPARE(o->property("d"), QVariant());
|
|
}
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\nExtendedInParentByIndirect {}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
|
|
// there are (visibly) no properties in this case (same as the previous)
|
|
QCOMPARE(o->property("b"), QVariant());
|
|
QCOMPARE(o->property("c"), QVariant());
|
|
QCOMPARE(o->property("d"), QVariant());
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::extensionRevision()
|
|
{
|
|
QQmlEngine engine;
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest 0.5\nExtendedWithRevisionOld { extension: 40\n}", QUrl());
|
|
QVERIFY(!c.isReady());
|
|
QRegularExpression error(
|
|
".*\"ExtendedWithRevisionOld.extension\" is not available in StaticTest 0.5.*");
|
|
QVERIFY2(error.match(c.errorString()).hasMatch(),
|
|
qPrintable(u"Unmatched error: "_s + c.errorString()));
|
|
}
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest 1.0\nExtendedWithRevisionNew { extension: 40\n}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::extendedGroupProperty()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData(R"(import StaticTest 1.0
|
|
ExtendedInGroup {
|
|
group.value: 42
|
|
group.value2: 42
|
|
}
|
|
)", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
|
|
ExtendedInGroup *extendedInGroup = qobject_cast<ExtendedInGroup *>(o.data());
|
|
QVERIFY(extendedInGroup);
|
|
QCOMPARE(extendedInGroup->group()->value(), 42);
|
|
QCOMPARE(extendedInGroup->group()->value2(), 42);
|
|
}
|
|
|
|
void tst_qqmllanguage::invalidInlineComponent()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import QtQuick 2.0\n"
|
|
"import QtQuick.Window 2.1\n"
|
|
"Window {\n"
|
|
" component TestPopup: Window {\n"
|
|
" visibility: Window.Windowed\n"
|
|
" }\n"
|
|
" TestPopup { color: \"blue\" }\n"
|
|
"}", QUrl());
|
|
QVERIFY(c.isError());
|
|
QVERIFY(c.errorString().contains("\"Window.visibility\" is not available in QtQuick 2.0."));
|
|
}
|
|
|
|
void tst_qqmllanguage::warnOnInjectedParameters()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
QTest::ignoreMessage(QtWarningMsg,
|
|
"qrc:/foo.qml:4:5 Parameter \"bar\" is not declared."
|
|
" Injection of parameters into signal handlers is deprecated."
|
|
" Use JavaScript functions with formal parameters instead.");
|
|
c.setData("import QtQml\n"
|
|
"QtObject {\n"
|
|
" signal foo(bar: string)\n"
|
|
" onFoo: print(bar)\n"
|
|
" Component.onCompleted: foo('baz')\n"
|
|
"}",
|
|
QUrl("qrc:/foo.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QTest::ignoreMessage(QtDebugMsg, "baz");
|
|
QScopedPointer<QObject> o(c.create());
|
|
}
|
|
|
|
#if QT_CONFIG(wheelevent)
|
|
void tst_qqmllanguage::warnOnInjectedParametersFromCppSignal()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
QTest::ignoreMessage(QtWarningMsg,
|
|
"qrc:/qtbug93987.qml:6:21 Parameter \"wheel\" is not declared."
|
|
" Injection of parameters into signal handlers is deprecated."
|
|
" Use JavaScript functions with formal parameters instead.");
|
|
c.setData(R"(
|
|
import QtQuick
|
|
MouseArea {
|
|
width: 640
|
|
height: 480
|
|
onWheel: print(wheel.angleDelta)
|
|
})",
|
|
QUrl("qrc:/qtbug93987.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
constexpr auto wheelDelta = 120; // copied from tst_QAbstractSlider. probably a default unit
|
|
QPoint p(100, 100);
|
|
QWheelEvent event(p, p, QPoint(), QPoint(0, wheelDelta * 1), Qt::NoButton, Qt::NoModifier,
|
|
Qt::NoScrollPhase, false);
|
|
QTest::ignoreMessage(QtDebugMsg, "QPoint(0, 120)");
|
|
QCoreApplication::sendEvent(o.get(), &event);
|
|
}
|
|
#endif
|
|
|
|
void tst_qqmllanguage::qtbug_86482()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine);
|
|
component.setData(QByteArray(R"(import QtQml 2.0
|
|
import StaticTest
|
|
QtObject {
|
|
id: root
|
|
property string result
|
|
property StringSignaler str: StringSignaler {
|
|
onSignal: function(value) { root.result = value; }
|
|
}
|
|
Component.onCompleted: str.call();
|
|
})"), QUrl());
|
|
VERIFY_ERRORS(0);
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QCOMPARE(o->property("result").toString(), QStringLiteral("Hello world!"));
|
|
}
|
|
|
|
void tst_qqmllanguage::qtbug_85615()
|
|
{
|
|
qmlRegisterSingletonType("Test.Singleton", 1, 0, "SingletonString", [](QQmlEngine *, QJSEngine *) -> QJSValue {
|
|
return QJSValue("Test");
|
|
});
|
|
qmlRegisterSingletonType("Test.Singleton", 1, 0, "SingletonInt", [](QQmlEngine *, QJSEngine *) -> QJSValue {
|
|
return QJSValue(123);
|
|
});
|
|
qmlRegisterSingletonType("Test.Singleton", 1, 0, "SingletonDouble", [](QQmlEngine *, QJSEngine *) -> QJSValue {
|
|
return QJSValue(1.23);
|
|
});
|
|
qmlRegisterSingletonType("Test.Singleton", 1, 0, "SingletonUndefined", [](QQmlEngine *, QJSEngine *) -> QJSValue {
|
|
return QJSValue(QJSValue::UndefinedValue);
|
|
});
|
|
qmlRegisterSingletonType("Test.Singleton", 1, 0, "SingletonNull", [](QQmlEngine *, QJSEngine *) -> QJSValue {
|
|
return QJSValue(QJSValue::NullValue);
|
|
});
|
|
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import QtQml 2.0\n"
|
|
"import Test.Singleton\n"
|
|
"QtObject {\n"
|
|
" property var resultString: SingletonString\n"
|
|
" property var resultInt: SingletonInt\n"
|
|
" property var resultDouble: SingletonDouble\n"
|
|
" property var resultUndefined: SingletonUndefined\n"
|
|
" property var resultNull: SingletonNull\n"
|
|
"}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
|
|
QScopedPointer<QObject> o(c.create());
|
|
QCOMPARE(o->property("resultString").toString(), "Test");
|
|
QCOMPARE(o->property("resultInt").toInt(), 123);
|
|
QCOMPARE(o->property("resultDouble").toDouble(), 1.23);
|
|
QVERIFY(!o->property("resultUndefined").isValid());
|
|
QCOMPARE(o->property("resultUndefined").metaType(), QMetaType(QMetaType::UnknownType));
|
|
QCOMPARE(o->property("resultNull").metaType(), QMetaType(QMetaType::Nullptr));
|
|
}
|
|
|
|
void tst_qqmllanguage::bareInlineComponent()
|
|
{
|
|
QQmlEngine engine;
|
|
|
|
QQmlComponent c(&engine, testFileUrl("bareInline.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QQmlMetaType::freeUnusedTypesAndCaches();
|
|
|
|
bool tab1Found = false;
|
|
const auto types = QQmlMetaType::qmlTypes();
|
|
for (const QQmlType &type : types) {
|
|
if (type.elementName() == QStringLiteral("Tab1")) {
|
|
QVERIFY(type.module().isEmpty());
|
|
tab1Found = true;
|
|
const auto ics = type.priv()->objectIdToICType;
|
|
QVERIFY(ics.size() > 0);
|
|
for (const QQmlType &ic : ics)
|
|
QVERIFY(ic.containingType() == type);
|
|
}
|
|
}
|
|
QVERIFY(tab1Found);
|
|
}
|
|
|
|
struct DummyDebugger : public QV4::Debugging::Debugger
|
|
{
|
|
bool pauseAtNextOpportunity() const final { return false; }
|
|
void maybeBreakAtInstruction() final { }
|
|
void enteringFunction() final { }
|
|
void leavingFunction(const QV4::ReturnedValue &) final { }
|
|
void aboutToThrow() final { }
|
|
};
|
|
|
|
void tst_qqmllanguage::hangOnWarning()
|
|
{
|
|
QQmlEngine engine;
|
|
|
|
// A debugger prevents the disk cache.
|
|
// If we load the file from disk cache we don't parse it and we don't see the warning.
|
|
engine.handle()->setDebugger(new DummyDebugger);
|
|
|
|
QTest::ignoreMessage(QtWarningMsg,
|
|
qPrintable(QStringLiteral("%1:3 : Ignored annotation")
|
|
.arg(testFileUrl("hangOnWarning.qml").toString())));
|
|
QQmlComponent component(&engine, testFileUrl("hangOnWarning.qml"));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object != nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::listEnumConversion()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
c.setData(R"(
|
|
import QtQml 2.0
|
|
import StaticTest 1.0
|
|
QtObject {
|
|
property EnumList enumList: EnumList {}
|
|
property var list: enumList.list()
|
|
property bool resultAlpha: EnumList.Alpha === list[0]
|
|
property bool resultBeta: EnumList.Beta === list[1]
|
|
property bool resultGamma: EnumList.Gamma === list[2]
|
|
property var resultEnumType: EnumList.Alpha
|
|
property var resultEnumListType: list[0]
|
|
})",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
|
|
QScopedPointer<QObject> o(c.create());
|
|
QCOMPARE(o->property("resultAlpha").toBool(), true);
|
|
QCOMPARE(o->property("resultBeta").toBool(), true);
|
|
QCOMPARE(o->property("resultGamma").toBool(), true);
|
|
QCOMPARE(o->property("resultEnumType").metaType(), QMetaType(QMetaType::Int));
|
|
QCOMPARE(o->property("resultEnumListType").metaType(), QMetaType(QMetaType::Int));
|
|
}
|
|
|
|
void tst_qqmllanguage::deepInlineComponentScriptBinding()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
c.loadUrl(testFileUrl("deepInlineComponentScriptBinding.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
}
|
|
|
|
void tst_qqmllanguage::propertyObserverOnReadonly()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine, testFileUrl("SelectionRange.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
|
|
QCOMPARE(o->property("zoomer").toDouble(), o->property("height").toDouble());
|
|
o->setProperty("height", QVariant::fromValue<double>(54.2));
|
|
QCOMPARE(o->property("zoomer").toDouble(), 54.2);
|
|
QCOMPARE(o->property("height").toDouble(), 54.2);
|
|
}
|
|
|
|
void tst_qqmllanguage::valueTypeWithEnum()
|
|
{
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import Test\n"
|
|
"ObjectTypeHoldingValueType1 {\n"
|
|
" vv.quality: ValueTypeWithEnum1.NormalQuality\n"
|
|
"}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
ObjectTypeHoldingValueType1 *holder = qobject_cast<ObjectTypeHoldingValueType1 *>(o.data());
|
|
|
|
QCOMPARE(holder->vv().quality(), ValueTypeWithEnum1::NormalQuality);
|
|
}
|
|
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&engine);
|
|
c.setData("import Test\n"
|
|
"ObjectTypeHoldingValueType2 {\n"
|
|
" vv.quality: ValueTypeWithEnum2.LowQuality\n"
|
|
"}", QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
ObjectTypeHoldingValueType2 *holder = qobject_cast<ObjectTypeHoldingValueType2 *>(o.data());
|
|
|
|
QCOMPARE(holder->vv().quality(), ValueTypeWithEnum2::LowQuality);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::propertyAndAliasMustHaveDistinctNames_data()
|
|
{
|
|
QTest::addColumn<QString>("fileName");
|
|
QTest::addColumn<QString>("error");
|
|
|
|
QTest::addRow("sameNamePropertyAlias") << "sameNamePropertyAlias.qml" << "Property duplicates alias name";
|
|
QTest::addRow("sameNameAliasProperty") << "sameNameAliasProperty.qml" << "Alias has same name as existing property";
|
|
}
|
|
|
|
void tst_qqmllanguage::propertyAndAliasMustHaveDistinctNames()
|
|
{
|
|
QFETCH(QString, fileName);
|
|
QFETCH(QString, error);
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl(fileName));
|
|
QVERIFY(!c.isReady());
|
|
auto actualError = c.errorString();
|
|
QVERIFY2(actualError.contains(error), qPrintable(actualError));
|
|
}
|
|
|
|
void tst_qqmllanguage::enumsFromRelatedTypes()
|
|
{
|
|
QQmlEngine e;
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import Test\n"
|
|
"ObjectTypeHoldingValueType1 {\n"
|
|
" vv.quality: ObjectTypeHoldingValueType1.NormalQuality\n"
|
|
"}", QUrl("Test1.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
ObjectTypeHoldingValueType1 *holder = qobject_cast<ObjectTypeHoldingValueType1 *>(o.data());
|
|
QCOMPARE(holder->q(), ValueTypeWithEnum1::NormalQuality);
|
|
}
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import Test\n"
|
|
"ObjectTypeHoldingValueType2 {\n"
|
|
" vv.quality: ObjectTypeHoldingValueType2.LowQuality\n"
|
|
"}", QUrl("Test2.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg,
|
|
"Test2.qml:3:5: Unable to assign [undefined] to ValueTypeWithEnum2::Quality");
|
|
QScopedPointer<QObject> o(c.create());
|
|
ObjectTypeHoldingValueType2 *holder = qobject_cast<ObjectTypeHoldingValueType2 *>(o.data());
|
|
QCOMPARE(holder->q(), ValueTypeWithEnum2::HighQuality);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::variantListConversion()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("variantListConversion.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
|
|
Foo *foo = qobject_cast<Foo *>(o.data());
|
|
QVERIFY(foo);
|
|
const QVariantList list = foo->getList();
|
|
QCOMPARE(list.size(), 3);
|
|
const Large l0 = qvariant_cast<Large>(list.at(0));
|
|
QCOMPARE(l0.a, 12ull);
|
|
const Large l1 = qvariant_cast<Large>(list.at(1));
|
|
QCOMPARE(l1.a, 13ull);
|
|
const QObject *attached = qvariant_cast<QObject *>(list.at(2));
|
|
QVERIFY(attached);
|
|
QCOMPARE(attached->metaObject(), &QQmlComponentAttached::staticMetaObject);
|
|
}
|
|
|
|
void tst_qqmllanguage::thisInArrowFunction()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("thisInArrow.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
|
|
QTest::ignoreMessage(QtDebugMsg, "43");
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QCOMPARE(qvariant_cast<QObject *>(o->property("arrowResult")), o.data());
|
|
QCOMPARE(qvariant_cast<QObject *>(o->property("funcResult")), o.data());
|
|
QCOMPARE(qvariant_cast<QObject *>(o->property("aResult")), o.data());
|
|
QCOMPARE(qvariant_cast<QObject *>(o->property("aaResult")), o.data());
|
|
|
|
QCOMPARE(qvariant_cast<QObject *>(o->property("fResult")), nullptr);
|
|
QCOMPARE(o->property("fResult").metaType(), QMetaType::fromType<QJSValue>());
|
|
QVERIFY(qvariant_cast<QJSValue>(o->property("fResult")).isObject());
|
|
|
|
QObject *child = qvariant_cast<QObject *>(o->property("child"));
|
|
QVERIFY(child != nullptr);
|
|
QCOMPARE(qvariant_cast<QObject *>(o->property("ffResult")), child);
|
|
}
|
|
|
|
void tst_qqmllanguage::jittedAsCast()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("jittedAsCast.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QCOMPARE(o->property("running").toBool(), true);
|
|
QTRY_COMPARE(o->property("running").toBool(), false);
|
|
QCOMPARE(o->property("interval").toInt(), 10);
|
|
}
|
|
|
|
void tst_qqmllanguage::propertyNecromancy()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("propertyNecromancy.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(qvariant_cast<QObject *>(o->property("notified")) != nullptr);
|
|
|
|
// It becomes null, not undefined.
|
|
QTRY_VERIFY(o->property("notified").isNull());
|
|
QVERIFY(o->property("notified").isValid());
|
|
}
|
|
|
|
void tst_qqmllanguage::generalizedGroupedProperty()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("generalizedGroupedProperty.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QCOMPARE(o->objectName(), QStringLiteral("foo"));
|
|
MyAttachedObject *rootAttached = static_cast<MyAttachedObject *>(
|
|
qmlAttachedPropertiesObject<MyQmlObject>(o.data()));
|
|
QVERIFY(rootAttached);
|
|
QCOMPARE(rootAttached->value(), 0);
|
|
QCOMPARE(rootAttached->value2(), 0);
|
|
|
|
ImmediateProperties *child = qvariant_cast<ImmediateProperties *>(o->property("child"));
|
|
QVERIFY(child);
|
|
QCOMPARE(child->objectName(), QStringLiteral("barrrrr"));
|
|
|
|
MyAttachedObject *childAttached = static_cast<MyAttachedObject *>(
|
|
qmlAttachedPropertiesObject<MyQmlObject>(child));
|
|
QVERIFY(childAttached);
|
|
QCOMPARE(childAttached->value(), 0);
|
|
|
|
qmlExecuteDeferred(child);
|
|
QCOMPARE(childAttached->value(), 4);
|
|
|
|
QCOMPARE(o->objectName(), QStringLiteral("barrrrr ..."));
|
|
QCOMPARE(rootAttached->value(), 10);
|
|
QCOMPARE(rootAttached->value2(), 0);
|
|
QCOMPARE(childAttached->value(), 4);
|
|
|
|
o->metaObject()->invokeMethod(o.data(), "something");
|
|
QCOMPARE(o->objectName(), QStringLiteral("rabrab ..."));
|
|
|
|
ImmediateProperties *meanChild = qvariant_cast<ImmediateProperties *>(o->property("meanChild"));
|
|
QVERIFY(meanChild);
|
|
qmlExecuteDeferred(meanChild);
|
|
QCOMPARE(child->objectName(), QStringLiteral("bar"));
|
|
QCOMPARE(o->objectName(), QStringLiteral("bar ..."));
|
|
QCOMPARE(childAttached->value(), 11);
|
|
|
|
ImmediateProperties *deferred = qvariant_cast<ImmediateProperties *>(o->property("deferred"));
|
|
QVERIFY(deferred);
|
|
QCOMPARE(deferred->objectName(), QStringLiteral("holz"));
|
|
qmlExecuteDeferred(deferred);
|
|
QCOMPARE(o->objectName(), QStringLiteral("holz ..."));
|
|
QCOMPARE(rootAttached->value(), 10);
|
|
QCOMPARE(rootAttached->value2(), 12);
|
|
|
|
ImmediateProperties *meanDeferred
|
|
= qvariant_cast<ImmediateProperties *>(o->property("meanDeferred"));
|
|
QVERIFY(meanDeferred);
|
|
qmlExecuteDeferred(meanDeferred);
|
|
QCOMPARE(deferred->objectName(), QStringLiteral("stein"));
|
|
QCOMPARE(o->objectName(), QStringLiteral("stein ..."));
|
|
|
|
{
|
|
QQmlComponent bad(&engine, testFileUrl("badGeneralizedGroupedProperty.qml"));
|
|
QVERIFY(bad.isError());
|
|
QVERIFY(bad.errorString().contains(
|
|
QStringLiteral("Cannot assign to non-existent property \"root\"")));
|
|
}
|
|
|
|
{
|
|
QQmlComponent bad(&engine, testFileUrl("badGeneralizedGroupedProperty2.qml"));
|
|
QVERIFY(bad.isError());
|
|
QVERIFY(bad.errorString().contains(QStringLiteral("Invalid grouped property access")));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::groupedAttachedProperty_data()
|
|
{
|
|
QTest::addColumn<QString>("file");
|
|
QTest::addRow("12") << QStringLiteral("validAttachedProperty.12.qml");
|
|
QTest::addRow("13") << QStringLiteral("validAttachedProperty.13.qml");
|
|
}
|
|
|
|
void tst_qqmllanguage::groupedAttachedProperty()
|
|
{
|
|
QFETCH(QString, file);
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl(file));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
MyTypeObject *typed = qobject_cast<MyTypeObject *>(o.data());
|
|
QVERIFY(typed != nullptr);
|
|
MyGroupedObject *grouped = typed->grouped();
|
|
QVERIFY(grouped != nullptr);
|
|
MyAttachedObject *attached = qobject_cast<MyAttachedObject *>(
|
|
qmlAttachedPropertiesObject<MyQmlObject>(grouped));
|
|
QVERIFY(attached != nullptr);
|
|
QCOMPARE(attached->value(), 10);
|
|
}
|
|
|
|
void tst_qqmllanguage::ambiguousContainingType()
|
|
{
|
|
// Need to do it twice, so that we load from disk cache the second time.
|
|
for (int i = 0; i < 2; ++i) {
|
|
QQmlEngine engine;
|
|
|
|
// Should not crash when loading the type
|
|
QQmlComponent c(&engine, testFileUrl("ambiguousBinding/ambiguousContainingType.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::objectAsBroken()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("asBroken.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
QVariant selfAsBroken = o->property("selfAsBroken");
|
|
QVERIFY(selfAsBroken.isValid());
|
|
QCOMPARE(selfAsBroken.metaType(), QMetaType::fromType<std::nullptr_t>());
|
|
|
|
QQmlComponent b(&engine, testFileUrl("Broken.qml"));
|
|
QVERIFY(b.isError());
|
|
}
|
|
|
|
void tst_qqmllanguage::customValueTypes()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("customValueTypes.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QCOMPARE(qvariant_cast<BaseValueType>(o->property("base")).content(), 27);
|
|
QCOMPARE(qvariant_cast<DerivedValueType>(o->property("derived")).content(), 28);
|
|
|
|
o->setObjectName(QStringLiteral("a"));
|
|
|
|
QCOMPARE(qvariant_cast<DerivedValueType>(o->property("derived")).content(), 14);
|
|
QCOMPARE(qvariant_cast<BaseValueType>(o->property("base")).content(), 13);
|
|
}
|
|
|
|
void tst_qqmllanguage::valueTypeList()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("valueTypeList.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
{
|
|
QCOMPARE(o->property("c").toInt(), 17);
|
|
QCOMPARE(qvariant_cast<QPointF>(o->property("d")), QPointF(3, 4));
|
|
QCOMPARE(qvariant_cast<DerivedValueType>(o->property("y")).content(), 29);
|
|
const QList<DerivedValueType> x = qvariant_cast<QList<DerivedValueType>>(o->property("x"));
|
|
QCOMPARE(x.size(), 3);
|
|
for (const DerivedValueType &d : x)
|
|
QCOMPARE(d.content(), 29);
|
|
|
|
const QList<BaseValueType> baseList
|
|
= qvariant_cast<QList<BaseValueType>>(o->property("baseList"));
|
|
QCOMPARE(baseList.size(), 3);
|
|
for (const BaseValueType &b : baseList)
|
|
QCOMPARE(b.content(), 29);
|
|
|
|
const QRectF f = qvariant_cast<QRectF>(o->property("f"));
|
|
QCOMPARE(f, QRectF(0, 2, 17, 1));
|
|
}
|
|
|
|
o->setObjectName(QStringLiteral("foo"));
|
|
{
|
|
QCOMPARE(qvariant_cast<QPointF>(o->property("d")), QPointF(12, 4));
|
|
|
|
// x is an actual value type list. We don't store references to y in there, but actual copies.
|
|
QCOMPARE(qvariant_cast<DerivedValueType>(o->property("y")).content(), 29);
|
|
|
|
const QList<DerivedValueType> x = qvariant_cast<QList<DerivedValueType>>(o->property("x"));
|
|
QCOMPARE(x.size(), 3);
|
|
QCOMPARE(x[0].content(), 29);
|
|
QCOMPARE(x[1].content(), 30);
|
|
QCOMPARE(x[2].content(), 29);
|
|
|
|
const QList<BaseValueType> baseList
|
|
= qvariant_cast<QList<BaseValueType>>(o->property("baseList"));
|
|
QCOMPARE(baseList.size(), 3);
|
|
for (const BaseValueType &b : baseList)
|
|
QCOMPARE(b.content(), 29);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::componentMix()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("componentMix.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
QObject *things = qvariant_cast<QObject *>(o->property("things"));
|
|
QVERIFY(things);
|
|
QObject *delegated = qvariant_cast<QObject *>(o->property("delegated"));
|
|
QVERIFY(delegated);
|
|
QObject *view = qvariant_cast<QObject *>(things->property("view"));
|
|
QVERIFY(view);
|
|
QObject *delegate = qvariant_cast<QObject *>(view->property("delegate"));
|
|
QVERIFY(delegate);
|
|
QCOMPARE(delegate->metaObject(), &QQmlComponent::staticMetaObject);
|
|
QObject *delegate2 = qvariant_cast<QObject *>(delegated->property("delegate"));
|
|
QVERIFY(delegate2);
|
|
QCOMPARE(delegate2->metaObject(), &QQmlComponent::staticMetaObject);
|
|
}
|
|
|
|
void tst_qqmllanguage::uncreatableAttached()
|
|
{
|
|
qmlRegisterTypesAndRevisions<ItemAttached>("ABC", 1);
|
|
QQmlEngine engine;
|
|
const QUrl url = testFileUrl("uncreatableAttached.qml");
|
|
QQmlComponent c(&engine, url);
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QTest::ignoreMessage(QtWarningMsg, "Only foo can have ItemAttached!");
|
|
QScopedPointer o(c.create());
|
|
QVERIFY(o.isNull());
|
|
QVERIFY(c.errorString().contains(
|
|
QLatin1String("Could not create attached properties object 'ItemAttached'")));
|
|
}
|
|
|
|
class MyGadget
|
|
{
|
|
Q_GADGET
|
|
Q_PROPERTY(int value READ value WRITE setValue RESET resetValue)
|
|
|
|
public:
|
|
int value() const { return m_value; }
|
|
void setValue(int value) { m_value = value; }
|
|
void resetValue() { setValue(25); }
|
|
|
|
bool operator==(const MyGadget &other) const { return m_value == other.m_value; }
|
|
bool operator!=(const MyGadget &other) const { return m_value != other.m_value; }
|
|
|
|
private:
|
|
int m_value = -1;
|
|
};
|
|
|
|
class MyObject : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_PROPERTY(MyGadget gadget MEMBER m_gadget RESET resetGadget NOTIFY gadgetChanged)
|
|
void resetGadget() { qDebug("FAIL"); }
|
|
|
|
public:
|
|
MyObject(QObject *parent = nullptr) : QObject(parent) { }
|
|
MyGadget m_gadget;
|
|
|
|
signals:
|
|
void gadgetChanged();
|
|
};
|
|
|
|
void tst_qqmllanguage::resetGadgetProperty()
|
|
{
|
|
qmlRegisterType<MyObject>("MyObject", 1, 0, "MyObject");
|
|
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine);
|
|
component.setData(
|
|
"import QtQml 2.0; import MyObject 1.0; MyObject { gadget.value: undefined }",
|
|
QUrl());
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> o(component.create());
|
|
QVERIFY2(!o.isNull(), qPrintable(component.errorString()));
|
|
|
|
MyObject *m = qobject_cast<MyObject *>(o.data());
|
|
QVERIFY(m);
|
|
QCOMPARE(m->m_gadget.value(), 25);
|
|
}
|
|
|
|
void tst_qqmllanguage::leakingAttributesQmlAttached()
|
|
{
|
|
// Check for leakage in the QML_ATTACHED macro
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQuick\n"
|
|
"Item {"
|
|
" OriginalQmlAttached.abc: true"
|
|
"}",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
QObject *attached = qmlAttachedPropertiesObject<OriginalQmlAttached>(o.data());
|
|
QVERIFY(attached);
|
|
QCOMPARE(attached->property("abc"), QVariant(true));
|
|
}
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQuick\n"
|
|
"Item {"
|
|
" LeakingQmlAttached.abc: true"
|
|
"}",
|
|
QUrl());
|
|
QVERIFY(!c.isReady());
|
|
}
|
|
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData("import StaticTest\n"
|
|
"import QtQuick\n"
|
|
"Item {"
|
|
" DerivedQmlAttached.anotherAbc: \"I am not a bool.\"\n"
|
|
" OriginalQmlAttached.abc: true"
|
|
"}",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
|
|
QObject *attached = qmlAttachedPropertiesObject<OriginalQmlAttached>(o.data());
|
|
QVERIFY(attached);
|
|
QCOMPARE(attached->property("abc"), QVariant(true));
|
|
|
|
attached = qmlAttachedPropertiesObject<DerivedQmlAttached>(o.data());
|
|
QVERIFY(attached);
|
|
QCOMPARE(attached->property("anotherAbc"), QVariant("I am not a bool."));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::leakingAttributesQmlSingleton()
|
|
{
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData(R"(
|
|
import StaticTest
|
|
import QtQuick
|
|
Item {
|
|
Text { text: OriginalSingleton.abc }
|
|
Text { text: OriginalSingleton.abc }
|
|
Text { id: check }
|
|
Component.onCompleted: {
|
|
OriginalSingleton.abc = "Updated string content!"
|
|
check.text = "onCompletedExecuted!"
|
|
}
|
|
})",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
QCOMPARE(o->children().size(), 4);
|
|
QCOMPARE(o->children()[2]->property("text"), QVariant("onCompletedExecuted!"));
|
|
QCOMPARE(o->children()[0]->property("text"), QVariant("Updated string content!"));
|
|
QCOMPARE(o->children()[1]->property("text"), QVariant("Updated string content!"));
|
|
}
|
|
{
|
|
QQmlComponent c(&engine);
|
|
|
|
c.setData(R"(
|
|
import StaticTest
|
|
import QtQuick
|
|
Item {
|
|
property var text: LeakingSingleton.abc
|
|
property var text2: LeakingSingleton.abc
|
|
Text { id: check }
|
|
Component.onCompleted: {
|
|
LeakingSingleton.abc = "Updated string content!"
|
|
check.text = "onCompletedExecuted!"
|
|
}
|
|
})",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
QCOMPARE(o->children().size(), 2);
|
|
QCOMPARE(o->children().front()->property("text"), QVariant("onCompletedExecuted!"));
|
|
// empty because not a singleton -> LeakingSingleton.abc is [undefined]
|
|
QVERIFY(!o->property("text").isValid());
|
|
QVERIFY(!o->property("text2").isValid());
|
|
}
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData(R"(
|
|
import StaticTest
|
|
import QtQuick
|
|
Item {
|
|
Text { text: DerivedSingleton.anotherAbc }
|
|
Text { text: DerivedSingleton.anotherAbc }
|
|
Text { id: check }
|
|
Component.onCompleted: {
|
|
DerivedSingleton.anotherAbc = "Updated string content!";
|
|
check.text = "onCompletedExecuted!"
|
|
}
|
|
})",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
QCOMPARE(o->children().size(), 4);
|
|
QCOMPARE(o->children()[2]->property("text"), QVariant("onCompletedExecuted!"));
|
|
QCOMPARE(o->children()[0]->property("text"), QVariant("Updated string content!"));
|
|
QCOMPARE(o->children()[1]->property("text"), QVariant("Updated string content!"));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::leakingAttributesQmlForeign()
|
|
{
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData(R"(
|
|
import StaticTest
|
|
ForeignerForeign {
|
|
abc: "Hello World"
|
|
})",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
QVERIFY(o->property("abc").isValid());
|
|
}
|
|
{
|
|
QQmlComponent c(&engine);
|
|
c.setData(R"(
|
|
import StaticTest
|
|
LeakingForeignerForeign {
|
|
})",
|
|
QUrl());
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o);
|
|
QVERIFY(o->property("anotherAbc").isValid());
|
|
QVERIFY(!o->property("abc").isValid());
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::attachedOwnProperties()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine);
|
|
c.setData(R"(
|
|
import QML
|
|
QtObject {
|
|
id: root
|
|
property list<string> props: Object.getOwnPropertyNames(root.Component)
|
|
}
|
|
)", QUrl(QStringLiteral("attachedOwn.qml")));
|
|
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
const QStringList expected {"objectName", "objectNameChanged", "completed", "destruction"};
|
|
QCOMPARE(o->property("props").value<QStringList>(), expected);
|
|
}
|
|
|
|
void tst_qqmllanguage::bindableOnly()
|
|
{
|
|
qmlRegisterTypesAndRevisions<BindableOnly>("ABC", 1);
|
|
QQmlEngine engine;
|
|
|
|
QQmlComponent c(&engine);
|
|
c.setData("import ABC\nBindableOnly {\n"
|
|
" property int a: score\n"
|
|
" data: \"sc\" + \"ore\"\n"
|
|
" objectName: data\n"
|
|
"}", QUrl(u"bindableOnly.qml"_s));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
BindableOnly *bindableOnly = qobject_cast<BindableOnly *>(o.data());
|
|
QCOMPARE(bindableOnly->scoreBindable().value(), 4);
|
|
QCOMPARE(o->property("a").toInt(), 4);
|
|
bindableOnly->scoreBindable().setValue(5);
|
|
QCOMPARE(bindableOnly->scoreBindable().value(), 5);
|
|
QCOMPARE(o->property("a").toInt(), 5);
|
|
QCOMPARE(o->property("data").value<QByteArray>(), QByteArray("score"));
|
|
QCOMPARE(o->objectName(), QStringLiteral("score"));
|
|
}
|
|
|
|
static void listsEqual(QObject *object, const char *method)
|
|
{
|
|
const QByteArray v4SequencePropertyName = QByteArray("v4Sequence") + method;
|
|
const QByteArray jsArrayPropertyName = QByteArray("jsArray") + method;
|
|
|
|
const QList<QRectF> v4SequenceProperty
|
|
= object->property(v4SequencePropertyName.constData()).value<QList<QRectF>>();
|
|
const QList<QRectF> jsArrayProperty
|
|
= object->property(jsArrayPropertyName.constData()).value<QList<QRectF>>();
|
|
|
|
const qsizetype v4SequenceCount = v4SequenceProperty.size();
|
|
QCOMPARE(v4SequenceCount, jsArrayProperty.size());
|
|
|
|
for (qsizetype i = 0; i < v4SequenceCount; ++i)
|
|
QCOMPARE(v4SequenceProperty.at(i), jsArrayProperty.at(i));
|
|
}
|
|
|
|
void tst_qqmllanguage::v4SequenceMethods()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("v4SequenceMethods.qml"));
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(!object.isNull());
|
|
|
|
QCOMPARE(object->property("v4SequenceToString"), object->property("jsArrayToString"));
|
|
QCOMPARE(object->property("v4SequenceToLocaleString"), object->property("jsArrayToLocaleString"));
|
|
|
|
QVERIFY(object->property("entriesMatch").toBool());
|
|
QVERIFY(object->property("keysMatch").toBool());
|
|
QVERIFY(object->property("valuesMatch").toBool());
|
|
|
|
listsEqual(object.data(), "Concat");
|
|
listsEqual(object.data(), "Pop");
|
|
listsEqual(object.data(), "Push");
|
|
listsEqual(object.data(), "Reverse");
|
|
listsEqual(object.data(), "Shift");
|
|
listsEqual(object.data(), "Unshift");
|
|
listsEqual(object.data(), "Filter");
|
|
listsEqual(object.data(), "Sort1");
|
|
listsEqual(object.data(), "Sort2");
|
|
|
|
QCOMPARE(object->property("v4SequenceFind"), object->property("jsArrayFind"));
|
|
QCOMPARE(object->property("v4SequenceFind").value<QRectF>().x(), 1.0);
|
|
|
|
QCOMPARE(object->property("v4SequenceFindIndex"), object->property("jsArrayFindIndex"));
|
|
QCOMPARE(object->property("v4SequenceFindIndex").toInt(), 1);
|
|
|
|
QCOMPARE(object->property("v4SequenceIncludes"), object->property("jsArrayIncludes"));
|
|
QVERIFY(object->property("v4SequenceIncludes").toBool());
|
|
|
|
QCOMPARE(object->property("v4SequenceJoin"), object->property("jsArrayJoin"));
|
|
QVERIFY(object->property("v4SequenceJoin").toString().contains(QStringLiteral("1")));
|
|
|
|
QCOMPARE(object->property("v4SequencePopped"), object->property("jsArrayPopped"));
|
|
QVERIFY(object->property("v4SequencePopped").value<QRectF>().x() != 1.0);
|
|
|
|
QCOMPARE(object->property("v4SequencePushed"), object->property("jsArrayPushed"));
|
|
QCOMPARE(object->property("v4SequencePushed").toInt(), 4);
|
|
|
|
QCOMPARE(object->property("v4SequenceShifted"), object->property("jsArrayShifted"));
|
|
QCOMPARE(object->property("v4SequenceShifted").value<QRectF>().x(), 1.0);
|
|
|
|
QCOMPARE(object->property("v4SequenceUnshifted"), object->property("jsArrayUnshifted"));
|
|
QCOMPARE(object->property("v4SequenceUnshifted").toInt(), 4);
|
|
|
|
QCOMPARE(object->property("v4SequenceIndexOf"), object->property("jsArrayIndexOf"));
|
|
QCOMPARE(object->property("v4SequenceIndexOf").toInt(), 1);
|
|
|
|
QCOMPARE(object->property("v4SequenceLastIndexOf"), object->property("jsArrayLastIndexOf"));
|
|
QCOMPARE(object->property("v4SequenceLastIndexOf").toInt(), 2);
|
|
|
|
QCOMPARE(object->property("v4SequenceEvery"), object->property("jsArrayEvery"));
|
|
QVERIFY(object->property("v4SequenceEvery").toBool());
|
|
|
|
QCOMPARE(object->property("v4SequenceSome"), object->property("jsArrayEvery"));
|
|
QVERIFY(object->property("v4SequenceSome").toBool());
|
|
|
|
QCOMPARE(object->property("v4SequenceForEach"), object->property("jsArrayForEach"));
|
|
QCOMPARE(object->property("v4SequenceForEach").toString(), QStringLiteral("-1--5--9-"));
|
|
|
|
QCOMPARE(object->property("v4SequenceMap").toStringList(), object->property("jsArrayMap").toStringList());
|
|
QCOMPARE(object->property("v4SequenceReduce").toString(), object->property("jsArrayReduce").toString());
|
|
|
|
QCOMPARE(object->property("v4SequenceOwnPropertyNames").toStringList(),
|
|
object->property("jsArrayOwnPropertyNames").toStringList());
|
|
}
|
|
|
|
void tst_qqmllanguage::v4SequenceMethodsWithParams_data()
|
|
{
|
|
QTest::addColumn<double>("i");
|
|
QTest::addColumn<double>("j");
|
|
QTest::addColumn<double>("k");
|
|
|
|
const double indices[] = {
|
|
double(std::numeric_limits<qsizetype>::min()),
|
|
double(std::numeric_limits<qsizetype>::min()) + 1,
|
|
double(std::numeric_limits<uint>::min()) - 1,
|
|
double(std::numeric_limits<uint>::min()),
|
|
double(std::numeric_limits<uint>::min()) + 1,
|
|
double(std::numeric_limits<int>::min()),
|
|
-10, -3, -2, -1, 0, 1, 2, 3, 10,
|
|
double(std::numeric_limits<int>::max()),
|
|
double(std::numeric_limits<uint>::max()) - 1,
|
|
double(std::numeric_limits<uint>::max()),
|
|
double(std::numeric_limits<uint>::max()) + 1,
|
|
double(std::numeric_limits<qsizetype>::max() - 1),
|
|
double(std::numeric_limits<qsizetype>::max()),
|
|
};
|
|
|
|
// We cannot test the full cross product. So, take a random sample instead.
|
|
const qsizetype numIndices = sizeof(indices) / sizeof(double);
|
|
qsizetype seed = QRandomGenerator::global()->generate();
|
|
const int numSamples = 4;
|
|
for (int i = 0; i < numSamples; ++i) {
|
|
seed = qHash(i, seed);
|
|
const double vi = indices[qAbs(seed) % numIndices];
|
|
for (int j = 0; j < numSamples; ++j) {
|
|
seed = qHash(j, seed);
|
|
const double vj = indices[qAbs(seed) % numIndices];
|
|
for (int k = 0; k < numSamples; ++k) {
|
|
seed = qHash(k, seed);
|
|
const double vk = indices[qAbs(seed) % numIndices];
|
|
const QString tag = QLatin1String("%1/%2/%3")
|
|
.arg(QString::number(vi), QString::number(vj), QString::number(vk));
|
|
QTest::newRow(qPrintable(tag)) << vi << vj << vk;
|
|
|
|
// output all the tags so that we can find out what combination caused a test to hang.
|
|
qDebug().noquote() << "scheduling" << tag;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::v4SequenceMethodsWithParams()
|
|
{
|
|
QFETCH(double, i);
|
|
QFETCH(double, j);
|
|
QFETCH(double, k);
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("v4SequenceMethodsWithParams.qml"));
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.createWithInitialProperties({
|
|
{QStringLiteral("i"), i},
|
|
{QStringLiteral("j"), j},
|
|
{QStringLiteral("k"), k}
|
|
}));
|
|
QVERIFY(!object.isNull());
|
|
|
|
listsEqual(object.data(), "CopyWithin");
|
|
listsEqual(object.data(), "Fill");
|
|
listsEqual(object.data(), "Slice");
|
|
listsEqual(object.data(), "Splice");
|
|
listsEqual(object.data(), "Spliced");
|
|
|
|
QCOMPARE(object->property("v4SequenceIndexOf"), object->property("jsArrayIndexOf"));
|
|
QCOMPARE(object->property("v4SequenceLastIndexOf"), object->property("jsArrayLastIndexOf"));
|
|
}
|
|
|
|
void tst_qqmllanguage::jsFunctionOverridesImport()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent component(&engine, testFileUrl("jsFunctionOverridesImport.qml"));
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QCOMPARE(object->objectName(), u"foo"_s);
|
|
}
|
|
|
|
void tst_qqmllanguage::bindingAliasToComponentUrl()
|
|
{
|
|
QQmlEngine engine;
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("bindingAliasToComponentUrl.qml"));
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object);
|
|
QCOMPARE(object->property("accessibleNormalUrl"), object->property("urlClone"));
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("bindingAliasToComponentUrl2.qml"));
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.create());
|
|
QVERIFY(object);
|
|
QCOMPARE(object->property("accessibleNormalProgress"), QVariant(1.0));
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::badGroupedProperty()
|
|
{
|
|
QQmlEngine engine;
|
|
const QUrl url = testFileUrl("badGroupedProperty.qml");
|
|
QQmlComponent c(&engine, url);
|
|
QVERIFY(c.isError());
|
|
QCOMPARE(c.errorString(),
|
|
QStringLiteral("%1:6 Cannot assign to non-existent property \"onComplete\"\n")
|
|
.arg(url.toString()));
|
|
}
|
|
|
|
void tst_qqmllanguage::functionInGroupedProperty()
|
|
{
|
|
QQmlEngine engine;
|
|
const QUrl url = testFileUrl("functionInGroupedProperty.qml");
|
|
QQmlComponent c(&engine, url);
|
|
QVERIFY(c.isError());
|
|
QCOMPARE(c.errorString(),
|
|
QStringLiteral("%1:6 Function declaration inside grouped property\n")
|
|
.arg(url.toString()));
|
|
}
|
|
|
|
void tst_qqmllanguage::signalInlineComponentArg()
|
|
{
|
|
QQmlEngine engine;
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("SignalInlineComponentArg.qml"));
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.create());
|
|
|
|
QCOMPARE(object->property("success"), u"Signal was called"_s);
|
|
}
|
|
{
|
|
QQmlComponent component(&engine, testFileUrl("signalInlineComponentArg1.qml"));
|
|
QVERIFY2(component.isReady(), qPrintable(component.errorString()));
|
|
QScopedPointer<QObject> object(component.create());
|
|
|
|
QCOMPARE(object->property("successFromOwnSignal"),
|
|
u"Own signal was called with component from another file"_s);
|
|
QCOMPARE(object->property("successFromSignalFromFile"),
|
|
u"Signal was called from another file"_s);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::functionSignatureEnforcement()
|
|
{
|
|
QQmlEngine engine;
|
|
|
|
QQmlComponent c1(&engine, testFileUrl("signatureIgnored.qml"));
|
|
QVERIFY2(c1.isReady(), qPrintable(c1.errorString()));
|
|
|
|
QScopedPointer<QObject> ignored(c1.create());
|
|
QCOMPARE(ignored->property("l").toInt(), 5);
|
|
QCOMPARE(ignored->property("m").toInt(), 77);
|
|
QCOMPARE(ignored->property("n").toInt(), 67);
|
|
|
|
QQmlComponent c2(&engine, testFileUrl("signatureEnforced.qml"));
|
|
QVERIFY2(c2.isReady(), qPrintable(c2.errorString()));
|
|
|
|
QScopedPointer<QObject> enforced(c2.create());
|
|
QCOMPARE(enforced->property("l").toInt(), 2); // strlen("no")
|
|
QCOMPARE(enforced->property("m").toInt(), 12);
|
|
QCOMPARE(enforced->property("n").toInt(), 99);
|
|
QCOMPARE(enforced->property("o").toInt(), 77);
|
|
}
|
|
|
|
void tst_qqmllanguage::importPrecedence()
|
|
{
|
|
QQmlEngine engine;
|
|
|
|
QQmlComponent c1(&engine, testFileUrl("importPrecedenceGood.qml"));
|
|
QVERIFY2(c1.isReady(), qPrintable(c1.errorString()));
|
|
QScopedPointer<QObject> o1(c1.create());
|
|
QVERIFY(!o1.isNull());
|
|
QVERIFY(o1->property("theAgent").value<QObject *>() != nullptr);
|
|
|
|
QUrl c2Url = testFileUrl("importPrecedenceBad.qml");
|
|
QQmlComponent c2(&engine, c2Url);
|
|
QVERIFY2(c2.isReady(), qPrintable(c2.errorString()));
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg,
|
|
qPrintable(c2Url.toString() + u":11: ReferenceError: agent is not defined"_s));
|
|
|
|
QScopedPointer<QObject> o2(c2.create());
|
|
QVERIFY(!o2.isNull());
|
|
QCOMPARE(o2->property("theAgent").value<QObject *>(), nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::nullIsNull()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("nullIsNull.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
QVERIFY(o->property("someProperty").value<QObject*>() != nullptr);
|
|
QTRY_COMPARE(o->property("someProperty").value<QObject*>(), nullptr);
|
|
}
|
|
|
|
void tst_qqmllanguage::multiRequired()
|
|
{
|
|
QQmlEngine engine;
|
|
const QUrl url = testFileUrl("multiRequired.qml");
|
|
QQmlComponent c(&engine, url);
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(o.isNull());
|
|
QCOMPARE(c.errorString(),
|
|
qPrintable(url.toString() + ":5 Required property description was not initialized\n"));
|
|
}
|
|
|
|
// QTBUG-111088
|
|
void tst_qqmllanguage::isNullOrUndefined()
|
|
{
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("isNullOrUndefined_interpreter.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVariant result = o.data()->property("result");
|
|
QVERIFY(result.isValid());
|
|
QCOMPARE(result.toInt(), 3);
|
|
}
|
|
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("isNullOrUndefined_jit.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVariant result = o.data()->property("result");
|
|
QVERIFY(result.isValid());
|
|
QCOMPARE(result.toInt(), 150);
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::objectAndGadgetMethodCallsRejectThisObject()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("objectAndGadgetMethodCallsRejectThisObject.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
|
|
const QList<int> lines = { 15, 19, 21, 22, 24, 25, 28, 31 };
|
|
for (int line : lines) {
|
|
const QString message
|
|
= ".*:%1: Calling C.. methods with 'this' objects different from the one "
|
|
"they were retrieved from is broken, due to historical reasons. The "
|
|
"original object is used as 'this' object. You can allow the given "
|
|
"'this' object to be used by setting "
|
|
"'pragma NativeMethodBehavior: AcceptThisObject'"_L1.arg(QString::number(line));
|
|
QTest::ignoreMessage(QtWarningMsg, QRegularExpression(message));
|
|
}
|
|
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QCOMPARE(o->property("badRect"), QRectF(1, 2, 3, 4));
|
|
QCOMPARE(o->property("goodRect1"), QRectF(1, 2, 3, 4));
|
|
QCOMPARE(o->property("goodRect2"), QRectF(1, 2, 3, 4));
|
|
|
|
QCOMPARE(o->property("badString"), QStringLiteral("27"));
|
|
QCOMPARE(o->property("goodString1"), QStringLiteral("27"));
|
|
QCOMPARE(o->property("goodString2"), QStringLiteral("27"));
|
|
QCOMPARE(o->property("goodString3"), QStringLiteral("27"));
|
|
|
|
QVERIFY(o->property("goodString4").value<QString>().startsWith("QObject_QML_"_L1));
|
|
QVERIFY(o->property("badString2").value<QString>().startsWith("QObject_QML_"_L1));
|
|
|
|
QCOMPARE(o->property("badInt"), 5);
|
|
QCOMPARE(o->property("goodInt1"), 5);
|
|
QCOMPARE(o->property("goodInt2"), 5);
|
|
QCOMPARE(o->property("goodInt3"), 5);
|
|
}
|
|
|
|
void tst_qqmllanguage::objectAndGadgetMethodCallsAcceptThisObject()
|
|
{
|
|
QQmlEngine engine;
|
|
QQmlComponent c(&engine, testFileUrl("objectAndGadgetMethodCallsAcceptThisObject.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg, QRegularExpression(
|
|
"objectAndGadgetMethodCallsAcceptThisObject.qml:16: Error: "
|
|
"Cannot call method QtObject::rect on QObject_QML_"));
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg, QRegularExpression(
|
|
"objectAndGadgetMethodCallsAcceptThisObject.qml:20: Error: "
|
|
"Cannot call method BaseValueType::report on QObject_QML_"));
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg, QRegularExpression(
|
|
"objectAndGadgetMethodCallsAcceptThisObject.qml:26: Error: "
|
|
"Cannot call method toString on QRectF"));
|
|
QTest::ignoreMessage(
|
|
QtWarningMsg, QRegularExpression(
|
|
"objectAndGadgetMethodCallsAcceptThisObject.qml:29: Error: "
|
|
"Cannot call method OriginalSingleton::mm on QObject_QML_"));
|
|
|
|
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
QCOMPARE(o->property("badRect"), QRectF());
|
|
QCOMPARE(o->property("goodRect1"), QRectF(1, 2, 3, 4));
|
|
QCOMPARE(o->property("goodRect2"), QRectF(1, 2, 3, 4));
|
|
|
|
QCOMPARE(o->property("badString"), QString());
|
|
QCOMPARE(o->property("goodString1"), QStringLiteral("27"));
|
|
QCOMPARE(o->property("goodString2"), QStringLiteral("27"));
|
|
QCOMPARE(o->property("goodString3"), QStringLiteral("28"));
|
|
|
|
QVERIFY(o->property("goodString4").value<QString>().startsWith("QtObject"_L1));
|
|
QCOMPARE(o->property("badString2"), QString());
|
|
|
|
QCOMPARE(o->property("badInt"), 0);
|
|
QCOMPARE(o->property("goodInt1"), 5);
|
|
QCOMPARE(o->property("goodInt2"), 5);
|
|
QCOMPARE(o->property("goodInt3"), 5);
|
|
}
|
|
|
|
void tst_qqmllanguage::longConversion()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&e, testFileUrl("longConversion.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
|
|
for (const char *prop : {
|
|
"testProp",
|
|
"testQProp",
|
|
"fromLocal",
|
|
"fromQLocal",
|
|
"fromBoolean",
|
|
"fromQBoolean"}) {
|
|
const QVariant val = o->property(prop);
|
|
QVERIFY(val.isValid());
|
|
QCOMPARE(val.metaType(), QMetaType::fromType<bool>());
|
|
QVERIFY(!val.toBool());
|
|
}
|
|
}
|
|
|
|
void tst_qqmllanguage::objectMethodClone()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&e, testFileUrl("objectMethodClone.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
QTRY_COMPARE(o->property("doneClicks").toInt(), 2);
|
|
}
|
|
|
|
void tst_qqmllanguage::unregisteredValueTypeConversion()
|
|
{
|
|
QQmlEngine e;
|
|
QQmlComponent c(&e, testFileUrl("unregisteredValueTypeConversion.qml"));
|
|
QVERIFY2(c.isReady(), qPrintable(c.errorString()));
|
|
QScopedPointer<QObject> o(c.create());
|
|
QVERIFY(!o.isNull());
|
|
UnregisteredValueTypeHandler *handler = qobject_cast<UnregisteredValueTypeHandler *>(o.data());
|
|
Q_ASSERT(handler);
|
|
QCOMPARE(handler->consumed, 2);
|
|
}
|
|
|
|
QTEST_MAIN(tst_qqmllanguage)
|
|
|
|
#include "tst_qqmllanguage.moc"
|