// std #include #include #include // Qt #include #include #include #include #include #include #include #include #include // DOtherSide #include "DOtherSide/DOtherSide.h" #include "DOtherSide/DosQObject.h" #include "DOtherSide/DosQMetaObject.h" #include "DOtherSide/DosQObject.h" #include "DOtherSide/DosQAbstractListModel.h" using namespace DOS; template bool ExecuteTest(int argc, char *argv[]) { Test test; return QTest::qExec(&test, argc, argv) == 0; } template bool ExecuteGuiTest(int argc, char *argv[]) { QApplication app(argc, argv); Test test; return QTest::qExec(&test, argc, argv) == 0; } /* * Test QGuiApplication */ class TestQGuiApplication : public QObject { Q_OBJECT private slots: void testExecution() { bool quit = false; dos_qguiapplication_create(); QTimer timer; QObject::connect(&timer, &QTimer::timeout, [&quit]() { quit = true; dos_qguiapplication_quit(); }); timer.start(100); dos_qguiapplication_exec(); QVERIFY(quit); dos_qguiapplication_delete(); } }; /* * Test QApplication */ class TestQApplication : public QObject { Q_OBJECT private slots: void testExecution() { bool quit = false; dos_qapplication_create(); QTimer timer; QObject::connect(&timer, &QTimer::timeout, [&quit]() { quit = true; dos_qapplication_quit(); }); timer.start(100); dos_qapplication_exec(); QVERIFY(quit); dos_qapplication_delete(); } }; /* * Test QQmlApplicationEngine */ class TestQQmlApplicationEngine : public QObject { Q_OBJECT private slots: void initTestCase() { m_engine = nullptr; } void cleanupTestCase() { QVERIFY(m_engine == nullptr); } void init() { QVERIFY(m_engine == nullptr); dos_qqmlapplicationengine_create(&m_engine); QVERIFY(m_engine != nullptr); } void cleanup() { dos_qqmlapplicationengine_delete(m_engine); m_engine = nullptr; } void testCreateAndDelete() { // Implicit by invoking init and cleanup } void testLoadUrl() { void *url = nullptr; dos_qurl_create(&url, "qrc:///main.qml", QUrl::TolerantMode); QVERIFY(url != nullptr); dos_qqmlapplicationengine_load_url(m_engine, url); QCOMPARE(engine()->rootObjects().size(), 1); QCOMPARE(engine()->rootObjects().front()->objectName(), QString::fromLocal8Bit("testWindow")); QVERIFY(engine()->rootObjects().front()->isWindowType()); dos_qurl_delete(url); } void testLoadData() { dos_qqmlapplicationengine_load_data(m_engine, "import QtQuick.Controls 1.4; ApplicationWindow { objectName: \"testWindow\"}"); QCOMPARE(engine()->rootObjects().size(), 1); QCOMPARE(engine()->rootObjects().front()->objectName(), QString::fromLocal8Bit("testWindow")); QVERIFY(engine()->rootObjects().front()->isWindowType()); } void testRootObjects() { void **rootObjects = nullptr; int length = 0; void *url = nullptr; dos_qurl_create(&url, "qrc:///main.qml", QUrl::TolerantMode); dos_qqmlapplicationengine_load_url(m_engine, url); dos_qurl_delete(url); dos_qqmlapplicationengine_rootObjects(m_engine, &rootObjects, &length); QCOMPARE(length, 1); QObject *window = reinterpret_cast(rootObjects[0]); QVERIFY(window->isWindowType()); dos_qobjectptr_array_delete(rootObjects); } private: QQmlApplicationEngine *engine() { return static_cast(m_engine); } void *m_engine; }; /* * Test QQmlContext */ class TestQQmlContext : public QObject { Q_OBJECT private slots: void initTestCase() { m_engine = nullptr; m_context = nullptr; } void cleanupTestCase() { QVERIFY(m_engine == nullptr); QVERIFY(m_context == nullptr); } void init() { dos_qqmlapplicationengine_create(&m_engine); dos_qqmlapplicationengine_context(m_engine, &m_context); QVERIFY(m_engine != nullptr); QVERIFY(m_context != nullptr); } void cleanup() { m_context = nullptr; dos_qqmlapplicationengine_delete(m_engine); m_engine = nullptr; } void testCreateAndDelete() { // Implicit by invoking init and cleanup } void testSetContextProperty() { QVariant testData("Test Message"); dos_qqmlcontext_setcontextproperty(m_context, "testData", &testData); engine()->loadData("import QtQuick 2.5; Text { objectName: \"label\"; text: testData } "); QObject *label = engine()->rootObjects().first(); QVERIFY(label != nullptr); QCOMPARE(label->objectName(), QString::fromLocal8Bit("label")); QCOMPARE(label->property("text").toString(), testData.toString()); } private: QQmlApplicationEngine *engine() { return static_cast(m_engine); } QQmlContext *context() { return static_cast(m_context); } void *m_engine; void *m_context; }; /* * Test QQmlContext */ class TestQObject : public QObject { Q_OBJECT private slots: void testPropertyInheritance() { DOS::SignalDefinitions signalDefinitions {DOS::SignalDefinition {"nameChanged", {}}}; DOS::SlotDefinitions slotDefinitions {DOS::SlotDefinition {"name", QMetaType::QString, {}}, DOS::SlotDefinition {"setName", QMetaType::Void, {QMetaType::QString}}}; DOS::PropertyDefinitions propertyDefinitions {DOS::PropertyDefinition{"name", QMetaType::QString, "name", "setName", "nameChanged"}}; auto mo = std::make_shared(std::make_shared(), "TestClass", signalDefinitions, slotDefinitions, propertyDefinitions); QString value = ""; auto ose = [&value](const QString & name, const std::vector &args) -> QVariant { if (name == "name") return value; else if (name == "setName") value = args.front().toString(); return QVariant(); }; DosQObject testObject(mo, ose); testObject.setObjectName("testObject"); testObject.setProperty("name", "foo"); /// Test property read QCOMPARE(testObject.property("objectName").toString(), QString("testObject")); QCOMPARE(testObject.property("name").toString(), QString("foo")); /// Test slot invokation QMetaObject::invokeMethod(&testObject, "setName", Q_ARG(QString, "bar")); QCOMPARE(testObject.property("name").toString(), QString("bar")); } }; /* * Test QQmlContext */ class TestQAbstractListModel : public QObject { Q_OBJECT private slots: void testPropertyInheritance() { DOS::SignalDefinitions signalDefinitions {DOS::SignalDefinition {"nameChanged", {}}}; DOS::SlotDefinitions slotDefinitions {DOS::SlotDefinition {"name", QMetaType::QString, {}}, DOS::SlotDefinition {"setName", QMetaType::Void, {QMetaType::QString}}}; DOS::PropertyDefinitions propertyDefinitions {DOS::PropertyDefinition{"name", QMetaType::QString, "name", "setName", "nameChanged"}}; auto mo = std::make_shared(std::make_shared(), "TestClass", signalDefinitions, slotDefinitions, propertyDefinitions); auto moh = std::make_unique(mo); QString value = ""; auto ose = [&value](const QString & name, const std::vector &args) -> QVariant { if (name == "name") return value; else if (name == "setName") value = args.front().toString(); return QVariant(); }; void *dPointer = nullptr; RowCountCallback rcc = nullptr; ColumnCountCallback ccc = nullptr; DataCallback dc = nullptr; SetDataCallback sdc = nullptr; RoleNamesCallback rnc = nullptr; FlagsCallback fc = nullptr; HeaderDataCallback hdc = nullptr; DosQAbstractListModel testObject(dPointer, moh->data(), ose, rcc, ccc, dc, sdc, rnc, fc, hdc); testObject.setObjectName("testObject"); testObject.setProperty("name", "foo"); /// Test property read QCOMPARE(testObject.property("objectName").toString(), QString("testObject")); QCOMPARE(testObject.property("name").toString(), QString("foo")); /// Test slot invokation QMetaObject::invokeMethod(&testObject, "setName", Q_ARG(QString, "bar")); QCOMPARE(testObject.property("name").toString(), QString("bar")); } }; int main(int argc, char *argv[]) { using namespace DOS; bool success = true; success &= ExecuteTest(argc, argv); success &= ExecuteTest(argc, argv); success &= ExecuteGuiTest(argc, argv); success &= ExecuteGuiTest(argc, argv); success &= ExecuteGuiTest(argc, argv); success &= ExecuteGuiTest(argc, argv); return success ? 0 : 1; } #include "test_dotherside.moc"