status-desktop/storybook/main.cpp

167 lines
6.1 KiB
C++

#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <QQmlContext>
#include <QQmlComponent>
#include <QDirIterator>
#include <QtWebView>
#include "cachecleaner.h"
#include "directorieswatcher.h"
#include "figmalinks.h"
#include "pagesmodel.h"
#include "sectionsdecoratormodel.h"
#include "testsrunner.h"
#include "systemutils.h"
#include <memory>
#include <StatusQ/typesregistration.h>
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<PagesModelInitialized>("Storybook", 1, 0, "PagesModel");
qmlRegisterType<SectionsDecoratorModel>("Storybook", 1, 0, "SectionsDecoratorModel");
qmlRegisterUncreatableType<FigmaLinks>("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<DirectoriesWatcher>(
"Storybook", 1, 0, "SourceWatcher", watcherFactory);
auto cleanerFactory = [](QQmlEngine* engine, QJSEngine*) {
return new CacheCleaner(engine);
};
qmlRegisterSingletonType<CacheCleaner>(
"Storybook", 1, 0, "CacheCleaner", cleanerFactory);
auto runnerFactory = [](QQmlEngine* engine, QJSEngine*) {
return new TestsRunner(
QCoreApplication::applicationDirPath() + QStringLiteral("/QmlTests"),
QML_IMPORT_ROOT + QStringLiteral("/qmlTests/tests"));
};
qmlRegisterSingletonType<TestsRunner>(
"Storybook", 1, 0, "TestsRunner", runnerFactory);
qmlRegisterSingletonType<SystemUtils>(
"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<QQmlComponent>(&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<QObject>(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);
}
}
}