/**************************************************************************** ** ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/ ** ** This file is part of the test suite of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** GNU Lesser General Public License Usage ** This file may be used under the terms of the GNU Lesser General Public ** License version 2.1 as published by the Free Software Foundation and ** appearing in the file LICENSE.LGPL included in the packaging of this ** file. Please review the following information to ensure the GNU Lesser ** General Public License version 2.1 requirements will be met: ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU General ** Public License version 3.0 as published by the Free Software Foundation ** and appearing in the file LICENSE.GPL included in the packaging of this ** file. Please review the following information to ensure the GNU General ** Public License version 3.0 requirements will be met: ** http://www.gnu.org/copyleft/gpl.html. ** ** Other Usage ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** ** ** ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #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 Q_DECLARE_METATYPE(qlonglong) Q_DECLARE_METATYPE(qulonglong) Q_DECLARE_METATYPE(QPointF) Q_DECLARE_METATYPE(QRectF) Q_DECLARE_METATYPE(QSize) Q_DECLARE_METATYPE(QSizeF) Q_DECLARE_METATYPE(QLine) Q_DECLARE_METATYPE(QLineF) Q_DECLARE_METATYPE(QPoint) Q_DECLARE_METATYPE(QRect) Q_DECLARE_METATYPE(QPixmap) Q_DECLARE_METATYPE(QBrush) Q_DECLARE_METATYPE(QFont) Q_DECLARE_METATYPE(QColor) Q_DECLARE_METATYPE(QKeySequence) class CustomNonQObject; class tst_QVariant : public QObject { Q_OBJECT public: tst_QVariant(QObject *parent = 0) : QObject(parent), customNonQObjectPointer(0) { } 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 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 toPointF_data(); void toPointF(); void toFont_data(); void toFont(); void toKeySequence_data(); void toKeySequence(); void toRectF_data(); void toRectF(); void toColor_data(); void toColor(); void toPixmap_data(); void toPixmap(); void toImage_data(); void toImage(); void toBrush_data(); void toBrush(); void qvariant_cast_QObject_data(); void qvariant_cast_QObject(); void qvariant_cast_QObject_derived(); void toLocale(); void toRegExp(); void toRegularExpression(); void matrix(); void transform(); void matrix4x4(); void vector2D(); void vector3D(); void vector4D(); void quaternion(); 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(); void operator_eq_eq_rhs(); void typeName_data(); void typeName(); void typeToName(); void streamInvalidVariant(); void podUserType(); void data(); void constData(); void saveLoadCustomTypes(); void globalColor(); void variantMap(); void variantHash(); void convertToQUint8() const; void invalidQColor() 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 convertBoolToByteArray() const; void convertBoolToByteArray_data() const; void convertByteArrayToBool() const; void convertByteArrayToBool_data() const; void toIntFromQString() const; void toIntFromDouble() const; void setValue(); void numericalConvert(); void moreCustomTypes(); void movabilityTest(); void variantInVariant(); void colorInteger(); void forwardDeclare(); void debugStream_data(); void debugStream(); void debugStreamType_data(); void debugStreamType(); void loadQt4Stream_data(); void loadQt4Stream(); void saveQt4Stream_data(); void saveQt4Stream(); void loadQt5Stream_data(); void loadQt5Stream(); void saveQt5Stream_data(); void saveQt5Stream(); void guiVariantAtExit(); void widgetsVariantAtExit(); private: void dataStream_data(QDataStream::Version version); void loadQVariantFromDataStream(QDataStream::Version version); void saveQVariantFromDataStream(QDataStream::Version version); CustomNonQObject *customNonQObjectPointer; QVector objectPointerTestData; }; Q_DECLARE_METATYPE(QDate) Q_DECLARE_METATYPE(QTime) Q_DECLARE_METATYPE(QDateTime) Q_DECLARE_METATYPE(QVariant) 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(QVariant::String); QCOMPARE(var3.typeName(), "QString"); QVERIFY(var3.isNull()); QVERIFY(var3.isValid()); QVariant var4(QVariant::Invalid); QCOMPARE(var4.type(), QVariant::Invalid); QVERIFY(var4.isNull()); QVERIFY(!var4.isValid()); QVariant var5(QLatin1String("hallo")); QCOMPARE(var5.type(), QVariant::String); QCOMPARE(var5.typeName(), "QString"); QVariant var6(qlonglong(0)); QCOMPARE(var6.type(), QVariant::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)); QTest::newRow("LastGuiType + 1") << uint(QMetaType::LastGuiType + 1); QVERIFY(!QMetaType::isRegistered(QMetaType::LastGuiType + 1)); QTest::newRow("LastWidgetsType + 1") << uint(QMetaType::LastWidgetsType + 1); QVERIFY(!QMetaType::isRegistered(QMetaType::LastWidgetsType + 1)); } struct MessageHandlerInvalidType { MessageHandlerInvalidType() : oldMsgHandler(qInstallMsgHandler(handler)) { ok = false; } ~MessageHandlerInvalidType() { qInstallMsgHandler(oldMsgHandler); } QtMsgHandler oldMsgHandler; static void handler(QtMsgType type, const char *txt) { QString msg = QString::fromLatin1(txt); // uint(-1) can be platform dependent so we check only beginning of the message. ok = msg.startsWith("Trying to construct an instance of an invalid type, type id:"); QVERIFY2(ok, (QString::fromLatin1("Message is not started correctly: '") + msg + '\'').toLatin1().constData()); } static bool ok; }; bool MessageHandlerInvalidType::ok; void tst_QVariant::constructor_invalid() { QFETCH(uint, typeId); { MessageHandlerInvalidType msg; QVariant variant(static_cast(typeId)); QVERIFY(!variant.isValid()); QVERIFY(variant.userType() == QMetaType::UnknownType); QVERIFY(msg.ok); } { MessageHandlerInvalidType msg; QVariant variant(typeId, /* copy */ 0); QVERIFY(!variant.isValid()); QVERIFY(variant.userType() == QMetaType::UnknownType); QVERIFY(msg.ok); } } void tst_QVariant::copy_constructor() { QVariant var7(QVariant::Int); QVariant var8(var7); QCOMPARE(var8.type(), QVariant::Int); QVERIFY(var8.isNull()); } void tst_QVariant::isNull() { QVariant var; QVERIFY( var.isNull() ); QString str1; QVariant var1( str1 ); QVERIFY( var1.isNull() ); QVariant var2( QString::null ); QVERIFY( var2.isNull() ); QVariant var3( QString( "blah" ) ); 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( QVariant::String ); QVERIFY( var6.isNull() ); QVariant varLL( (qlonglong)0 ); QVERIFY( !varLL.isNull() ); QVariant var7(QString::null); QVERIFY(var7.isNull()); } void tst_QVariant::swap() { QVariant v1 = 1, v2 = 2.0; v1.swap(v2); QCOMPARE(v1.type(),QVariant::Double); QCOMPARE(v1.toDouble(),2.0); QCOMPARE(v2.type(),QVariant::Int); QCOMPARE(v2.toInt(),1); } void tst_QVariant::canConvert_data() { QTest::addColumn("val"); QTest::addColumn("BitArrayCast"); QTest::addColumn("BitmapCast"); QTest::addColumn("BoolCast"); QTest::addColumn("BrushCast"); QTest::addColumn("ByteArrayCast"); QTest::addColumn("ColorCast"); QTest::addColumn("CursorCast"); QTest::addColumn("DateCast"); QTest::addColumn("DateTimeCast"); QTest::addColumn("DoubleCast"); QTest::addColumn("FontCast"); QTest::addColumn("ImageCast"); QTest::addColumn("IntCast"); QTest::addColumn("InvalidCast"); QTest::addColumn("KeySequenceCast"); QTest::addColumn("ListCast"); QTest::addColumn("LongLongCast"); QTest::addColumn("MapCast"); QTest::addColumn("PaletteCast"); QTest::addColumn("PenCast"); QTest::addColumn("PixmapCast"); QTest::addColumn("PointCast"); QTest::addColumn("RectCast"); QTest::addColumn("RegionCast"); QTest::addColumn("SizeCast"); QTest::addColumn("SizePolicyCast"); QTest::addColumn("StringCast"); QTest::addColumn("StringListCast"); QTest::addColumn("TimeCast"); QTest::addColumn("UIntCast"); QTest::addColumn("ULongLongCast"); #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 = qVariantFromValue(QBitmap()); QTest::newRow("Bitmap") << var << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N; var = qVariantFromValue(QBrush()); QTest::newRow("Brush") << var << N << N << N << Y << N << Y << 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; var = QVariant(QByteArray()); QTest::newRow("ByteArray") << var << N << N << Y << N << Y << Y << 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 = qVariantFromValue(QColor()); QTest::newRow("Color") << var << N << N << N << Y << Y << Y << 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; #ifndef QT_NO_CURSOR var = qVariantFromValue(QCursor()); QTest::newRow("Cursor") << var << N << N << N << N << N << N << 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; #endif 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 = qVariantFromValue(QFont()); QTest::newRow("Font") << var << 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 << Y << N << N << N << N; var = qVariantFromValue(QIcon()); QTest::newRow("Icon") << 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 = qVariantFromValue(QImage()); QTest::newRow("Image") << var << N << Y << N << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N; 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 << Y << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N << N; var = qVariantFromValue(QKeySequence()); QTest::newRow("KeySequence") << var << N << N << N << N << N << N << N << N << N << N << N << N << Y << N << Y << N << N << N << N << N << N << N << N << N << N << N << Y << 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 = qVariantFromValue(QPalette()); QTest::newRow("Palette") << var << 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 << N << N << N; var = qVariantFromValue(QPen()); QTest::newRow("Pen") << var << 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 << N << N; var = qVariantFromValue(QPixmap()); QTest::newRow("Pixmap") << var << N << Y << N << Y << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << Y << N << N << N << N << N << N << N << N << N << N; var = qVariantFromValue(QPolygon()); QTest::newRow("PointArray") << 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(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 = qVariantFromValue(QRegion()); QTest::newRow("Region") << var << 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 << 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 = qVariantFromValue(QSizePolicy()); QTest::newRow("SizePolicy") << 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 << Y << 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 << N << 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; #undef N #undef Y } void tst_QVariant::canConvert() { QFETCH(QVariant, val); QFETCH(bool, BitArrayCast); QFETCH(bool, BitmapCast); QFETCH(bool, BoolCast); QFETCH(bool, BrushCast); QFETCH(bool, ByteArrayCast); QFETCH(bool, ColorCast); QFETCH(bool, CursorCast); QFETCH(bool, DateCast); QFETCH(bool, DateTimeCast); QFETCH(bool, DoubleCast); QFETCH(bool, FontCast); QFETCH(bool, ImageCast); QFETCH(bool, IntCast); QFETCH(bool, InvalidCast); QFETCH(bool, KeySequenceCast); QFETCH(bool, ListCast); QFETCH(bool, LongLongCast); QFETCH(bool, MapCast); QFETCH(bool, PaletteCast); QFETCH(bool, PenCast); QFETCH(bool, PixmapCast); QFETCH(bool, PointCast); QFETCH(bool, RectCast); QFETCH(bool, RegionCast); QFETCH(bool, SizeCast); QFETCH(bool, SizePolicyCast); QFETCH(bool, StringCast); QFETCH(bool, StringListCast); QFETCH(bool, TimeCast); QFETCH(bool, UIntCast); QFETCH(bool, ULongLongCast); QCOMPARE(val.canConvert(QVariant::BitArray), BitArrayCast); QCOMPARE(val.canConvert(QVariant::Bitmap), BitmapCast); QCOMPARE(val.canConvert(QVariant::Bool), BoolCast); QCOMPARE(val.canConvert(QVariant::Brush), BrushCast); QCOMPARE(val.canConvert(QVariant::ByteArray), ByteArrayCast); QCOMPARE(val.canConvert(QVariant::Color), ColorCast); QCOMPARE(val.canConvert(QVariant::Cursor), CursorCast); QCOMPARE(val.canConvert(QVariant::Date), DateCast); QCOMPARE(val.canConvert(QVariant::DateTime), DateTimeCast); QCOMPARE(val.canConvert(QVariant::Double), DoubleCast); QCOMPARE(val.canConvert(QVariant::Type(QMetaType::Float)), DoubleCast); QCOMPARE(val.canConvert(QVariant::Font), FontCast); QCOMPARE(val.canConvert(QVariant::Image), ImageCast); QCOMPARE(val.canConvert(QVariant::Int), IntCast); QCOMPARE(val.canConvert(QVariant::Invalid), InvalidCast); QCOMPARE(val.canConvert(QVariant::KeySequence), KeySequenceCast); QCOMPARE(val.canConvert(QVariant::List), ListCast); QCOMPARE(val.canConvert(QVariant::LongLong), LongLongCast); QCOMPARE(val.canConvert(QVariant::Map), MapCast); QCOMPARE(val.canConvert(QVariant::Palette), PaletteCast); QCOMPARE(val.canConvert(QVariant::Pen), PenCast); QCOMPARE(val.canConvert(QVariant::Pixmap), PixmapCast); QCOMPARE(val.canConvert(QVariant::Point), PointCast); QCOMPARE(val.canConvert(QVariant::Rect), RectCast); QCOMPARE(val.canConvert(QVariant::Region), RegionCast); QCOMPARE(val.canConvert(QVariant::Size), SizeCast); QCOMPARE(val.canConvert(QVariant::SizePolicy), SizePolicyCast); QCOMPARE(val.canConvert(QVariant::String), StringCast); QCOMPARE(val.canConvert(QVariant::StringList), StringListCast); QCOMPARE(val.canConvert(QVariant::Time), TimeCast); QCOMPARE(val.canConvert(QVariant::UInt), UIntCast); QCOMPARE(val.canConvert(QVariant::ULongLong), ULongLongCast); // Invalid type ids QCOMPARE(val.canConvert(-1), false); QCOMPARE(val.canConvert(-23), false); QCOMPARE(val.canConvert(-23876), false); QCOMPARE(val.canConvert(23876), false); } 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( "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; QTest::newRow( "keysequence" ) << qVariantFromValue( QKeySequence( Qt::Key_A ) ) << 65 << 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; } void tst_QVariant::toInt() { QFETCH( QVariant, value ); QFETCH( int, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() == value.canConvert( QVariant::Int ) ); 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( "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; } void tst_QVariant::toUInt() { QFETCH( QVariant, value ); QFETCH( uint, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::UInt ) ); 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() ); QVERIFY( value.canConvert( QVariant::Size ) ); 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() ); QVERIFY( value.canConvert( QVariant::SizeF ) ); 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() ); QVERIFY( value.canConvert( QVariant::Line ) ); 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() ); QVERIFY( value.canConvert( QVariant::LineF ) ); 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() ); QVERIFY( value.canConvert( QVariant::Point ) ); 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() ); QVERIFY( value.canConvert( QVariant::Rect ) ); 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() ); QVERIFY( value.canConvert( QVariant::Char ) ); 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; } void tst_QVariant::toBool() { QFETCH( QVariant, value ); QFETCH( bool, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Bool ) ); 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() ); QVERIFY( value.canConvert( QVariant::PointF ) ); 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() ); QVERIFY( value.canConvert( QVariant::RectF ) ); QRectF d = value.toRectF(); QCOMPARE( d, result ); } void tst_QVariant::toColor_data() { QTest::addColumn("value"); QTest::addColumn("result"); QColor c("red"); QTest::newRow( "string" ) << QVariant( QString( "red" ) ) << c; QTest::newRow( "solid brush" ) << QVariant( QBrush(c) ) << c; } void tst_QVariant::toColor() { QFETCH( QVariant, value ); QFETCH( QColor, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Color ) ); QColor d = qVariantValue(value); QCOMPARE( d, result ); } void tst_QVariant::toPixmap_data() { QTest::addColumn("value"); QTest::addColumn("result"); QPixmap pm(30, 30); pm.fill(Qt::red); QTest::newRow( "image" ) << QVariant( pm ) << pm; QBitmap bm(30, 30); bm.fill(Qt::color1); QTest::newRow( "bitmap" ) << QVariant( bm ) << QPixmap(bm); } void tst_QVariant::toPixmap() { QFETCH( QVariant, value ); QFETCH( QPixmap, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Pixmap ) ); QPixmap d = qVariantValue(value); QCOMPARE( d, result ); } void tst_QVariant::toImage_data() { QTest::addColumn("value"); QTest::addColumn("result"); QImage im(30, 30, QImage::Format_ARGB32); im.fill(0x7fff0000); QTest::newRow( "image" ) << QVariant( im ) << im; } void tst_QVariant::toImage() { QFETCH( QVariant, value ); QFETCH( QImage, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Image ) ); QImage d = qVariantValue(value); QCOMPARE( d, result ); } void tst_QVariant::toBrush_data() { QTest::addColumn("value"); QTest::addColumn("result"); QColor c(Qt::red); QTest::newRow( "color" ) << QVariant( c ) << QBrush(c); QPixmap pm(30, 30); pm.fill(c); QTest::newRow( "pixmap" ) << QVariant( pm ) << QBrush(pm); } void tst_QVariant::toBrush() { QFETCH( QVariant, value ); QFETCH( QBrush, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Brush ) ); QBrush d = qVariantValue(value); QCOMPARE( d, result ); } void tst_QVariant::toFont_data() { QTest::addColumn("value"); QTest::addColumn("result"); QFont f("times",12,-1,false); QTest::newRow( "string" ) << QVariant( QString( "times,12,-1,5,50,0,0,0,0,0" ) ) << f; } void tst_QVariant::toFont() { QFETCH( QVariant, value ); QFETCH( QFont, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Font ) ); QFont d = qVariantValue(value); QCOMPARE( d, result ); } void tst_QVariant::toKeySequence_data() { QTest::addColumn("value"); QTest::addColumn("result"); QTest::newRow( "int" ) << QVariant( 67108929 ) << QKeySequence( Qt::CTRL + Qt::Key_A ); QTest::newRow( "qstring" ) << QVariant( QString( "Ctrl+A" ) ) << QKeySequence( Qt::CTRL + Qt::Key_A ); } void tst_QVariant::toKeySequence() { QFETCH( QVariant, value ); QFETCH( QKeySequence, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::KeySequence ) ); QKeySequence d = qVariantValue(value); 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; } void tst_QVariant::toDouble() { QFETCH( QVariant, value ); QFETCH( double, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Double ) ); bool ok; double d = value.toDouble( &ok ); QCOMPARE( d, result ); QVERIFY( 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; } void tst_QVariant::toLongLong() { QFETCH( QVariant, value ); QFETCH( qlonglong, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::LongLong ) ); 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; } void tst_QVariant::toULongLong() { QFETCH( QVariant, value ); QFETCH( qulonglong, result ); QFETCH( bool, valueOK ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::ULongLong ) ); 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" ); QTest::newRow( "float" ) << QVariant( 123.456f ) << QByteArray( "123.456" ); QTest::newRow( "longlong" ) << QVariant( (qlonglong)34 ) << QByteArray( "34" ); QTest::newRow( "ulonglong" ) << QVariant( (qulonglong)34 ) << QByteArray( "34" ); } void tst_QVariant::toByteArray() { QFETCH( QVariant, value ); QFETCH( QByteArray, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::ByteArray ) ); QByteArray ba = value.toByteArray(); QCOMPARE( ba, 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" ); QTest::newRow( "float" ) << QVariant( 123.456f ) << QString( "123.456" ); 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" ); QTest::newRow( "qdatetime" ) << QVariant( QDateTime( QDate( 2002, 1, 1 ), QTime( 12, 34, 56 ) ) ) << QString( "2002-01-01T12:34:56" ); QTest::newRow( "qkeysequence" ) << qVariantFromValue( QKeySequence( Qt::CTRL + Qt::Key_A ) ) #ifndef Q_OS_MAC << QString( "Ctrl+A" ); #else << QString(QChar(0x2318)) + "A"; #endif QFont font( "times", 12 ); QTest::newRow( "qfont" ) << qVariantFromValue( font ) << QString("times,12,-1,5,50,0,0,0,0,0"); QTest::newRow( "qcolor" ) << qVariantFromValue( QColor( 10, 10, 10 ) ) << QString( "#0a0a0a" ); QTest::newRow( "llong" ) << QVariant( (qlonglong)Q_INT64_C(123456789012) ) << QString( "123456789012" ); } void tst_QVariant::toString() { QFETCH( QVariant, value ); QFETCH( QString, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::String ) ); QString str = value.toString(); QCOMPARE( str, 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() ); QVERIFY( value.canConvert( QVariant::Date ) ); 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 ); } void tst_QVariant::toTime() { QFETCH( QVariant, value ); QFETCH( QTime, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::Time ) ); 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 ) ); } void tst_QVariant::toDateTime() { QFETCH( QVariant, value ); QFETCH( QDateTime, result ); QVERIFY( value.isValid() ); QVERIFY( value.canConvert( QVariant::DateTime ) ); QCOMPARE( value.toDateTime(), result ); } void tst_QVariant::toLocale() { QVariant variant; QLocale loc = variant.toLocale(); variant = QLocale::system(); loc = variant.toLocale(); } void tst_QVariant::toRegExp() { QVariant variant; QRegExp rx = variant.toRegExp(); variant = QRegExp("foo"); rx = variant.toRegExp(); } 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+")); } void tst_QVariant::matrix() { QVariant variant; QMatrix matrix = qVariantValue(variant); QVERIFY(matrix.isIdentity()); qVariantSetValue(variant, QMatrix().rotate(90)); QCOMPARE(QMatrix().rotate(90), qVariantValue(variant)); void *mmatrix = QMetaType::create(QVariant::Matrix, 0); QVERIFY(mmatrix); QMetaType::destroy(QVariant::Matrix, mmatrix); } void tst_QVariant::matrix4x4() { QVariant variant; QMatrix4x4 matrix = qVariantValue(variant); QVERIFY(matrix.isIdentity()); QMatrix4x4 m; m.scale(2.0f); qVariantSetValue(variant, m); QCOMPARE(m, qVariantValue(variant)); void *mmatrix = QMetaType::create(QVariant::Matrix4x4, 0); QVERIFY(mmatrix); QMetaType::destroy(QVariant::Matrix4x4, mmatrix); } void tst_QVariant::transform() { QVariant variant; QTransform matrix = qVariantValue(variant); QVERIFY(matrix.isIdentity()); qVariantSetValue(variant, QTransform().rotate(90)); QCOMPARE(QTransform().rotate(90), qVariantValue(variant)); void *mmatrix = QMetaType::create(QVariant::Transform, 0); QVERIFY(mmatrix); QMetaType::destroy(QVariant::Transform, mmatrix); } void tst_QVariant::vector2D() { QVariant variant; QVector2D vector = qVariantValue(variant); QVERIFY(vector.isNull()); qVariantSetValue(variant, QVector2D(0.1, 0.2)); QCOMPARE(QVector2D(0.1, 0.2), qVariantValue(variant)); void *pvector = QMetaType::create(QVariant::Vector2D, 0); QVERIFY(pvector); QMetaType::destroy(QVariant::Vector2D, pvector); } void tst_QVariant::vector3D() { QVariant variant; QVector3D vector = qVariantValue(variant); QVERIFY(vector.isNull()); qVariantSetValue(variant, QVector3D(0.1, 0.2, 0.3)); QCOMPARE(QVector3D(0.1, 0.2, 0.3), qVariantValue(variant)); void *pvector = QMetaType::create(QVariant::Vector3D, 0); QVERIFY(pvector); QMetaType::destroy(QVariant::Vector3D, pvector); } void tst_QVariant::vector4D() { QVariant variant; QVector4D vector = qVariantValue(variant); QVERIFY(vector.isNull()); qVariantSetValue(variant, QVector4D(0.1, 0.2, 0.3, 0.4)); QCOMPARE(QVector4D(0.1, 0.2, 0.3, 0.4), qVariantValue(variant)); void *pvector = QMetaType::create(QVariant::Vector4D, 0); QVERIFY(pvector); QMetaType::destroy(QVariant::Vector4D, pvector); } void tst_QVariant::quaternion() { QVariant variant; QQuaternion quaternion = qVariantValue(variant); QVERIFY(quaternion.isIdentity()); qVariantSetValue(variant, QQuaternion(0.1, 0.2, 0.3, 0.4)); QCOMPARE(QQuaternion(0.1, 0.2, 0.3, 0.4), qVariantValue(variant)); void *pquaternion = QMetaType::create(QVariant::Quaternion, 0); QVERIFY(pquaternion); QMetaType::destroy(QVariant::Quaternion, pquaternion); } struct CustomStreamableClass { int i; bool operator==(const CustomStreamableClass& other) const { return i == other.i; } }; Q_DECLARE_METATYPE(CustomStreamableClass); QDataStream &operator<<(QDataStream &out, const CustomStreamableClass &myObj) { return out << myObj.i; } QDataStream &operator>>(QDataStream &in, CustomStreamableClass &myObj) { return in >> myObj.i; } void tst_QVariant::writeToReadFromDataStream_data() { qRegisterMetaTypeStreamOperators(); QTest::addColumn("writeVariant"); QTest::addColumn("isNull"); { typedef QList variantsList; variantsList 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( QBitArray() ) << true; QBitArray bitarray( 3 ); bitarray[0] = 0; bitarray[1] = 1; bitarray[2] = 0; QTest::newRow( "bitarray_valid" ) << QVariant( bitarray ) << false; QTest::newRow( "bytearray_invalid" ) << QVariant( QByteArray() ) << true; QTest::newRow( "int_invalid") << QVariant(QVariant::Int) << 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( "bitmap_invalid" ) << qVariantFromValue( QBitmap() ) << true; QBitmap bitmap( 10, 10 ); bitmap.fill( Qt::red ); QTest::newRow( "bitmap_valid" ) << qVariantFromValue( bitmap ) << false; QTest::newRow( "brush_valid" ) << qVariantFromValue( QBrush( Qt::red ) ) << false; QTest::newRow( "color_valid" ) << qVariantFromValue( QColor( Qt::red ) ) << false; #ifndef QT_NO_CURSOR QTest::newRow( "cursor_valid" ) << qVariantFromValue( QCursor( Qt::PointingHandCursor ) ) << false; #endif QTest::newRow( "date_invalid" ) << QVariant( QDate() ) << true; QTest::newRow( "date_valid" ) << QVariant( QDate( 2002, 07, 06 ) ) << false; QTest::newRow( "datetime_invalid" ) << QVariant( QDateTime() ) << true; 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( "font_valid" ) << qVariantFromValue( QFont( "times", 12 ) ) << false; QTest::newRow( "pixmap_invalid" ) << qVariantFromValue( QPixmap() ) << true; QPixmap pixmap( 10, 10 ); pixmap.fill( Qt::red ); QTest::newRow( "pixmap_valid" ) << qVariantFromValue( pixmap ) << false; // QTest::newRow( "iconset_valid" ) << QVariant( QIcon( pixmap ) ) << false; QTest::newRow( "image_invalid" ) << qVariantFromValue( QImage() ) << true; QTest::newRow( "keysequence_valid" ) << qVariantFromValue( QKeySequence( Qt::CTRL + Qt::Key_A ) ) << false; QTest::newRow( "int_valid" ) << QVariant( -123 ) << false; typedef QMap variantsMap; variantsMap 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( "palette_valid" ) << qVariantFromValue(QPalette(QColor("turquoise"))) << false; QTest::newRow( "pen_valid" ) << qVariantFromValue( QPen( Qt::red ) ) << false; QTest::newRow( "pointarray_invalid" ) << qVariantFromValue( QPolygon() ) << true; QTest::newRow( "pointarray_valid" ) << qVariantFromValue( QPolygon( QRect( 10, 10, 20, 20 ) ) ) << false; QTest::newRow( "region_invalid" ) << qVariantFromValue( QRegion() ) << true; QTest::newRow( "region_valid" ) << qVariantFromValue( QRegion( 10, 10, 20, 20 ) ) << false; QTest::newRow( "sizepolicy_valid" ) << qVariantFromValue( QSizePolicy( QSizePolicy::Fixed, QSizePolicy::Fixed ) ) << false; QTest::newRow( "point_invalid" ) << qVariantFromValue( QPoint() ) << true; QTest::newRow( "point_valid" ) << qVariantFromValue( QPoint( 10, 10 ) ) << false; QTest::newRow( "rect_invalid" ) << QVariant( QRect() ) << true; QTest::newRow( "rect_valid" ) << QVariant( QRect( 10, 10, 20, 20 ) ) << false; QTest::newRow( "size_invalid" ) << QVariant( QSize( 0, 0 ) ) << true; QTest::newRow( "size_valid" ) << QVariant( QSize( 10, 10 ) ) << false; QTest::newRow( "string_invalid" ) << QVariant( QString() ) << true; 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( QTime() ) << true; 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)) << true; QTest::newRow( "regexp" ) << QVariant(QRegExp("foo", Qt::CaseInsensitive)) << false; QTest::newRow( "regexp_empty" ) << QVariant(QRegExp()) << 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::Long, (void *) 0) << false; long longInt = -1l; QTest::newRow("QMetaType::Long") << QVariant(QMetaType::Long, &longInt) << false; QTest::newRow("QMetaType::Short invalid") << QVariant(QMetaType::Short, (void *) 0) << false; short shortInt = 1; QTest::newRow("QMetaType::Short") << QVariant(QMetaType::Short, &shortInt) << false; QTest::newRow("QMetaType::Char invalid") << QVariant(QMetaType::Char, (void *) 0) << false; char ch = 'c'; QTest::newRow("QMetaType::Char") << QVariant(QMetaType::Char, &ch) << false; QTest::newRow("QMetaType::ULong invalid") << QVariant(QMetaType::ULong, (void *) 0) << false; ulong ulongInt = 1ul; QTest::newRow("QMetaType::ULong") << QVariant(QMetaType::ULong, &ulongInt) << false; QTest::newRow("QMetaType::UShort invalid") << QVariant(QMetaType::UShort, (void *) 0) << false; ushort ushortInt = 1u; QTest::newRow("QMetaType::UShort") << QVariant(QMetaType::UShort, &ushortInt) << false; QTest::newRow("QMetaType::UChar invalid") << QVariant(QMetaType::UChar, (void *) 0) << false; uchar uch = 0xf0; QTest::newRow("QMetaType::UChar") << QVariant(QMetaType::UChar, &uch) << false; QTest::newRow("QMetaType::Float invalid") << QVariant(QMetaType::Float, (void *) 0) << false; float f = 1.234f; QTest::newRow("QMetaType::Float") << QVariant(QMetaType::Float, &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 != QVariant::Invalid && writeType != QVariant::Bitmap && writeType != QVariant::Pixmap && writeType != QVariant::Image) { 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); QVERIFY(qIsNaN(readFloat) == qIsNaN(writtenFloat)); if (!qIsNaN(readFloat)) QVERIFY(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() { QTest::ignoreMessage(QtWarningMsg, "Trying to construct an instance of an invalid type, type id: 49"); const QByteArray settingsHex("00000031ffffffffff"); 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.type(), QVariant::Invalid); } 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(QVariant::Int) << false; // 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 mUInt(42u); QVariant mUIntString(QByteArray("42")); QVariant mUIntQString(QString("42")); QVariant mDouble(42.11); QVariant mDoubleString(QByteArray("42.11")); QVariant mDoubleQString(QString("42.11")); QVariant mFloat(42.11f); QVariant mFloatString(QByteArray("42.11")); QVariant mFloatQString(QString("42.11")); 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")); 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 << true; QTest::newRow( "mIntString_mInt" ) << mIntString << mInt << true; QTest::newRow( "mInt_mIntQString" ) << mInt << mIntQString << true; QTest::newRow( "mIntQString_mInt" ) << mIntQString << mInt << true; QTest::newRow( "mUInt_mUIntString" ) << mUInt << mUIntString << true; QTest::newRow( "mUIntString_mUInt" ) << mUIntString << mUInt << true; QTest::newRow( "mUInt_mUIntQString" ) << mUInt << mUIntQString << true; QTest::newRow( "mUIntQString_mUInt" ) << mUIntQString << mUInt << true; QTest::newRow( "mDouble_mDoubleString" ) << mDouble << mDoubleString << true; QTest::newRow( "mDoubleString_mDouble" ) << mDoubleString << mDouble << true; QTest::newRow( "mDouble_mDoubleQString" ) << mDouble << mDoubleQString << true; QTest::newRow( "mDoubleQString_mDouble" ) << mDoubleQString << mDouble << true; QTest::newRow( "mFloat_mFloatString" ) << mFloat << mFloatString << true; QTest::newRow( "mFloatString_mFloat" ) << mFloatString << mFloat << true; QTest::newRow( "mFloat_mFloatQString" ) << mFloat << mFloatQString << true; QTest::newRow( "mFloatQString_mFloat" ) << mFloatQString << mFloat << true; QTest::newRow( "mLongLong_mLongLongString" ) << mLongLong << mLongLongString << true; QTest::newRow( "mLongLongString_mLongLong" ) << mLongLongString << mLongLong << true; QTest::newRow( "mLongLong_mLongLongQString" ) << mLongLong << mLongLongQString << true; QTest::newRow( "mLongLongQString_mLongLong" ) << mLongLongQString << mLongLong << true; QTest::newRow( "mULongLong_mULongLongString" ) << mULongLong << mULongLongString << true; QTest::newRow( "mULongLongString_mULongLong" ) << mULongLongString << mULongLong << true; QTest::newRow( "mULongLong_mULongLongQString" ) << mULongLong << mULongLongQString << true; QTest::newRow( "mULongLongQString_mULongLong" ) << mULongLongQString << mULongLong << true; QTest::newRow( "mBool_mBoolString" ) << mBool << mBoolString << true; QTest::newRow( "mBoolString_mBool" ) << mBoolString << mBool << true; 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(0) << false; QTest::newRow("invalidConversionR") << QVariant(0) << QVariant(QString("bubu")) << false; // ### many other combinations missing { 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 ); QEXPECT_FAIL("nullint", "See QTBUG-22933", Continue); QCOMPARE( left == right, equal ); } void tst_QVariant::operator_eq_eq_rhs() { QVariant v = 42; QVERIFY(v == 42); QVERIFY(42 == v); #if 0 /* This should _not_ compile */ QStringList list; QDateTime dt; QVERIFY(dt == list); #endif } 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"); 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("41") << int(QVariant::RegExp) << QByteArray("QRegExp"); QTest::newRow("43") << int(QVariant::Matrix) << QByteArray("QMatrix"); 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++ ) { 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 QCOMPARE( QVariant::typeToName( QVariant::Type(max+1) ), (char*)0 ); // 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); } void tst_QVariant::streamInvalidVariant() { int writeX = 1; int writeY = 2; int readX; int readY; QVariant writeVariant; QVariant readVariant; QVERIFY( writeVariant.type() == QVariant::Invalid ); 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.type() == QVariant::Invalid ); 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() { --instanceCount; } int number; const char *text; }; Q_DECLARE_METATYPE(MyType) Q_DECLARE_METATYPE(MyType*) void tst_QVariant::userType() { { MyType data(1, "eins"); MyType data2(2, "zwei"); { QVariant userVar; qVariantSetValue(userVar, data); QCOMPARE(userVar.type(), QVariant::UserType); QCOMPARE(userVar.userType(), qMetaTypeId()); QCOMPARE(userVar.typeName(), "MyType"); QVERIFY(!userVar.isNull()); QVERIFY(!userVar.canConvert(QVariant::String)); QVariant userVar2(userVar); QVERIFY(userVar == userVar2); qVariantSetValue(userVar2, 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; qVariantSetValue(userVar3, data2); userVar3 = userVar2; QVERIFY(userVar2 == userVar3); } // At this point all QVariants got destroyed but we have 2 MyType instances. QCOMPARE(instanceCount, 2); { QVariant userVar; qVariantSetValue(userVar, &data); QCOMPARE(userVar.type(), QVariant::UserType); QCOMPARE(userVar.userType(), qMetaTypeId()); QCOMPARE(userVar.typeName(), "MyType*"); QVERIFY(!userVar.isNull()); QVERIFY(!userVar.canConvert(QVariant::String)); QVariant userVar2(userVar); QVERIFY(userVar == userVar2); qVariantSetValue(userVar2, &data2); QVERIFY(userVar != userVar2); MyType * const*varData = reinterpret_cast(userVar.constData()); QVERIFY(varData); QCOMPARE(*varData, &data); QVariant userVar3; qVariantSetValue(userVar3, &data2); /* This check is correct now. userVar2 contains a pointer to data2 and so * does userVar3. */ QVERIFY(userVar2 == userVar3); userVar3 = userVar2; QVERIFY(userVar2 == userVar3); } QCOMPARE(instanceCount, 2); QVariant myCarrier; qVariantSetValue(myCarrier, data); QCOMPARE(instanceCount, 3); { QVariant second = myCarrier; QCOMPARE(instanceCount, 3); second.detach(); QCOMPARE(instanceCount, 4); } QCOMPARE(instanceCount, 3); MyType data3(0, "null"); data3 = qVariantValue(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; qVariantSetValue(myCarrier, 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; qVariantSetValue(myCarrier, s); QCOMPARE((int)qvariant_cast(myCarrier), 42); } { qlonglong ll = Q_INT64_C(42); QVariant myCarrier; qVariantSetValue(myCarrier, 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; QVariant pod_as_variant = qVariantFromValue(pod); MyTypePOD pod2 = qvariant_cast(pod_as_variant); QCOMPARE(pod.a, pod2.a); QCOMPARE(pod.b, pod2.b); qVariantSetValue(pod_as_variant, pod); pod2 = qVariantValue(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::Int, &i); } QCOMPARE(v.type(), QVariant::Int); QCOMPARE(v.toInt(), 7); { QString s("foo"); v = QVariant(QMetaType::QString, &s); } QCOMPARE(v.type(), QVariant::String); QCOMPARE(v.toString(), QString("foo")); { double d = 4.4; v = QVariant(QMetaType::Double, &d); } QCOMPARE(v.type(), QVariant::Double); QCOMPARE(v.toDouble(), 4.4); { float f = 4.5f; v = QVariant(QMetaType::Float, &f); } QCOMPARE(v.userType(), int(QMetaType::Float)); QCOMPARE(v.toDouble(), 4.5); { QByteArray ba("bar"); v = QVariant(QMetaType::QByteArray, &ba); } QCOMPARE(v.type(), QVariant::ByteArray); 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()), qVariantValue(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"); qVariantSetValue(v3, sl); Foo foo; foo.i = 42; qVariantSetValue(v4, 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 = qVariantFromValue(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 = qVariantFromValue(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(qVariantFromValue(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 }; int tp = qRegisterMetaType("Blah"); QVariant v = QVariant(tp, &i); qRegisterMetaTypeStreamOperators("Blah"); QCOMPARE(v.userType(), tp); QVERIFY(v.type() == QVariant::UserType); { QDataStream stream(&data, QIODevice::WriteOnly); stream << v; } v = QVariant(); { QDataStream stream(data); stream >> v; } QCOMPARE(int(v.userType()), QMetaType::type("Blah")); int value = *(int*)v.constData(); QCOMPARE(value, 42); } void tst_QVariant::url() { QString str("http://qt.nokia.com"); 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(qVariantCanConvert(v3)); QCOMPARE(v2.toUrl(), v3.toUrl()); QVERIFY(qVariantCanConvert(v2)); QCOMPARE(v2.toString(), str); QCOMPARE(v3.toString(), str); } void tst_QVariant::globalColor() { QVariant variant(Qt::blue); QVERIFY(variant.type() == QVariant::Color); QVERIFY(qVariantValue(variant) == QColor(Qt::blue)); } void tst_QVariant::variantMap() { QMap map; map["test"] = 42; QVariant v = map; QVariantMap map2 = qvariant_cast(v); QCOMPARE(map2.value("test").toInt(), 42); QVariant v2 = QVariant(QMetaType::type("QVariantMap"), &map); QCOMPARE(qvariant_cast(v2).value("test").toInt(), 42); QVariant v3 = QVariant(QMetaType::type("QMap"), &map); QCOMPARE(qvariant_cast(v3).value("test").toInt(), 42); } void tst_QVariant::variantHash() { QHash hash; hash["test"] = 42; QVariant v = hash; QVariantHash hash2 = qvariant_cast(v); QCOMPARE(hash2.value("test").toInt(), 42); QVariant v2 = QVariant(QMetaType::type("QVariantHash"), &hash); QCOMPARE(qvariant_cast(v2).value("test").toInt(), 42); QVariant v3 = QVariant(QMetaType::type("QHash"), &hash); QCOMPARE(qvariant_cast(v3).value("test").toInt(), 42); } void tst_QVariant::invalidQColor() const { QVariant va("An invalid QColor::name() value."); QVERIFY(va.canConvert(QVariant::Color)); QVERIFY(!va.convert(QVariant::Color)); QVERIFY(!qvariant_cast(va).isValid()); } class CustomQObject : public QObject { Q_OBJECT public: CustomQObject(QObject *parent = 0) : QObject(parent) {} }; class CustomQWidget : public QWidget { Q_OBJECT public: CustomQWidget(QWidget *parent = 0) : QWidget(parent) {} }; Q_DECLARE_METATYPE(CustomQObject*) Q_DECLARE_METATYPE(CustomQWidget*) 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"); QObject *obj = new QObject; obj->setObjectName(QString::fromLatin1("Hello")); QTest::newRow("from QObject") << QVariant(QMetaType::QObjectStar, &obj) << true; QTest::newRow("from QObject2") << QVariant::fromValue(obj) << true; QTest::newRow("from String") << QVariant(QLatin1String("1, 2, 3")) << false; QTest::newRow("from int") << QVariant((int) 123) << false; QWidget *widget = new QWidget; widget->setObjectName(QString::fromLatin1("Hello")); QTest::newRow("from QWidget") << QVariant::fromValue(widget) << true; CustomQObject *customObject = new CustomQObject(this); customObject->setObjectName(QString::fromLatin1("Hello")); QTest::newRow("from Derived QObject") << QVariant::fromValue(customObject) << true; CustomQWidget *customWidget = new CustomQWidget; customWidget->setObjectName(QString::fromLatin1("Hello")); QTest::newRow("from Derived QWidget") << QVariant::fromValue(customWidget) << true; QTest::newRow("from custom Object") << QVariant::fromValue(CustomNonQObject()) << false; // Deleted in cleanupTestCase. customNonQObjectPointer = new CustomNonQObject; QTest::newRow("from custom ObjectStar") << QVariant::fromValue(customNonQObjectPointer) << false; // Deleted in cleanupTestCase. objectPointerTestData.push_back(obj); objectPointerTestData.push_back(widget); objectPointerTestData.push_back(customObject); objectPointerTestData.push_back(customWidget); } void tst_QVariant::qvariant_cast_QObject() { QFETCH(QVariant, data); QFETCH(bool, success); QObject *o = qvariant_cast(data); QCOMPARE(o != 0, success); if (success) { QCOMPARE(o->objectName(), QString::fromLatin1("Hello")); } } class CustomQObjectDerived : public CustomQObject { Q_OBJECT public: CustomQObjectDerived(QObject *parent = 0) : CustomQObject(parent) {} }; Q_DECLARE_METATYPE(CustomQObjectDerived*) class CustomQObjectDerivedNoMetaType : public CustomQObject { Q_OBJECT public: CustomQObjectDerivedNoMetaType(QObject *parent = 0) : CustomQObject(parent) {} }; void tst_QVariant::qvariant_cast_QObject_derived() { { CustomQObjectDerivedNoMetaType *object = new CustomQObjectDerivedNoMetaType(this); QVariant data = QVariant::fromValue(object); QVERIFY(data.userType() == qMetaTypeId()); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); QVERIFY(data.value() == 0); } { CustomQObjectDerived *object = new CustomQObjectDerived(this); QVariant data = QVariant::fromValue(object); QVERIFY(data.userType() == qMetaTypeId()); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); QCOMPARE(data.value(), object); QVERIFY(data.value() == 0); } { CustomQWidget customWidget; QWidget *widget = &customWidget; QVariant data = QVariant::fromValue(widget); QVERIFY(data.userType() == QMetaType::QWidgetStar); QCOMPARE(data.value(), widget); QCOMPARE(data.value(), widget); QCOMPARE(data.value(), widget); } } Q_DECLARE_METATYPE(qint8); 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(qVariantCanConvert(v0)); 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(qVariantCanConvert(v0)); 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(qVariantCanConvert(v0)); 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(qVariantCanConvert(v0)); QCOMPARE(int(qvariant_cast(v0)), 32); QCOMPARE(int(v0.toUInt()), 32); QCOMPARE(v0.toString(), QString("32")); } } void tst_QVariant::comparePointers() const { class MyClass { }; MyClass myClass; QVariant v = qVariantFromValue(&myClass); QVariant v2 = qVariantFromValue(&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 = qVariantFromValue(p1); v2 = v1; QVERIFY(v1 == v2); v2 = qVariantFromValue(p2); QVERIFY(v1 == v2); p2 = 0; v2 = qVariantFromValue(p2); QVERIFY(v1 != v2); } void tst_QVariant::dataStar() const { qRegisterMetaType(); Data *p1 = new Data; QVariant v1 = qVariantFromValue(p1); QCOMPARE(v1.userType(), qMetaTypeId()); QCOMPARE(qvariant_cast(v1), p1); QVariant v2 = v1; QVERIFY(v1 == v2); v2 = qVariantFromValue(p1); QVERIFY(v1 == v2); delete p1; } void tst_QVariant::canConvertQStringList() const { QFETCH(bool, canConvert); QFETCH(QStringList, input); QFETCH(QString, result); QVariant v(input); QCOMPARE(v.canConvert(QVariant::String), 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") << false << 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") << false << l << QString(); l << "c"; QTest::newRow("Three items") << false << l << QString(); } template void convertMetaType() { QVERIFY(qVariantFromValue(10).isValid()); QVERIFY(qVariantFromValue(10).canConvert(QVariant::Int)); QCOMPARE(qVariantFromValue(10).toInt(), 10); QCOMPARE(qVariantFromValue(10), qVariantFromValue(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(QVariant::DateTime); 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(QVariant::DateTime)); } 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, 0, 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, "Trying to construct an instance of an invalid type, type id: 127"); 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(QVariant::Date)); variant = foo; QVERIFY(!variant.convert(QVariant::DateTime)); variant = foo; QVERIFY(!variant.convert(QVariant::Time)); variant = foo; QVERIFY(!variant.convert(QVariant::Int)); variant = foo; QVERIFY(!variant.convert(QVariant::Double)); variant = foo; QVERIFY(!variant.convert(QVariant::Type(QMetaType::Float))); } struct WontCompare { int x,y,z,q,w,e,r,t; }; Q_DECLARE_METATYPE(WontCompare); void tst_QVariant::compareCustomTypes() const { qRegisterMetaType("WontCompare"); WontCompare f1; f1.x = 0; const QVariant variant1(qVariantFromValue(f1)); WontCompare f2; f2.x = 0; const QVariant variant2(qVariantFromValue(f2)); /* We compare pointers. */ QVERIFY(variant1 != variant2); QVERIFY(variant1 == variant1); QVERIFY(variant2 == variant2); } void tst_QVariant::timeToDateTime() const { const QVariant val(QTime::currentTime()); QVERIFY(!val.canConvert(QVariant::DateTime)); QVERIFY(!val.toDateTime().isValid()); } Q_DECLARE_METATYPE(QHostAddress) void tst_QVariant::copyingUserTypes() const { QVariant var; QVariant var3; const QHostAddress ha("127.0.0.1"); qVariantSetValue(var, ha); var3 = var; QCOMPARE(qVariantValue(var3), ha); } void tst_QVariant::convertBoolToByteArray() const { QFETCH(QByteArray, input); QFETCH(bool, canConvert); QFETCH(bool, value); const QVariant variant(input); QCOMPARE(qVariantCanConvert(variant), 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.type(), QVariant::Bool); QCOMPARE(variant.toBool(), input); QVERIFY(qVariantCanConvert(variant)); 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"); } /*! 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 QVERIFY((int)d == 2147483630); QVariant var(d); QVERIFY( var.canConvert( QVariant::Int ) ); bool ok; int result = var.toInt(&ok); QVERIFY( ok == true ); QCOMPARE(result, 2147483630); } void tst_QVariant::setValue() { QTransform t; //we just take a value so that we're sure that it will be shared QVariant v1 = t; QVERIFY( v1.isDetached() ); QVariant v2 = v1; QVERIFY( !v1.isDetached() ); QVERIFY( !v2.isDetached() ); qVariantSetValue(v2, 3); //set an integer value QVERIFY( v1.isDetached() ); QVERIFY( v2.isDetached() ); } void tst_QVariant::numericalConvert() { #if defined(Q_OS_LINUX) && defined(Q_CC_GNU) && !defined(__x86_64__) QSKIP("Known to fail due to a GCC bug on at least Ubuntu 10.04 32-bit - check QTBUG-8959"); #endif QVariant vfloat(float(5.3)); QVariant vdouble(double(5.3)); QVariant vreal(qreal(5.3)); QVariant vint(int(5)); QVariant vuint(uint(5)); QVariant vshort(short(5)); QVariant vlonglong(quint64(5)); QVariant vstringint(QString::fromLatin1("5")); QVariant vstring(QString::fromLatin1("5.3")); QVector vect; vect << &vfloat << &vdouble << &vreal << &vint << &vuint << &vshort<< &vlonglong << &vstringint << &vstring; for(int i = 0; i < vect.size(); i++) { double num = 5.3; if (i >= 3 && i <= 7) num = 5; QVariant *v = vect.at(i); QCOMPARE(v->toFloat() , float(num)); QCOMPARE(float(v->toReal()) , float(num)); QCOMPARE(float(v->toDouble()) , float(num)); if(i != 8) { QCOMPARE(v->toInt() , int(num)); QCOMPARE(v->toUInt() , uint(num)); QCOMPARE(v->toULongLong() , quint64(num)); } QCOMPARE(v->toString() , QString::number(num)); } } 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::isStatic && QTypeInfo::isComplex)) { // 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::isStatic && QTypeInfo::isComplex)) { // 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 = qVariantFromValue(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()); QCOMPARE(QVariant::typeToName(QVariant::Type(v.userType())), QMetaType::typeName(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++; } 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; } }; int MyNotMovable::count = 0; struct MyShared : QSharedData { MyMovable movable; }; QT_BEGIN_NAMESPACE Q_DECLARE_TYPEINFO(MyMovable, Q_MOVABLE_TYPE); QT_END_NAMESPACE Q_DECLARE_METATYPE(QList) Q_DECLARE_METATYPE(MyPrimitive) Q_DECLARE_METATYPE(MyData) Q_DECLARE_METATYPE(MyMovable) Q_DECLARE_METATYPE(MyNotMovable) Q_DECLARE_METATYPE(QList) Q_DECLARE_METATYPE(QList) Q_DECLARE_METATYPE(QList) Q_DECLARE_METATYPE(QList) Q_DECLARE_METATYPE(MyPrimitive *) Q_DECLARE_METATYPE(MyData *) Q_DECLARE_METATYPE(MyMovable *) Q_DECLARE_METATYPE(MyNotMovable *) Q_DECLARE_METATYPE(QSharedDataPointer) 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, true, 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); { PLAY_WITH_VARIANT(12.12, false, "12.12", 12.12, true); PLAY_WITH_VARIANT(12.12f, false, "12.12", 12.12f, true); PLAY_WITH_VARIANT('a', false, "a", 'a', true); PLAY_WITH_VARIANT((unsigned char)('a'), false, "a", 'a', true); PLAY_WITH_VARIANT( quint8(12), false, "\xc", 12, true); PLAY_WITH_VARIANT( qint8(13), false, "\xd", 13, 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( qint8(-12), false, "\xf4", -12, 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), false, 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 << (QVector() << 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); } { QList > data; PLAY_WITH_VARIANT(data, false, QString(), 0, false); data << (QLinkedList() << 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].type(), QVariant::Invalid); buffer[0].~QVariant(); memcpy(buffer, &variant, sizeof(QVariant)); QCOMPARE(buffer[0].type(), QVariant::UserType); 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.type(), QVariant::Int); QVariant var2 = var1; QCOMPARE(var2, var1); QCOMPARE(var2.type(), QVariant::Int); QVariant var3 = QVariant::fromValue(var1); QCOMPARE(var3, var1); QCOMPARE(var3.type(), QVariant::Int); QVariant var4 = qvariant_cast(var1); QCOMPARE(var4, var1); QCOMPARE(var4.type(), QVariant::Int); QVariant var5; var5 = var1; QCOMPARE(var5, var1); QCOMPARE(var5.type(), QVariant::Int); QVariant var6; var6.setValue(var1); QCOMPARE(var6, var1); QCOMPARE(var6.type(), QVariant::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((int)var8.type(), (int)QVariant::String); QCOMPARE(qvariant_cast(QVariant(qvariant_cast(var8))), str); QVariant var9(qMetaTypeId(), &var1); QCOMPARE(var9.userType(), qMetaTypeId()); QCOMPARE(qvariant_cast(var9), var1); } void tst_QVariant::colorInteger() { QVariant v = QColor(Qt::red); QCOMPARE(v.type(), QVariant::Color); QCOMPARE(v.value(), QColor(Qt::red)); v.setValue(1000); QCOMPARE(v.type(), QVariant::Int); QCOMPARE(v.toInt(), 1000); v.setValue(QColor(Qt::yellow)); QCOMPARE(v.type(), QVariant::Color); QCOMPARE(v.value(), QColor(Qt::yellow)); } class Forward; Q_DECLARE_OPAQUE_POINTER(Forward*) Q_DECLARE_METATYPE(Forward*) void tst_QVariant::forwardDeclare() { Forward *f = 0; 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; foreach (const QFileInfo &fileInfo, dir.entryInfoList(QStringList() << "*.bin")) { 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::type(typeName.toLatin1()); if (id == QMetaType::Void) { // Void type is not supported by QVariant return; } QVariant constructedVariant(static_cast(id)); QCOMPARE(constructedVariant.userType(), id); QCOMPARE(QMetaType::typeName(loadedVariant.userType()), 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::type(typeName.toLatin1()); 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(static_cast(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()); } class MessageHandler { public: MessageHandler(const int typeId, QtMsgHandler msgHandler = handler) : oldMsgHandler(qInstallMsgHandler(msgHandler)) { currentId = typeId; } ~MessageHandler() { qInstallMsgHandler(oldMsgHandler); } bool testPassed() const { return ok; } protected: static void handler(QtMsgType, const char *txt) { QString msg = QString::fromLatin1(txt); // 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()); ok &= (currentId == QMetaType::UnknownType ? msg.contains("Invalid") : msg.contains(QMetaType::typeName(currentId))); QVERIFY2(ok, (QString::fromLatin1("Message doesn't contain type name: '") + msg + '\'').toLatin1().constData()); if (currentId == QMetaType::Char || currentId == QMetaType::QChar) { // Chars insert '\0' into the qdebug stream, it is not possible to find a real string length return; } if (QMetaType::typeFlags(currentId) & QMetaType::PointerToQObject) { QByteArray currentName = QMetaType::typeName(currentId); currentName.chop(1); ok &= (msg.contains(", " + currentName) || msg.contains(", 0x0")); } ok &= msg.endsWith(") "); QVERIFY2(ok, (QString::fromLatin1("Message is not correctly finished: '") + msg + '\'').toLatin1().constData()); } QtMsgHandler oldMsgHandler; static int currentId; static bool ok; }; bool MessageHandler::ok; int MessageHandler::currentId; void tst_QVariant::debugStream_data() { QTest::addColumn("variant"); QTest::addColumn("typeId"); for (int id = 0; id < QMetaType::User; ++id) { const char *tagName = QMetaType::typeName(id); if (!tagName) continue; if (id != QMetaType::Void) { QTest::newRow(tagName) << QVariant(static_cast(id)) << id; } } QTest::newRow("QBitArray(111)") << QVariant(QBitArray(3, true)) << qMetaTypeId(); QTest::newRow("CustomStreamableClass") << QVariant(qMetaTypeId(), 0) << qMetaTypeId(); QTest::newRow("MyClass") << QVariant(qMetaTypeId(), 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()); } struct MessageHandlerType : public MessageHandler { MessageHandlerType(const int typeId) : MessageHandler(typeId, handler) {} static void handler(QtMsgType, const char *txt) { QString msg = QString::fromLatin1(txt); // 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()); ok &= (currentId == QMetaType::UnknownType ? msg.contains("Invalid") : msg.contains(QMetaType::typeName(currentId))); 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); qDebug() << QVariant::Type(typeId); QVERIFY(msgHandler.testPassed()); } void tst_QVariant::guiVariantAtExit() { // crash test, it should not crash at QGuiApplication exit static QVariant cursor = QCursor(); static QVariant point = QPoint(); static QVariant image = QImage(); static QVariant pallete = QPalette(); Q_UNUSED(cursor); Q_UNUSED(point); Q_UNUSED(image); Q_UNUSED(pallete); QVERIFY(true); } void tst_QVariant::widgetsVariantAtExit() { // crash test, it should not crash at QGuiApplication exit static QVariant icon= QIcon(); static QVariant sizePolicy = QSizePolicy(); Q_UNUSED(icon); Q_UNUSED(sizePolicy); QVERIFY(true); } QTEST_MAIN(tst_QVariant) #include "tst_qvariant.moc"