#include #include #include #include #include #include #include "cachecleaner.h" #include "directorieswatcher.h" #include "figmalinks.h" #include "pagesmodel.h" #include "sectionsdecoratormodel.h" #include "testsrunner.h" #include "systemutils.h" #include #include struct PagesModelInitialized : public PagesModel { explicit PagesModelInitialized(QObject *parent = nullptr) : PagesModel(QML_IMPORT_ROOT + QStringLiteral("/pages"), parent) {} }; void loadContextPropertiesMocks(const char* storybookRoot, QQmlApplicationEngine& engine); int main(int argc, char *argv[]) { // Required by the WalletConnectSDK view QtWebView::initialize(); #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling); #endif QCoreApplication::setAttribute(Qt::AA_ShareOpenGLContexts); QGuiApplication app(argc, argv); QGuiApplication::setOrganizationName(QStringLiteral("Status")); QGuiApplication::setOrganizationDomain(QStringLiteral("status.im")); QGuiApplication::setApplicationName(QStringLiteral("Status Desktop Storybook")); QCommandLineParser cmdParser; cmdParser.addHelpOption(); cmdParser.addPositionalArgument(QStringLiteral("page"), QStringLiteral("Open the given page on startup")); if (!cmdParser.parse(app.arguments())) { qWarning() << "Error (ignored) while parsing cmd line arguments:" << cmdParser.errorText(); } qputenv("QT_QUICK_CONTROLS_HOVER_ENABLED", QByteArrayLiteral("1")); auto chromiumFlags = qgetenv("QTWEBENGINE_CHROMIUM_FLAGS"); if(!chromiumFlags.contains("--disable-seccomp-filter-sandbox")) { chromiumFlags +=" --disable-seccomp-filter-sandbox"; } qputenv("QTWEBENGINE_CHROMIUM_FLAGS", chromiumFlags); QQmlApplicationEngine engine; const QStringList additionalImportPaths { STATUSQ_MODULE_IMPORT_PATH, QML_IMPORT_ROOT + QStringLiteral("/../ui/app"), QML_IMPORT_ROOT + QStringLiteral("/../ui/imports"), QML_IMPORT_ROOT + QStringLiteral("/src"), QML_IMPORT_ROOT + QStringLiteral("/pages"), QML_IMPORT_ROOT + QStringLiteral("/stubs") }; for (const auto& path : additionalImportPaths) engine.addImportPath(path); engine.rootContext()->setContextProperty(QStringLiteral("pagesFolder"), QML_IMPORT_ROOT + QStringLiteral("/pages")); qmlRegisterType("Storybook", 1, 0, "PagesModel"); qmlRegisterType("Storybook", 1, 0, "SectionsDecoratorModel"); qmlRegisterUncreatableType("Storybook", 1, 0, "FigmaLinks", {}); auto watcherFactory = [additionalImportPaths](QQmlEngine*, QJSEngine*) { auto watcher = new DirectoriesWatcher(); watcher->addPaths(additionalImportPaths); // Test path added here as a temporary solution. Ideally, tests should // be observed separately. watcher->addPaths({ QML_IMPORT_ROOT + QStringLiteral("/qmlTests/tests") }); return watcher; }; qmlRegisterSingletonType( "Storybook", 1, 0, "SourceWatcher", watcherFactory); auto cleanerFactory = [](QQmlEngine* engine, QJSEngine*) { return new CacheCleaner(engine); }; qmlRegisterSingletonType( "Storybook", 1, 0, "CacheCleaner", cleanerFactory); auto runnerFactory = [](QQmlEngine* engine, QJSEngine*) { return new TestsRunner( QCoreApplication::applicationDirPath() + QStringLiteral("/QmlTests"), QML_IMPORT_ROOT + QStringLiteral("/qmlTests/tests")); }; qmlRegisterSingletonType( "Storybook", 1, 0, "TestsRunner", runnerFactory); qmlRegisterSingletonType( "Storybook", 1, 0, "SystemUtils", [](QQmlEngine*, QJSEngine*) { return new SystemUtils; }); registerStatusQTypes(); loadContextPropertiesMocks(QML_IMPORT_ROOT, engine); #ifdef Q_OS_WIN const QUrl url(QUrl::fromLocalFile(QML_IMPORT_ROOT + QStringLiteral("/main.qml"))); #else const QUrl url(QML_IMPORT_ROOT + QStringLiteral("/main.qml")); #endif QObject::connect(&engine, &QQmlApplicationEngine::objectCreated, &app, [url](QObject *obj, const QUrl &objUrl) { if (!obj && url == objUrl) QCoreApplication::exit(-1); }, Qt::QueuedConnection); const auto args = cmdParser.positionalArguments(); if (!args.isEmpty()) engine.setInitialProperties({{QStringLiteral("currentPage"), args.constFirst()}}); engine.load(url); return QGuiApplication::exec(); } void loadContextPropertiesMocks(const char* storybookRoot, QQmlApplicationEngine& engine) { QDirIterator it(QML_IMPORT_ROOT + QStringLiteral("/stubs/nim/sectionmocks"), QDirIterator::Subdirectories); while (it.hasNext()) { it.next(); if (it.fileInfo().isFile() && it.fileInfo().suffix() == QStringLiteral("qml")) { auto component = std::make_unique(&engine, QUrl::fromLocalFile(it.filePath())); if (component->status() != QQmlComponent::Ready) { qWarning() << "Failed to load mock for" << it.filePath() << component->errorString(); continue; } auto objPtr = std::unique_ptr(component->create()); if(!objPtr) { qWarning() << "Failed to create mock for" << it.filePath(); continue; } if(!objPtr->property("contextPropertyName").isValid()) { qInfo() << "Not a mock, missing property name \"contextPropertyName\""; continue; } auto contextPropertyName = objPtr->property("contextPropertyName").toString(); auto obj = objPtr.release(); obj->setParent(&engine); engine.rootContext()->setContextProperty(contextPropertyName, obj); } } }