// Copyright (C) 2021 The Qt Company Ltd. // Copyright (C) 2016 Olivier Goffart // Copyright (C) 2016 Intel Corporation. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include // don't assume template constexpr inline bool my_is_same_v = false; template constexpr inline bool my_is_same_v = true; #define CHECK_IMPL(func, arg, Variant, cvref, R) \ static_assert(my_is_same_v(std::declval< Variant cvref >())), R cvref >) #define CHECK_GET_IF(Variant, cvref) \ CHECK_IMPL(get_if, int, Variant, cvref *, int) #define CHECK_GET(Variant, cvref) \ CHECK_IMPL(get, int, Variant, cvref, int) CHECK_GET_IF(QVariant, /* unadorned */); CHECK_GET_IF(QVariant, const); CHECK_GET(QVariant, &); CHECK_GET(QVariant, const &); CHECK_GET(QVariant, &&); CHECK_GET(QVariant, const &&); // check for a type derived from QVariant: struct MyVariant : QVariant { using QVariant::QVariant; }; CHECK_GET_IF(MyVariant, /* unadorned */); CHECK_GET_IF(MyVariant, const); CHECK_GET(MyVariant, &); CHECK_GET(MyVariant, const &); CHECK_GET(MyVariant, &&); CHECK_GET(MyVariant, const &&); #undef CHECK_GET_IF #undef CHECK_GET #undef CHECK_IMPL #include // Please stick to alphabetic order. #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "tst_qvariant_common.h" #include #include #include #include #include using namespace Qt::StringLiterals; class CustomNonQObject; struct NonDefaultConstructible; template struct QVariantFromValueCompiles { static inline constexpr bool value = false; }; template struct QVariantFromValueCompiles()))>> { static inline constexpr bool value = true; }; static_assert(QVariantFromValueCompiles::value); static_assert(!QVariantFromValueCompiles::value); enum EnumTest_Enum0 { EnumTest_Enum0_value = 42, EnumTest_Enum0_negValue = -8 }; Q_DECLARE_METATYPE(EnumTest_Enum0) enum EnumTest_Enum1 : qint64 { EnumTest_Enum1_value = 42, EnumTest_Enum1_bigValue = (Q_INT64_C(1) << 33) + 50 }; Q_DECLARE_METATYPE(EnumTest_Enum1) enum EnumTest_Enum3 : qint64 { EnumTest_Enum3_value = -47, EnumTest_Enum3_bigValue = (Q_INT64_C(1) << 56) + 5 }; Q_DECLARE_METATYPE(EnumTest_Enum3) enum EnumTest_Enum4 : quint64 { EnumTest_Enum4_value = 47, EnumTest_Enum4_bigValue = (Q_INT64_C(1) << 52) + 45 }; Q_DECLARE_METATYPE(EnumTest_Enum4) enum EnumTest_Enum5 : uint { EnumTest_Enum5_value = 47 }; Q_DECLARE_METATYPE(EnumTest_Enum5) enum EnumTest_Enum6 : uchar { EnumTest_Enum6_value = 47 }; Q_DECLARE_METATYPE(EnumTest_Enum6) enum class EnumTest_Enum7 { EnumTest_Enum7_value = 47, ensureSignedEnum7 = -1 }; Q_DECLARE_METATYPE(EnumTest_Enum7) enum EnumTest_Enum8 : short { EnumTest_Enum8_value = 47 }; Q_DECLARE_METATYPE(EnumTest_Enum8) template int qToUnderlying(QFlags f) { return f.toInt(); } class tst_QVariant : public QObject { Q_OBJECT static void runTestFunction() { QFETCH(QFunctionPointer, testFunction); testFunction(); } public: tst_QVariant(QObject *parent = nullptr) : QObject(parent), customNonQObjectPointer(0) { } enum MetaEnumTest_Enum0 { MetaEnumTest_Enum0_dummy = 2, MetaEnumTest_Enum0_value = 42, MetaEnsureSignedEnum0 = -1 }; Q_ENUM(MetaEnumTest_Enum0) enum MetaEnumTest_Enum1 : qint64 { MetaEnumTest_Enum1_value = 42, MetaEnumTest_Enum1_bigValue = (Q_INT64_C(1) << 33) + 50 }; Q_ENUM(MetaEnumTest_Enum1) enum MetaEnumTest_Enum3 : qint64 { MetaEnumTest_Enum3_value = -47, MetaEnumTest_Enum3_bigValue = (Q_INT64_C(1) << 56) + 5, MetaEnumTest_Enum3_bigNegValue = -(Q_INT64_C(1) << 56) - 3 }; Q_ENUM(MetaEnumTest_Enum3) enum MetaEnumTest_Enum4 : quint64 { MetaEnumTest_Enum4_value = 47, MetaEnumTest_Enum4_bigValue = (Q_INT64_C(1) << 52) + 45 }; Q_ENUM(MetaEnumTest_Enum4) enum MetaEnumTest_Enum5 : uint { MetaEnumTest_Enum5_value = 47 }; Q_ENUM(MetaEnumTest_Enum5) enum MetaEnumTest_Enum6 : uchar { MetaEnumTest_Enum6_value = 47 }; Q_ENUM(MetaEnumTest_Enum6) enum MetaEnumTest_Enum8 : short { MetaEnumTest_Enum8_value = 47 }; Q_ENUM(MetaEnumTest_Enum8) private slots: void cleanupTestCase(); void constructor(); void copy_constructor(); void constructor_invalid_data(); void constructor_invalid(); void isNull(); void swap(); void canConvert_data(); void canConvert(); void canConvertAndConvert_ReturnFalse_WhenConvertingBetweenPointerAndValue_data(); void canConvertAndConvert_ReturnFalse_WhenConvertingBetweenPointerAndValue(); void canConvertAndConvert_ReturnFalse_WhenConvertingQObjectBetweenPointerAndValue(); void convert(); void toSize_data(); void toSize(); void toSizeF_data(); void toSizeF(); void toPoint_data(); void toPoint(); void toRect_data(); void toRect(); void toChar_data(); void toChar(); void toLine_data(); void toLine(); void toLineF_data(); void toLineF(); void toInt_data(); void toInt(); void toUInt_data(); void toUInt(); void toBool_data(); void toBool(); void toLongLong_data(); void toLongLong(); void toULongLong_data(); void toULongLong(); void toByteArray_data(); void toByteArray(); void toString_data(); void toString(); void toDate_data(); void toDate(); void toTime_data(); void toTime(); void toDateTime_data(); void toDateTime(); void toDouble_data(); void toDouble(); void toFloat_data(); void toFloat(); void toPointF_data(); void toPointF(); void toRectF_data(); void toRectF(); void qvariant_cast_QObject_data(); void qvariant_cast_QObject(); void qvariant_cast_QObject_derived(); void qvariant_cast_QObject_wrapper(); void qvariant_cast_QSharedPointerQObject(); void qvariant_cast_const(); void toLocale(); void toRegularExpression(); void url(); void userType(); void basicUserType(); void variant_to(); void writeToReadFromDataStream_data(); void writeToReadFromDataStream(); void writeToReadFromOldDataStream(); void checkDataStream(); void operator_eq_eq_data(); void operator_eq_eq(); #if QT_DEPRECATED_SINCE(6, 0) void typeName_data(); void typeName(); void typeToName(); #endif void streamInvalidVariant(); void podUserType(); void data(); void constData(); void saveLoadCustomTypes(); void variantMap(); void variantHash(); void convertToQUint8() const; void compareNumerics_data() const; void compareNumerics() const; void comparePointers() const; void voidStar() const; void dataStar() const; void canConvertQStringList() const; void canConvertQStringList_data() const; void canConvertMetaTypeToInt() const; void variantToDateTimeWithoutWarnings() const; void invalidDateTime() const; void loadUnknownUserType(); void loadBrokenUserType(); void invalidDate() const; void compareCustomTypes() const; void timeToDateTime() const; void copyingUserTypes() const; void valueClassHierarchyConversion() const; void convertBoolToByteArray() const; void convertBoolToByteArray_data() const; void convertByteArrayToBool() const; void convertByteArrayToBool_data() const; void convertIterables() const; void convertConstNonConst() const; void toIntFromQString() const; void toIntFromDouble() const; void setValue(); void fpStringRoundtrip_data() const; void fpStringRoundtrip() const; void numericalConvert_data(); void numericalConvert(); void moreCustomTypes(); void movabilityTest(); void variantInVariant(); void userConversion(); void modelIndexConversion(); void forwardDeclare(); void debugStream_data(); void debugStream(); #if QT_DEPRECATED_SINCE(6, 0) void debugStreamType_data(); void debugStreamType(); #endif void loadQt4Stream_data(); void loadQt4Stream(); void saveQt4Stream_data(); void saveQt4Stream(); void loadQt5Stream_data(); void loadQt5Stream(); void saveQt5Stream_data(); void saveQt5Stream(); void saveInvalid_data(); void saveInvalid(); void saveNewBuiltinWithOldStream(); void implicitConstruction(); void iterateSequentialContainerElements_data(); void iterateSequentialContainerElements() { runTestFunction(); } void iterateAssociativeContainerElements_data(); void iterateAssociativeContainerElements() { runTestFunction(); } void iterateContainerElements(); void pairElements_data(); void pairElements() { runTestFunction(); } void enums_data(); void enums() { runTestFunction(); } void metaEnums_data(); void metaEnums() { runTestFunction(); } void nullConvert(); void accessSequentialContainerKey(); void shouldDeleteVariantDataWorksForSequential(); void shouldDeleteVariantDataWorksForAssociative(); void fromStdVariant(); void qt4UuidDataStream(); void sequentialIterableEndianessSanityCheck(); void sequentialIterableAppend(); void preferDirectConversionOverInterfaces(); void mutableView(); void canViewAndView_ReturnFalseAndDefault_WhenConvertingBetweenPointerAndValue(); void moveOperations(); void equalsWithoutMetaObject(); void constructFromIncompatibleMetaType_data(); void constructFromIncompatibleMetaType(); void constructFromQtLT65MetaType(); void copyNonDefaultConstructible(); void inplaceConstruct(); void emplace(); void getIf_int() { getIf_impl(42); } void getIf_QString() { getIf_impl(u"string"_s); }; void getIf_NonDefaultConstructible(); void getIfSpecial(); void get_int() { get_impl(42); } void get_QString() { get_impl(u"string"_s); } void get_NonDefaultConstructible(); private: using StdVariant = std::variant; template void getIf_impl(T t) const; template void get_impl(T t) const; template void canViewAndView_ReturnFalseAndDefault_WhenConvertingBetweenPointerAndValue_impl(const QByteArray &typeName); void dataStream_data(QDataStream::Version version); void loadQVariantFromDataStream(QDataStream::Version version); void saveQVariantFromDataStream(QDataStream::Version version); CustomNonQObject *customNonQObjectPointer; QList objectPointerTestData; }; const qlonglong intMax1 = (qlonglong)INT_MAX + 1; const qulonglong uintMax1 = (qulonglong)UINT_MAX + 1; void tst_QVariant::constructor() { QVariant variant; QVERIFY( !variant.isValid() ); QVERIFY( variant.isNull() ); QVariant var2(variant); QVERIFY( !var2.isValid() ); QVERIFY( variant.isNull() ); QVariant varll(intMax1); QVariant varll2(varll); QCOMPARE(varll2, varll); QVariant var3 {QMetaType::fromType()}; QCOMPARE(var3.typeName(), "QString"); QVERIFY(var3.isNull()); QVERIFY(var3.isValid()); QVariant var3a = QVariant::fromMetaType(QMetaType::fromType()); QCOMPARE(var3a.typeName(), "QString"); QVERIFY(var3a.isNull()); QVERIFY(var3a.isValid()); QVariant var4 {QMetaType()}; QCOMPARE(var4.typeId(), QMetaType::UnknownType); QVERIFY(var4.isNull()); QVERIFY(!var4.isValid()); QVariant var4a = QVariant::fromMetaType(QMetaType()); QCOMPARE(var4a.typeId(), QMetaType::UnknownType); QVERIFY(var4a.isNull()); QVERIFY(!var4a.isValid()); QVariant var5(QLatin1String("hallo")); QCOMPARE(var5.typeId(), QMetaType::QString); QCOMPARE(var5.typeName(), "QString"); QVariant var6(qlonglong(0)); QCOMPARE(var6.typeId(), QMetaType::LongLong); QCOMPARE(var6.typeName(), "qlonglong"); QVariant var7 = 5; QVERIFY(var7.isValid()); QVERIFY(!var7.isNull()); QVariant var8; var8.setValue(5); QVERIFY(var8.isValid()); QVERIFY(!var8.isNull()); } void tst_QVariant::constructor_invalid_data() { QTest::addColumn("typeId"); QTest::newRow("-1") << uint(-1); QTest::newRow("-122234567") << uint(-122234567); QTest::newRow("0xfffffffff") << uint(0xfffffffff); QTest::newRow("LastCoreType + 1") << uint(QMetaType::LastCoreType + 1); QVERIFY(!QMetaType::isRegistered(QMetaType::LastCoreType + 1)); } void tst_QVariant::constructor_invalid() { QFETCH(uint, typeId); { QTest::ignoreMessage(QtWarningMsg, QRegularExpression("^Trying to construct an instance of an invalid type")); QVariant variant {QMetaType(typeId)}; QVERIFY(!variant.isValid()); QVERIFY(variant.isNull()); QCOMPARE(variant.typeId(), int(QMetaType::UnknownType)); QCOMPARE(variant.userType(), int(QMetaType::UnknownType)); } { QTest::ignoreMessage(QtWarningMsg, QRegularExpression("^Trying to construct an instance of an invalid type")); QVariant variant = QVariant::fromMetaType(QMetaType(typeId)); QVERIFY(!variant.isValid()); QVERIFY(variant.isNull()); QCOMPARE(variant.typeId(), int(QMetaType::UnknownType)); QCOMPARE(variant.userType(), int(QMetaType::UnknownType)); } { QTest::ignoreMessage(QtWarningMsg, QRegularExpression("^Trying to construct an instance of an invalid type")); QVariant variant(QMetaType(typeId), /* copy */ nullptr); QVERIFY(!variant.isValid()); QVERIFY(variant.isNull()); QCOMPARE(variant.userType(), int(QMetaType::UnknownType)); } } void tst_QVariant::copy_constructor() { QVariant var7 {QMetaType::fromType()}; QVariant var8(var7); QCOMPARE(var8.typeId(), QMetaType::Int); QVERIFY(var8.isNull()); } Q_DECLARE_METATYPE(int*) void tst_QVariant::isNull() { QVariant var; QVERIFY( var.isNull() ); QVariant empty = QString(); QVERIFY(empty.toString().isNull()); QVERIFY(!empty.isNull()); QVERIFY(empty.isValid()); QCOMPARE(empty.typeName(), "QString"); QVariant var3( QString( "blah" ) ); QVERIFY( !var3.isNull() ); var3 = QVariant(QMetaType::fromType()); QVERIFY( var3.isNull() ); QVariant var4( 0 ); QVERIFY( !var4.isNull() ); QVariant var5 = QString(); QVERIFY( !var5.isNull() ); QVariant var6( QString( "blah" ) ); QVERIFY( !var6.isNull() ); var6 = QVariant(); QVERIFY( var6.isNull() ); var6.convert(QMetaType::fromType()); QVERIFY( var6.isNull() ); QVariant varLL( (qlonglong)0 ); QVERIFY( !varLL.isNull() ); QVariant var8(QMetaType::fromType(), nullptr); QVERIFY(var8.isNull()); var8 = QVariant::fromValue(nullptr); QVERIFY(var8.isNull()); QVariant var9 = QVariant(QJsonValue(QJsonValue::Null)); QVERIFY(!var9.isNull()); var9 = QVariant::fromValue(QJsonValue(QJsonValue::Null)); QVERIFY(!var9.isNull()); QVariant var10(QMetaType::fromType(), nullptr); QVERIFY(var10.isNull()); var10 = QVariant::fromValue(nullptr); QVERIFY(var10.isNull()); QVariant var11(QMetaType::fromType(), nullptr); QVERIFY(var11.isNull()); var11 = QVariant::fromValue(nullptr); QVERIFY(var11.isNull()); QVERIFY(QVariant::fromValue(nullptr).isNull()); QVariant var12(QVariant::fromValue(QString())); QVERIFY(!var12.isNull()); } void tst_QVariant::swap() { QVariant v1 = 1, v2 = 2.0; v1.swap(v2); QCOMPARE(v1.typeId(), QMetaType::Double); QCOMPARE(v1.toDouble(),2.0); QCOMPARE(v2.typeId(), QMetaType::Int); QCOMPARE(v2.toInt(),1); } void tst_QVariant::canConvert_data() { TST_QVARIANT_CANCONVERT_DATATABLE_HEADERS #ifdef Y #undef Y #endif #ifdef N #undef N #endif #define Y true #define N false // bita bitm bool brsh byta col curs date dt dbl font img int inv kseq list ll map pal pen pix pnt rect reg size sp str strl time uint ull QVariant var(QBitArray(0)); QTest::newRow("BitArray") << var << Y << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N; var = QVariant(QByteArray()); QTest::newRow("ByteArray") << var << N << N << Y << N << Y << Y << N << N << N << Y << N << N << Y << N << N << Y << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant(QDate()); QTest::newRow("Date") << var << N << N << N << N << N << N << N << Y << Y << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N; var = QVariant(QDateTime()); QTest::newRow("DateTime") << var << N << N << N << N << N << N << N << Y << Y << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << Y << N << N; var = QVariant((double)0.1); QTest::newRow("Double") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant(0.1f); QTest::newRow("Float") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant((int)1); QTest::newRow("Int") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << Y << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant(); QTest::newRow("Invalid") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N; var = QVariant(QList()); QTest::newRow("List") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N; var = QVariant((qlonglong)1); QTest::newRow("LongLong") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant(QMap()); QTest::newRow("Map") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N << N << N << N; var = QVariant(QPoint()); QTest::newRow("Point") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N; var = QVariant(QRect()); QTest::newRow("Rect") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N; var = QVariant(QSize()); QTest::newRow("Size") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N; var = QVariant(QString()); QTest::newRow("String") << var << N << N << Y << N << Y << Y << N << Y << Y << Y << Y << N << Y << N << Y << Y << Y << N << N << N << N << N << N << N << N << N << Y << Y << Y << Y << Y; var = QVariant(QStringList("entry")); QTest::newRow("StringList") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N << Y << Y << N << N << N; var = QVariant(QTime()); QTest::newRow("Time") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << Y << N << N; var = QVariant((uint)1); QTest::newRow("UInt") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant((qulonglong)1); QTest::newRow("ULongLong") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant::fromValue('a'); QTest::newRow("Char") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant::fromValue(-1); QTest::newRow("SChar") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant::fromValue((short)-3); QTest::newRow("Short") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant::fromValue((ushort)7); QTest::newRow("UShort") << var << N << N << Y << N << Y << N << N << N << N << Y << N << N << Y << N << N << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant::fromValue(QJsonValue(QStringLiteral("hello"))); QTest::newRow("JsonValue") << var << N << N << Y << N << N << N << N << N << N << Y << N << N << Y << N << N << Y << Y << Y << N << N << N << N << N << N << N << N << Y << N << N << Y << Y; var = QVariant::fromValue(QJsonArray()); QTest::newRow("JsonArray") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N; var = QVariant::fromValue(QJsonObject()); QTest::newRow("JsonObject") << var << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N << N << N << N; #undef N #undef Y } void tst_QVariant::canConvert() { TST_QVARIANT_CANCONVERT_FETCH_DATA // This test links against QtGui but not QtWidgets, so QSizePolicy isn't real for it. QTest::ignoreMessage(QtWarningMsg, // QSizePolicy's id is 0x2000, a.k.a. 8192 "Trying to construct an instance of an invalid type, type id: 8192"); TST_QVARIANT_CANCONVERT_COMPARE_DATA #if QT_DEPRECATED_SINCE(6, 0) QT_WARNING_PUSH QT_WARNING_DISABLE_DEPRECATED // Invalid type ids QTest::ignoreMessage(QtWarningMsg, "Trying to construct an instance of an invalid type, type id: -1"); QCOMPARE(val.canConvert(-1), false); QTest::ignoreMessage(QtWarningMsg, "Trying to construct an instance of an invalid type, type id: -23"); QCOMPARE(val.canConvert(-23), false); QTest::ignoreMessage(QtWarningMsg, "Trying to construct an instance of an invalid type, type id: -23876"); QCOMPARE(val.canConvert(-23876), false); QTest::ignoreMessage(QtWarningMsg, "Trying to construct an instance of an invalid type, type id: 23876"); QCOMPARE(val.canConvert(23876), false); QT_WARNING_POP #endif // QT_DEPRECATED_SINCE(6, 0) } namespace { // Used for testing canConvert/convert of QObject derived types struct QObjectDerived : QObject { Q_OBJECT }; // Adds a test table row for checking value <-> pointer conversion // If type is a pointer, the target type is value type and vice versa. template void addRowForPointerValueConversion() { using ValueType = std::remove_pointer_t; if constexpr (!std::is_same_v) { static ValueType instance{}; // static since we may need a pointer to a valid object QVariant variant; if constexpr (std::is_pointer_v) variant = QVariant::fromValue(&instance); else variant = QVariant::fromValue(instance); // Toggle pointer/value type using TargetType = std::conditional_t, ValueType, T *>; const QMetaType fromType = QMetaType::fromType(); const QMetaType toType = QMetaType::fromType(); QTest::addRow("%s->%s", fromType.name(), toType.name()) << variant << QMetaType::fromType(); } } } // namespace void tst_QVariant::canConvertAndConvert_ReturnFalse_WhenConvertingBetweenPointerAndValue_data() { QTest::addColumn("variant"); QTest::addColumn("targetType"); #define ADD_ROW(typeName, typeNameId, realType) \ addRowForPointerValueConversion(); \ addRowForPointerValueConversion(); // Add rows for static primitive types QT_FOR_EACH_STATIC_PRIMITIVE_NON_VOID_TYPE(ADD_ROW) // Add rows for static core types QT_FOR_EACH_STATIC_CORE_CLASS(ADD_ROW) #undef ADD_ROW } void tst_QVariant::canConvertAndConvert_ReturnFalse_WhenConvertingBetweenPointerAndValue() { QFETCH(QVariant, variant); QFETCH(QMetaType, targetType); QVERIFY(!variant.canConvert(targetType)); QVERIFY(!variant.convert(targetType)); // As per the documentation, when QVariant::convert fails, the // QVariant is cleared and changed to the requested type. QVERIFY(variant.isNull()); QCOMPARE(variant.metaType(), targetType); } void tst_QVariant::canConvertAndConvert_ReturnFalse_WhenConvertingQObjectBetweenPointerAndValue() { // Types derived from QObject are non-copyable and require their own test. // We only test pointer -> value conversion, because constructing a QVariant // from a non-copyable object will just set the QVariant to null. QObjectDerived object; QVariant variant = QVariant::fromValue(&object); constexpr QMetaType targetType = QMetaType::fromType(); QVERIFY(!variant.canConvert(targetType)); QTest::ignoreMessage( QtWarningMsg, QRegularExpression(".*does not support destruction and copy construction")); QVERIFY(!variant.convert(targetType)); // When the QVariant::convert fails, the QVariant is cleared, and since the target type is // invalid for QVariant, the QVariant's type is also cleared to an unknown type. QVERIFY(variant.isNull()); QCOMPARE(variant.metaType(), QMetaType()); } void tst_QVariant::convert() { // verify that after convert(), the variant's type has been changed QVariant var = QVariant::fromValue(QString("A string")); var.convert(QMetaType::fromType()); QCOMPARE(var.metaType(), QMetaType::fromType()); QCOMPARE(var.toInt(), 0); } void tst_QVariant::toInt_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::addColumn("valueOK"); QTest::newRow( "invalid" ) << QVariant() << 0 << false; QTest::newRow( "int" ) << QVariant( 123 ) << 123 << true; QTest::newRow( "char" ) << QVariant::fromValue('a') << int('a') << true; signed char signedChar = -13; QTest::newRow( "signed char" ) << QVariant::fromValue(signedChar) << -13 << true; QTest::newRow( "short" ) << QVariant::fromValue(short(-7)) << int(-7) << true; QTest::newRow( "ushort" ) << QVariant::fromValue(ushort(30000)) << 30000 << true; QTest::newRow( "double" ) << QVariant( 3.1415927 ) << 3 << true; QTest::newRow( "float" ) << QVariant( 3.1415927f ) << 3 << true; QTest::newRow( "uint" ) << QVariant( 123u ) << 123 << true; QTest::newRow( "int-string" ) << QVariant( QString("123") ) << 123 << true; QTest::newRow( "string" ) << QVariant( QString("Unicode String") ) << 0 << false; QTest::newRow( "longlong0" ) << QVariant( (qlonglong)34 ) << 34 << true; QTest::newRow( "longlong1" ) << QVariant( intMax1 ) << (int)INT_MIN << true; QTest::newRow( "ulonglong0" ) << QVariant( (qulonglong)34 ) << 34 << true; QTest::newRow( "ulonglong1" ) << QVariant( uintMax1 ) << 0 << true; QTest::newRow( "signedint" ) << QVariant( -123 ) << -123 << true; QTest::newRow( "signeddouble" ) << QVariant( -3.1415927 ) << -3 << true; QTest::newRow( "signedfloat" ) << QVariant( -3.1415927f ) << -3 << true; QTest::newRow( "signedint-string" ) << QVariant( QString("-123") ) << -123 << true; QTest::newRow( "signedlonglong0" ) << QVariant( (qlonglong)-34 ) << -34 << true; QTest::newRow( "QChar" ) << QVariant(QChar('a')) << int('a') << true; QByteArray bytearray(4, ' '); bytearray[0] = 'T'; bytearray[1] = 'e'; bytearray[2] = 's'; bytearray[3] = 't'; QTest::newRow( "QByteArray1" ) << QVariant( bytearray ) << 0 << false; bytearray[0] = '4'; bytearray[1] = '5'; bytearray[2] = '0'; bytearray[3] = '0'; QTest::newRow( "QByteArray2" ) << QVariant( bytearray ) << 4500 << true; QTest::newRow("int-QJsonValue") << QVariant(QJsonValue(321)) << 321 << true; QTest::newRow("undefined-QJsonValue") << QVariant(QJsonValue(QJsonValue::Undefined)) << 0 << false; } #if QT_DEPRECATED_SINCE(6, 0) # define EXEC_DEPRECATED_CALL(x) QT_IGNORE_DEPRECATIONS(x) #else # define EXEC_DEPRECATED_CALL(x) #endif void tst_QVariant::toInt() { QFETCH( QVariant, value ); QFETCH( int, result ); QFETCH( bool, valueOK ); EXEC_DEPRECATED_CALL(QVERIFY( value.isValid() == value.canConvert( QVariant::Int ) );) QVERIFY( value.isValid() == value.canConvert(QMetaType::fromType()) ); bool ok; int i = value.toInt( &ok ); QCOMPARE( i, result ); QVERIFY( ok == valueOK ); } void tst_QVariant::toUInt_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::addColumn("valueOK"); QTest::newRow( "int" ) << QVariant( 123 ) << (uint)123 << true; QTest::newRow( "char" ) << QVariant::fromValue('a') << uint('a') << true; signed char signedChar = 12; QTest::newRow( "signed char" ) << QVariant::fromValue(signedChar) << uint(12) << true; QTest::newRow( "double" ) << QVariant( 3.1415927 ) << (uint)3 << true; QTest::newRow( "float" ) << QVariant( 3.1415927f ) << (uint)3 << true; QTest::newRow( "uint" ) << QVariant( 123u ) << (uint)123 << true; QTest::newRow( "int-string" ) << QVariant( QString("123") ) << (uint)123 << true; QTest::newRow( "string" ) << QVariant( QString("Unicode String") ) << (uint)0 << false; QTest::newRow( "string2" ) << QVariant( QString("4") ) << (uint)4 << true; QTest::newRow( "longlong0" ) << QVariant( (qlonglong)34 ) << (uint)34 << true; QTest::newRow( "longlong1" ) << QVariant( intMax1 ) << (uint)INT_MIN << true; QTest::newRow( "ulonglong0" ) << QVariant( (qulonglong)34 ) << (uint)34 << true; QTest::newRow( "ulonglong1" ) << QVariant( uintMax1 ) << (uint)0 << true; QTest::newRow( "negativeint" ) << QVariant( -123 ) << (uint)-123 << true; QTest::newRow( "negativedouble" ) << QVariant( -3.1415927 ) << (uint)-3 << true; QTest::newRow( "negativefloat" ) << QVariant( -3.1415927f ) << (uint)-3 << true; QTest::newRow( "negativeint-string" ) << QVariant( QString("-123") ) << (uint)0 << false; QTest::newRow( "negativelonglong0" ) << QVariant( (qlonglong)-34 ) << (uint)-34 << true; QTest::newRow( "QChar" ) << QVariant(QChar('a')) << uint('a') << true; QByteArray bytearray(4, ' '); bytearray[0] = '4'; bytearray[1] = '3'; bytearray[2] = '2'; bytearray[3] = '1'; QTest::newRow( "QByteArray" ) << QVariant( bytearray ) << (uint)4321 << true; QTest::newRow("int-QJsonValue") << QVariant(QJsonValue(321)) << (uint)321 << true; QTest::newRow("null-QJsonValue") << QVariant(QJsonValue(QJsonValue::Null)) << (uint)0 << false; } void tst_QVariant::toUInt() { QFETCH( QVariant, value ); QFETCH( uint, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::UInt ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); bool ok; uint i = value.toUInt( &ok ); QVERIFY( ok == valueOK ); QCOMPARE( i, result ); } void tst_QVariant::toSize_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "qsizef4" ) << QVariant( QSizeF(4, 2) ) << QSize(4, 2); QTest::newRow( "qsizef1" ) << QVariant( QSizeF(0, 0) ) << QSize(0, 0); QTest::newRow( "qsizef2" ) << QVariant( QSizeF(-5, -1) ) << QSize(-5, -1); QTest::newRow( "qsizef3" ) << QVariant( QSizeF() ) << QSize(); } void tst_QVariant::toSize() { QFETCH( QVariant, value ); QFETCH( QSize, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Size ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QSize i = value.toSize(); QCOMPARE( i, result ); } void tst_QVariant::toSizeF_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "qsize1" ) << QVariant( QSize(0, 0) ) << QSizeF(0, 0); QTest::newRow( "qsize2" ) << QVariant( QSize(-5, -1) ) << QSizeF(-5, -1); QTest::newRow( "qsize3" ) << QVariant( QSize() ) << QSizeF(); QTest::newRow( "qsize4" ) << QVariant(QSize(4,2)) << QSizeF(4,2); } void tst_QVariant::toSizeF() { QFETCH( QVariant, value ); QFETCH( QSizeF, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::SizeF ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QSizeF i = value.toSizeF(); QCOMPARE( i, result ); } void tst_QVariant::toLine_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "linef1" ) << QVariant( QLineF(1, 2, 3, 4) ) << QLine(1, 2, 3, 4); QTest::newRow( "linef2" ) << QVariant( QLineF(-1, -2, -3, -4) ) << QLine(-1, -2, -3, -4); QTest::newRow( "linef3" ) << QVariant( QLineF(0, 0, 0, 0) ) << QLine(0, 0, 0, 0); QTest::newRow( "linef4" ) << QVariant( QLineF() ) << QLine(); } void tst_QVariant::toLine() { QFETCH( QVariant, value ); QFETCH( QLine, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Line ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QLine i = value.toLine(); QCOMPARE( i, result ); } void tst_QVariant::toLineF_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "line1" ) << QVariant( QLine(-1, -2, -3, -4) ) << QLineF(-1, -2, -3, -4); QTest::newRow( "line2" ) << QVariant( QLine(0, 0, 0, 0) ) << QLineF(0, 0, 0, 0); QTest::newRow( "line3" ) << QVariant( QLine() ) << QLineF(); QTest::newRow( "line4" ) << QVariant( QLine(1, 2, 3, 4) ) << QLineF(1, 2, 3, 4); } void tst_QVariant::toLineF() { QFETCH( QVariant, value ); QFETCH( QLineF, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::LineF ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QLineF i = value.toLineF(); QCOMPARE( i, result ); } void tst_QVariant::toPoint_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "pointf1" ) << QVariant( QPointF(4, 2) ) << QPoint(4, 2); QTest::newRow( "pointf2" ) << QVariant( QPointF(0, 0) ) << QPoint(0, 0); QTest::newRow( "pointf3" ) << QVariant( QPointF(-4, -2) ) << QPoint(-4, -2); QTest::newRow( "pointf4" ) << QVariant( QPointF() ) << QPoint(); QTest::newRow( "pointf5" ) << QVariant( QPointF(-4.2f, -2.3f) ) << QPoint(-4, -2); } void tst_QVariant::toPoint() { QFETCH( QVariant, value ); QFETCH( QPoint, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Point ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QPoint i = value.toPoint(); QCOMPARE( i, result ); } void tst_QVariant::toRect_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "rectf1" ) << QVariant(QRectF(1, 2, 3, 4)) << QRect(1, 2, 3, 4); QTest::newRow( "rectf2" ) << QVariant(QRectF(0, 0, 0, 0)) << QRect(0, 0, 0, 0); QTest::newRow( "rectf3" ) << QVariant(QRectF(-1, -2, -3, -4)) << QRect(-1, -2, -3, -4); QTest::newRow( "rectf4" ) << QVariant(QRectF(-1.3f, 0, 3.9f, -4.0)) << QRect(-1, 0, 4, -4); QTest::newRow( "rectf5" ) << QVariant(QRectF()) << QRect(); } void tst_QVariant::toRect() { QFETCH( QVariant, value ); QFETCH( QRect, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Rect ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QRect i = value.toRect(); QCOMPARE( i, result ); } void tst_QVariant::toChar_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "longlong" ) << QVariant(qlonglong('6')) << QChar('6'); QTest::newRow( "ulonglong" ) << QVariant(qulonglong('7')) << QChar('7'); } void tst_QVariant::toChar() { QFETCH( QVariant, value ); QFETCH( QChar, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Char ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QChar i = value.toChar(); QCOMPARE( i, result ); } void tst_QVariant::toBool_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "int0" ) << QVariant( 0 ) << false; QTest::newRow( "int1" ) << QVariant( 123 ) << true; QTest::newRow( "uint0" ) << QVariant( 0u ) << false; QTest::newRow( "uint1" ) << QVariant( 123u ) << true; QTest::newRow( "double0" ) << QVariant( 0.0 ) << false; QTest::newRow( "float0" ) << QVariant( 0.0f ) << false; QTest::newRow( "double1" ) << QVariant( 3.1415927 ) << true; QTest::newRow( "float1" ) << QVariant( 3.1415927f ) << true; QTest::newRow( "string0" ) << QVariant( QString("3") ) << true; QTest::newRow( "string1" ) << QVariant( QString("true") ) << true; QTest::newRow( "string2" ) << QVariant( QString("0") ) << false; QTest::newRow( "string3" ) << QVariant( QString("fAlSe") ) << false; QTest::newRow( "longlong0" ) << QVariant( (qlonglong)0 ) << false; QTest::newRow( "longlong1" ) << QVariant( (qlonglong)1 ) << true; QTest::newRow( "ulonglong0" ) << QVariant( (qulonglong)0 ) << false; QTest::newRow( "ulonglong1" ) << QVariant( (qulonglong)1 ) << true; QTest::newRow( "QChar" ) << QVariant(QChar('a')) << true; QTest::newRow( "Null_QChar" ) << QVariant(QChar(0)) << false; QTest::newRow("QJsonValue(true)") << QVariant(QJsonValue(true)) << true; QTest::newRow("QJsonValue(false)") << QVariant(QJsonValue(false)) << false; } void tst_QVariant::toBool() { QFETCH( QVariant, value ); QFETCH( bool, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Bool ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); bool i = value.toBool(); QCOMPARE( i, result ); } void tst_QVariant::toPointF_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "QPoint" ) << QVariant( QPointF( 19, 84) ) << QPointF( 19, 84 ); } void tst_QVariant::toPointF() { QFETCH( QVariant, value ); QFETCH( QPointF, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::PointF ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QPointF d = value.toPointF(); QCOMPARE( d, result ); } void tst_QVariant::toRectF_data() { QTest::addColumn("value"); QTest::addColumn("result"); QRect r( 1, 9, 8, 4 ); QRectF rf( 1.0, 9.0, 8.0, 4.0 ); QTest::newRow( "QRect" ) << QVariant( r ) << rf; } void tst_QVariant::toRectF() { QFETCH( QVariant, value ); QFETCH( QRectF, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::RectF ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QRectF d = value.toRectF(); QCOMPARE( d, result ); } void tst_QVariant::toDouble_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::addColumn("valueOK"); QByteArray bytearray(4, ' '); bytearray[0] = '3'; bytearray[1] = '2'; bytearray[2] = '.'; bytearray[3] = '1'; QTest::newRow( "bytearray" ) << QVariant( bytearray ) << 32.1 << true; QTest::newRow("double-QJsonValue") << QVariant(QJsonValue(32.1)) << 32.1 << true; QTest::newRow("null-QJsonValue") << QVariant(QJsonValue(QJsonValue::Null)) << 0.0 << false; } void tst_QVariant::toDouble() { QFETCH( QVariant, value ); QFETCH( double, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Double ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); bool ok; double d = value.toDouble( &ok ); QCOMPARE( d, result ); QVERIFY( ok == valueOK ); } void tst_QVariant::toFloat_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::addColumn("valueOK"); QByteArray bytearray(4, ' '); bytearray[0] = '3'; bytearray[1] = '2'; bytearray[2] = '.'; bytearray[3] = '1'; QTest::newRow("QByteArray") << QVariant(bytearray) << float(32.1) << true; QTest::newRow("double-QJsonValue") << QVariant(QJsonValue(32.1)) << float(32.1) << true; QTest::newRow("undefined-QJsonValue") << QVariant(QJsonValue(QJsonValue::Undefined)) << float(0.0) << false; } void tst_QVariant::toFloat() { QFETCH(QVariant, value ); QFETCH(float, result); QFETCH(bool, valueOK); QVERIFY(value.isValid()); EXEC_DEPRECATED_CALL(QVERIFY(value.canConvert(QMetaType::Float));) QVERIFY(value.canConvert(QMetaType::fromType())); bool ok; float d = value.toFloat(&ok); QCOMPARE(d, result); QCOMPARE(ok, valueOK); } void tst_QVariant::toLongLong_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::addColumn("valueOK"); QTest::newRow( "int0" ) << QVariant( 123 ) << (qlonglong)123 << true; QTest::newRow( "double" ) << QVariant( 3.1415927 ) << (qlonglong)3 << true; QTest::newRow( "float" ) << QVariant( 3.1415927f ) << (qlonglong)3 << true; QTest::newRow( "uint" ) << QVariant( 123u ) << (qlonglong)123 << true; QTest::newRow( "int-string" ) << QVariant( QString("123") ) << (qlonglong)123 << true; QTest::newRow( "string" ) << QVariant( QString("Unicode fun") ) << (qlonglong)0 << false; QTest::newRow( "longlong" ) << QVariant( intMax1 ) << intMax1 << true; QTest::newRow( "ulonglong" ) << QVariant( uintMax1 ) << (qlonglong)uintMax1 << true; QTest::newRow( "QChar" ) << QVariant(QChar('a')) << qlonglong('a') << true; QByteArray bytearray(4, ' '); bytearray[0] = '3'; bytearray[1] = '2'; bytearray[2] = '0'; bytearray[3] = '0'; QTest::newRow( "QByteArray" ) << QVariant( bytearray ) << (qlonglong) 3200 << true; QTest::newRow("int-QJsonValue") << QVariant(QJsonValue(321)) << (qlonglong)321 << true; QTest::newRow("string-QJsonValue") << QVariant(QJsonValue(QString("string"))) << (qlonglong)0 << false; qint64 value64 = (Q_INT64_C(12) << 35) + 8; QTest::newRow("qint64") << QVariant::fromValue(value64) << qlonglong(value64) << true; QTest::newRow("-qint64") << QVariant::fromValue(-value64) << qlonglong(-value64) << true; QTest::newRow("long") << QVariant::fromValue(long(464646)) << qlonglong(464646) << true; QTest::newRow("LONG_MAX") << QVariant::fromValue( LONG_MAX ) << qlonglong(LONG_MAX) << true; QTest::newRow("LONG_MIN") << QVariant::fromValue( LONG_MIN ) << qlonglong(LONG_MIN) << true; QTest::newRow( "short" ) << QVariant(short(12)) << qlonglong(12) << true; QTest::newRow( "-short" ) << QVariant(short(-24)) << qlonglong(-24) << true; QTest::newRow( "ushort" ) << QVariant(ushort(15)) << qlonglong(15) << true; } void tst_QVariant::toLongLong() { QFETCH( QVariant, value ); QFETCH( qlonglong, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::LongLong ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); bool ok; qlonglong ll = value.toLongLong( &ok ); QCOMPARE( ll, result ); QVERIFY( ok == valueOK ); } void tst_QVariant::toULongLong_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::addColumn("valueOK"); QTest::newRow( "int0" ) << QVariant( 123 ) << (qulonglong)123 << true; QTest::newRow( "double" ) << QVariant( 3.1415927 ) << (qulonglong)3 << true; QTest::newRow( "float" ) << QVariant( 3.1415927f ) << (qulonglong)3 << true; QTest::newRow( "uint" ) << QVariant( 123u ) << (qulonglong)123 << true; QTest::newRow( "int-string" ) << QVariant( QString("123") ) << (qulonglong)123 << true; QTest::newRow( "string" ) << QVariant( QString("Unicode fun") ) << (qulonglong)0 << false; QTest::newRow( "ulonglong-string" ) << QVariant( QString("18446744073709551615") ) << Q_UINT64_C(18446744073709551615) << true; QTest::newRow( "bytaa-string" ) << QVariant( QString("18446744073709551615") ) << Q_UINT64_C(18446744073709551615) << true; QTest::newRow( "longlong" ) << QVariant( intMax1 ) << (qulonglong)intMax1 << true; QTest::newRow( "ulonglong" ) << QVariant( uintMax1 ) << uintMax1 << true; QTest::newRow( "QChar" ) << QVariant(QChar('a')) << qulonglong('a') << true; QByteArray bytearray(4, ' '); bytearray[0] = '3'; bytearray[1] = '2'; bytearray[2] = '0'; bytearray[3] = '1'; QTest::newRow( "QByteArray" ) << QVariant( bytearray ) << (qulonglong) 3201 << true; QTest::newRow("int-QJsonValue") << QVariant(QJsonValue(321)) << (qulonglong)321 << true; QTest::newRow("bool-QJsonValue") << QVariant(QJsonValue(true)) << (qulonglong)0 << false; quint64 value64 = (Q_INT64_C(12) << 35) + 8; QTest::newRow("qint64") << QVariant::fromValue(value64) << qulonglong(value64) << true; QTest::newRow("long") << QVariant::fromValue(long(464646)) << qulonglong(464646) << true; QTest::newRow("LONG_MAX") << QVariant::fromValue( LONG_MAX ) << qulonglong(LONG_MAX) << true; QTest::newRow("ULONG_MAX") << QVariant::fromValue( ULONG_MAX ) << qulonglong(ULONG_MAX) << true; QTest::newRow( "short" ) << QVariant(short(12)) << qulonglong(12) << true; QTest::newRow( "-short" ) << QVariant(short(-24)) << qulonglong(-24) << true; QTest::newRow( "ushort" ) << QVariant(ushort(15)) << qulonglong(15) << true; } void tst_QVariant::toULongLong() { QFETCH( QVariant, value ); QFETCH( qulonglong, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::ULongLong ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); bool ok; qulonglong ll = value.toULongLong( &ok ); QCOMPARE( ll, result ); QVERIFY( ok == valueOK ); } void tst_QVariant::toByteArray_data() { QTest::addColumn("value"); QTest::addColumn("result"); QByteArray ba(5, ' '); ba[0] = 'T'; ba[1] = 'e'; ba[2] = 's'; ba[3] = 't'; ba[4] = '\0'; QByteArray variantBa = ba; QTest::newRow( "qbytearray" ) << QVariant( variantBa ) << ba; QTest::newRow( "int" ) << QVariant( -123 ) << QByteArray( "-123" ); QTest::newRow( "uint" ) << QVariant( (uint)123 ) << QByteArray( "123" ); QTest::newRow( "double" ) << QVariant( 123.456 ) << QByteArray( "123.456" ); // Conversion from float to double adds bits of which the double-to-string converter doesn't // know they're insignificant QTest::newRow( "float" ) << QVariant( 123.456f ) << QByteArray( "123.45600128173828" ); QTest::newRow( "longlong" ) << QVariant( (qlonglong)34 ) << QByteArray( "34" ); QTest::newRow( "ulonglong" ) << QVariant( (qulonglong)34 ) << QByteArray( "34" ); QTest::newRow( "nullptr" ) << QVariant::fromValue(nullptr) << QByteArray(); } void tst_QVariant::toByteArray() { QFETCH( QVariant, value ); QFETCH( QByteArray, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::ByteArray ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QByteArray ba = value.toByteArray(); QCOMPARE( ba.isNull(), result.isNull() ); QCOMPARE( ba, result ); QVERIFY( value.convert(QMetaType::fromType()) ); QCOMPARE( value.toByteArray(), result ); } void tst_QVariant::toString_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "qstring" ) << QVariant( QString( "Test" ) ) << QString( "Test" ); QTest::newRow( "charstar" ) << QVariant(QLatin1String("Test")) << QString("Test"); QTest::newRow( "qbytearray") << QVariant( QByteArray( "Test\0" ) ) << QString( "Test" ); QTest::newRow( "int" ) << QVariant( -123 ) << QString( "-123" ); QTest::newRow( "uint" ) << QVariant( (uint)123 ) << QString( "123" ); QTest::newRow( "double" ) << QVariant( 123.456 ) << QString( "123.456" ); // Conversion from float to double adds bits of which the double-to-string converter doesn't // know they're insignificant QTest::newRow( "float" ) << QVariant( 123.456f ) << QString( "123.45600128173828" ); QTest::newRow( "bool" ) << QVariant( true ) << QString( "true" ); QTest::newRow( "qdate" ) << QVariant( QDate( 2002, 1, 1 ) ) << QString( "2002-01-01" ); QTest::newRow( "qtime" ) << QVariant( QTime( 12, 34, 56 ) ) << QString( "12:34:56.000" ); QTest::newRow( "qtime-with-ms" ) << QVariant( QTime( 12, 34, 56, 789 ) ) << QString( "12:34:56.789" ); QTest::newRow( "qdatetime" ) << QVariant( QDateTime( QDate( 2002, 1, 1 ), QTime( 12, 34, 56, 789 ) ) ) << QString( "2002-01-01T12:34:56.789" ); QTest::newRow( "llong" ) << QVariant( (qlonglong)Q_INT64_C(123456789012) ) << QString( "123456789012" ); QTest::newRow("QJsonValue") << QVariant(QJsonValue(QString("hello"))) << QString("hello"); QTest::newRow("QJsonValue(Null)") << QVariant(QJsonValue(QJsonValue::Null)) << QString(); QTest::newRow("nullptr") << QVariant::fromValue(nullptr) << QString(); } void tst_QVariant::toString() { QFETCH( QVariant, value ); QFETCH( QString, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::String ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QString str = value.toString(); QCOMPARE( str.isNull(), result.isNull() ); QCOMPARE( str, result ); QVERIFY( value.convert(QMetaType::fromType()) ); QCOMPARE( value.toString(), result ); } void tst_QVariant::toDate_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "qdate" ) << QVariant( QDate( 2002, 10, 10 ) ) << QDate( 2002, 10, 10 ); QTest::newRow( "qdatetime" ) << QVariant( QDateTime( QDate( 2002, 10, 10 ), QTime( 12, 34, 56 ) ) ) << QDate( 2002, 10, 10 ); QTest::newRow( "qstring" ) << QVariant( QString( "2002-10-10" ) ) << QDate( 2002, 10, 10 ); } void tst_QVariant::toDate() { QFETCH( QVariant, value ); QFETCH( QDate, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Date ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QCOMPARE( value.toDate(), result ); } void tst_QVariant::toTime_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "qtime" ) << QVariant( QTime( 12, 34, 56 ) ) << QTime( 12, 34, 56 ); QTest::newRow( "qdatetime" ) << QVariant( QDateTime( QDate( 2002, 10, 10 ), QTime( 12, 34, 56 ) ) ) << QTime( 12, 34, 56 ); QTest::newRow( "qstring" ) << QVariant( QString( "12:34:56" ) ) << QTime( 12, 34, 56 ); QTest::newRow( "qstring-with-ms" ) << QVariant( QString( "12:34:56.789" ) ) << QTime( 12, 34, 56, 789 ); } void tst_QVariant::toTime() { QFETCH( QVariant, value ); QFETCH( QTime, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::Time ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QCOMPARE( value.toTime(), result ); } void tst_QVariant::toDateTime_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "qdatetime" ) << QVariant( QDateTime( QDate( 2002, 10, 10 ), QTime( 12, 34, 56 ) ) ) << QDateTime( QDate( 2002, 10, 10 ), QTime( 12, 34, 56 ) ); QTest::newRow( "qdate" ) << QVariant( QDate( 2002, 10, 10 ) ) << QDateTime( QDate( 2002, 10, 10 ), QTime( 0, 0, 0 ) ); QTest::newRow( "qstring" ) << QVariant( QString( "2002-10-10T12:34:56" ) ) << QDateTime( QDate( 2002, 10, 10 ), QTime( 12, 34, 56 ) ); QTest::newRow("qstring-utc") << QVariant(QString("2002-10-10T12:34:56Z")) << QDateTime(QDate(2002, 10, 10), QTime(12, 34, 56), QTimeZone::UTC); QTest::newRow( "qstring-with-ms" ) << QVariant( QString( "2002-10-10T12:34:56.789" ) ) << QDateTime( QDate( 2002, 10, 10 ), QTime( 12, 34, 56, 789 ) ); } void tst_QVariant::toDateTime() { QFETCH( QVariant, value ); QFETCH( QDateTime, result ); QVERIFY( value.isValid() ); EXEC_DEPRECATED_CALL(QVERIFY( value.canConvert( QVariant::DateTime ) );) QVERIFY( value.canConvert(QMetaType::fromType()) ); QCOMPARE( value.toDateTime(), result ); } #undef EXEC_DEPRECATED_CALL void tst_QVariant::toLocale() { QVariant variant; QLocale loc = variant.toLocale(); variant = QLocale::system(); loc = variant.toLocale(); } void tst_QVariant::toRegularExpression() { QVariant variant; QRegularExpression re = variant.toRegularExpression(); QCOMPARE(re, QRegularExpression()); variant = QRegularExpression("abc.*def"); re = variant.toRegularExpression(); QCOMPARE(re, QRegularExpression("abc.*def")); variant = QVariant::fromValue(QRegularExpression("[ab]\\w+")); re = variant.value(); QCOMPARE(re, QRegularExpression("[ab]\\w+")); } struct CustomStreamableClass { int i; bool operator==(const CustomStreamableClass& other) const { return i == other.i; } }; QDataStream &operator<<(QDataStream &out, const CustomStreamableClass &myObj) { return out << myObj.i; } QDataStream &operator>>(QDataStream &in, CustomStreamableClass &myObj) { return in >> myObj.i; } Q_DECLARE_METATYPE(CustomStreamableClass); void tst_QVariant::writeToReadFromDataStream_data() { QTest::addColumn("writeVariant"); QTest::addColumn("isNull"); { QVariantList valuelist; valuelist << QVariant( 1 ) << QVariant( QString("Two") ) << QVariant( 3.45 ); QVariant var(valuelist); QTest::newRow( "list_valid" ) << var << false; } QTest::newRow( "invalid" ) << QVariant() << true; QTest::newRow( "bitarray_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "bitarray_empty" ) << QVariant( QBitArray() ) << false; QBitArray bitarray( 3 ); bitarray[0] = 0; bitarray[1] = 1; bitarray[2] = 0; QTest::newRow( "bitarray_valid" ) << QVariant( bitarray ) << false; QTest::newRow( "bytearray_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "bytearray_empty" ) << QVariant( QByteArray() ) << false; QTest::newRow( "int_invalid") << QVariant(QMetaType::fromType()) << true; QByteArray bytearray(5, ' '); bytearray[0] = 'T'; bytearray[1] = 'e'; bytearray[2] = 's'; bytearray[3] = 't'; bytearray[4] = '\0'; QTest::newRow( "bytearray_valid" ) << QVariant( bytearray ) << false; QTest::newRow( "date_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "date_empty" ) << QVariant( QDate() ) << false; QTest::newRow( "date_valid" ) << QVariant( QDate( 2002, 07, 06 ) ) << false; QTest::newRow( "datetime_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "datetime_empty" ) << QVariant( QDateTime() ) << false; QTest::newRow( "datetime_valid" ) << QVariant( QDateTime( QDate( 2002, 07, 06 ), QTime( 14, 0, 0 ) ) ) << false; QTest::newRow( "double_valid" ) << QVariant( 123.456 ) << false; QTest::newRow( "float_valid" ) << QVariant( 123.456f ) << false; QTest::newRow( "int_valid" ) << QVariant( -123 ) << false; QVariantMap vMap; vMap.insert( "int", QVariant( 1 ) ); vMap.insert( "string", QVariant( QString("Two") ) ); vMap.insert( "double", QVariant( 3.45 ) ); vMap.insert( "float", QVariant( 3.45f ) ); QTest::newRow( "map_valid" ) << QVariant( vMap ) << false; QTest::newRow( "point_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "point_empty" ) << QVariant::fromValue( QPoint() ) << false; QTest::newRow( "point_valid" ) << QVariant::fromValue( QPoint( 10, 10 ) ) << false; QTest::newRow( "rect_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "rect_empty" ) << QVariant( QRect() ) << false; QTest::newRow( "rect_valid" ) << QVariant( QRect( 10, 10, 20, 20 ) ) << false; QTest::newRow( "size_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "size_empty" ) << QVariant( QSize( 0, 0 ) ) << false; QTest::newRow( "size_valid" ) << QVariant( QSize( 10, 10 ) ) << false; QTest::newRow( "string_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "string_empty" ) << QVariant( QString() ) << false; QTest::newRow( "string_valid" ) << QVariant( QString( "Test" ) ) << false; QStringList stringlist; stringlist << "One" << "Two" << "Three"; QTest::newRow( "stringlist_valid" ) << QVariant( stringlist ) << false; QTest::newRow( "time_invalid" ) << QVariant(QMetaType::fromType()) << true; QTest::newRow( "time_empty" ) << QVariant( QTime() ) << false; QTest::newRow( "time_valid" ) << QVariant( QTime( 14, 0, 0 ) ) << false; QTest::newRow( "uint_valid" ) << QVariant( (uint)123 ) << false; QTest::newRow( "qchar" ) << QVariant(QChar('a')) << false; QTest::newRow( "qchar_null" ) << QVariant(QChar(0)) << false; QTest::newRow( "regularexpression" ) << QVariant(QRegularExpression("abc.*def")) << false; QTest::newRow( "regularexpression_empty" ) << QVariant(QRegularExpression()) << false; // types known to QMetaType, but not part of QVariant::Type QTest::newRow("QMetaType::Long invalid") << QVariant(QMetaType::fromType(), nullptr) << true; long longInt = -1l; QTest::newRow("QMetaType::Long") << QVariant(QMetaType::fromType(), &longInt) << false; QTest::newRow("QMetaType::Short invalid") << QVariant(QMetaType::fromType(), nullptr) << true; short shortInt = 1; QTest::newRow("QMetaType::Short") << QVariant(QMetaType::fromType(), &shortInt) << false; QTest::newRow("QMetaType::Char invalid") << QVariant(QMetaType::fromType(), nullptr) << true; char ch = 'c'; QTest::newRow("QMetaType::Char") << QVariant(QMetaType::fromType(), &ch) << false; QTest::newRow("QMetaType::ULong invalid") << QVariant(QMetaType::fromType(), nullptr) << true; ulong ulongInt = 1ul; QTest::newRow("QMetaType::ULong") << QVariant(QMetaType::fromType(), &ulongInt) << false; QTest::newRow("QMetaType::UShort invalid") << QVariant(QMetaType::fromType(), nullptr) << true; ushort ushortInt = 1u; QTest::newRow("QMetaType::UShort") << QVariant(QMetaType::fromType(), &ushortInt) << false; QTest::newRow("QMetaType::UChar invalid") << QVariant(QMetaType::fromType(), nullptr) << true; uchar uch = 0xf0; QTest::newRow("QMetaType::UChar") << QVariant(QMetaType::fromType(), &uch) << false; QTest::newRow("QMetaType::Float invalid") << QVariant(QMetaType::fromType(), nullptr) << true; float f = 1.234f; QTest::newRow("QMetaType::Float") << QVariant(QMetaType::fromType(), &f) << false; CustomStreamableClass custom = {123}; QTest::newRow("Custom type") << QVariant::fromValue(custom) << false; } void tst_QVariant::writeToReadFromDataStream() { QFETCH( QVariant, writeVariant ); QFETCH( bool, isNull ); QByteArray data; QDataStream writeStream( &data, QIODevice::WriteOnly ); writeStream << writeVariant; QVariant readVariant; QDataStream readStream( &data, QIODevice::ReadOnly ); readStream >> readVariant; QVERIFY( readVariant.isNull() == isNull ); // Best way to confirm the readVariant contains the same data? // Since only a few won't match since the serial numbers are different // I won't bother adding another bool in the data test. const int writeType = writeVariant.userType(); if (writeType == qMetaTypeId()) { QCOMPARE(qvariant_cast(readVariant), qvariant_cast(writeVariant)); } else if ( writeType != QMetaType::UnknownType && writeType != QMetaType::QBitmap && writeType != QMetaType::QPixmap && writeType != QMetaType::QImage) { switch (writeType) { default: QCOMPARE( readVariant, writeVariant ); break; // compare types know by QMetaType but not QVariant (QVariant::operator==() knows nothing about them) case QMetaType::Long: QCOMPARE(qvariant_cast(readVariant), qvariant_cast(writeVariant)); break; case QMetaType::ULong: QCOMPARE(qvariant_cast(readVariant), qvariant_cast(writeVariant)); break; case QMetaType::Short: QCOMPARE(qvariant_cast(readVariant), qvariant_cast(writeVariant)); break; case QMetaType::UShort: QCOMPARE(qvariant_cast(readVariant), qvariant_cast(writeVariant)); break; case QMetaType::Char: QCOMPARE(qvariant_cast(readVariant), qvariant_cast(writeVariant)); break; case QMetaType::UChar: QCOMPARE(qvariant_cast(readVariant), qvariant_cast(writeVariant)); break; case QMetaType::Float: { // the uninitialized float can be NaN (observed on Windows Mobile 5 ARMv4i) float readFloat = qvariant_cast(readVariant); float writtenFloat = qvariant_cast(writeVariant); QCOMPARE(qIsNaN(readFloat), qIsNaN(writtenFloat)); if (!qIsNaN(readFloat)) QCOMPARE(readFloat, writtenFloat); } break; } } } void tst_QVariant::writeToReadFromOldDataStream() { QVariant writeVariant = QString("hello"); QByteArray data; QDataStream writeStream(&data, QIODevice::WriteOnly); writeStream.setVersion(QDataStream::Qt_2_1); writeStream << writeVariant; QVariant readVariant; QDataStream readStream(&data, QIODevice::ReadOnly); readStream.setVersion(QDataStream::Qt_2_1); readStream >> readVariant; QCOMPARE(writeVariant.userType(), readVariant.userType()); QCOMPARE(writeVariant, readVariant); } void tst_QVariant::checkDataStream() { const int typeId = QMetaType::LastCoreType + 1; QVERIFY(!QMetaType::isRegistered(typeId)); QTest::ignoreMessage(QtWarningMsg, QRegularExpression("^Trying to construct an instance of an invalid type")); QByteArray settingsHex("000000"); settingsHex.append(QByteArray::number(typeId, 16)); settingsHex.append("ffffffffff"); const QByteArray settings = QByteArray::fromHex(settingsHex); QDataStream in(settings); QVariant v; in >> v; // the line below has been left out for now since the data stream // is not necessarily considered corrupt when an invalid QVariant is // constructed. However, it might be worth considering changing that behavior // in the future. // QCOMPARE(in.status(), QDataStream::ReadCorruptData); QCOMPARE(v.typeId(), QMetaType::UnknownType); } void tst_QVariant::operator_eq_eq_data() { QTest::addColumn("left"); QTest::addColumn("right"); QTest::addColumn("equal"); // left == right ? QVariant inv; QVariant i0( int(0) ); QVariant i1( int(1) ); // Invalid QTest::newRow( "invinv" ) << inv << inv << true; // Int QTest::newRow( "int1int1" ) << i1 << i1 << true; QTest::newRow( "int1int0" ) << i1 << i0 << false; QTest::newRow( "nullint" ) << i0 << QVariant(QMetaType::fromType()) << true; // LongLong and ULongLong QVariant ll1( (qlonglong)1 ); QVariant lln2( (qlonglong)-2 ); QVariant ull1( (qulonglong)1 ); QVariant ull3( (qulonglong)3 ); QTest::newRow( "ll1ll1" ) << ll1 << ll1 << true; QTest::newRow( "ll1lln2" ) << ll1 << lln2 << false; QTest::newRow( "ll1ull1" ) << ull1 << ull1 << true; QTest::newRow( "ll1i1" ) << ull1 << i1 << true; QTest::newRow( "ull1ull1" ) << ull1 << ull1 << true; QTest::newRow( "ull1i1" ) << ull1 << ull1 << true; QVariant mInt(-42); QVariant mIntString(QByteArray("-42")); QVariant mIntQString(QString("-42")); QVariant mIntZero(0); QVariant mIntStringZero(QByteArray("0")); QVariant mIntQStringZero(QString("0")); QVariant mUInt(42u); QVariant mUIntString(QByteArray("42")); QVariant mUIntQString(QString("42")); QVariant mDouble(42.11); #ifdef QT_NO_DOUBLECONVERSION // Without libdouble-conversion we don't get the shortest possible representation. QVariant mDoubleString(QByteArray("42.109999999999999")); QVariant mDoubleQString(QByteArray("42.109999999999999")); #else // You cannot fool the double-to-string conversion into producing insignificant digits with // libdouble-conversion. You can, of course, add insignificant digits to the string and fool // the double-to-double comparison after converting the string to a double. QVariant mDoubleString(QByteArray("42.11")); QVariant mDoubleQString(QString("42.11")); #endif // Float-to-double conversion produces insignificant extra bits. QVariant mFloat(42.11f); #ifdef QT_NO_DOUBLECONVERSION // The trailing '2' is not significant, but snprintf doesn't know this. QVariant mFloatString(QByteArray("42.110000610351562")); QVariant mFloatQString(QString("42.110000610351562")); #else QVariant mFloatString(QByteArray("42.11000061035156")); QVariant mFloatQString(QString("42.11000061035156")); #endif QVariant mLongLong((qlonglong)-42); QVariant mLongLongString(QByteArray("-42")); QVariant mLongLongQString(QString("-42")); QVariant mULongLong((qulonglong)42); QVariant mULongLongString(QByteArray("42")); QVariant mULongLongQString(QString("42")); QVariant mBool(false); QVariant mBoolString(QByteArray("false")); QVariant mBoolQString(QString("false")); QVariant mTextString(QByteArray("foobar")); QVariant mTextQString(QString("foobar")); QTest::newRow( "double_int" ) << QVariant(42.0) << QVariant(42) << true; QTest::newRow( "float_int" ) << QVariant(42.f) << QVariant(42) << true; QTest::newRow( "mInt_mIntString" ) << mInt << mIntString << false; QTest::newRow( "mIntString_mInt" ) << mIntString << mInt << false; QTest::newRow( "mInt_mIntQString" ) << mInt << mIntQString << true; QTest::newRow( "mIntQString_mInt" ) << mIntQString << mInt << true; QTest::newRow( "mIntZero_mIntStringZero" ) << mIntZero << mIntStringZero << false; QTest::newRow( "mIntStringZero_mIntZero" ) << mIntStringZero << mIntZero << false; QTest::newRow( "mIntZero_mIntQStringZero" ) << mIntZero << mIntQStringZero << true; QTest::newRow( "mIntQStringZero_mIntZero" ) << mIntQStringZero << mIntZero << true; QTest::newRow( "mInt_mTextString" ) << mInt << mTextString << false; QTest::newRow( "mTextString_mInt" ) << mTextString << mInt << false; QTest::newRow( "mInt_mTextQString" ) << mInt << mTextQString << false; QTest::newRow( "mTextQString_mInt" ) << mTextQString << mInt << false; QTest::newRow( "mIntZero_mTextString" ) << mIntZero << mTextString << false; QTest::newRow( "mTextString_mIntZero" ) << mTextString << mIntZero << false; QTest::newRow( "mIntZero_mTextQString" ) << mIntZero << mTextQString << false; QTest::newRow( "mTextQString_mIntZero" ) << mTextQString << mIntZero << false; QTest::newRow( "mUInt_mUIntString" ) << mUInt << mUIntString << false; QTest::newRow( "mUIntString_mUInt" ) << mUIntString << mUInt << false; QTest::newRow( "mUInt_mUIntQString" ) << mUInt << mUIntQString << true; QTest::newRow( "mUIntQString_mUInt" ) << mUIntQString << mUInt << true; QTest::newRow( "mDouble_mDoubleString" ) << mDouble << mDoubleString << false; QTest::newRow( "mDoubleString_mDouble" ) << mDoubleString << mDouble << false; QTest::newRow( "mDouble_mDoubleQString" ) << mDouble << mDoubleQString << true; QTest::newRow( "mDoubleQString_mDouble" ) << mDoubleQString << mDouble << true; QTest::newRow( "mDouble_mTextString" ) << mDouble << mTextString << false; QTest::newRow( "mTextString_mDouble" ) << mTextString << mDouble << false; QTest::newRow( "mDouble_mTextQString" ) << mDouble << mTextQString << false; QTest::newRow( "mTextQString_mDouble" ) << mTextQString << mDouble << false; QTest::newRow( "mFloat_mFloatString" ) << mFloat << mFloatString << false; QTest::newRow( "mFloatString_mFloat" ) << mFloatString << mFloat << false; QTest::newRow( "mFloat_mFloatQString" ) << mFloat << mFloatQString << true; QTest::newRow( "mFloatQString_mFloat" ) << mFloatQString << mFloat << true; QTest::newRow( "mLongLong_mLongLongString" ) << mLongLong << mLongLongString << false; QTest::newRow( "mLongLongString_mLongLong" ) << mLongLongString << mLongLong << false; QTest::newRow( "mLongLong_mLongLongQString" ) << mLongLong << mLongLongQString << true; QTest::newRow( "mLongLongQString_mLongLong" ) << mLongLongQString << mLongLong << true; QTest::newRow( "mULongLong_mULongLongString" ) << mULongLong << mULongLongString << false; QTest::newRow( "mULongLongString_mULongLong" ) << mULongLongString << mULongLong << false; QTest::newRow( "mULongLong_mULongLongQString" ) << mULongLong << mULongLongQString << true; QTest::newRow( "mULongLongQString_mULongLong" ) << mULongLongQString << mULongLong << true; QTest::newRow( "mBool_mBoolString" ) << mBool << mBoolString << false; QTest::newRow( "mBoolString_mBool" ) << mBoolString << mBool << false; QTest::newRow( "mBool_mBoolQString" ) << mBool << mBoolQString << true; QTest::newRow( "mBoolQString_mBool" ) << mBoolQString << mBool << true; QTest::newRow("ba2qstring") << QVariant(QLatin1String("hallo")) << QVariant(QString("hallo")) << true; QTest::newRow("qstring2ba") << QVariant(QString("hallo")) << QVariant(QLatin1String("hallo")) << true; QTest::newRow("char_char") << QVariant(QChar('a')) << QVariant(QChar('a')) << true; QTest::newRow("char_char2") << QVariant(QChar('a')) << QVariant(QChar('b')) << false; QTest::newRow("invalidConversion") << QVariant(QString("bubu")) << QVariant() << false; QTest::newRow("invalidConversionR") << QVariant() << QVariant(QString("bubu")) << false; // ### many other combinations missing { QUuid uuid(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10); QTest::newRow("uuidstring") << QVariant(uuid) << QVariant(uuid.toString()) << false; QTest::newRow("stringuuid") << QVariant(uuid.toString()) << QVariant(uuid) << false; QTest::newRow("uuidbytearray") << QVariant(uuid) << QVariant(uuid.toByteArray()) << false; QTest::newRow("bytearrayuuid") << QVariant(uuid.toByteArray()) << QVariant(uuid) << false; } { QMap map1; map1.insert( "X", 1 ); QMap map2; map2.insert( "Y", 1 ); QTest::newRow("TwoItemsInEqual") << QVariant(map1) << QVariant(map2) << false; } { QMap map1; map1.insert( "X", 1 ); QMap map2; map2.insert( "X", 1 ); QTest::newRow("TwoItemsEqual") << QVariant(map1) << QVariant(map2) << true; } { QMap map1; map1.insert( "X", 1 ); QMap map2; QTest::newRow("PopulatedEmptyMap") << QVariant(map1) << QVariant(map2) << false; } { QMap map1; QMap map2; map2.insert( "X", 1 ); QTest::newRow("EmptyPopulatedMap") << QVariant(map1) << QVariant(map2) << false; } { QMap map1; map1.insert( "X", 1 ); map1.insert( "Y", 1 ); QMap map2; map2.insert( "X", 1 ); QTest::newRow("FirstLarger") << QVariant(map1) << QVariant(map2) << false; } { QMap map1; map1.insert( "X", 1 ); QMap map2; map2.insert( "X", 1 ); map2.insert( "Y", 1 ); QTest::newRow("SecondLarger") << QVariant(map1) << QVariant(map2) << false; } // same thing with hash { QHash hash1; hash1.insert( "X", 1 ); QHash hash2; hash2.insert( "Y", 1 ); QTest::newRow("HashTwoItemsInEqual") << QVariant(hash1) << QVariant(hash2) << false; } { QHash hash1; hash1.insert( "X", 1 ); QHash hash2; hash2.insert( "X", 1 ); QTest::newRow("HashTwoItemsEqual") << QVariant(hash1) << QVariant(hash2) << true; } { QHash hash1; hash1.insert( "X", 1 ); QHash hash2; QTest::newRow("HashPopulatedEmptyHash") << QVariant(hash1) << QVariant(hash2) << false; } { QHash hash1; QHash hash2; hash2.insert( "X", 1 ); QTest::newRow("EmptyPopulatedHash") << QVariant(hash1) << QVariant(hash2) << false; } { QHash hash1; hash1.insert( "X", 1 ); hash1.insert( "Y", 1 ); QHash hash2; hash2.insert( "X", 1 ); QTest::newRow("HashFirstLarger") << QVariant(hash1) << QVariant(hash2) << false; } { QHash hash1; hash1.insert( "X", 1 ); QHash hash2; hash2.insert( "X", 1 ); hash2.insert( "Y", 1 ); QTest::newRow("HashSecondLarger") << QVariant(hash1) << QVariant(hash2) << false; } } void tst_QVariant::operator_eq_eq() { QFETCH( QVariant, left ); QFETCH( QVariant, right ); QFETCH( bool, equal ); QCOMPARE( left == right, equal ); } #if QT_DEPRECATED_SINCE(6, 0) QT_WARNING_PUSH QT_WARNING_DISABLE_DEPRECATED void tst_QVariant::typeName_data() { QTest::addColumn("type"); QTest::addColumn("res"); QTest::newRow("0") << int(QVariant::Invalid) << QByteArray(""); QTest::newRow("1") << int(QVariant::Map) << QByteArray("QVariantMap"); QTest::newRow("2") << int(QVariant::List) << QByteArray("QVariantList"); QTest::newRow("3") << int(QVariant::String) << QByteArray("QString"); QTest::newRow("4") << int(QVariant::StringList) << QByteArray("QStringList"); QTest::newRow("5") << int(QVariant::Font) << QByteArray("QFont"); QTest::newRow("6") << int(QVariant::Pixmap) << QByteArray("QPixmap"); QTest::newRow("7") << int(QVariant::Brush) << QByteArray("QBrush"); QTest::newRow("8") << int(QVariant::Rect) << QByteArray("QRect"); QTest::newRow("9") << int(QVariant::Size) << QByteArray("QSize"); QTest::newRow("10") << int(QVariant::Color) << QByteArray("QColor"); QTest::newRow("11") << int(QVariant::Palette) << QByteArray("QPalette"); QTest::newRow("12") << int(QVariant::Point) << QByteArray("QPoint"); QTest::newRow("13") << int(QVariant::Image) << QByteArray("QImage"); QTest::newRow("14") << int(QVariant::Int) << QByteArray("int"); QTest::newRow("15") << int(QVariant::UInt) << QByteArray("uint"); QTest::newRow("16") << int(QVariant::Bool) << QByteArray("bool"); QTest::newRow("17") << int(QVariant::Double) << QByteArray("double"); QTest::newRow("18") << int(QMetaType::Float) << QByteArray("float"); QTest::newRow("19") << int(QVariant::Polygon) << QByteArray("QPolygon"); QTest::newRow("20") << int(QVariant::Region) << QByteArray("QRegion"); QTest::newRow("21") << int(QVariant::Bitmap) << QByteArray("QBitmap"); QTest::newRow("22") << int(QVariant::Cursor) << QByteArray("QCursor"); // The test below doesn't work as long as we don't link against widgets // QTest::newRow("23") << int(QVariant::SizePolicy) << QByteArray("QSizePolicy"); QTest::newRow("24") << int(QVariant::Date) << QByteArray("QDate"); QTest::newRow("25") << int(QVariant::Time) << QByteArray("QTime"); QTest::newRow("26") << int(QVariant::DateTime) << QByteArray("QDateTime"); QTest::newRow("27") << int(QVariant::ByteArray) << QByteArray("QByteArray"); QTest::newRow("28") << int(QVariant::BitArray) << QByteArray("QBitArray"); QTest::newRow("29") << int(QVariant::KeySequence) << QByteArray("QKeySequence"); QTest::newRow("30") << int(QVariant::Pen) << QByteArray("QPen"); QTest::newRow("31") << int(QVariant::LongLong) << QByteArray("qlonglong"); QTest::newRow("32") << int(QVariant::ULongLong) << QByteArray("qulonglong"); QTest::newRow("33") << int(QVariant::Char) << QByteArray("QChar"); QTest::newRow("34") << int(QVariant::Url) << QByteArray("QUrl"); QTest::newRow("35") << int(QVariant::TextLength) << QByteArray("QTextLength"); QTest::newRow("36") << int(QVariant::TextFormat) << QByteArray("QTextFormat"); QTest::newRow("37") << int(QVariant::Locale) << QByteArray("QLocale"); QTest::newRow("38") << int(QVariant::LineF) << QByteArray("QLineF"); QTest::newRow("39") << int(QVariant::RectF) << QByteArray("QRectF"); QTest::newRow("40") << int(QVariant::PointF) << QByteArray("QPointF"); QTest::newRow("44") << int(QVariant::Transform) << QByteArray("QTransform"); QTest::newRow("45") << int(QVariant::Hash) << QByteArray("QVariantHash"); QTest::newRow("46") << int(QVariant::Matrix4x4) << QByteArray("QMatrix4x4"); QTest::newRow("47") << int(QVariant::Vector2D) << QByteArray("QVector2D"); QTest::newRow("48") << int(QVariant::Vector3D) << QByteArray("QVector3D"); QTest::newRow("49") << int(QVariant::Vector4D) << QByteArray("QVector4D"); QTest::newRow("50") << int(QVariant::Quaternion) << QByteArray("QQuaternion"); QTest::newRow("51") << int(QVariant::RegularExpression) << QByteArray("QRegularExpression"); } void tst_QVariant::typeName() { QFETCH( int, type ); QFETCH( QByteArray, res ); QCOMPARE(QString::fromLatin1(QVariant::typeToName((QVariant::Type)type)), QString::fromLatin1(res.constData())); } // test nameToType as well void tst_QVariant::typeToName() { QVariant v; QCOMPARE( QVariant::typeToName( v.type() ), (const char*)0 ); // Invalid // assumes that QVariant::Type contains consecutive values int max = QVariant::LastGuiType; for (int t = 1; t <= max; ++t) { if (!QMetaType::isRegistered(t)) { QTest::ignoreMessage(QtWarningMsg, QRegularExpression( "^Trying to construct an instance of an invalid type")); } const char *n = QVariant::typeToName( (QVariant::Type)t ); if (n) QCOMPARE( int(QVariant::nameToType( n )), t ); } QCOMPARE(QVariant::typeToName(QVariant::Int), "int"); // not documented but we return 0 if the type is out of range // by testing this we catch cases where QVariant is extended // but type_map is not updated accordingly QTest::ignoreMessage(QtWarningMsg, QRegularExpression( "^Trying to construct an instance of an invalid type")); QCOMPARE(QVariant::typeToName(QVariant::Type(max + 1)), (const char *)nullptr); // invalid type names QVERIFY( QVariant::nameToType( 0 ) == QVariant::Invalid ); QVERIFY( QVariant::nameToType( "" ) == QVariant::Invalid ); QVERIFY( QVariant::nameToType( "foo" ) == QVariant::Invalid ); QCOMPARE(QVariant::nameToType("UserType"), QVariant::Invalid); // We don't support these old (Qt3) types anymore. QCOMPARE(QVariant::nameToType("QIconSet"), QVariant::Invalid); QCOMPARE(QVariant::nameToType("Q3CString"), QVariant::Invalid); QCOMPARE(QVariant::nameToType("Q_LLONG"), QVariant::Invalid); QCOMPARE(QVariant::nameToType("Q_ULLONG"), QVariant::Invalid); } QT_WARNING_POP #endif // QT_DEPRECATED_SINCE(6, 0) void tst_QVariant::streamInvalidVariant() { int writeX = 1; int writeY = 2; int readX; int readY; QVariant writeVariant; QVariant readVariant; QVERIFY( writeVariant.typeId() == QMetaType::UnknownType ); QByteArray data; QDataStream writeStream( &data, QIODevice::WriteOnly ); writeStream << writeX << writeVariant << writeY; QDataStream readStream( &data, QIODevice::ReadOnly ); readStream >> readX >> readVariant >> readY; QVERIFY( readX == writeX ); // Two invalid QVariant's aren't necessarily the same, so == will // return false if one is invalid, so check the type() instead QVERIFY( readVariant.typeId() == QMetaType::UnknownType ); QVERIFY( readY == writeY ); } static int instanceCount = 0; struct MyType { MyType(int n = 0, const char *t=0): number(n), text(t) { ++instanceCount; } MyType(const MyType &other) : number(other.number), text(other.text) { ++instanceCount; } MyType &operator=(const MyType &other) { number = other.number; text = other.text; return *this; } ~MyType() { --instanceCount; } int number; const char *text; }; bool operator==(const MyType &a, const MyType &b) { return a.number == b.number && a.text == b.text; } static_assert(QTypeTraits::has_operator_equal_v); Q_DECLARE_METATYPE(MyType) Q_DECLARE_METATYPE(MyType*) void tst_QVariant::userType() { { MyType data(1, "eins"); MyType data2(2, "zwei"); { QVariant userVar; userVar.setValue(data); QVERIFY(QMetaType::fromName("MyType").isValid()); QCOMPARE(QMetaType::fromName("MyType"), QMetaType::fromType()); QVERIFY(userVar.typeId() > QMetaType::User); QCOMPARE(userVar.userType(), qMetaTypeId()); QCOMPARE(userVar.typeName(), "MyType"); QVERIFY(!userVar.isNull()); QVERIFY(!userVar.canConvert()); QVariant userVar2(userVar); QCOMPARE(userVar, userVar2); userVar2.setValue(data2); QVERIFY(userVar != userVar2); const MyType *varData = static_cast(userVar.constData()); QVERIFY(varData); QCOMPARE(varData->number, data.number); QCOMPARE(varData->text, data.text); QVariant userVar3; userVar3.setValue(data2); userVar3 = userVar2; QCOMPARE(userVar2, userVar3); } // At this point all QVariants got destroyed but we have 2 MyType instances. QCOMPARE(instanceCount, 2); { QVariant userVar; userVar.setValue(&data); QVERIFY(userVar.typeId() > QMetaType::User); QCOMPARE(userVar.userType(), qMetaTypeId()); QCOMPARE(userVar.typeName(), "MyType*"); QVERIFY(!userVar.isNull()); QVERIFY(!userVar.canConvert()); QVariant userVar2(userVar); QCOMPARE(userVar, userVar2); userVar2.setValue(&data2); QVERIFY(userVar != userVar2); MyType * const*varData = reinterpret_cast(userVar.constData()); QVERIFY(varData); QCOMPARE(*varData, &data); QVariant userVar3; userVar3.setValue(&data2); /* This check is correct now. userVar2 contains a pointer to data2 and so * does userVar3. */ QCOMPARE(userVar2, userVar3); userVar3 = userVar2; QCOMPARE(userVar2, userVar3); } QCOMPARE(instanceCount, 2); QVariant myCarrier; myCarrier.setValue(data); QCOMPARE(instanceCount, 3); { QVariant second = myCarrier; QCOMPARE(instanceCount, 3); second.detach(); QCOMPARE(instanceCount, 4); } QCOMPARE(instanceCount, 3); MyType data3(0, "null"); data3 = qvariant_cast(myCarrier); QCOMPARE(data3.number, 1); QCOMPARE(data3.text, (const char *)"eins"); #ifndef Q_CC_SUN QCOMPARE(instanceCount, 4); #endif } { const MyType data(3, "drei"); QVariant myCarrier; myCarrier.setValue(data); QCOMPARE(myCarrier.typeName(), "MyType"); const MyType data2 = qvariant_cast(myCarrier); QCOMPARE(data2.number, 3); QCOMPARE(data2.text, (const char *)"drei"); } { short s = 42; QVariant myCarrier; myCarrier.setValue(s); QCOMPARE((int)qvariant_cast(myCarrier), 42); } { qlonglong ll = Q_INT64_C(42); QVariant myCarrier; myCarrier.setValue(ll); QCOMPARE(qvariant_cast(myCarrier), 42); } // At this point all QVariants got destroyed and MyType objects too. QCOMPARE(instanceCount, 0); } struct MyTypePOD { int a; int b; }; Q_DECLARE_METATYPE(MyTypePOD) void tst_QVariant::podUserType() { MyTypePOD pod; pod.a = 10; pod.b = 20; // one of these two must register the type // (QVariant::fromValue calls QMetaType::fromType) QVariant pod_as_variant = QVariant::fromValue(pod); QMetaType mt = QMetaType::fromType(); QCOMPARE(pod_as_variant.metaType(), mt); QCOMPARE(pod_as_variant.metaType().name(), mt.name()); QCOMPARE(QMetaType::fromName(mt.name()), mt); QCOMPARE_NE(pod_as_variant.typeId(), 0); MyTypePOD pod2 = qvariant_cast(pod_as_variant); QCOMPARE(pod.a, pod2.a); QCOMPARE(pod.b, pod2.b); pod_as_variant.setValue(pod); pod2 = qvariant_cast(pod_as_variant); QCOMPARE(pod.a, pod2.a); QCOMPARE(pod.b, pod2.b); } void tst_QVariant::basicUserType() { QVariant v; { int i = 7; v = QVariant(QMetaType::fromType(), &i); } QCOMPARE(v.typeId(), QMetaType::Int); QCOMPARE(v.toInt(), 7); { QString s("foo"); v = QVariant(QMetaType::fromType(), &s); } QCOMPARE(v.typeId(), QMetaType::QString); QCOMPARE(v.toString(), QString("foo")); { double d = 4.4; v = QVariant(QMetaType::fromType(), &d); } QCOMPARE(v.typeId(), QMetaType::Double); QCOMPARE(v.toDouble(), 4.4); { float f = 4.5f; v = QVariant(QMetaType::fromType(), &f); } QCOMPARE(v.userType(), int(QMetaType::Float)); QCOMPARE(v.toDouble(), 4.5); { QByteArray ba("bar"); v = QVariant(QMetaType::fromType(), &ba); } QCOMPARE(v.typeId(), QMetaType::QByteArray); QCOMPARE(v.toByteArray(), QByteArray("bar")); } void tst_QVariant::data() { QVariant v; QVariant i = 1; QVariant d = 1.12; QVariant f = 1.12f; QVariant ll = (qlonglong)2; QVariant ull = (qulonglong)3; QVariant s(QString("hallo")); QVariant r(QRect(1,2,3,4)); v = i; QVERIFY(v.data()); QCOMPARE(*static_cast(v.data()), i.toInt()); v = d; QVERIFY(v.data()); QCOMPARE(*static_cast(v.data()), d.toDouble()); v = f; QVERIFY(v.data()); QCOMPARE(*static_cast(v.data()), qvariant_cast(v)); v = ll; QVERIFY(v.data()); QCOMPARE(*static_cast(v.data()), ll.toLongLong()); v = ull; QVERIFY(v.data()); QCOMPARE(*static_cast(v.data()), ull.toULongLong()); v = s; QVERIFY(v.data()); QCOMPARE(*static_cast(v.data()), s.toString()); v = r; QVERIFY(v.data()); QCOMPARE(*static_cast(v.data()), r.toRect()); } void tst_QVariant::constData() { QVariant v; int i = 1; double d = 1.12; float f = 1.12f; qlonglong ll = 2; qulonglong ull = 3; QString s("hallo"); QRect r(1,2,3,4); v = QVariant(i); QVERIFY(v.constData()); QCOMPARE(*static_cast(v.constData()), i); v = QVariant(d); QVERIFY(v.constData()); QCOMPARE(*static_cast(v.constData()), d); v = QVariant(f); QVERIFY(v.constData()); QCOMPARE(*static_cast(v.constData()), f); v = QVariant(ll); QVERIFY(v.constData()); QCOMPARE(*static_cast(v.constData()), ll); v = QVariant(ull); QVERIFY(v.constData()); QCOMPARE(*static_cast(v.constData()), ull); v = QVariant(s); QVERIFY(v.constData()); QCOMPARE(*static_cast(v.constData()), s); v = QVariant(r); QVERIFY(v.constData()); QCOMPARE(*static_cast(v.constData()), r); } struct Foo { Foo(): i(0) {} int i; }; Q_DECLARE_METATYPE(Foo) void tst_QVariant::variant_to() { QVariant v1(4.2); QVariant v2(5); QVariant v3; QVariant v4; QStringList sl; sl << QLatin1String("blah"); v3.setValue(sl); Foo foo; foo.i = 42; v4.setValue(foo); QCOMPARE(qvariant_cast(v1), 4.2); QCOMPARE(qvariant_cast(v1), 4.2f); QCOMPARE(qvariant_cast(v2), 5); QCOMPARE(qvariant_cast(v3), sl); QCOMPARE(qvariant_cast(v3), QString::fromLatin1("blah")); QCOMPARE(qvariant_cast(v4).i, 42); QVariant v5; QCOMPARE(qvariant_cast(v5).i, 0); QCOMPARE(qvariant_cast(v1), 4); QVariant n = QVariant::fromValue(42); QCOMPARE(qvariant_cast(n), 42); QCOMPARE(qvariant_cast(n), 42u); QCOMPARE(qvariant_cast(n), 42.0); QCOMPARE(qvariant_cast(n), 42.f); QCOMPARE(qvariant_cast(n), short(42)); QCOMPARE(qvariant_cast(n), ushort(42)); n = QVariant::fromValue(43l); QCOMPARE(qvariant_cast(n), 43); QCOMPARE(qvariant_cast(n), 43u); QCOMPARE(qvariant_cast(n), 43.0); QCOMPARE(qvariant_cast(n), 43.f); QCOMPARE(qvariant_cast(n), 43l); n = QLatin1String("44"); QCOMPARE(qvariant_cast(n), 44); QCOMPARE(qvariant_cast(n), 44ul); QCOMPARE(qvariant_cast(n), 44.0f); QCOMPARE(QVariant::fromValue(0.25f).toDouble(), 0.25); } struct Blah { int i; }; QDataStream& operator>>(QDataStream& s, Blah& c) { return (s >> c.i); } QDataStream& operator<<(QDataStream& s, const Blah& c) { return (s << c.i); } void tst_QVariant::saveLoadCustomTypes() { QByteArray data; Blah i = { 42 }; auto tp = QMetaType::fromType(); QVariant v = QVariant(tp, &i); QCOMPARE(v.userType(), tp.id()); QVERIFY(v.typeId() > QMetaType::User); { QDataStream stream(&data, QIODevice::WriteOnly); stream << v; } v = QVariant(); { QDataStream stream(data); stream >> v; } QCOMPARE(int(v.userType()), QMetaType::fromName("Blah").id()); int value = *(int*)v.constData(); QCOMPARE(value, 42); } void tst_QVariant::url() { QString str("http://qt-project.org"); QUrl url(str); QVariant v(url); //built with a QUrl QVariant v2 = v; QVariant v3(str); //built with a QString QCOMPARE(v2.toUrl(), url); QVERIFY(v3.canConvert()); QCOMPARE(v2.toUrl(), v3.toUrl()); QVERIFY(v2.canConvert()); QCOMPARE(v2.toString(), str); QCOMPARE(v3.toString(), str); } void tst_QVariant::variantMap() { QMap map; map["test"] = 42; QVariant v = map; QVariantMap map2 = qvariant_cast(v); QCOMPARE(map2.value("test").toInt(), 42); QCOMPARE(map2, map); map2 = v.toMap(); QCOMPARE(map2.value("test").toInt(), 42); QCOMPARE(map2, map); QVariant v2 = QVariant(QMetaType::fromType(), &map); QCOMPARE(qvariant_cast(v2).value("test").toInt(), 42); QVariant v3 = QVariant(QMetaType::fromType>(), &map); QCOMPARE(qvariant_cast(v3).value("test").toInt(), 42); QHash hash; hash["test"] = 42; QCOMPARE(hash, v.toHash()); } void tst_QVariant::variantHash() { QHash hash; hash["test"] = 42; QVariant v = hash; QVariantHash hash2 = qvariant_cast(v); QCOMPARE(hash2.value("test").toInt(), 42); QCOMPARE(hash2, hash); hash2 = v.toHash(); QCOMPARE(hash2.value("test").toInt(), 42); QCOMPARE(hash2, hash); QVariant v2 = QVariant(QMetaType::fromType(), &hash); QCOMPARE(qvariant_cast(v2).value("test").toInt(), 42); QVariant v3 = QVariant(QMetaType::fromType>(), &hash); QCOMPARE(qvariant_cast(v3).value("test").toInt(), 42); QMap map; map["test"] = 42; QCOMPARE(map, v.toMap()); } class CustomQObject : public QObject { Q_OBJECT public: CustomQObject(QObject *parent = nullptr) : QObject(parent) {} }; Q_DECLARE_METATYPE(CustomQObject*) class CustomNonQObject { }; Q_DECLARE_METATYPE(CustomNonQObject) Q_DECLARE_METATYPE(CustomNonQObject*) void tst_QVariant::cleanupTestCase() { delete customNonQObjectPointer; qDeleteAll(objectPointerTestData); } void tst_QVariant::qvariant_cast_QObject_data() { QTest::addColumn("data"); QTest::addColumn("success"); QTest::addColumn("isNull"); QObject *obj = new QObject; obj->setObjectName(QString::fromLatin1("Hello")); QTest::newRow("from QObject") << QVariant(QMetaType::fromType(), &obj) << true << false; QTest::newRow("from QObject2") << QVariant::fromValue(obj) << true << false; QTest::newRow("from String") << QVariant(QLatin1String("1, 2, 3")) << false << false; QTest::newRow("from int") << QVariant((int) 123) << false << false; CustomQObject *customObject = new CustomQObject(this); customObject->setObjectName(QString::fromLatin1("Hello")); QTest::newRow("from Derived QObject") << QVariant::fromValue(customObject) << true << false; QTest::newRow("from custom Object") << QVariant::fromValue(CustomNonQObject()) << false << false; // Deleted in cleanupTestCase. customNonQObjectPointer = new CustomNonQObject; QTest::newRow("from custom ObjectStar") << QVariant::fromValue(customNonQObjectPointer) << false << false; // Deleted in cleanupTestCase. objectPointerTestData.push_back(obj); objectPointerTestData.push_back(customObject); QTest::newRow("null QObject") << QVariant::fromValue(0) << true << true; QTest::newRow("null derived QObject") << QVariant::fromValue(0) << true << true; QTest::newRow("null custom object") << QVariant::fromValue(0) << false << true; QTest::newRow("zero int") << QVariant::fromValue(0) << false << false; } void tst_QVariant::qvariant_cast_QObject() { QFETCH(QVariant, data); QFETCH(bool, success); QFETCH(bool, isNull); QObject *o = qvariant_cast(data); QCOMPARE(o != nullptr, success && !isNull); if (success) { if (!isNull) QCOMPARE(o->objectName(), QString::fromLatin1("Hello")); QVERIFY(data.canConvert()); QVERIFY(data.canConvert(QMetaType::fromType())); QVERIFY(data.canConvert(QMetaType(::qMetaTypeId()))); QCOMPARE(data.value() == nullptr, isNull); QVERIFY(data.convert(QMetaType::fromType())); QCOMPARE(data.userType(), int(QMetaType::QObjectStar)); } else { QVERIFY(!data.canConvert()); QVERIFY(!data.canConvert(QMetaType::fromType())); QVERIFY(!data.canConvert(QMetaType(::qMetaTypeId()))); QVERIFY(!data.value()); QVERIFY(data.userType() != QMetaType::QObjectStar); QVERIFY(!data.convert(QMetaType::fromType())); } } class CustomQObjectDerived : public CustomQObject { Q_OBJECT public: CustomQObjectDerived(QObject *parent = nullptr) : CustomQObject(parent) {} }; Q_DECLARE_METATYPE(CustomQObjectDerived*) class CustomQObjectDerivedNoMetaType : public CustomQObject { Q_OBJECT public: CustomQObjectDerivedNoMetaType(QObject *parent = nullptr) : CustomQObject(parent) {} }; void tst_QVariant::qvariant_cast_QObject_derived() { { CustomQObjectDerivedNoMetaType *object = new CustomQObjectDerivedNoMetaType(this); QVariant data = QVariant::fromValue(object); QCOMPARE(data.userType(), qMetaTypeId()); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); } { CustomQObjectDerived *object = new CustomQObjectDerived(this); QVariant data = QVariant::fromValue(object); QCOMPARE(data.userType(), qMetaTypeId()); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); } { QObject *object = new CustomQObjectDerivedNoMetaType(this); QVariant data = QVariant::fromValue(object); QVERIFY(data.canConvert()); QVERIFY(data.convert(QMetaType(qMetaTypeId()))); QCOMPARE(data.value(), object); QCOMPARE(data.isNull(), false); } } struct QObjectWrapper { explicit QObjectWrapper(QObject *o = nullptr) : obj(o) {} QObject* getObject() const { return obj; } private: QObject *obj; }; Q_DECLARE_METATYPE(QObjectWrapper) struct Converter { Converter() {} QObject* operator()(const QObjectWrapper &f) const { return f.getObject(); } }; namespace MyNS { template class SmartPointer { T* pointer; public: typedef T element_type; explicit SmartPointer(T *t = nullptr) : pointer(t) { } T* operator->() const { return pointer; } }; template struct SequentialContainer { typedef T value_type; typedef const T* const_iterator; T t; const_iterator begin() const { return &t; } const_iterator end() const { return &t + 1; } }; template struct AssociativeContainer : public std::map { }; } Q_DECLARE_SMART_POINTER_METATYPE(MyNS::SmartPointer) Q_DECLARE_SEQUENTIAL_CONTAINER_METATYPE(MyNS::SequentialContainer) Q_DECLARE_ASSOCIATIVE_CONTAINER_METATYPE(MyNS::AssociativeContainer) // Test that explicit declaration does not degrade features. Q_DECLARE_METATYPE(MyNS::SmartPointer) Q_DECLARE_METATYPE(MyNS::SmartPointer) Q_DECLARE_METATYPE(QSharedPointer) void tst_QVariant::qvariant_cast_QObject_wrapper() { QMetaType::registerConverter(&QObjectWrapper::getObject); CustomQObjectDerived *object = new CustomQObjectDerived(this); QObjectWrapper wrapper(object); QVariant v = QVariant::fromValue(wrapper); QCOMPARE(v.value(), object); v.convert(QMetaType(qMetaTypeId())); QCOMPARE(v.value(), object); MyNS::SequentialContainer sc; sc.t = 47; MyNS::AssociativeContainer ac; QVariant::fromValue(sc); QVariant::fromValue(ac); { QFile *f = new QFile(this); MyNS::SmartPointer sp(f); QVariant spVar = QVariant::fromValue(sp); QVERIFY(spVar.canConvert()); QCOMPARE(f, spVar.value()); } { QFile *f = new QFile(this); QPointer sp(f); QVariant spVar = QVariant::fromValue(sp); QVERIFY(spVar.canConvert()); QCOMPARE(f, spVar.value()); } { QFile *f = new QFile(this); QSharedPointer sp(f); QWeakPointer wp = sp; QVariant wpVar = QVariant::fromValue(wp); QVERIFY(wpVar.canConvert()); QCOMPARE(f, wpVar.value()); } { QFile *f = new QFile(this); QSharedPointer sp(f); QWeakPointer wp = sp.toWeakRef(); QVariant wpVar = QVariant::fromValue(wp); QVERIFY(wpVar.canConvert()); QCOMPARE(f, wpVar.value()); } { QFile *f = new QFile(this); QSharedPointer sp(f); QVariant spVar = QVariant::fromValue(sp); QVERIFY(spVar.canConvert()); QCOMPARE(f, spVar.value()); } { QIODevice *f = new QFile(this); MyNS::SmartPointer sp(f); QVariant spVar = QVariant::fromValue(sp); QVERIFY(spVar.canConvert()); QCOMPARE(f, spVar.value()); } { QIODevice *f = new QFile(this); QSharedPointer sp(f); QVariant spVar = QVariant::fromValue(sp); QVERIFY(spVar.canConvert()); QCOMPARE(f, spVar.value()); } // Compile tests: qRegisterMetaType >(); // Not declared as a metatype: qRegisterMetaType >("MyNS::SmartPointer"); } void tst_QVariant::qvariant_cast_QSharedPointerQObject() { // ensure no problems between this form and the auto-registering in QVariant::fromValue qRegisterMetaType >("QSharedPointer"); QObject *rawptr = new QObject; QSharedPointer strong(rawptr); QWeakPointer weak(strong); QPointer qptr(rawptr); QVariant v = QVariant::fromValue(strong); QCOMPARE(v.value >(), strong); // clear our QSP; the copy inside the variant should keep the object alive strong.clear(); // check that the object didn't get deleted QVERIFY(!weak.isNull()); QVERIFY(!qptr.isNull()); strong = qvariant_cast >(v); QCOMPARE(strong.data(), rawptr); QVERIFY(strong == weak); // now really delete the object and verify strong.clear(); v.clear(); QVERIFY(weak.isNull()); QVERIFY(qptr.isNull()); // compile test: // QVariant::fromValue has already called this function qRegisterMetaType >(); } void tst_QVariant::qvariant_cast_const() { int i = 42; QVariant v = QVariant::fromValue(&i); QVariant vConst = QVariant::fromValue(const_cast(&i)); QCOMPARE(v.value(), &i); QCOMPARE(v.value(), &i); QCOMPARE(vConst.value(), nullptr); QCOMPARE(vConst.value(), &i); } void tst_QVariant::convertToQUint8() const { /* qint8. */ { const qint8 anInt = 32; /* QVariant(int) gets invoked here so the QVariant has nothing with qint8 to do. * It's of type QVariant::Int. */ const QVariant v0 = anInt; QVERIFY(v0.canConvert()); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(v0.toInt()), 32); QCOMPARE(v0.toString(), QString("32")); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(qvariant_cast(v0)), 32); } /* quint8. */ { const quint8 anInt = 32; const QVariant v0 = anInt; QVERIFY(v0.canConvert()); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(v0.toUInt()), 32); QCOMPARE(v0.toString(), QString("32")); } /* qint16. */ { const qint16 anInt = 32; const QVariant v0 = anInt; QVERIFY(v0.canConvert()); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(v0.toInt()), 32); QCOMPARE(v0.toString(), QString("32")); } /* quint16. */ { const quint16 anInt = 32; const QVariant v0 = anInt; QVERIFY(v0.canConvert()); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(v0.toUInt()), 32); QCOMPARE(v0.toString(), QString("32")); } } void tst_QVariant::compareNumerics_data() const { QTest::addColumn("v1"); QTest::addColumn("v2"); QTest::addColumn("result"); QTest::addRow("invalid-invalid") << QVariant() << QVariant() << QPartialOrdering::Unordered; static const auto asString = [](const QVariant &v) { if (v.isNull()) return QStringLiteral("null"); if (v.metaType().flags() & QMetaType::IsEnumeration) return v.metaType().flags() & QMetaType::IsUnsignedEnumeration ? QString::number(v.toULongLong()) : QString::number(v.toLongLong()); switch (v.typeId()) { case QMetaType::Char: case QMetaType::Char16: case QMetaType::Char32: case QMetaType::UChar: return QString::number(v.toUInt()); case QMetaType::SChar: return QString::number(v.toInt()); } return v.toString(); }; auto addCompareToInvalid = [](auto value) { QVariant v = QVariant::fromValue(value); QTest::addRow("invalid-%s(%s)", v.typeName(), qPrintable(asString(v))) << QVariant() << v << QPartialOrdering::Unordered; QTest::addRow("%s(%s)-invalid", v.typeName(), qPrintable(asString(v))) << v << QVariant() << QPartialOrdering::Unordered; }; addCompareToInvalid(false); addCompareToInvalid(true); addCompareToInvalid(char(0)); addCompareToInvalid(qint8(0)); addCompareToInvalid(quint8(0)); addCompareToInvalid(short(0)); addCompareToInvalid(ushort(0)); addCompareToInvalid(int(0)); addCompareToInvalid(uint(0)); addCompareToInvalid(long(0)); addCompareToInvalid(ulong(0)); addCompareToInvalid(qint64(0)); addCompareToInvalid(quint64(0)); addCompareToInvalid(0.f); addCompareToInvalid(0.0); addCompareToInvalid(QCborSimpleType{}); QT_WARNING_PUSH QT_WARNING_DISABLE_CLANG("-Wsign-compare") QT_WARNING_DISABLE_GCC("-Wsign-compare") QT_WARNING_DISABLE_MSVC(4018) // '<': signed/unsigned mismatch static const auto addComparePairWithResult = [](auto value1, auto value2, QPartialOrdering order) { QVariant v1 = QVariant::fromValue(value1); QVariant v2 = QVariant::fromValue(value2); QTest::addRow("%s(%s)-%s(%s)", v1.typeName(), qPrintable(asString(v1)), v2.typeName(), qPrintable(asString(v2))) << v1 << v2 << order; }; static const auto addComparePair = [](auto value1, auto value2) { QPartialOrdering order = QPartialOrdering::Unordered; if (value1 == value2) order = QPartialOrdering::Equivalent; else if (value1 < value2) order = QPartialOrdering::Less; else if (value1 > value2) order = QPartialOrdering::Greater; addComparePairWithResult(value1, value2, order); }; QT_WARNING_POP // homogeneous first static const auto addList = [](auto list) { for (auto v1 : list) for (auto v2 : list) addComparePair(v1, v2); }; auto addSingleType = [](auto zero) { using T = decltype(zero); T one = T(zero + 1); T min = std::numeric_limits::min(); T max = std::numeric_limits::max(); T mid = max / 2 + 1; if (min != zero) addList(std::array{zero, one, min, mid, max}); else addList(std::array{zero, one, mid, max}); }; addList(std::array{ false, true }); addList(std::array{ QCborSimpleType{}, QCborSimpleType::False, QCborSimpleType(0xff) }); addSingleType(char(0)); addSingleType(char16_t(0)); addSingleType(char32_t(0)); addSingleType(qint8(0)); addSingleType(quint8(0)); addSingleType(qint16(0)); addSingleType(quint16(0)); addSingleType(qint32(0)); addSingleType(quint32(0)); addSingleType(qint64(0)); addSingleType(quint64(0)); addSingleType(0.f); addSingleType(0.0); addList(std::array{ EnumTest_Enum0{}, EnumTest_Enum0_value, EnumTest_Enum0_negValue }); addList(std::array{ EnumTest_Enum1{}, EnumTest_Enum1_value, EnumTest_Enum1_bigValue }); addList(std::array{ EnumTest_Enum7{}, EnumTest_Enum7::EnumTest_Enum7_value, EnumTest_Enum7::ensureSignedEnum7 }); addList(std::array{ Qt::AlignRight|Qt::AlignHCenter, Qt::AlignCenter|Qt::AlignVCenter }); // heterogeneous addComparePair(char(0), qint8(-127)); addComparePair(char(127), qint8(127)); addComparePair(char(127), quint8(127)); addComparePair(qint8(-1), quint8(255)); addComparePair(char16_t(256), qint8(-1)); addComparePair(char16_t(256), short(-1)); addComparePair(char16_t(256), int(-1)); addComparePair(char32_t(256), int(-1)); addComparePair(0U, -1); addComparePair(~0U, -1); addComparePair(Q_UINT64_C(0), -1); addComparePair(~Q_UINT64_C(0), -1); addComparePair(Q_UINT64_C(0), Q_INT64_C(-1)); addComparePair(~Q_UINT64_C(0), Q_INT64_C(-1)); addComparePair(INT_MAX, uint(INT_MAX)); addComparePair(INT_MAX, qint64(INT_MAX) + 1); addComparePair(INT_MAX, UINT_MAX); addComparePair(INT_MAX, qint64(UINT_MAX)); addComparePair(INT_MAX, qint64(UINT_MAX) + 1); addComparePair(INT_MAX, quint64(UINT_MAX)); addComparePair(INT_MAX, quint64(UINT_MAX) + 1); addComparePair(INT_MAX, LONG_MIN); addComparePair(INT_MAX, LONG_MAX); addComparePair(INT_MAX, LLONG_MIN); addComparePair(INT_MAX, LLONG_MAX); addComparePair(INT_MIN, uint(INT_MIN)); addComparePair(INT_MIN, uint(INT_MIN) + 1); addComparePair(INT_MIN + 1, uint(INT_MIN)); addComparePair(INT_MIN + 1, uint(INT_MIN) + 1); addComparePair(INT_MIN, qint64(INT_MIN) - 1); addComparePair(INT_MIN + 1, qint64(INT_MIN) + 1); addComparePair(INT_MIN + 1, qint64(INT_MIN) - 1); addComparePair(INT_MIN, UINT_MAX); addComparePair(INT_MIN, qint64(UINT_MAX)); addComparePair(INT_MIN, qint64(UINT_MAX) + 1); addComparePair(INT_MIN, quint64(UINT_MAX)); addComparePair(INT_MIN, quint64(UINT_MAX) + 1); addComparePair(UINT_MAX, qint64(UINT_MAX) + 1); addComparePair(UINT_MAX, quint64(UINT_MAX) + 1); addComparePair(UINT_MAX, qint64(INT_MIN) - 1); addComparePair(UINT_MAX, quint64(INT_MIN) + 1); addComparePair(LLONG_MAX, quint64(LLONG_MAX)); addComparePair(LLONG_MAX, quint64(LLONG_MAX) + 1); addComparePair(LLONG_MIN, quint64(LLONG_MAX)); addComparePair(LLONG_MIN, quint64(LLONG_MAX) + 1); addComparePair(LLONG_MIN, quint64(LLONG_MIN) + 1); addComparePair(LLONG_MIN + 1, quint64(LLONG_MIN) + 1); addComparePair(LLONG_MIN, LLONG_MAX - 1); addComparePair(LLONG_MIN, LLONG_MAX); // floating point addComparePair(0.f, 0); addComparePair(0.f, 0U); addComparePair(0.f, Q_INT64_C(0)); addComparePair(0.f, Q_UINT64_C(0)); addComparePair(0.f, 0.); addComparePair(0.f, 1.); addComparePair(0.f, 1.); addComparePair(float(1 << 24), 1 << 24); addComparePair(float(1 << 24) - 1, (1 << 24) - 1); addComparePair(-float(1 << 24), 1 << 24); addComparePair(-float(1 << 24) + 1, -(1 << 24) + 1); addComparePair(HUGE_VALF, qInf()); addComparePair(HUGE_VALF, -qInf()); addComparePair(qQNaN(), std::numeric_limits::quiet_NaN()); if (sizeof(qreal) == sizeof(double)) { addComparePair(std::numeric_limits::min(), std::numeric_limits::min()); addComparePair(std::numeric_limits::min(), std::numeric_limits::min()); addComparePair(std::numeric_limits::max(), std::numeric_limits::min()); addComparePair(std::numeric_limits::max(), std::numeric_limits::max()); addComparePair(double(Q_INT64_C(1) << 53), Q_INT64_C(1) << 53); addComparePair(double(Q_INT64_C(1) << 53) - 1, (Q_INT64_C(1) << 53) - 1); addComparePair(-double(Q_INT64_C(1) << 53), Q_INT64_C(1) << 53); addComparePair(-double(Q_INT64_C(1) << 53) + 1, (Q_INT64_C(1) << 53) + 1); } // enums vs integers addComparePair(EnumTest_Enum0_value, 0); addComparePair(EnumTest_Enum0_value, 0U); addComparePair(EnumTest_Enum0_value, 0LL); addComparePair(EnumTest_Enum0_value, 0ULL); addComparePair(EnumTest_Enum0_value, int(EnumTest_Enum0_value)); addComparePair(EnumTest_Enum0_value, qint64(EnumTest_Enum0_value)); addComparePair(EnumTest_Enum0_value, quint64(EnumTest_Enum0_value)); addComparePair(EnumTest_Enum0_negValue, int(EnumTest_Enum0_value)); addComparePair(EnumTest_Enum0_negValue, qint64(EnumTest_Enum0_value)); addComparePair(EnumTest_Enum0_negValue, quint64(EnumTest_Enum0_value)); addComparePair(EnumTest_Enum0_negValue, int(EnumTest_Enum0_negValue)); addComparePair(EnumTest_Enum0_negValue, qint64(EnumTest_Enum0_negValue)); addComparePair(EnumTest_Enum0_negValue, quint64(EnumTest_Enum0_negValue)); addComparePair(EnumTest_Enum1_value, 0); addComparePair(EnumTest_Enum1_value, 0U); addComparePair(EnumTest_Enum1_value, 0LL); addComparePair(EnumTest_Enum1_value, 0ULL); addComparePair(EnumTest_Enum1_value, int(EnumTest_Enum1_value)); addComparePair(EnumTest_Enum1_value, qint64(EnumTest_Enum1_value)); addComparePair(EnumTest_Enum1_value, quint64(EnumTest_Enum1_value)); addComparePair(EnumTest_Enum1_bigValue, int(EnumTest_Enum1_value)); addComparePair(EnumTest_Enum1_bigValue, qint64(EnumTest_Enum1_value)); addComparePair(EnumTest_Enum1_bigValue, quint64(EnumTest_Enum1_value)); addComparePair(EnumTest_Enum1_bigValue, int(EnumTest_Enum1_bigValue)); addComparePair(EnumTest_Enum1_bigValue, qint64(EnumTest_Enum1_bigValue)); addComparePair(EnumTest_Enum1_bigValue, quint64(EnumTest_Enum1_bigValue)); addComparePair(EnumTest_Enum3_value, 0); addComparePair(EnumTest_Enum3_value, 0U); addComparePair(EnumTest_Enum3_value, 0LL); addComparePair(EnumTest_Enum3_value, 0ULL); addComparePair(EnumTest_Enum3_value, int(EnumTest_Enum3_value)); addComparePair(EnumTest_Enum3_value, qint64(EnumTest_Enum3_value)); addComparePair(EnumTest_Enum3_value, quint64(EnumTest_Enum3_value)); addComparePair(EnumTest_Enum3_bigValue, int(EnumTest_Enum3_value)); addComparePair(EnumTest_Enum3_bigValue, qint64(EnumTest_Enum3_value)); addComparePair(EnumTest_Enum3_bigValue, quint64(EnumTest_Enum3_value)); addComparePair(EnumTest_Enum3_bigValue, int(EnumTest_Enum3_bigValue)); addComparePair(EnumTest_Enum3_bigValue, qint64(EnumTest_Enum3_bigValue)); addComparePair(EnumTest_Enum3_bigValue, quint64(EnumTest_Enum3_bigValue)); // enums of different types always compare as unordered addComparePairWithResult(EnumTest_Enum0_value, EnumTest_Enum1_value, QPartialOrdering::Unordered); } void tst_QVariant::compareNumerics() const { QFETCH(QVariant, v1); QFETCH(QVariant, v2); QFETCH(QPartialOrdering, result); QCOMPARE(QVariant::compare(v1, v2), result); QEXPECT_FAIL("invalid-invalid", "needs fixing", Continue); if (result == QPartialOrdering::Equivalent) QCOMPARE_EQ(v1, v2); else QCOMPARE_NE(v1, v2); } void tst_QVariant::comparePointers() const { class MyClass { }; MyClass myClass; QVariant v = QVariant::fromValue(&myClass); QVariant v2 = QVariant::fromValue(&myClass); QCOMPARE(v, v2); } struct Data {}; Q_DECLARE_METATYPE(Data*) void tst_QVariant::voidStar() const { char c; void *p1 = &c; void *p2 = p1; QVariant v1, v2; v1 = QVariant::fromValue(p1); v2 = v1; QCOMPARE(v1, v2); v2 = QVariant::fromValue(p2); QCOMPARE(v1, v2); p2 = nullptr; v2 = QVariant::fromValue(p2); QVERIFY(v1 != v2); } void tst_QVariant::dataStar() const { qRegisterMetaType(); Data *p1 = new Data; QVariant v1 = QVariant::fromValue(p1); QCOMPARE(v1.userType(), qMetaTypeId()); QCOMPARE(qvariant_cast(v1), p1); QVariant v2 = v1; QCOMPARE(v1, v2); v2 = QVariant::fromValue(p1); QCOMPARE(v1, v2); delete p1; } void tst_QVariant::canConvertQStringList() const { QFETCH(bool, canConvert); QFETCH(QStringList, input); QFETCH(QString, result); QVariant v(input); QCOMPARE(v.canConvert(), canConvert); QCOMPARE(v.toString(), result); } void tst_QVariant::canConvertQStringList_data() const { QTest::addColumn("canConvert"); QTest::addColumn("input"); QTest::addColumn("result"); QTest::newRow("An empty list") << true << QStringList() << QString(); QTest::newRow("A single item") << true << QStringList(QLatin1String("foo")) << QString::fromLatin1("foo"); QTest::newRow("A single, but empty item") << true << QStringList(QString()) << QString(); QStringList l; l << "a" << "b"; QTest::newRow("Two items") << true << l << QString(); l << "c"; QTest::newRow("Three items") << true << l << QString(); } template void convertMetaType() { QVERIFY(QVariant::fromValue(10).isValid()); QVERIFY(QVariant::fromValue(10).template canConvert()); QCOMPARE(QVariant::fromValue(10).toInt(), 10); QCOMPARE(QVariant::fromValue(10), QVariant::fromValue(10)); } #define CONVERT_META_TYPE(Type) \ convertMetaType(); \ if (QTest::currentTestFailed()) \ QFAIL("convertMetaType<" #Type "> failed"); void tst_QVariant::canConvertMetaTypeToInt() const { CONVERT_META_TYPE(long); CONVERT_META_TYPE(short); CONVERT_META_TYPE(short); CONVERT_META_TYPE(unsigned short); CONVERT_META_TYPE(ushort); CONVERT_META_TYPE(ulong); CONVERT_META_TYPE(unsigned long); CONVERT_META_TYPE(uchar); CONVERT_META_TYPE(unsigned char); CONVERT_META_TYPE(char); CONVERT_META_TYPE(uint); CONVERT_META_TYPE(unsigned int); } #undef CONVERT_META_TYPE /*! These calls should not produce any warnings. */ void tst_QVariant::variantToDateTimeWithoutWarnings() const { { const QVariant variant(QLatin1String("An invalid QDateTime string")); const QDateTime dateTime(variant.toDateTime()); QVERIFY(!dateTime.isValid()); } { QVariant v1(QLatin1String("xyz")); v1.convert(QMetaType::fromType()); QVariant v2(QLatin1String("xyz")); QDateTime dt1(v2.toDateTime()); const QVariant v3(QLatin1String("xyz")); const QDateTime dt2(v3.toDateTime()); } } void tst_QVariant::invalidDateTime() const { QVariant variant(QString::fromLatin1("Invalid date time string")); QVERIFY(!variant.toDateTime().isValid()); QVERIFY(!variant.convert(QMetaType::fromType())); } struct MyClass { MyClass() : myValue(0) {} int myValue; }; Q_DECLARE_METATYPE( MyClass ) void tst_QVariant::loadUnknownUserType() { qRegisterMetaType("MyClass"); QTest::ignoreMessage(QtWarningMsg, "QVariant::load: unable to load type " + QByteArray::number(qMetaTypeId()) +"."); char data[] = {0, QMetaType::User >> 16, char(QMetaType::User >> 8) , char(QMetaType::User), 0, 0, 0, 0, 8, 'M', 'y', 'C', 'l', 'a', 's', 's', 0}; QByteArray ba(data, sizeof(data)); QDataStream ds(&ba, QIODevice::ReadOnly); QVariant var; var.load(ds); QCOMPARE(ds.status(), QDataStream::ReadCorruptData); } void tst_QVariant::loadBrokenUserType() { QTest::ignoreMessage(QtWarningMsg, QRegularExpression("^Trying to construct an instance of an invalid type")); char data[] = {0, 0, 0, 127, 0 }; QByteArray ba(data, sizeof(data)); QDataStream ds(&ba, QIODevice::ReadOnly); QVariant var; var.load(ds); QCOMPARE(ds.status(), QDataStream::Ok); } void tst_QVariant::invalidDate() const { QString foo("Hello"); QVariant variant(foo); QVERIFY(!variant.convert(QMetaType::fromType())); variant = foo; QVERIFY(!variant.convert(QMetaType::fromType())); variant = foo; QVERIFY(!variant.convert(QMetaType::fromType())); variant = foo; QVERIFY(!variant.convert(QMetaType::fromType())); variant = foo; QVERIFY(!variant.convert(QMetaType::fromType())); variant = foo; QVERIFY(!variant.convert(QMetaType::fromType())); } struct WontCompare { int x; }; Q_DECLARE_METATYPE(WontCompare); struct WillCompare { int x; }; bool operator==(const WillCompare &a, const WillCompare &b) { return a.x == b.x; } Q_DECLARE_METATYPE(WillCompare); void tst_QVariant::compareCustomTypes() const { { WontCompare f1{0}; const QVariant variant1(QVariant::fromValue(f1)); WontCompare f2{1}; const QVariant variant2(QVariant::fromValue(f2)); /* No comparison operator exists. */ QVERIFY(variant1 != variant2); QVERIFY(variant1 != variant1); QVERIFY(variant2 != variant2); } { WillCompare f1{0}; const QVariant variant1(QVariant::fromValue(f1)); WillCompare f2 {1}; const QVariant variant2(QVariant::fromValue(f2)); QVERIFY(variant1 != variant2); QCOMPARE(variant1, variant1); QCOMPARE(variant2, variant2); } } void tst_QVariant::timeToDateTime() const { const QVariant val(QTime::currentTime()); QVERIFY(!val.canConvert()); QVERIFY(!val.toDateTime().isValid()); } struct CustomComparable { CustomComparable(int value = 0) : myValue(value) {} int myValue; bool operator==(const CustomComparable &other) const { return other.myValue == myValue; } }; Q_DECLARE_METATYPE(CustomComparable) void tst_QVariant::copyingUserTypes() const { QVariant var; QVariant varCopy; const CustomComparable userType = CustomComparable(42); var.setValue(userType); varCopy = var; const CustomComparable copiedType = qvariant_cast(varCopy); QCOMPARE(copiedType, userType); QCOMPARE(copiedType.myValue, 42); } struct NonQObjectBase {}; struct NonQObjectDerived : NonQObjectBase {}; void tst_QVariant::valueClassHierarchyConversion() const { { // QVariant allows downcasting QScopedPointer derived {new CustomQObjectDerived}; QVariant var = QVariant::fromValue(derived.get()); CustomQObject *object = var.value(); QVERIFY(object); } { // QVariant supports upcasting to the correct dynamic type for QObjects QScopedPointer derived {new CustomQObjectDerived}; QVariant var = QVariant::fromValue(derived.get()); CustomQObjectDerived *object = var.value(); QVERIFY(object); } { // QVariant forbids unsafe upcasting QScopedPointer base {new CustomQObject}; QVariant var = QVariant::fromValue(base.get()); CustomQObjectDerived *object = var.value(); QVERIFY(!object); } { // QVariant does not support upcastingfor non-QObjects QScopedPointer derived {new NonQObjectDerived}; QVariant var = QVariant::fromValue(derived.get()); NonQObjectDerived *object = var.value(); QVERIFY(!object); } } void tst_QVariant::convertBoolToByteArray() const { QFETCH(QByteArray, input); QFETCH(bool, canConvert); QFETCH(bool, value); const QVariant variant(input); QCOMPARE(variant.canConvert(), canConvert); if(canConvert) { /* Just call this function so we run the code path. */ QCOMPARE(variant.toBool(), value); } } void tst_QVariant::convertBoolToByteArray_data() const { QTest::addColumn("input"); QTest::addColumn("canConvert"); QTest::addColumn("value"); QTest::newRow("false") << QByteArray("false") << true << false; QTest::newRow("FALSE") << QByteArray("FALSE") << true << false; QTest::newRow("falSE") << QByteArray("FALSE") << true << false; QTest::newRow("") << QByteArray("") << true << false; QTest::newRow("null QByteArray") << QByteArray() << true << false; QTest::newRow("any-content") << QByteArray("any-content") << true << true; QTest::newRow("true") << QByteArray("true") << true << true; QTest::newRow("TRUE") << QByteArray("TRUE") << true << true; QTest::newRow("trUE") << QByteArray("trUE") << true << true; } void tst_QVariant::convertByteArrayToBool() const { QFETCH(bool, input); QFETCH(QByteArray, output); const QVariant variant(input); QCOMPARE(variant.typeId(), QMetaType::Bool); QCOMPARE(variant.toBool(), input); QVERIFY(variant.canConvert()); QCOMPARE(variant.toByteArray(), output); } void tst_QVariant::convertByteArrayToBool_data() const { QTest::addColumn("input"); QTest::addColumn("output"); QTest::newRow("false") << false << QByteArray("false"); QTest::newRow("true") << true << QByteArray("true"); } void tst_QVariant::convertIterables() const { { QStringList list; list.append("Hello"); QCOMPARE(QVariant::fromValue(list).value().size(), list.size()); } { QByteArrayList list; list.append("Hello"); QCOMPARE(QVariant::fromValue(list).value().size(), list.size()); } { QVariantList list; list.append("World"); QCOMPARE(QVariant::fromValue(list).value().size(), list.size()); } { QMap map; map.insert("3", 4); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); map.insert("4", 5); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); } { QVariantMap map; map.insert("3", 4); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); map.insert("4", 5); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); QCOMPARE(QVariant::fromValue(map).value().size(), map.size()); } { QHash hash; hash.insert("3", 4); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); hash.insert("4", 5); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); } { QVariantHash hash; hash.insert("3", 4); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); hash.insert("4", 5); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); QCOMPARE(QVariant::fromValue(hash).value().size(), hash.size()); } } struct Derived : QObject { Q_OBJECT }; void tst_QVariant::convertConstNonConst() const { Derived *derived = new Derived; QObject *obj = derived; QObject const *unrelatedConstObj = new QObject; auto cleanUp = qScopeGuard([&] { delete unrelatedConstObj; delete derived; }); QObject const *constObj = obj; Derived const *constDerived = derived; QCOMPARE(QVariant::fromValue(constObj).value(), obj); QCOMPARE(QVariant::fromValue(obj).value(), constObj); QCOMPARE(QVariant::fromValue(constDerived).value(), derived); QCOMPARE(QVariant::fromValue(derived).value(), derived); QObject const *derivedAsConstObject = derived; // up cast and remove const is possible, albeit dangerous QCOMPARE(QVariant::fromValue(derivedAsConstObject).value(), derived); QCOMPARE(QVariant::fromValue(unrelatedConstObj).value(), nullptr); } /*! We verify that: 1. Converting the string "9.9" to int fails. This is the behavior of toLongLong() and hence also QVariant, since it uses it. 2. Converting the QVariant containing the double 9.9 to int works. Rationale: "9.9" is not a valid int. However, doubles are by definition not ints and therefore it makes more sense to perform conversion for those. */ void tst_QVariant::toIntFromQString() const { QVariant first("9.9"); bool ok; QCOMPARE(first.toInt(&ok), 0); QVERIFY(!ok); QCOMPARE(QString("9.9").toLongLong(&ok), qlonglong(0)); QVERIFY(!ok); QVariant v(9.9); QCOMPARE(v.toInt(&ok), 10); QVERIFY(ok); } /*! We verify that: 1. Conversion from (64 bit) double to int works (no overflow). 2. Same conversion works for QVariant::convert. Rationale: if 2147483630 is set in float and then converted to int, there will be overflow and the result will be -2147483648. */ void tst_QVariant::toIntFromDouble() const { double d = 2147483630; // max int 2147483647 QCOMPARE((int)d, 2147483630); QVariant var(d); QVERIFY(var.canConvert()); bool ok; int result = var.toInt(&ok); QVERIFY( ok == true ); QCOMPARE(result, 2147483630); } void tst_QVariant::fpStringRoundtrip_data() const { QTest::addColumn("number"); QTest::newRow("float") << QVariant(1 + FLT_EPSILON); QTest::newRow("double") << QVariant(1 + DBL_EPSILON); } void tst_QVariant::fpStringRoundtrip() const { QFETCH(QVariant, number); QVariant converted = number; QVERIFY(converted.convert(QMetaType::fromType())); QVERIFY(converted.convert(QMetaType(number.typeId()))); QCOMPARE(converted, number); converted = number; QVERIFY(converted.convert(QMetaType::fromType())); QVERIFY(converted.convert(QMetaType(number.typeId()))); QCOMPARE(converted, number); } void tst_QVariant::numericalConvert_data() { QTest::addColumn("v"); QTest::addColumn("isInteger"); QTest::newRow("float") << QVariant(float(5.3)) << false; QTest::newRow("double") << QVariant(double(5.3)) << false; QTest::newRow("qreal") << QVariant(qreal(5.3)) << false; QTest::newRow("int") << QVariant(int(5)) << true; QTest::newRow("uint") << QVariant(uint(5)) << true; QTest::newRow("short") << QVariant(short(5)) << true; QTest::newRow("longlong") << QVariant(quint64(5)) << true; QTest::newRow("long") << QVariant::fromValue(long(5)) << true; QTest::newRow("stringint") << QVariant(QString::fromLatin1("5")) << true; QTest::newRow("string") << QVariant(QString::fromLatin1("5.30000019")) << false; } void tst_QVariant::numericalConvert() { QFETCH(QVariant, v); QFETCH(bool, isInteger); double num = isInteger ? 5 : 5.3; QCOMPARE(v.toFloat() , float(num)); QCOMPARE(float(v.toReal()) , float(num)); QCOMPARE(float(v.toDouble()) , float(num)); if (isInteger) { QCOMPARE(v.toInt() , int(num)); QCOMPARE(v.toUInt() , uint(num)); QCOMPARE(v.toULongLong() , quint64(num)); QCOMPARE(v.value() , ulong(num)); QCOMPARE(v.value() , ushort(num)); } switch (v.userType()) { case QMetaType::Double: QCOMPARE(v.toString() , QString::number(num, 'g', QLocale::FloatingPointShortest)); break; case QMetaType::Float: QCOMPARE(v.toString() , QString::number(float(num), 'g', QLocale::FloatingPointShortest)); break; } } template void playWithVariant(const T &orig, bool isNull, const QString &toString, double toDouble, bool toBool) { QVariant v = QVariant::fromValue(orig); QVERIFY(v.isValid()); QCOMPARE(v.isNull(), isNull); QCOMPARE(v.toString(), toString); QCOMPARE(v.toDouble(), toDouble); QCOMPARE(v.toBool(), toBool); QCOMPARE(qvariant_cast(v), orig); { QVariant v2 = v; if (QTypeInfo::isRelocatable) { // Type is movable so standard comparison algorithm in QVariant should work // In a custom type QVariant is not aware of ==operator so it won't be called, // which may cause problems especially visible when using a not-movable type QCOMPARE(v2, v); } QVERIFY(v2.isValid()); QCOMPARE(v2.isNull(), isNull); QCOMPARE(v2.toString(), toString); QCOMPARE(v2.toDouble(), toDouble); QCOMPARE(v2.toBool(), toBool); QCOMPARE(qvariant_cast(v2), orig); QVariant v3; v = QVariant(); QCOMPARE(v3, v); v = v2; if (QTypeInfo::isRelocatable) { // Type is movable so standard comparison algorithm in QVariant should work // In a custom type QVariant is not aware of ==operator so it won't be called, // which may cause problems especially visible when using a not-movable type QCOMPARE(v2, v); } QCOMPARE(qvariant_cast(v2), qvariant_cast(v)); QCOMPARE(v2.toString(), toString); v3 = QVariant::fromValue(orig); QVERIFY(v3.isValid()); QCOMPARE(v3.isNull(), isNull); QCOMPARE(v3.toString(), toString); QCOMPARE(v3.toDouble(), toDouble); QCOMPARE(v3.toBool(), toBool); QCOMPARE(qvariant_cast(v3), qvariant_cast(v)); } QVERIFY(v.isValid()); QCOMPARE(v.isNull(), isNull); QCOMPARE(v.toString(), toString); QCOMPARE(v.toDouble(), toDouble); QCOMPARE(v.toBool(), toBool); QCOMPARE(qvariant_cast(v), orig); if (qMetaTypeId() != qMetaTypeId()) QCOMPARE(v.userType(), qMetaTypeId()); } #define PLAY_WITH_VARIANT(Orig, IsNull, ToString, ToDouble, ToBool) \ playWithVariant(Orig, IsNull, ToString, ToDouble, ToBool);\ if (QTest::currentTestFailed())\ QFAIL("playWithVariant failed"); struct MyPrimitive { char x, y; bool operator==(const MyPrimitive &o) const { return x == o.x && y == o.y; } }; QT_BEGIN_NAMESPACE Q_DECLARE_TYPEINFO(MyPrimitive, Q_PRIMITIVE_TYPE); QT_END_NAMESPACE struct MyData { void *ptr; MyData() : ptr(this) {} ~MyData() { if (ptr != this) qWarning("%s: object has moved", Q_FUNC_INFO); } MyData(const MyData& o) : ptr(this) { if (o.ptr != &o) qWarning("%s: other object has moved", Q_FUNC_INFO); } MyData &operator=(const MyData &o) { if (ptr != this) qWarning("%s: object has moved", Q_FUNC_INFO); if (o.ptr != &o) qWarning("%s: other object has moved", Q_FUNC_INFO); return *this; } bool operator==(const MyData &o) const { if (ptr != this) qWarning("%s: object has moved", Q_FUNC_INFO); if (o.ptr != &o) qWarning("%s: other object has moved", Q_FUNC_INFO); return true; } }; struct MyMovable { static int count; int v; MyMovable() { v = count++; } ~MyMovable() { count--; } MyMovable(const MyMovable &o) : v(o.v) { count++; } MyMovable &operator=(const MyMovable &o) { v = o.v; return *this; } bool operator==(const MyMovable &o) const { return v == o.v; } }; int MyMovable::count = 0; struct MyNotMovable { static int count; MyNotMovable *that; MyNotMovable() : that(this) { count++; } ~MyNotMovable() { QCOMPARE(that, this); count--; } MyNotMovable(const MyNotMovable &o) : that(this) { QCOMPARE(o.that, &o); count++; } MyNotMovable &operator=(const MyNotMovable &o) { bool ok = that == this && o.that == &o; if (!ok) qFatal("MyNotMovable has been moved"); return *this; } //PLAY_WITH_VARIANT test that they are equal, but never that they are not equal // so it would be fine just to always return true bool operator==(const MyNotMovable &o) const { bool ok = that == this && o.that == &o; if (!ok) qFatal("MyNotMovable has been moved"); return ok; } // Make it too big to store it in the variant itself void *dummy[4]; }; int MyNotMovable::count = 0; struct MyShared : QSharedData { MyMovable movable; }; QT_BEGIN_NAMESPACE Q_DECLARE_TYPEINFO(MyMovable, Q_RELOCATABLE_TYPE); QT_END_NAMESPACE Q_DECLARE_METATYPE(MyPrimitive) Q_DECLARE_METATYPE(MyData) Q_DECLARE_METATYPE(MyMovable) Q_DECLARE_METATYPE(MyNotMovable) Q_DECLARE_METATYPE(MyPrimitive *) Q_DECLARE_METATYPE(MyData *) Q_DECLARE_METATYPE(MyMovable *) Q_DECLARE_METATYPE(MyNotMovable *) Q_DECLARE_METATYPE(QSharedDataPointer) void tst_QVariant::setValue() { MyNotMovable t; //we just take a value so that we're sure that it will be shared QVariant v1 = QVariant::fromValue(t); QVERIFY( v1.isDetached() ); QVariant v2 = v1; QVERIFY( !v1.isDetached() ); QVERIFY( !v2.isDetached() ); v2.setValue(3); //set an integer value QVERIFY( v1.isDetached() ); QVERIFY( v2.isDetached() ); } void tst_QVariant::moreCustomTypes() { { QList listSize; PLAY_WITH_VARIANT(listSize, false, QString(), 0, false); listSize << QSize(4,5) << QSize(89,23) << QSize(5,6); PLAY_WITH_VARIANT(listSize, false, QString(), 0, false); } { QString str; PLAY_WITH_VARIANT(str, false, QString(), 0, false); str = QString::fromLatin1("123456789.123"); PLAY_WITH_VARIANT(str, false, str, 123456789.123, true); } { QSize size; PLAY_WITH_VARIANT(size, false, QString(), 0, false); PLAY_WITH_VARIANT(QSize(45,78), false, QString(), 0, false); } { MyData d; PLAY_WITH_VARIANT(d, false, QString(), 0, false); PLAY_WITH_VARIANT(&d, false, QString(), 0, false); QList l; PLAY_WITH_VARIANT(l, false, QString(), 0, false); l << MyData() << MyData(); PLAY_WITH_VARIANT(l, false, QString(), 0, false); } { MyPrimitive d = { 4, 5 }; PLAY_WITH_VARIANT(d, false, QString(), 0, false); PLAY_WITH_VARIANT(&d, false, QString(), 0, false); QList l; PLAY_WITH_VARIANT(l, false, QString(), 0, false); l << d; PLAY_WITH_VARIANT(l, false, QString(), 0, false); } { MyMovable d; PLAY_WITH_VARIANT(d, false, QString(), 0, false); PLAY_WITH_VARIANT(&d, false, QString(), 0, false); QList l; PLAY_WITH_VARIANT(l, false, QString(), 0, false); l << MyMovable() << d; PLAY_WITH_VARIANT(l, false, QString(), 0, false); } QCOMPARE(MyMovable::count, 0); QCOMPARE(MyNotMovable::count, 0); { MyNotMovable d; PLAY_WITH_VARIANT(d, false, QString(), 0, false); PLAY_WITH_VARIANT(&d, false, QString(), 0, false); QList l; PLAY_WITH_VARIANT(l, false, QString(), 0, false); l << MyNotMovable() << d; PLAY_WITH_VARIANT(l, false, QString(), 0, false); } QCOMPARE(MyNotMovable::count, 0); { #ifdef QT_NO_DOUBLECONVERSION // snprintf cannot do "shortest" conversion and always adds noise. PLAY_WITH_VARIANT(12.12, false, "12.119999999999999", 12.12, true); #else // Double can be printed exactly with libdouble-conversion PLAY_WITH_VARIANT(12.12, false, "12.12", 12.12, true); #endif // Float is converted to double, adding insignificant bits PLAY_WITH_VARIANT(12.12f, false, "12.119999885559082", 12.12f, true); PLAY_WITH_VARIANT(quint16(14), false, "14", 14, true); PLAY_WITH_VARIANT( qint16(15), false, "15", 15, true); PLAY_WITH_VARIANT(quint32(16), false, "16", 16, true); PLAY_WITH_VARIANT( qint32(17), false, "17", 17, true); PLAY_WITH_VARIANT(quint64(18), false, "18", 18, true); PLAY_WITH_VARIANT( qint64(19), false, "19", 19, true); PLAY_WITH_VARIANT( qint16(-13), false, "-13", -13, true); PLAY_WITH_VARIANT( qint32(-14), false, "-14", -14, true); PLAY_WITH_VARIANT( qint64(-15), false, "-15", -15, true); PLAY_WITH_VARIANT(quint64(0), false, "0", 0, false); PLAY_WITH_VARIANT( true, false, "true", 1, true); PLAY_WITH_VARIANT( false, false, "false", 0, false); PLAY_WITH_VARIANT(QString("hello\n"), false, "hello\n", 0, true); } { int i = 5; PLAY_WITH_VARIANT((void *)(&i), false, QString(), 0, false); PLAY_WITH_VARIANT((void *)(0), true, QString(), 0, false); } { QVariant v1 = QVariant::fromValue(5); QVariant v2 = QVariant::fromValue(5.0); QVariant v3 = QVariant::fromValue(quint16(5)); QVariant v4 = 5; QVariant v5 = QVariant::fromValue(MyPrimitive()); QVariant v6 = QVariant::fromValue(MyMovable()); QVariant v7 = QVariant::fromValue(MyData()); PLAY_WITH_VARIANT(v1, false, "5", 5, true); PLAY_WITH_VARIANT(v2, false, "5", 5, true); PLAY_WITH_VARIANT(v3, false, "5", 5, true); PLAY_WITH_VARIANT(v4, false, "5", 5, true); PLAY_WITH_VARIANT(v5, false, QString(), 0, false); } QCOMPARE(MyMovable::count, 0); { QSharedDataPointer d(new MyShared); PLAY_WITH_VARIANT(d, false, QString(), 0, false); } QCOMPARE(MyMovable::count, 0); { QList > data; PLAY_WITH_VARIANT(data, false, QString(), 0, false); data << (QList() << 42); PLAY_WITH_VARIANT(data, false, QString(), 0, false); } { QList > data; PLAY_WITH_VARIANT(data, false, QString(), 0, false); data << (QList() << 42); PLAY_WITH_VARIANT(data, false, QString(), 0, false); } { QList > data; PLAY_WITH_VARIANT(data, false, QString(), 0, false); data << (QSet() << 42); PLAY_WITH_VARIANT(data, false, QString(), 0, false); } } void tst_QVariant::movabilityTest() { // This test checks if QVariant is movable even if an internal data is not movable. QVERIFY(!MyNotMovable::count); { QVariant variant = QVariant::fromValue(MyNotMovable()); QVERIFY(MyNotMovable::count); // prepare destination memory space to which variant will be moved QVariant buffer[1]; QCOMPARE(buffer[0].typeId(), QMetaType::UnknownType); buffer[0].~QVariant(); memcpy(static_cast(buffer), static_cast(&variant), sizeof(QVariant)); QVERIFY(buffer[0].typeId() > QMetaType::User); QCOMPARE(buffer[0].userType(), qMetaTypeId()); MyNotMovable tmp(buffer[0].value()); new (&variant) QVariant(); } QVERIFY(!MyNotMovable::count); } void tst_QVariant::variantInVariant() { QVariant var1 = 5; QCOMPARE(var1.typeId(), QMetaType::Int); QVariant var2 = var1; QCOMPARE(var2, var1); QCOMPARE(var2.typeId(), QMetaType::Int); QVariant var3 = QVariant::fromValue(var1); QCOMPARE(var3, var1); QCOMPARE(var3.typeId(), QMetaType::Int); QVariant var4 = qvariant_cast(var1); QCOMPARE(var4, var1); QCOMPARE(var4.typeId(), QMetaType::Int); QVariant var5; var5 = var1; QCOMPARE(var5, var1); QCOMPARE(var5.typeId(), QMetaType::Int); QVariant var6; var6.setValue(var1); QCOMPARE(var6, var1); QCOMPARE(var6.typeId(), QMetaType::Int); QCOMPARE(QVariant::fromValue(var1), QVariant::fromValue(var2)); QCOMPARE(qvariant_cast(var3), QVariant::fromValue(var4)); QCOMPARE(qvariant_cast(var5), qvariant_cast(var6)); QString str("hello"); QVariant var8 = qvariant_cast(QVariant::fromValue(QVariant::fromValue(str))); QCOMPARE(var8.typeId(), QMetaType::QString); QCOMPARE(qvariant_cast(QVariant(qvariant_cast(var8))), str); QVariant var9(QMetaType::fromType(), &var1); QCOMPARE(var9.userType(), qMetaTypeId()); QCOMPARE(qvariant_cast(var9), var1); } struct Convertible { double d; operator int() const { return (int)d; } operator double() const { return d; } operator QString() const { return QString::number(d); } }; Q_DECLARE_METATYPE(Convertible); struct BigConvertible { double d; double dummy[sizeof(QVariant) / sizeof(double)]; operator int() const { return (int)d; } operator double() const { return d; } operator QString() const { return QString::number(d); } }; Q_DECLARE_METATYPE(BigConvertible); static_assert(sizeof(BigConvertible) > sizeof(QVariant)); void tst_QVariant::userConversion() { { QVERIFY(!(QMetaType::hasRegisteredConverterFunction())); QVERIFY(!(QMetaType::hasRegisteredConverterFunction())); QVERIFY(!(QMetaType::hasRegisteredConverterFunction())); Convertible c = { 123 }; QVariant v = QVariant::fromValue(c); bool ok; v.toInt(&ok); QVERIFY(!ok); v.toDouble(&ok); QVERIFY(!ok); QString s = v.toString(); QVERIFY(s.isEmpty()); QMetaType::registerConverter(); QMetaType::registerConverter(); QMetaType::registerConverter(); int i = v.toInt(&ok); QVERIFY(ok); QCOMPARE(i, 123); double d = v.toDouble(&ok); QVERIFY(ok); QCOMPARE(d, 123.); s = v.toString(); QCOMPARE(s, QString::fromLatin1("123")); } { QVERIFY(!(QMetaType::hasRegisteredConverterFunction())); QVERIFY(!(QMetaType::hasRegisteredConverterFunction())); QVERIFY(!(QMetaType::hasRegisteredConverterFunction())); BigConvertible c = { 123, { 0, 0 } }; QVariant v = QVariant::fromValue(c); bool ok; v.toInt(&ok); QVERIFY(!ok); v.toDouble(&ok); QVERIFY(!ok); QString s = v.toString(); QVERIFY(s.isEmpty()); QMetaType::registerConverter(); QMetaType::registerConverter(); QMetaType::registerConverter(); int i = v.toInt(&ok); QVERIFY(ok); QCOMPARE(i, 123); double d = v.toDouble(&ok); QVERIFY(ok); QCOMPARE(d, 123.); s = v.toString(); QCOMPARE(s, QString::fromLatin1("123")); } } void tst_QVariant::modelIndexConversion() { QVariant modelIndexVariant = QModelIndex(); QVERIFY(modelIndexVariant.canConvert()); QVERIFY(modelIndexVariant.convert(QMetaType::fromType())); QCOMPARE(modelIndexVariant.typeId(), QMetaType::QPersistentModelIndex); QVERIFY(modelIndexVariant.canConvert(QMetaType::fromType())); QVERIFY(modelIndexVariant.convert(QMetaType::fromType())); QCOMPARE(modelIndexVariant.typeId(), QMetaType::QModelIndex); } class Forward; Q_DECLARE_OPAQUE_POINTER(Forward*) Q_DECLARE_METATYPE(Forward*) void tst_QVariant::forwardDeclare() { Forward *f = nullptr; QVariant v = QVariant::fromValue(f); QCOMPARE(qvariant_cast(v), f); } void tst_QVariant::loadQt5Stream_data() { dataStream_data(QDataStream::Qt_5_0); } void tst_QVariant::loadQt5Stream() { loadQVariantFromDataStream(QDataStream::Qt_5_0); } void tst_QVariant::saveQt5Stream_data() { dataStream_data(QDataStream::Qt_5_0); } void tst_QVariant::saveQt5Stream() { saveQVariantFromDataStream(QDataStream::Qt_5_0); } void tst_QVariant::loadQt4Stream_data() { dataStream_data(QDataStream::Qt_4_9); } void tst_QVariant::loadQt4Stream() { loadQVariantFromDataStream(QDataStream::Qt_4_9); } void tst_QVariant::saveQt4Stream_data() { dataStream_data(QDataStream::Qt_4_9); } void tst_QVariant::saveQt4Stream() { saveQVariantFromDataStream(QDataStream::Qt_4_9); } void tst_QVariant::dataStream_data(QDataStream::Version version) { QTest::addColumn("fileName"); QString path; switch (version) { case QDataStream::Qt_4_9: path = QString::fromLatin1("qt4.9"); break; case QDataStream::Qt_5_0: path = QString::fromLatin1("qt5.0"); break; default: Q_UNIMPLEMENTED(); } path = path.prepend(":/stream/").append("/"); QDir dir(path); uint i = 0; const auto entries = dir.entryInfoList(QStringList{u"*.bin"_s}); for (const QFileInfo &fileInfo : entries) { QTest::newRow((path + fileInfo.fileName()).toLatin1()) << fileInfo.filePath(); i += 1; } QVERIFY(i > 10); } void tst_QVariant::loadQVariantFromDataStream(QDataStream::Version version) { QFETCH(QString, fileName); QFile file(fileName); QVERIFY(file.open(QIODevice::ReadOnly)); QDataStream stream(&file); stream.setVersion(version); QString typeName; QVariant loadedVariant; stream >> typeName >> loadedVariant; const int id = QMetaType::fromName(typeName.toLatin1()).id(); if (id == QMetaType::Void) { // Void type is not supported by QVariant return; } QVariant constructedVariant {QMetaType(id)}; QCOMPARE(constructedVariant.userType(), id); QCOMPARE(QMetaType(loadedVariant.userType()).name(), typeName.toLatin1().constData()); QCOMPARE(loadedVariant.userType(), constructedVariant.userType()); } void tst_QVariant::saveQVariantFromDataStream(QDataStream::Version version) { QFETCH(QString, fileName); QFile file(fileName); QVERIFY(file.open(QIODevice::ReadOnly)); QDataStream dataFileStream(&file); QString typeName; dataFileStream >> typeName; QByteArray data = file.readAll(); const int id = QMetaType::fromName(typeName.toLatin1()).id(); if (id == QMetaType::Void) { // Void type is not supported by QVariant return; } QBuffer buffer; buffer.open(QIODevice::ReadWrite); QDataStream stream(&buffer); stream.setVersion(version); QVariant constructedVariant {QMetaType(id)}; QCOMPARE(constructedVariant.userType(), id); stream << constructedVariant; // We are testing QVariant there is no point in testing full array. QCOMPARE(buffer.data().left(5), data.left(5)); buffer.seek(0); QVariant recunstructedVariant; stream >> recunstructedVariant; QCOMPARE(recunstructedVariant.userType(), constructedVariant.userType()); } void tst_QVariant::debugStream_data() { QTest::addColumn("variant"); QTest::addColumn("typeId"); for (int id = 0; id < QMetaType::LastCoreType + 1; ++id) { if (id && !QMetaType::isRegistered(id)) { QTest::ignoreMessage(QtWarningMsg, QRegularExpression( "^Trying to construct an instance of an invalid type")); } const char *tagName = QMetaType(id).name(); if (tagName && id != QMetaType::Void) QTest::newRow(tagName) << QVariant(QMetaType(id)) << id; } QTest::newRow("QBitArray(111)") << QVariant(QBitArray(3, true)) << qMetaTypeId(); QTest::newRow("CustomStreamableClass") << QVariant(QMetaType::fromType(), 0) << qMetaTypeId(); QTest::newRow("MyClass") << QVariant(QMetaType::fromType(), 0) << qMetaTypeId(); QTest::newRow("InvalidVariant") << QVariant() << int(QMetaType::UnknownType); QTest::newRow("CustomQObject") << QVariant::fromValue(this) << qMetaTypeId(); } void tst_QVariant::debugStream() { QFETCH(QVariant, variant); QFETCH(int, typeId); MessageHandler msgHandler(typeId); qDebug() << variant; QVERIFY(msgHandler.testPassed()); } #if QT_DEPRECATED_SINCE(6, 0) struct MessageHandlerType : public MessageHandler { MessageHandlerType(const int typeId) : MessageHandler(typeId, handler) {} static void handler(QtMsgType, const QMessageLogContext &, const QString &msg) { // Format itself is not important, but basic data as a type name should be included in the output ok = msg.startsWith("QVariant::"); QVERIFY2(ok, (QString::fromLatin1("Message is not started correctly: '") + msg + '\'').toLatin1().constData()); QT_WARNING_PUSH QT_WARNING_DISABLE_DEPRECATED ok &= (currentId == QMetaType::UnknownType ? msg.contains("Invalid") : msg.contains(QMetaType::typeName(currentId))); QT_WARNING_POP QVERIFY2(ok, (QString::fromLatin1("Message doesn't contain type name: '") + msg + '\'').toLatin1().constData()); } }; void tst_QVariant::debugStreamType_data() { debugStream_data(); } void tst_QVariant::debugStreamType() { QFETCH(QVariant, variant); QFETCH(int, typeId); MessageHandlerType msgHandler(typeId); QT_IGNORE_DEPRECATIONS(qDebug() << QVariant::Type(typeId);) QVERIFY(msgHandler.testPassed()); } #endif // QT_DEPRECATED_SINCE(6, 0) void tst_QVariant::implicitConstruction() { // This is a compile-time test QVariant v; #define FOR_EACH_CORE_CLASS(F) \ F(Char) \ F(String) \ F(StringList) \ F(ByteArray) \ F(BitArray) \ F(Date) \ F(Time) \ F(DateTime) \ F(Url) \ F(Locale) \ F(Rect) \ F(RectF) \ F(Size) \ F(SizeF) \ F(Line) \ F(LineF) \ F(Point) \ F(PointF) \ F(EasingCurve) \ F(Uuid) \ F(ModelIndex) \ F(PersistentModelIndex) \ F(RegularExpression) \ F(JsonValue) \ F(JsonObject) \ F(JsonArray) \ F(JsonDocument) \ #define CONSTRUCT(TYPE) \ { \ Q##TYPE t; \ v = t; \ t = v.to##TYPE(); \ QVERIFY(true); \ } FOR_EACH_CORE_CLASS(CONSTRUCT) #undef CONSTRUCT #undef FOR_EACH_CORE_CLASS } void tst_QVariant::saveInvalid_data() { QTest::addColumn("version"); for (unsigned version = QDataStream::Qt_5_0; version > QDataStream::Qt_1_0; --version) QTest::newRow(QString::number(version).toUtf8()) << version; } void tst_QVariant::saveInvalid() { QFETCH(unsigned, version); QByteArray data; QDataStream stream(&data, QIODevice::WriteOnly); stream.setVersion(version); stream << QVariant(); QCOMPARE(stream.status(), QDataStream::Ok); QVERIFY(data.size() >= 4); QCOMPARE(int(data.constData()[0]), 0); QCOMPARE(int(data.constData()[1]), 0); QCOMPARE(int(data.constData()[2]), 0); QCOMPARE(int(data.constData()[3]), 0); } void tst_QVariant::saveNewBuiltinWithOldStream() { QByteArray data; QDataStream stream(&data, QIODevice::WriteOnly); stream.setVersion(QDataStream::Qt_3_1); stream << QVariant::fromValue(123); // QJsonValue class was introduced in Qt5 QCOMPARE(stream.status(), QDataStream::Ok); QVERIFY(data.size() >= 4); QCOMPARE(int(data.constData()[0]), 0); QCOMPARE(int(data.constData()[1]), 0); QCOMPARE(int(data.constData()[2]), 0); QCOMPARE(int(data.constData()[3]), 0); } template struct ContainerAPI { static void insert(Container &container, typename Container::value_type value) { container.push_back(value); } static bool compare(const QVariant &variant, typename Container::value_type value) { return variant.value() == value; } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; template struct ContainerAPI { static void insert(Container &container, int value) { container.push_back(QVariant::fromValue(value)); } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; template struct ContainerAPI { static void insert(Container &container, int value) { container.push_back(QString::number(value)); } static bool compare(const QVariant &variant, QString value) { return variant.value() == value; } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; template struct ContainerAPI { static void insert(Container &container, int value) { container.push_back(QByteArray::number(value)); } static bool compare(const QVariant &variant, QByteArray value) { return variant.value() == value; } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; template struct ContainerAPI { static void insert(Container &container, int value) { container.push_back(QChar::fromLatin1(char(value) + '0')); } static bool compare(const QVariant &variant, QChar value) { return variant.value() == value; } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; template struct ContainerAPI { static void insert(Container &container, int value) { container.push_back(char(value) + '0'); } static bool compare(const QVariant &variant, char value) { return variant.value() == value; } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; #ifdef __has_include # if __has_include() # define TEST_FORWARD_LIST # include Q_DECLARE_SEQUENTIAL_CONTAINER_METATYPE(std::forward_list) // Test that explicit declaration does not degrade features. Q_DECLARE_METATYPE(std::forward_list) template struct ContainerAPI > { static void insert(std::forward_list &container, Value_Type value) { container.push_front(value); } static bool compare(const QVariant &variant, Value_Type value) { return variant.value() == value; } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; template<> struct ContainerAPI > { static void insert(std::forward_list &container, int value) { container.push_front(QVariant::fromValue(value)); } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; template<> struct ContainerAPI > { static void insert(std::forward_list &container, int value) { container.push_front(QString::number(value)); } static bool compare(const QVariant &variant, QString value) { return variant.value() == value; } static bool compare(QVariant variant, const QVariant &value) { return variant == value; } }; # endif // __has_include() #endif // __has_include template struct KeyGetter { static const typename Container::key_type & get(const typename Container::const_iterator &it) { return it.key(); } static const typename Container::mapped_type & value(const typename Container::const_iterator &it) { return it.value(); } }; template struct KeyGetter > { static const T & get(const typename std::map::const_iterator &it) { return it->first; } static const U & value(const typename std::map::const_iterator &it) { return it->second; } }; typedef std::unordered_map StdUnorderedMap_int_bool; Q_DECLARE_ASSOCIATIVE_CONTAINER_METATYPE(std::unordered_map) Q_DECLARE_METATYPE(StdUnorderedMap_int_bool) template struct KeyGetter > { static const T & get(const typename std::unordered_map::const_iterator &it) { return it->first; } static const U & value(const typename std::unordered_map::const_iterator &it) { return it->second; } }; template void sortIterable(QSequentialIterable *iterable) { std::sort(Iterator(iterable->mutableBegin()), Iterator(iterable->mutableEnd()), [&](const QVariant &a, const QVariant &b) { return a.toInt() < b.toInt(); }); } template static void testSequentialIteration() { int numSeen = 0; Container sequence; ContainerAPI::insert(sequence, 1); ContainerAPI::insert(sequence, 2); ContainerAPI::insert(sequence, 3); QVariant listVariant = QVariant::fromValue(sequence); QVERIFY(listVariant.canConvert()); QVariantList varList = listVariant.value(); QCOMPARE(varList.size(), (int)std::distance(sequence.begin(), sequence.end())); QSequentialIterable listIter = listVariant.view(); QCOMPARE(varList.size(), listIter.size()); typename Container::iterator containerIter = sequence.begin(); const typename Container::iterator containerEnd = sequence.end(); for (int i = 0; i < listIter.size(); ++i, ++containerIter, ++numSeen) { QVERIFY(ContainerAPI::compare(listIter.at(i), *containerIter)); QVERIFY(ContainerAPI::compare(listIter.at(i), varList.at(i))); } QCOMPARE(numSeen, (int)std::distance(sequence.begin(), sequence.end())); QCOMPARE(containerIter, containerEnd); numSeen = 0; containerIter = sequence.begin(); for (QVariant v : listIter) { QVERIFY(ContainerAPI::compare(v, *containerIter)); QVERIFY(ContainerAPI::compare(v, varList.at(numSeen))); ++containerIter; ++numSeen; } QCOMPARE(numSeen, (int)std::distance(sequence.begin(), sequence.end())); auto compareLists = [&]() { int numSeen = 0; auto varList = listVariant.value(); auto varIter = varList.begin(); for (const QVariant &v : std::as_const(listIter)) { QVERIFY(ContainerAPI::compare(v, *varIter)); ++varIter; ++numSeen; } QCOMPARE(varIter, varList.end()); numSeen = 0; auto constVarIter = varList.constBegin(); for (QVariant v : listIter) { QVERIFY(ContainerAPI::compare(v, *constVarIter)); ++constVarIter; ++numSeen; } QCOMPARE(numSeen, (int)std::distance(varList.begin(), varList.end())); }; compareLists(); QVariant first = listIter.at(0); QVariant second = listIter.at(1); QVariant third = listIter.at(2); compareLists(); listIter.addValue(third); compareLists(); listIter.addValue(second); compareLists(); listIter.addValue(first); compareLists(); QCOMPARE(listIter.size(), 6); if (listIter.canRandomAccessIterate()) sortIterable(&listIter); else if (listIter.canReverseIterate()) sortIterable(&listIter); else if (listIter.canForwardIterate()) return; // std::sort cannot sort with only forward iterators. else QFAIL("The container has no meaningful iterators"); compareLists(); QCOMPARE(listIter.size(), 6); QCOMPARE(listIter.at(0), first); QCOMPARE(listIter.at(1), first); QCOMPARE(listIter.at(2), second); QCOMPARE(listIter.at(3), second); QCOMPARE(listIter.at(4), third); QCOMPARE(listIter.at(5), third); if (listIter.metaContainer().canRemoveValue()) { listIter.removeValue(); compareLists(); QCOMPARE(listIter.size(), 5); QCOMPARE(listIter.at(0), first); QCOMPARE(listIter.at(1), first); QCOMPARE(listIter.at(2), second); QCOMPARE(listIter.at(3), second); QCOMPARE(listIter.at(4), third); } else { // QString and QByteArray have no pop_back or pop_front and it's unclear what other // method we should use to remove an item. QVERIFY((std::is_same_v || std::is_same_v)); } auto i = listIter.mutableBegin(); QVERIFY(i != listIter.mutableEnd()); *i = QStringLiteral("17"); if (listIter.metaContainer().valueMetaType() == QMetaType::fromType()) QCOMPARE(listIter.at(0).toInt(), 17); else if (listIter.metaContainer().valueMetaType() == QMetaType::fromType()) QCOMPARE(listIter.at(0).toBool(), false); *i = QStringLiteral("true"); if (listIter.metaContainer().valueMetaType() == QMetaType::fromType()) QCOMPARE(listIter.at(0).toInt(), 0); else if (listIter.metaContainer().valueMetaType() == QMetaType::fromType()) QCOMPARE(listIter.at(0).toBool(), true); } template static void testAssociativeIteration() { using Key = typename Container::key_type; using Mapped = typename Container::mapped_type; int numSeen = 0; Container mapping; mapping[5] = true; mapping[15] = false; QVariant mappingVariant = QVariant::fromValue(mapping); QVariantMap varMap = mappingVariant.value(); QVariantMap varHash = mappingVariant.value(); QAssociativeIterable mappingIter = mappingVariant.view(); typename Container::const_iterator containerIter = mapping.begin(); const typename Container::const_iterator containerEnd = mapping.end(); for ( ;containerIter != containerEnd; ++containerIter, ++numSeen) { Mapped expected = KeyGetter::value(containerIter); Key key = KeyGetter::get(containerIter); Mapped actual = qvariant_cast(mappingIter.value(key)); QCOMPARE(qvariant_cast(varMap.value(QString::number(key))), expected); QCOMPARE(qvariant_cast(varHash.value(QString::number(key))), expected); QCOMPARE(actual, expected); const QAssociativeIterable::const_iterator it = mappingIter.find(key); QVERIFY(it != mappingIter.end()); QCOMPARE(it.value().value(), expected); } QCOMPARE(numSeen, (int)std::distance(mapping.begin(), mapping.end())); QCOMPARE(containerIter, containerEnd); QVERIFY(mappingIter.find(10) == mappingIter.end()); auto i = mappingIter.mutableFind(QStringLiteral("nonono")); QCOMPARE(i, mappingIter.mutableEnd()); i = mappingIter.mutableFind(QStringLiteral("5")); QVERIFY(i != mappingIter.mutableEnd()); *i = QStringLiteral("17"); if (mappingIter.metaContainer().mappedMetaType() == QMetaType::fromType()) QCOMPARE(mappingIter.value(5).toInt(), 17); else if (mappingIter.metaContainer().mappedMetaType() == QMetaType::fromType()) QCOMPARE(mappingIter.value(5).toBool(), true); *i = QStringLiteral("true"); if (mappingIter.metaContainer().mappedMetaType() == QMetaType::fromType()) QCOMPARE(mappingIter.value(5).toInt(), 0); else if (mappingIter.metaContainer().mappedMetaType() == QMetaType::fromType()) QCOMPARE(mappingIter.value(5).toBool(), true); QVERIFY(mappingIter.containsKey("5")); mappingIter.removeKey(QStringLiteral("5")); QCOMPARE(mappingIter.find(5), mappingIter.end()); mappingIter.setValue(5, 44); if (mappingIter.metaContainer().mappedMetaType() == QMetaType::fromType()) QCOMPARE(mappingIter.value(5).toInt(), 44); else if (mappingIter.metaContainer().mappedMetaType() == QMetaType::fromType()) QCOMPARE(mappingIter.value(5).toBool(), true); // Test that find() does not coerce auto container = Container(); container[0] = true; QVariant containerVariant = QVariant::fromValue(container); QAssociativeIterable iter = containerVariant.value(); auto f = iter.constFind(QStringLiteral("anything")); QCOMPARE(f, iter.constEnd()); } void tst_QVariant::iterateSequentialContainerElements_data() { QTest::addColumn("testFunction"); #define ADD(T) QTest::newRow(#T) << &testSequentialIteration ADD(QQueue); ADD(QQueue); ADD(QQueue); ADD(QList); ADD(QList); ADD(QList); ADD(QList); ADD(QStack); ADD(QStack); ADD(QStack); ADD(std::vector); ADD(std::vector); ADD(std::vector); ADD(std::list); ADD(std::list); ADD(std::list); ADD(QStringList); ADD(QByteArrayList); ADD(QString); ADD(QByteArray); #ifdef TEST_FORWARD_LIST ADD(std::forward_list); ADD(std::forward_list); ADD(std::forward_list); #endif #undef ADD } void tst_QVariant::iterateAssociativeContainerElements_data() { QTest::addColumn("testFunction"); #define ADD(C, K, V) QTest::newRow(#C #K #V) << &testAssociativeIteration>; ADD(QHash, int, bool); ADD(QHash, int, int); ADD(QMap, int, bool); ADD(std::map, int, bool); ADD(std::unordered_map, int, bool); #undef ADD } void tst_QVariant::iterateContainerElements() { { QVariantList ints; ints << 1 << 2 << 3; QVariant var = QVariant::fromValue(ints); QSequentialIterable iter = var.value(); QSequentialIterable::const_iterator it = iter.begin(); QSequentialIterable::const_iterator end = iter.end(); QCOMPARE(ints.at(1), *(it + 1)); int i = 0; for ( ; it != end; ++it, ++i) { QCOMPARE(ints.at(i), *it); } it = iter.begin(); QVariantList intsCopy; intsCopy << *(it++); intsCopy << *(it++); intsCopy << *(it++); QCOMPARE(ints, intsCopy); } { QMap mapping; mapping.insert(1, "one"); mapping.insert(2, "two"); mapping.insert(3, "three"); QVariant var = QVariant::fromValue(mapping); QAssociativeIterable iter = var.value(); QAssociativeIterable::const_iterator it = iter.begin(); QAssociativeIterable::const_iterator end = iter.end(); QCOMPARE(*(++mapping.begin()), (*(it + 1)).toString()); int i = 0; for ( ; it != end; ++it, ++i) { QCOMPARE(*(std::next(mapping.begin(), i)), (*it).toString()); } QVariantList nums; nums << "one" << "two" << "three"; it = iter.begin(); QVariantList numsCopy; numsCopy << *(it++); numsCopy << *(it++); numsCopy << *(it++); QCOMPARE(nums, numsCopy); } { auto container = QVariantMap(); container["one"] = 1; auto containerVariant = QVariant::fromValue(container); auto iter = containerVariant.value(); auto value = iter.value("one"); QCOMPARE(value, QVariant(1)); auto f = iter.constFind("one"); QCOMPARE(*f, QVariant(1)); } } template static void testVariantPairElements() { QFETCH(std::function, makeValue); Pair p; makeValue(&p); QVariant v = QVariant::fromValue(p); QVERIFY(v.canConvert()); QVariantPair pi = v.value(); QCOMPARE(pi.first, QVariant::fromValue(p.first)); QCOMPARE(pi.second, QVariant::fromValue(p.second)); } void tst_QVariant::pairElements_data() { QTest::addColumn("testFunction"); QTest::addColumn>("makeValue"); static auto makeString = [](auto &&value) -> QString { using T = std::decay_t; if constexpr (std::is_integral_v || std::is_floating_point_v) { return QString::number(value); } else if constexpr (std::is_same_v) { return value.toString(); } else { return value; } }; auto addRow = [](auto &&first, auto &&second) { using Pair = std::pair, std::decay_t>; std::function makeValue = [=](void *pair) { *static_cast(pair) = Pair{first, second}; }; QTest::addRow("%s", qPrintable(makeString(first) + u',' + makeString(second))) << &testVariantPairElements << makeValue; }; addRow(4, 5); addRow(QStringLiteral("one"), QStringLiteral("two")); addRow(QVariant(4), QVariant(5)); addRow(QVariant(41), 65); addRow(41, QVariant(15)); } template static void testVariantEnum() { using Enum = decltype(value); auto canLosslesslyConvert = [=](auto zero) { return sizeof(value) <= sizeof(zero) || value == Enum(decltype(zero)(qToUnderlying(value))); }; bool losslessConvertToInt = canLosslesslyConvert(int{}); QVariant var = QVariant::fromValue(value); QCOMPARE(var.userType(), qMetaTypeId()); QVERIFY(var.canConvert()); QVERIFY(var.canConvert()); QVERIFY(var.canConvert()); QVERIFY(var.canConvert()); QVERIFY(var.canConvert()); QVERIFY(var.canConvert()); QVERIFY(var.canConvert()); QCOMPARE(var.value(), value); QCOMPARE(var.value(), static_cast(value)); QCOMPARE(var.value(), static_cast(value)); QCOMPARE(var.value(), static_cast(value)); QCOMPARE(var.value(), static_cast(value)); QCOMPARE(var.value(), static_cast(value)); QCOMPARE(var.value(), static_cast(value)); QVariant var2 = var; QVERIFY(var2.convert(QMetaType::fromType())); QCOMPARE(var2.value(), static_cast(value)); QVariant strVar = QString::number(qToUnderlying(value)); QVariant baVar = QByteArray::number(qToUnderlying(value)); QCOMPARE(strVar.value(), value); QCOMPARE(baVar.value(), value); QCOMPARE(var.value(), strVar); QCOMPARE(var.value(), baVar); // unary + to silence gcc warning if (losslessConvertToInt) { int intValue = static_cast(value); QVariant intVar = intValue; QVERIFY(intVar.canConvert()); QCOMPARE(intVar.value(), value); } qint64 longValue = static_cast(value); QVERIFY(QVariant(longValue).canConvert()); QCOMPARE(QVariant(longValue).value(), value); auto value2 = Enum(qToUnderlying(value) + 1); var2 = QVariant::fromValue(value2); QCOMPARE_EQ(var, var); QCOMPARE_NE(var, var2); QCOMPARE(QVariant::compare(var, var), QPartialOrdering::Equivalent); QCOMPARE(QVariant::compare(var, var2), QPartialOrdering::Less); QCOMPARE(QVariant::compare(var2, var), QPartialOrdering::Greater); QCOMPARE_EQ(var, static_cast(value)); QCOMPARE_EQ(var, static_cast(value)); QCOMPARE_EQ(static_cast(value), var); QCOMPARE_EQ(static_cast(value), var); QCOMPARE_NE(var2, static_cast(value)); QCOMPARE_NE(var2, static_cast(value)); QCOMPARE_NE(static_cast(value), var2); QCOMPARE_NE(static_cast(value), var2); if (losslessConvertToInt) { QCOMPARE_EQ(var, int(value)); QCOMPARE_EQ(int(value), var); QCOMPARE_NE(var2, int(value)); QCOMPARE_NE(int(value), var2); } if (canLosslesslyConvert(uint{})) { QCOMPARE_EQ(var, uint(value)); QCOMPARE_EQ(uint(value), var); QCOMPARE_NE(var2, uint(value)); QCOMPARE_NE(uint(value), var2); } if (canLosslesslyConvert(short{})) { QCOMPARE_EQ(var, short(value)); QCOMPARE_EQ(short(value), var); QCOMPARE_NE(var2, short(value)); QCOMPARE_NE(short(value), var2); } if (canLosslesslyConvert(ushort{})) { QCOMPARE_EQ(var, ushort(value)); QCOMPARE_EQ(ushort(value), var); QCOMPARE_NE(var2, ushort(value)); QCOMPARE_NE(ushort(value), var2); } if (canLosslesslyConvert(char{})) { QCOMPARE_EQ(var, char(value)); QCOMPARE_EQ(char(value), var); QCOMPARE_NE(var2, char(value)); QCOMPARE_NE(char(value), var2); } if (canLosslesslyConvert(uchar{})) { QCOMPARE_EQ(var, uchar(value)); QCOMPARE_EQ(uchar(value), var); QCOMPARE_NE(var2, uchar(value)); QCOMPARE_NE(uchar(value), var2); } if (canLosslesslyConvert(qint8{})) { QCOMPARE_EQ(var, qint8(value)); QCOMPARE_EQ(qint8(value), var); QCOMPARE_NE(var2, qint8(value)); QCOMPARE_NE(qint8(value), var2); } // string compares too (of the values in decimal) QCOMPARE_EQ(var, QString::number(qToUnderlying(value))); QCOMPARE_EQ(QString::number(qToUnderlying(value)), var); QCOMPARE_NE(var, QString::number(qToUnderlying(value2))); QCOMPARE_NE(QString::number(qToUnderlying(value2)), var); } void tst_QVariant::enums_data() { QTest::addColumn("testFunction"); #define ADD(V) QTest::newRow(#V) << &testVariantEnum ADD(EnumTest_Enum0_value); ADD(EnumTest_Enum0_negValue); ADD(EnumTest_Enum1_value); ADD(EnumTest_Enum1_bigValue); ADD(EnumTest_Enum3::EnumTest_Enum3_value); ADD(EnumTest_Enum3::EnumTest_Enum3_bigValue); ADD(EnumTest_Enum4::EnumTest_Enum4_value); ADD(EnumTest_Enum4::EnumTest_Enum4_bigValue); ADD(EnumTest_Enum5::EnumTest_Enum5_value); ADD(EnumTest_Enum6::EnumTest_Enum6_value); ADD(EnumTest_Enum7::EnumTest_Enum7_value); ADD(EnumTest_Enum8::EnumTest_Enum8_value); ADD(EnumTest_Enum3::EnumTest_Enum3_value); #undef ADD } // ### C++20: this would be easier if QFlags were a structural type template static void testVariantMetaEnum() { Enum value(Value); QFETCH(QString, string); QVariant var = QVariant::fromValue(value); QVERIFY(var.canConvert()); QVERIFY(var.canConvert()); QCOMPARE(var.value(), string); QCOMPARE(var.value(), string.toLatin1()); QVariant strVar = string; QVERIFY(strVar.canConvert()); // unary + to silence gcc warning if ((+static_cast(value) > INT_MAX) || (+static_cast(value) < INT_MIN)) { QEXPECT_FAIL("", "QMetaEnum api uses 'int' as return type QTBUG-27451", Abort); } QCOMPARE(strVar.value(), value); strVar = string.toLatin1(); QVERIFY(strVar.canConvert()); QCOMPARE(strVar.value(), value); } void tst_QVariant::metaEnums_data() { QTest::addColumn("testFunction"); QTest::addColumn("string"); #define METAENUMS_TEST(Value) \ QTest::newRow(#Value) << &testVariantMetaEnum << #Value; METAENUMS_TEST(MetaEnumTest_Enum0_value); METAENUMS_TEST(MetaEnumTest_Enum1_value); METAENUMS_TEST(MetaEnumTest_Enum1_bigValue); METAENUMS_TEST(MetaEnumTest_Enum3_value); METAENUMS_TEST(MetaEnumTest_Enum3_bigValue); METAENUMS_TEST(MetaEnumTest_Enum3_bigNegValue); METAENUMS_TEST(MetaEnumTest_Enum4_value); METAENUMS_TEST(MetaEnumTest_Enum4_bigValue); METAENUMS_TEST(MetaEnumTest_Enum5_value); METAENUMS_TEST(MetaEnumTest_Enum6_value); METAENUMS_TEST(MetaEnumTest_Enum8_value); { using namespace Qt; METAENUMS_TEST(RichText); } #undef METAENUMS_TEST QTest::newRow("AlignBottom") << &testVariantMetaEnum << "AlignBottom"; constexpr auto AlignHCenterBottom = Qt::AlignmentFlag((Qt::AlignHCenter | Qt::AlignBottom).toInt()); QTest::newRow("AlignHCenter|AlignBottom") << &testVariantMetaEnum << "AlignHCenter|AlignBottom"; } void tst_QVariant::nullConvert() { // null variant with no initialized value QVariant nullVar {QMetaType::fromType()}; QVERIFY(nullVar.isValid()); QVERIFY(nullVar.isNull()); // We can not convert a variant with no value QVERIFY(!nullVar.convert(QMetaType::fromType())); QCOMPARE(nullVar.typeId(), QMetaType::QUrl); QVERIFY(nullVar.isNull()); } void tst_QVariant::accessSequentialContainerKey() { QString nameResult; { QMap mapping; QString name = QString::fromLatin1("Seven"); mapping.insert(name, nullptr); QVariant variant = QVariant::fromValue(mapping); QAssociativeIterable iterable = variant.value(); QAssociativeIterable::const_iterator iit = iterable.begin(); const QAssociativeIterable::const_iterator end = iterable.end(); for ( ; iit != end; ++iit) { nameResult += iit.key().toString(); } } // Destroy mapping // Regression test for QTBUG-52246 - no memory corruption/double deletion // of the string key. QCOMPARE(nameResult, QStringLiteral("Seven")); } void tst_QVariant::shouldDeleteVariantDataWorksForSequential() { QCOMPARE(instanceCount, 0); { QtMetaContainerPrivate::QMetaSequenceInterface metaSequence {}; metaSequence.iteratorCapabilities = QtMetaContainerPrivate::RandomAccessCapability | QtMetaContainerPrivate::BiDirectionalCapability | QtMetaContainerPrivate::ForwardCapability | QtMetaContainerPrivate::InputCapability; metaSequence.sizeFn = [](const void *) { return qsizetype(1); }; metaSequence.createConstIteratorFn = [](const void *, QtMetaContainerPrivate::QMetaSequenceInterface::Position) -> void* { return nullptr; }; metaSequence.addValueFn = [](void *, const void *, QtMetaContainerPrivate::QMetaSequenceInterface::Position) {}; metaSequence.advanceConstIteratorFn = [](void *, qsizetype) {}; metaSequence.destroyConstIteratorFn = [](const void *){}; metaSequence.compareConstIteratorFn = [](const void *, const void *) { return true; // all iterators are nullptr }; metaSequence.copyConstIteratorFn = [](void *, const void *){}; metaSequence.diffConstIteratorFn = [](const void *, const void *) -> qsizetype { return 0; }; metaSequence.valueAtIndexFn = [](const void *, qsizetype, void *dataPtr) -> void { MyType mytype {1, "eins"}; *static_cast(dataPtr) = mytype; }; metaSequence.valueAtConstIteratorFn = [](const void *, void *dataPtr) -> void { MyType mytype {2, "zwei"}; *static_cast(dataPtr) = mytype; }; metaSequence.valueMetaType = QtPrivate::qMetaTypeInterfaceForType(); QSequentialIterable iterable(QMetaSequence(&metaSequence), nullptr); QVariant value1 = iterable.at(0); QVERIFY(value1.canConvert()); QCOMPARE(value1.value().number, 1); QVariant value2 = *iterable.begin(); QVERIFY(value2.canConvert()); QCOMPARE(value2.value().number, 2); } QCOMPARE(instanceCount, 0); } void tst_QVariant::shouldDeleteVariantDataWorksForAssociative() { QCOMPARE(instanceCount, 0); { QtMetaContainerPrivate::QMetaAssociationInterface iterator {}; iterator.sizeFn = [](const void *) -> qsizetype {return 1;}; iterator.mappedMetaType = QtPrivate::qMetaTypeInterfaceForType(); iterator.keyMetaType = QtPrivate::qMetaTypeInterfaceForType(); iterator.createConstIteratorFn = []( const void *, QtMetaContainerPrivate::QMetaContainerInterface::Position) -> void * { return nullptr; }; iterator.advanceConstIteratorFn = [](void *, qsizetype) {}; iterator.destroyConstIteratorFn = [](const void *){}; iterator.compareConstIteratorFn = [](const void *, const void *) { return true; /*all iterators are nullptr*/ }; iterator.createConstIteratorAtKeyFn = [](const void *, const void *) -> void * { return reinterpret_cast(quintptr(42)); }; iterator.copyConstIteratorFn = [](void *, const void *) {}; iterator.diffConstIteratorFn = [](const void *, const void *) -> qsizetype { return 0; }; iterator.keyAtConstIteratorFn = [](const void *iterator, void *dataPtr) -> void { MyType mytype {1, "key"}; if (reinterpret_cast(iterator) == 42) { mytype.number = 42; mytype.text = "find_key"; } *static_cast(dataPtr) = mytype; }; iterator.mappedAtConstIteratorFn = [](const void *iterator, void *dataPtr) -> void { MyType mytype {2, "value"}; if (reinterpret_cast(iterator) == 42) { mytype.number = 42; mytype.text = "find_value"; } *static_cast(dataPtr) = mytype; }; QAssociativeIterable iterable(QMetaAssociation(&iterator), nullptr); auto it = iterable.begin(); QVariant value1 = it.key(); QVERIFY(value1.canConvert()); QCOMPARE(value1.value().number, 1); QCOMPARE(value1.value().text, "key"); QVariant value2 = it.value(); QVERIFY(value2.canConvert()); QCOMPARE(value2.value().number, 2); auto findIt = iterable.find(QVariant::fromValue(MyType {})); value1 = findIt.key(); QCOMPARE(value1.value().number, 42); QCOMPARE(value1.value().text, "find_key"); value2 = findIt.value(); QCOMPARE(value2.value().number, 42); QCOMPARE(value2.value().text, "find_value"); } QCOMPARE(instanceCount, 0); } void tst_QVariant::fromStdVariant() { #define CHECK_EQUAL(lhs, rhs, type) do { \ QCOMPARE(lhs.typeId(), rhs.typeId()); \ if (lhs.isNull()) { \ QVERIFY(rhs.isNull()); \ } else { \ QVERIFY(!rhs.isNull()); \ QCOMPARE(get< type >(lhs), get< type >(rhs)); \ } \ } while (false) { typedef std::variant intorbool_t; intorbool_t stdvar = 5; QVariant qvar = QVariant::fromStdVariant(stdvar); QVERIFY(!qvar.isNull()); QCOMPARE(qvar.typeId(), QMetaType::Int); QCOMPARE(qvar.value(), std::get(stdvar)); { const auto qv2 = QVariant::fromStdVariant(std::move(stdvar)); CHECK_EQUAL(qv2, qvar, int); } stdvar = true; qvar = QVariant::fromStdVariant(stdvar); QVERIFY(!qvar.isNull()); QCOMPARE(qvar.typeId(), QMetaType::Bool); QCOMPARE(qvar.value(), std::get(stdvar)); { const auto qv2 = QVariant::fromStdVariant(std::move(stdvar)); CHECK_EQUAL(qv2, qvar, bool); } } { std::variant stdvar; QVariant qvar = QVariant::fromStdVariant(stdvar); QVERIFY(!qvar.isValid()); { const auto qv2 = QVariant::fromStdVariant(std::move(stdvar)); CHECK_EQUAL(qv2, qvar, int); // fake type, they're empty } stdvar = -4; qvar = QVariant::fromStdVariant(stdvar); QVERIFY(!qvar.isNull()); QCOMPARE(qvar.typeId(), QMetaType::Int); QCOMPARE(qvar.value(), std::get(stdvar)); { const auto qv2 = QVariant::fromStdVariant(std::move(stdvar)); CHECK_EQUAL(qv2, qvar, int); } } { std::variant stdvar = QChar::fromLatin1(' '); QVariant qvar = QVariant::fromStdVariant(stdvar); QVERIFY(!qvar.isNull()); QCOMPARE(qvar.typeId(), QMetaType::QChar); QCOMPARE(qvar.value(), std::get(stdvar)); { const auto qv2 = QVariant::fromStdVariant(std::move(stdvar)); CHECK_EQUAL(qv2, qvar, QChar); } } // rvalue fromStdVariant() actually moves: { const auto foo = u"foo"_s; std::variant stdvar = foo; QVariant qvar = QVariant::fromStdVariant(std::move(stdvar)); const auto ps = get_if(&stdvar); QVERIFY(ps); QVERIFY(ps->isNull()); // QString was moved from QVERIFY(!qvar.isNull()); QCOMPARE(qvar.typeId(), QMetaType::QString); QCOMPARE(get(qvar), foo); } #undef CHECK_EQUAL } void tst_QVariant::qt4UuidDataStream() { QByteArray data; QDataStream stream(&data, QIODevice::WriteOnly); stream.setVersion(QDataStream::Qt_4_8); QUuid source(0x12345678,0x1234,0x1234,0x12,0x23,0x34,0x45,0x56,0x67,0x78,0x89); stream << QVariant::fromValue(source); const QByteArray qt4Data = QByteArray::fromHex("0000007f000000000651557569640012345678123412341223344556677889"); QCOMPARE(data, qt4Data); QDataStream input(&data, QIODevice::ReadOnly); input.setVersion(QDataStream::Qt_4_8); QVariant result; input >> result; QCOMPARE(result.value(), source); } void tst_QVariant::sequentialIterableEndianessSanityCheck() { namespace QMTP = QtMetaContainerPrivate; QMTP::IteratorCapabilities oldIteratorCaps = QMTP::InputCapability | QMTP::ForwardCapability | QMTP::BiDirectionalCapability | QMTP::RandomAccessCapability; QMTP::QMetaSequenceInterface seqImpl {}; QCOMPARE(seqImpl.revision, 0u); memcpy(&seqImpl.iteratorCapabilities, &oldIteratorCaps, sizeof(oldIteratorCaps)); QCOMPARE(seqImpl.revision, 0u); } void tst_QVariant::sequentialIterableAppend() { { QList container { 1, 2 }; auto variant = QVariant::fromValue(container); QVERIFY(variant.canConvert()); QSequentialIterable asIterable = variant.view(); const int i = 3, j = 4; void *mutableIterable = asIterable.mutableIterable(); asIterable.metaContainer().addValueAtEnd(mutableIterable, &i); asIterable.metaContainer().addValueAtEnd(mutableIterable, &j); QCOMPARE(variant.value>(), QList ({ 1, 2, 3, 4 })); asIterable.metaContainer().addValueAtBegin(mutableIterable, &i); asIterable.metaContainer().addValueAtBegin(mutableIterable, &j); QCOMPARE(variant.value>(), QList ({ 4, 3, 1, 2, 3, 4 })); asIterable.metaContainer().removeValueAtBegin(mutableIterable); QCOMPARE(variant.value>(), QList ({ 3, 1, 2, 3, 4 })); asIterable.metaContainer().removeValueAtEnd(mutableIterable); QCOMPARE(variant.value>(), QList ({ 3, 1, 2, 3 })); } { QSet container { QByteArray{"hello"}, QByteArray{"world"} }; auto variant = QVariant::fromValue(std::move(container)); QVERIFY(variant.canConvert()); QSequentialIterable asIterable = variant.view(); QByteArray qba1 {"goodbye"}; QByteArray qba2 { "moon" }; void *mutableIterable = asIterable.mutableIterable(); asIterable.metaContainer().addValue(mutableIterable, &qba1); asIterable.metaContainer().addValue(mutableIterable, &qba2); QSet reference { "hello", "world", "goodbye", "moon" }; QCOMPARE(variant.value>(), reference); asIterable.metaContainer().addValue(mutableIterable, &qba1); asIterable.metaContainer().addValue(mutableIterable, &qba2); QCOMPARE(variant.value>(), reference); } } void tst_QVariant::preferDirectConversionOverInterfaces() { using namespace QtMetaTypePrivate; bool calledCorrectConverter = false; QMetaType::registerConverter([](const MyType &) { return QSequentialIterable {}; }); QMetaType::registerConverter([&calledCorrectConverter](const MyType &) { calledCorrectConverter = true; return QVariantList {}; }); QMetaType::registerConverter([](const MyType &) { return QAssociativeIterable {}; }); QMetaType::registerConverter([&calledCorrectConverter](const MyType &) { calledCorrectConverter = true; return QVariantHash {}; }); QMetaType::registerConverter([&calledCorrectConverter](const MyType &) { calledCorrectConverter = true; return QVariantMap {}; }); auto holder = QVariant::fromValue(MyType {}); QVERIFY(holder.canConvert()); QVERIFY(holder.canConvert()); QVERIFY(holder.canConvert()); QVERIFY(holder.canConvert()); QVERIFY(holder.canConvert()); holder.value(); QVERIFY(calledCorrectConverter); calledCorrectConverter = false; holder.value(); QVERIFY(calledCorrectConverter); calledCorrectConverter = false; holder.value(); QVERIFY(calledCorrectConverter); } struct MyTypeView { MyType *data; }; void tst_QVariant::mutableView() { bool calledView = false; const bool success = QMetaType::registerMutableView([&](MyType &data) { calledView = true; return MyTypeView { &data }; }); QVERIFY(success); QTest::ignoreMessage( QtWarningMsg, "Mutable view on type already registered from type MyType to type MyTypeView"); const bool shouldFail = QMetaType::registerMutableView([&](MyType &) { return MyTypeView { nullptr }; }); QVERIFY(!shouldFail); auto original = QVariant::fromValue(MyType {}); QVERIFY(original.canView()); QVERIFY(!original.canConvert()); MyTypeView view = original.view(); QVERIFY(calledView); const char *txt = "lll"; view.data->number = 113; view.data->text = txt; MyType extracted = original.view(); QCOMPARE(extracted.number, 0); QCOMPARE(extracted.text, nullptr); } template void tst_QVariant::canViewAndView_ReturnFalseAndDefault_WhenConvertingBetweenPointerAndValue_impl( const QByteArray &typeName) { T instance{}; // Value -> Pointer QVariant value = QVariant::fromValue(instance); QVERIFY2(!value.canView(), typeName); QCOMPARE(value.view(), nullptr); // Expect default constructed pointer // Pointer -> Value QVariant pointer = QVariant::fromValue(&instance); QVERIFY2(!pointer.canView(), typeName); QCOMPARE(pointer.view(), T{}); // Expect default constructed. Note: Weak test since instance // is default constructed, but we detect data corruption } void tst_QVariant::canViewAndView_ReturnFalseAndDefault_WhenConvertingBetweenPointerAndValue() { #define ADD_TEST_IMPL(typeName, typeNameId, realType) \ canViewAndView_ReturnFalseAndDefault_WhenConvertingBetweenPointerAndValue_impl( \ #typeName); // Add tests for static primitive types QT_FOR_EACH_STATIC_PRIMITIVE_NON_VOID_TYPE(ADD_TEST_IMPL) // Add tests for static core types QT_FOR_EACH_STATIC_CORE_CLASS(ADD_TEST_IMPL) #undef ADD_TEST_IMPL } struct MoveTester { bool wasMoved = false; MoveTester() = default; MoveTester(const MoveTester &) {}; // non-trivial on purpose MoveTester(MoveTester &&other) { other.wasMoved = true; } MoveTester& operator=(const MoveTester &) = default; MoveTester& operator=(MoveTester &&other) {other.wasMoved = true; return *this;} }; void tst_QVariant::moveOperations() { { QString str = "Hello"; auto v = QVariant::fromValue(str); QVariant v2(std::move(v)); QCOMPARE(v2.toString(), str); v = QVariant::fromValue(str); v2 = std::move(v); QCOMPARE(v2.toString(), str); } std::list list; auto v = QVariant::fromValue(list); QVariant v2(std::move(v)); QVERIFY(v2.value>() == list); v = QVariant::fromValue(list); v2 = std::move(v); QVERIFY(v2.value>() == list); { MoveTester tester; QVariant::fromValue(tester); QVERIFY(!tester.wasMoved); QVariant::fromValue(std::move(tester)); QVERIFY(tester.wasMoved); } { const MoveTester tester; QVariant::fromValue(std::move(tester)); QVERIFY(!tester.wasMoved); // we don't want to move from const variables } { QVariant var(std::in_place_type); const auto p = get_if(&var); QVERIFY(p); auto &tester = *p; QVERIFY(!tester.wasMoved); [[maybe_unused]] auto copy = var.value(); QVERIFY(!tester.wasMoved); [[maybe_unused]] auto moved = std::move(var).value(); QVERIFY(tester.wasMoved); } } class NoMetaObject : public QObject {}; void tst_QVariant::equalsWithoutMetaObject() { using T = NoMetaObject*; QtPrivate::QMetaTypeInterface d = { /*.revision=*/ 0, /*.alignment=*/ alignof(T), /*.size=*/ sizeof(T), /*.flags=*/ QtPrivate::QMetaTypeTypeFlags::Flags, /*.typeId=*/ 0, /*.metaObject=*/ nullptr, // on purpose. /*.name=*/ "NoMetaObject*", /*.defaultCtr=*/ [](const QtPrivate::QMetaTypeInterface *, void *addr) { new (addr) T(); }, /*.copyCtr=*/ [](const QtPrivate::QMetaTypeInterface *, void *addr, const void *other) { new (addr) T(*reinterpret_cast(other)); }, /*.moveCtr=*/ [](const QtPrivate::QMetaTypeInterface *, void *addr, void *other) { new (addr) T(std::move(*reinterpret_cast(other))); }, /*.dtor=*/ [](const QtPrivate::QMetaTypeInterface *, void *addr) { reinterpret_cast(addr)->~T(); }, /*.equals*/ nullptr, /*.lessThan*/ nullptr, /*.debugStream=*/ nullptr, /*.dataStreamOut=*/ nullptr, /*.dataStreamIn=*/ nullptr, /*.legacyRegisterOp=*/ nullptr }; QMetaType noMetaObjectMetaType(&d); QMetaType qobjectMetaType = QMetaType::fromType(); QVERIFY(noMetaObjectMetaType.flags() & QMetaType::PointerToQObject); QVERIFY(qobjectMetaType.flags() & QMetaType::PointerToQObject); QVariant noMetaObjectVariant(noMetaObjectMetaType, nullptr); QVariant qobjectVariant(qobjectMetaType, nullptr); // Shouldn't crash QVERIFY(noMetaObjectVariant != qobjectVariant); QVERIFY(qobjectVariant != noMetaObjectVariant); } struct NonDefaultConstructible { NonDefaultConstructible(int i) :i(i) {} int i; friend bool operator==(NonDefaultConstructible l, NonDefaultConstructible r) { return l.i == r.i; } }; template <> char *QTest::toString(const NonDefaultConstructible &ndc) { return qstrdup('{' + QByteArray::number(ndc.i) + '}'); } struct Indestructible { Indestructible() {} Indestructible(const Indestructible &) {} Indestructible &operator=(const Indestructible &) { return *this; } private: ~Indestructible() {} }; struct NotCopyable { NotCopyable() = default; NotCopyable(const NotCopyable&) = delete; NotCopyable &operator=(const NotCopyable &) = delete; }; void tst_QVariant::constructFromIncompatibleMetaType_data() { QTest::addColumn("type"); auto addRow = [](QMetaType meta) { QTest::newRow(meta.name()) << meta; }; addRow(QMetaType::fromType()); addRow(QMetaType::fromType()); addRow(QMetaType::fromType()); addRow(QMetaType::fromType()); addRow(QMetaType::fromType()); } void tst_QVariant::constructFromIncompatibleMetaType() { QFETCH(QMetaType, type); const auto anticipate = [type]() { // In that case, we run into a different condition (size == 0), and do not warn if (type == QMetaType::fromType()) { QTest::ignoreMessage(QtWarningMsg, "QVariant: Cannot create type 'NonDefaultConstructible' without a " "default constructor"); } else if (type != QMetaType::fromType()) { QTest::ignoreMessage( QtWarningMsg, "QVariant: Provided metatype for '" + QByteArray(type.name()) + "' does not support destruction and copy construction"); } }; anticipate(); QVariant var(type, nullptr); QVERIFY(!var.isValid()); QVERIFY(!var.metaType().isValid()); anticipate(); QVariant regular(1.0); QVERIFY(!var.canView(type)); QVERIFY(!var.canConvert(type)); QVERIFY(!QVariant(regular).convert(type)); } void tst_QVariant::constructFromQtLT65MetaType() { auto qsizeIface = QtPrivate::qMetaTypeInterfaceForType(); QtPrivate::QMetaTypeInterface qsize64Iface = { /*revision*/0, 8, 8, QMetaType::NeedsConstruction | QMetaType::NeedsDestruction, 0, qsizeIface->metaObjectFn, "FakeQSize", qsizeIface->defaultCtr, qsizeIface->copyCtr, qsizeIface->moveCtr, /*dtor =*/ nullptr, qsizeIface->equals, qsizeIface->lessThan, qsizeIface->debugStream, qsizeIface->dataStreamOut, qsizeIface->dataStreamIn, /*legacyregop =*/ nullptr }; QVariant var{ QMetaType(&qsize64Iface) }; QVERIFY(var.isValid()); } void tst_QVariant::copyNonDefaultConstructible() { NonDefaultConstructible ndc(42); QVariant var = QVariant::fromValue(ndc); QVERIFY(var.isDetached()); QCOMPARE(var.metaType(), QMetaType::fromType()); QVERIFY(var.constData() != &ndc); // qvariant_cast and QVariant::value don't compile QCOMPARE(get(std::as_const(var)), ndc); QVariant var2 = var; var2.detach(); // force another copy QVERIFY(var2.isDetached()); QVERIFY(var2.constData() != var.constData()); QCOMPARE(get(std::as_const(var2)), get(std::as_const(var))); QCOMPARE(var2, var); } void tst_QVariant::inplaceConstruct() { { NonDefaultConstructible ndc(42); QVariant var(std::in_place_type, 42); QVERIFY(get_if(&var)); QCOMPARE(get(var), ndc); } { std::vector vec {1, 2, 3, 4}; QVariant var(std::in_place_type>, {1, 2, 3, 4}); QVERIFY(get_if>(&var)); QCOMPARE(get>(var), vec); } } struct LargerThanInternalQVariantStorage { char data[6 * sizeof(void *)]; }; struct alignas(256) LargerThanInternalQVariantStorageOveraligned { char data[6 * sizeof(void *)]; }; struct alignas(128) SmallerAlignmentEvenLargerSize { char data[17 * sizeof(void *)]; }; void tst_QVariant::emplace() { { // can emplace non default constructible + can emplace on null variant NonDefaultConstructible ndc(42); QVariant var; var.emplace(42); QVERIFY(get_if(&var)); QCOMPARE(get(var), ndc); } { // can emplace using ctor taking initializer_list QVariant var; var.emplace>({0, 1, 2, 3, 4}); auto vecPtr = get_if>(&var); QVERIFY(vecPtr); QCOMPARE(vecPtr->size(), 5U); for (int i = 0; i < 5; ++i) QCOMPARE(vecPtr->at(size_t(i)), i); } // prequisites for the test QCOMPARE_LE(sizeof(std::vector), sizeof(std::string)); QCOMPARE(alignof(std::vector), alignof(std::string)); { // emplace can reuse storage auto var = QVariant::fromValue(std::string{}); QVERIFY(var.data_ptr().is_shared); auto data = var.constData(); std::vector &vec = var.emplace>(3, 42); /* alignment is the same, so the pointer is exactly the same; no offset change */ auto expected = std::vector{42, 42, 42}; QCOMPARE(get_if>(&var), &vec); QCOMPARE(get>(var), expected); QCOMPARE(var.constData(), data); } { // emplace can't reuse storage if the variant is shared auto var = QVariant::fromValue(std::string{}); [[maybe_unused]] QVariant causesSharing = var; QVERIFY(var.data_ptr().is_shared); auto data = var.constData(); var.emplace>(3, 42); auto expected = std::vector{42, 42, 42}; QVERIFY(get_if>(&var)); QCOMPARE(get>(var), expected); QCOMPARE_NE(var.constData(), data); } { // emplace puts element into the correct place - non-shared QVERIFY(QVariant::Private::canUseInternalSpace(QMetaType::fromType().iface())); QVariant var; var.emplace(QChar('x')); QVERIFY(!var.data_ptr().is_shared); } { // emplace puts element into the correct place - shared QVERIFY(!QVariant::Private::canUseInternalSpace(QMetaType::fromType().iface())); QVariant var; var.emplace(42, 'x'); QVERIFY(var.data_ptr().is_shared); } { // emplace does not reuse the storage if alignment is too large auto iface = QMetaType::fromType().iface(); QVERIFY(!QVariant::Private::canUseInternalSpace(iface)); auto var = QVariant::fromValue(LargerThanInternalQVariantStorage{}); auto data = var.constData(); var.emplace(); QCOMPARE_NE(var.constData(), data); } { // emplace does reuse the storage if new alignment and size are together small enough auto iface = QMetaType::fromType().iface(); QVERIFY(!QVariant::Private::canUseInternalSpace(iface)); auto var = QVariant::fromValue(LargerThanInternalQVariantStorageOveraligned{}); auto data = var.constData(); var.emplace(); // no exact match below - the alignment is after all different QCOMPARE_LE(quintptr(var.constData()), quintptr(data)); QCOMPARE_LE(quintptr(var.constData()), quintptr(data) + sizeof(LargerThanInternalQVariantStorageOveraligned)); } } void tst_QVariant::getIf_NonDefaultConstructible() { getIf_impl(NonDefaultConstructible{42}); } void tst_QVariant::getIfSpecial() { QVariant v{QString{}}; // used to be a null QVariant in Qt 5 QCOMPARE_NE(get_if(&v), nullptr); // not anymore... } void tst_QVariant::get_NonDefaultConstructible() { get_impl(NonDefaultConstructible{42}); } template T mutate(const T &t) { return t + t; } template <> NonDefaultConstructible mutate(const NonDefaultConstructible &t) { return NonDefaultConstructible{t.i + t.i}; } template QVariant make_null_QVariant_of_type() { return QVariant(QMetaType::fromType()); } template void tst_QVariant::getIf_impl(T t) const { QVariant v = QVariant::fromValue(t); QVariant null; QVERIFY(null.isNull()); [[maybe_unused]] QVariant nulT; if constexpr (std::is_default_constructible_v) { // typed null QVariants don't work with non-default-constuctable types nulT = make_null_QVariant_of_type(); QVERIFY(nulT.isNull()); } QVariant date = QDate(2023, 3, 3); static_assert(!std::is_same_v); // for behavioral comparison: StdVariant stdn = {}, stdv = t; // returns nullptr on type mismatch: { // const QCOMPARE_EQ(get_if(&std::as_const(stdn)), nullptr); QCOMPARE_EQ(get_if(&std::as_const(date)), nullptr); // mutable QCOMPARE_EQ(get_if(&stdn), nullptr); QCOMPARE_EQ(get_if(&date), nullptr); } // returns nullptr on null variant (QVariant only): { QCOMPARE_EQ(get_if(&std::as_const(null)), nullptr); QCOMPARE_EQ(get_if(&null), nullptr); if constexpr (std::is_default_constructible_v) { // const access return nullptr QCOMPARE_EQ(get_if(&std::as_const(nulT)), nullptr); // but mutable access makes typed null QVariants non-null (like data()) QCOMPARE_NE(get_if(&nulT), nullptr); QVERIFY(!nulT.isNull()); nulT = make_null_QVariant_of_type(); // reset to null state } } // const access: { auto ps = get_if(&std::as_const(stdv)); static_assert(std::is_same_v); QCOMPARE_NE(ps, nullptr); QCOMPARE_EQ(*ps, t); auto pv = get_if(&std::as_const(v)); static_assert(std::is_same_v); QCOMPARE_NE(pv, nullptr); QCOMPARE_EQ(*pv, t); } // mutable access: { T t2 = mutate(t); auto ps = get_if(&stdv); static_assert(std::is_same_v); QCOMPARE_NE(ps, nullptr); QCOMPARE_EQ(*ps, t); *ps = t2; auto ps2 = get_if(&stdv); QCOMPARE_NE(ps2, nullptr); QCOMPARE_EQ(*ps2, t2); auto pv = get_if(&v); static_assert(std::is_same_v); QCOMPARE_NE(pv, nullptr); QCOMPARE_EQ(*pv, t); *pv = t2; auto pv2 = get_if(&v); QCOMPARE_NE(pv2, nullptr); QCOMPARE_EQ(*pv2, t2); // typed null QVariants become non-null (data() behavior): if constexpr (std::is_default_constructible_v) { QVERIFY(nulT.isNull()); auto pn = get_if(&nulT); QVERIFY(!nulT.isNull()); static_assert(std::is_same_v); QCOMPARE_NE(pn, nullptr); QCOMPARE_EQ(*pn, T{}); *pn = t2; auto pn2 = get_if(&nulT); QCOMPARE_NE(pn2, nullptr); QCOMPARE_EQ(*pn2, t2); } } } template void tst_QVariant::get_impl(T t) const { QVariant v = QVariant::fromValue(t); // for behavioral comparison: StdVariant stdv = t; #define FOR_EACH_CVREF(op) \ op(/*unadorned*/, &&) \ op(&, &) \ op(&&, &&) \ op(const, const &&) \ op(const &, const &) \ op(const &&, const &&) \ /* end */ #define CHECK_RETURN_TYPE_OF(Variant, cvref_in, cvref_out) \ static_assert(std::is_same_v< \ decltype(get(std::declval())), \ T cvref_out \ >); \ /* end */ #define CHECK_RETURN_TYPE(cvref_in, cvref_out) \ CHECK_RETURN_TYPE_OF(StdVariant, cvref_in, cvref_out) \ CHECK_RETURN_TYPE_OF(QVariant, cvref_in, cvref_out) \ /* end */ FOR_EACH_CVREF(CHECK_RETURN_TYPE) #undef CHECK_RETURN_TYPE #undef FOR_EACH_CVREF // const access: { auto &&rs = get(std::as_const(stdv)); QCOMPARE_EQ(rs, t); auto &&rv = get(std::as_const(v)); QCOMPARE_EQ(rv, t); } // mutable access: { T t2 = mutate(t); auto &&rs = get(stdv); QCOMPARE_EQ(rs, t); rs = t2; auto &&rs2 = get(stdv); QCOMPARE_EQ(rs2, t2); auto &&rv = get(v); QCOMPARE_EQ(rv, t); rv = t2; auto &&rv2 = get(v); QCOMPARE_EQ(rv2, t2); } } QTEST_MAIN(tst_QVariant) #include "tst_qvariant.moc"