QML动态对象管理
创始人
2024-05-29 03:01:20
0

QML中有多种方式来动态创建和管理QML对象:

  • Loader (加载器)
  • Repeater(复制器)
  • ListView,GridWiew,PethView(视图) (之后会介绍)

 使用加载器:

Rectangle{id:rect1width: 100height: 100color: "blue"Component{id:com1Rectangle{height: 50width: 50color: "red"}}}Loader{sourceComponent: com1}//加载一个com1

使用复制器:

Grid{spacing: 10columns: 5//五列//复制器Repeater{model:25//生成25个Rectangle{width:50;height: 50color: "lightGreen"Text{anchors.centerIn: parentfont{bold: true;pixelSize: 20}text:index//获取编号}}}}

 

使用JavaScript代码创建动态QML对象:

QML支持从JavaScript中动态创建对象。这对于在必要时延迟对象的实例化非常有用,从而缩短应用程序启动时间。它还允许动态创建视觉对象并将其添加到场景中,以响应用户输入或其他事件

动态创建对象的方法:

  • 调用Qt.createComponent()来动态创建Component对象
  • 使用Qt.createQmlObject() 从QML字符串创建对象,
  • 如果在 QML 文档中定义了现有组件,并且希望动态创建该组件的实例,则创建组件会更好。否则,当对象 QML 本身在运行时生成时,从 QML 字符串创建对象很有用。

动态创建组件:

要动态加载 QML 文件中定义的组件,请在 Qt对象中调用 Qt.createComponent() 函数。此函数将 QML 文件的 URL 作为其唯一参数,并从此 URL 创建Component对象。

拥有组件之后,可以调用createObject()函数创建该组件的一个实例,此函数可以接收一个或两个参数:

  1.  指定新对象的父对象:父对象可以是图形对象或非图形对象。只有具有图形父对象的图形对象才会渲染到 QtQuick 可视化画布。如果您希望稍后设置父级,则可以安全地设置null作为函数参数。 
  2. (可选参数)是定义对象的初始任何属性值的属性-值对的映射。此参数指定的属性值在对象创建完成之前应用于对象,从而避免在必须初始化特定属性以启用其他属性绑定时可能发生的绑定错误。此外,与在创建对象后定义属性值和绑定相比,性能优势很小。

使用时需要注意的情况: 

 QML文件可以有两种:

  1. 本地文件的话可以直接创建实例,不需要等待
  2. 如果是网上加载的文件的话,需要判断component的状态,就绪后才能创建实例

创建一个 myWidget.qml文件

import QtQuick 2.9Rectangle{width: 100;height: 100;color: "red"
}

主qml文件

import QtQuick 2.9
import "textJs.js" as Logic//导入js文件
import QtQuick.Window 2.2Window {id:window1visible: truewidth: 640height: 480title: qsTr("Hello World")Rectangle{id:appWindowwidth: 300;height: 300color: "lightBlue"Component.onCompleted: Logic.createSpriteObjects()//调用函数}}

textJs.js文件

var component;//接收创建的对象
var sprite;//接收实例function createSpriteObjects() {component = Qt.createComponent("myWidget.qml");//创建一个对象if (component.status == Component.Ready)//如果就绪finishCreation();//直接创建实例elsecomponent.statusChanged.connect(finishCreation);//等待就绪后创建实例
}function finishCreation() {if (component.status == Component.Ready) {sprite = component.createObject(appWindow, {x: 100, y: 100});//创建实例if (sprite == null) {// Error Handlingconsole.log("Error creating object");}} else if (component.status == Component.Error) {// Error Handlingconsole.log("Error loading component:", component.errorString());}
}

运行结果 :

从字符串中创建对象:

QML 直到运行时才定义,则可以使用 Qt.createQmlObject() 函数从 QML 字符串创建 QML 对象

Qt.createQmlObject(参数1,参数2,参数3) 

  •  参数1:创建的QML字符
  • 参数2:父类
  • 参数3:要与新对象关联的文件路径;这用于错误报告
var newObject = Qt.createQmlObject
('import QtQuick 2.0; Rectangle {color: "red"; width: 20; height: 20}',parentItem,"dynamicSnippet1");

维护动态创建的对象:

管理动态创建的对象时,必须确保创建上下文的寿命超过创建的对象。否则,如果首先销毁创建上下文,则动态对象中的绑定和信号处理程序将不再工作。

创建上下文取决于对象的创建方式:

  • 使用Qt.createComponent(),则创建上下文是调用此方法的QQmlContext
  • 使用Qt.createQmlObject(),则创建上下文是传递给此方法的父对象的上下文
  • 如果定义了一个Conponent{},然后在其上调了createObject(),创建上下文就是该Component

另外,请注意,虽然动态创建的对象可能与其他对象相同,但它们在 QML 中没有 id。

动态删除对象:

在许多用户界面中,将可视对象的不透明度设置为 0 或将可视对象移出屏幕而不是将其删除就足够了。但是,如果您有大量动态创建的对象,则删除未使用的对象可能会获得有价值的性能优势。

请注意,切勿手动删除由方便的 QML 对象工厂(如加载程序和转发器)动态创建的对象。此外,还应避免删除不是您自己动态创建的对象。

使用destroy()函数来删除对象,这个函数有一个可选的参数,可以用来设置在销毁该对象前的延迟时间(毫秒),默认为0。

例子:

application.qml

import QtQuick 2.0Item {id: containerwidth: 500; height: 100Component.onCompleted: {var component = Qt.createComponent("SelfDestroyingRect.qml");//创建对象for (var i=0; i<5; i++) {var object = component.createObject(container);//创建实例object.x = (object.width + 10) * i;}}
}

SelfDestroyingRect.qml

import QtQuick 2.0Rectangle {id: rectwidth: 80; height: 80color: "red"NumberAnimation on opacity {to: 0duration: 1000onRunningChanged: {if (!running) {  //当不在运行时console.log("Destroying...")rect.destroy();//销毁对象}}}
}

使用:Qt.createQmlObject()创建的对象同样可以使用destroy()销毁

var newObject = Qt.createQmlObject('import QtQuick 2.0; Rectangle {color: "red"; width: 20; height: 20}',parentItem,"dynamicSnippet1");
newObject.destroy(1000);

参考文档:

Dynamic QML Object Creation from JavaScript | Qt QML 5.15.12 

相关内容

热门资讯

苹果系统安卓爱思助手,系统兼容... 你有没有发现,手机的世界里,苹果系统和安卓系统就像是一对欢喜冤家,总是各有各的粉丝,各有各的拥趸。而...
安卓系统占用很大内存,揭秘内存... 手机里的安卓系统是不是让你感觉内存不够用,就像你的房间堆满了杂物,总是找不到地方放新东西?别急,今天...
安卓系统p30,安卓系统下的摄... 你有没有发现,最近安卓系统P30在手机圈里可是火得一塌糊涂呢!这不,我就来给你好好扒一扒这款手机的那...
siri被安卓系统进入了,智能... 你知道吗?最近科技圈可是炸开了锅,因为一个大家伙——Siri,竟然悄悄地溜进了安卓系统!这可不是什么...
最强挂机系统和安卓区别,揭秘安... 亲爱的读者,你是否曾在游戏中遇到过这样的困扰:一边想要享受游戏带来的乐趣,一边又不想放弃手中的零食或...
安卓系统为什么设系统盘,保障稳... 你有没有想过,为什么安卓系统里会有一个叫做“系统盘”的东西呢?这可不是随便设置的,背后可是有大学问的...
王者怎么加安卓系统的,轻松提升... 你有没有想过,你的手机里那款超酷的王者荣耀,怎么才能让它更好地在你的安卓系统上运行呢?别急,今天就来...
安卓手机系统怎么开热点,共享网... 你有没有想过,当你身处一个没有Wi-Fi信号的地方,而你的安卓手机里却存满了精彩视频和游戏时,是不是...
安卓系统11的平板电脑,性能升... 你有没有发现,最近平板电脑市场又热闹起来了?没错,安卓系统11的新一代平板电脑正在悄悄地走进我们的生...
安卓手机系统创始人,安卓手机系... 你有没有想过,那些陪伴我们每天生活的安卓手机,它们的灵魂是谁赋予的呢?没错,就是那位神秘而又传奇的安...
安卓11系统速度提升,体验再升... 你知道吗?最近安卓系统又升级啦!这次可是直接跳到了安卓11,听说速度提升了不少呢!是不是很心动?那就...
安卓5.1原生系统设置apk,... 你有没有想过,你的安卓手机里那些看似普通的设置,其实隐藏着不少小秘密呢?今天,就让我带你一探究竟,揭...
手机安卓系统玩音游,畅享指尖音... 你有没有发现,现在手机上的游戏种类越来越丰富,尤其是音游,简直让人爱不释手!今天,就让我来给你详细介...
安卓系统与win10,系统融合... 你有没有想过,为什么你的手机里装的是安卓系统,而电脑上却是Windows 10呢?这两种操作系统,就...
苹果系统王者安卓系统可以登吗,... 你有没有想过,为什么苹果系统的手机那么受欢迎,而安卓系统的手机却也能在市场上占有一席之地呢?今天,咱...
安卓系统怎么重制系统还原,安卓... 手机用久了是不是感觉卡得要命,想给它来个大变身?别急,今天就来教你怎么给安卓手机重置系统,让它焕然一...
安卓9系统怎样应用分身,轻松实... 你有没有发现,手机里的APP越来越多,有时候一个APP里还要处理好多任务,分身功能简直就是救星啊!今...
获取安卓系统的ip地址,轻松获... 你有没有想过,你的安卓手机里隐藏着一个神秘的IP地址?没错,就是那个能让你在网络世界里找到自己的小秘...
LG彩电安卓系统升级,畅享智能... 你家的LG彩电是不是最近有点儿“闹别扭”,屏幕上时不时地跳出个升级提示?别急,今天就来给你详细说说这...
阴阳师安卓苹果系统,安卓与苹果... 亲爱的玩家们,你是否曾在深夜里,手握手机,沉浸在阴阳师的神秘世界?今天,就让我带你一起探索这款风靡全...