从内存而不是URL动态加载QML

时间:2017-01-19 04:19:01

标签: c++ qt qml

是否可以从内存中动态加载QML而不是使用文件?我已经看到很多代码总是需要一个URL,如下所示:

<!DOCTYPE html>
<html ng-app="plunker">

<head>
  <meta charset="utf-8" />
  <title>AngularJS Plunker</title>
  <script>
    document.write('<base href="' + document.location + '" />');
  </script>
  <link rel="stylesheet" href="style.css" />
  <script data-require="angular.js@1.5.x" src="https://code.angularjs.org/1.5.8/angular.js" data-semver="1.5.8"></script>
  <script src="app.js"></script>
</head>

<body ng-controller="MainCtrl">
  <table>
    <tr ng-repeat="item in list" ng-cloak>
      <td>
        <select ng-model="item" ng-change="change()" ng-options="vendor.vendorName for vendor in vendorList track by vendor.vendorNo"></select>
      </td>
    </tr>
  </table>
</body>

</html>

在上面的示例中,将加载QGuiApplication app(argc, argv); QQmlEngine* engine = new QQmlEngine; QQmlComponent component(engine, QUrl(QStringLiteral("qrc:/main.qml"))); // ignoring error checking (make sure component is ready, compiles w/o errors, etc) QObject* object = component.create(); app.exec(); ,QML的神奇之处将确保解析任何类型。特别是,Qt认为文件的名称是QML类型,它可用于同一目录中的任何其他QML文件。因此,如果main.qml使用类型main.qml,它会查找文件Simple并将其用作类型定义(http://doc.qt.io/qt-5/qtqml-documents-definetypes.html)。

现在我会变得危险:我想使用内存中的QML而不是使用文件。这就是它的工作方式(请暂时忽略糟糕的设计,内存泄漏等):

Simple.qml

所以QByteArray SimpleQml() { return QByteArray("import QtQuick 2.7; Text {") + "text: \"Hello, World!\"}"; } QByteArray TextExampleQml() { return QByteArray("import QtQuick 2.7; Simple{") + "color: \"Red\"}"; } QObject* createObjectFromMemory(QByteArray qmlData, QQmlEngine* engine, QQmlComponent* componentOut) { componentOut = new QQmlComponent(engine); // note this line: load data from memory (no URL specified) componentOut->setData(qmlData, QUrl()); if (componentOut->status() != QQmlComponent::Status::Ready) { qDebug() << "Component status is not ready"; foreach(QQmlError err, componentOut->errors()) { qDebug() << "Description: " << err.description(); qDebug() << err.toString(); } qDebug() << componentOut->errorString(); return nullptr; } else { return component.create(); } } int main(int argc, char* argv[]) { QGuiApplication app(argc, argv); QQuickView view; QQmlComponent* component = nullptr; // works great: shows a small window with the text "Hello, World!" QObject* object = createObjectFromMemory(SimpleQml(), view.engine(), component); // problem: Simple is not a type // QObject* object = createObjectFromMemory(TextExampleQml(), view.engine(), component); // setContent() is marked as an internal Qt method (but it's public) // see source for qmlscene for why I'm using it here view.setContent(QUrl(), component, object); view.show(); app.exec(); // don't forget to clean up (if you're not doing a demo snippet of code) //delete object; //delete component; } 只是一个Text对象,带有&#34; Hello,World!&#34;作为文本。当使用SimpleQml()作为数据源调用createObjectFromMemory()时,会出现一个小窗口,其中包含文本&#34; Hello,World!&#34;。

我使用QQuickView :: setContent(),这是一个在Qt源中被标记为内部的函数,但它在qmlscene中使用,我试图实现一个这里的效果类似(https://code.woboq.org/qt5/qtdeclarative/tools/qmlscene/main.cpp.html#main)。

显然,问题是当我尝试使用SimpleQml()作为数据源来调用createObjectFromMemory()时。 QML引擎不知道TextExampleQml()应该是一种类型 - 如何告诉QML这种类型?我认为Simplehttp://doc.qt.io/qt-5/qqmlengine.html#qmlRegisterType)可能是一个很好的候选人,但我找不到一种方法来注册来自内存中的一大块QML的类型。我想说,&#34;嘿,我有这个QML代码块,它的类型是&#39;简单&#39;。&#34;

假设我可以轻松预测所有类型的依赖关系,并且能够知道加载QML内存块的正确顺序。我可以将SimpleQml()返回的QML代码注册为类型qmlRegisterType(将其保存在内存中,而不是将其保存到文件中)吗?如果有的话,我想了解QML引擎如何做到这一点。

我意识到我打破了一个黄金法则:不要从C ++进入QML(而是从QML进入C ++)。因此,如果任何人都有一个更清洁的JavaScript解决方案,我可以接受建议,但请记住我尝试从内存加载,而不是从文件中加载。

更新 正如评论中所指出的,有一个名为createQmlObject()的JavaScript函数可以从字符串创建一个QML对象,这与我在上面的C ++中所做的非常相似。但是,这会遇到同样的问题 - Simple创建的类型是......好吧,它不是一个类型,它是一个对象。我喜欢的是一种将内存中QML字符串注册为QML类型的方法。 所以我的问题仍然存在。

1 个答案:

答案 0 :(得分:7)

一种选择是为您的QML引擎提供自定义QNetworkAccessManager,并通过createRequest()实施自定义网址方案的处理。

E.g。你的createRequest()实现将检查传递的URL是否有你的方案,让我们说“内存”,如果是,它需要URL的其余部分来决定调用哪个函数或传递哪些数据。

除此之外,它只是调用基础实现。

然后可以像这样加载您的主要QML文件

QQuickView view;
view.setSource(QUrl("memory://main.qml"));

如果未指定,QML中的所有网址都与当前文件相关,否则Simple的查找应查找memory://Simple.qml