是否可以从内存中动态加载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这种类型?我认为Simple
(http://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类型的方法。 所以我的问题仍然存在。
答案 0 :(得分:7)
一种选择是为您的QML引擎提供自定义QNetworkAccessManager
,并通过createRequest()
实施自定义网址方案的处理。
E.g。你的createRequest()
实现将检查传递的URL是否有你的方案,让我们说“内存”,如果是,它需要URL的其余部分来决定调用哪个函数或传递哪些数据。
除此之外,它只是调用基础实现。
然后可以像这样加载您的主要QML文件
QQuickView view;
view.setSource(QUrl("memory://main.qml"));
如果未指定,QML中的所有网址都与当前文件相关,否则Simple
的查找应查找memory://Simple.qml