aboutsummaryrefslogtreecommitdiffstats
path: root/include/iqqmlengine.h
blob: 9aaa5e0e2087302edd9dda9583bb965189618d51 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
/***************************************************************************************************
 Copyright (C) 2024 The Qt Company Ltd.
 SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
***************************************************************************************************/

#pragma once

#ifndef QT_QUICK_LIB
struct IQQmlEngine {};
#else

#include "qdotnetinterface.h"
#include "qdotnetadapter.h"

#ifdef __GNUC__
#   pragma GCC diagnostic push
#   pragma GCC diagnostic ignored "-Wconversion"
#endif
#include <QElapsedTimer>
#include <QQmlEngine>
#include <QThread>
#ifdef __GNUC__
#   pragma GCC diagnostic pop
#endif

#include <functional>

struct IQQmlEngine : public QDotNetNativeInterface<QQmlEngine>
{
    static inline const QString &AssemblyQualifiedName =
        QStringLiteral("Qt.Quick.IQQmlEngine, Qt.DotNet.Adapter");

    bool exited = false;
    int exitCode = -1;

    IQQmlEngine()
        : QDotNetNativeInterface<QQmlEngine>(AssemblyQualifiedName,
            QDotNetAdapter::instance().qmlEngine(), false)
    {
        init();
    }

    void init() {
        const auto *engine = QDotNetAdapter::instance().qmlEngine();
        if (engine == nullptr)
            return;

        QObject::connect(engine, &QQmlEngine::exit,
            [this](int code)
            {
                exitCode = code;
                exited = true;
            });
        QObject::connect(engine, &QQmlEngine::quit,
            [this]()
            {
                exitCode = 0;
                exited = true;
            });
        setCallback<void, QString, QString>("LoadFromModule", [this](void *data,
            const QString &uri, const QString &typeName)
            {
                auto *qmlEngine = reinterpret_cast<QQmlEngine *>(data);
                if (!qmlEngine)
                    return;
                QMetaObject::invokeMethod(qmlEngine, "loadFromModule", Qt::BlockingQueuedConnection,
                    Q_ARG(QAnyStringView, uri), Q_ARG(QAnyStringView, typeName));
            });
        setCallback<bool, int>("WaitForExit", [this](void *data, int timeout)
            {
                if (timeout == 0)
                    return false;
                QElapsedTimer timer;
                timer.start();
                while (!timer.hasExpired(timeout) && !IQQmlEngine::exited)
                    QThread::usleep(100);
                return IQQmlEngine::exited;
            });
        setCallback<void>("ProcessEvents", [this](void *data)
            {
                QCoreApplication::processEvents();
            });
    }

    static void staticInit(QDotNetInterface *sta)
    {
        static IQQmlEngine qmlEngine;
        sta->setCallback<IQQmlEngine>("QQmlEngine_Get",
            [](void *) { return IQQmlEngine(qmlEngine); });
    }
};
#endif