如何从Qt QuickWorkerScript
调用Python方法?
序言:
我已经设法通过注册一个QObject
子类从我的QML应用程序访问我的Python方法:
class Foo(QtCore.QObject):
@QtCore.pyqtSlot()
def bar(self):
pass # do stuff here
app = QtGui.QGuiApplication([])
QtQml.qmlRegisterType(Foo, 'Foo', 1, 0, 'Foo')
# more boilerplate Qt/Qt Quick
因此,从我的 QML 应用程序中,我可以成功调用Foo.bar()
。
据我了解,我应该从WorkerScript
调用任何长时间运行的函数。
问题:
如何使用WorkerThread
在后台线程中运行Foo.bar()
?
根据文档,WorkerScript
不能使用import
语法,所以我不确定如何在不导入的情况下访问Foo
。
更新:
我需要 UI 能够显示Foo.bar()
进度,因为该功能需要一点时间并执行多项操作。
向WorkerScript
发送信息的唯一方法是通过sendMessage()
:
sendMessage(jsobject message)
将给定的消息发送到另一个线程中的工作线程脚本处理程序。 其他工作线程脚本处理程序可以通过 onMessage() handler.
消息对象只能包含以下类型的值:
- 布尔值、数字、字符串
- JavaScript 对象和数组
- ListModel 对象(不允许使用任何其他类型的 QObject*)
所有对象和数组都将复制到消息中。除了 ListModel 对象之外,其他线程对消息中传递的对象所做的任何修改都不会反映在原始对象中。
但是,由于它读取所有元素都被复制(ListModel 类型的元素除外),因此无法使用从 QObject 类或其方法继承的任何对象。
你可以使进度成为pyqtProperty
,以便你可以将其公开给QML
,并使用一个槽从另一个线程更新其值,通过QThreadPool
QRunnable
,更新是通过QMetaObject::invokeMethod()
class Runnable(QRunnable):
def __init__(self, obj):
QRunnable.__init__(self)
# main thread
self.obj = obj
def run(self):
# another thread
self.obj.bar()
class Foo(QObject):
def __init__(self, *args, **kwags):
QObject.__init__(self, *args, **kwags)
self._progress = 0
@pyqtSlot()
def run_bar(self):
self.runnable = Runnable(self)
QThreadPool.globalInstance().start(self.runnable)
progressChanged = pyqtSignal(int)
@pyqtProperty(int, notify=progressChanged)
def progress(self):
return self._progress
@pyqtSlot(int)
def updateProgress(self, value):
if self._progress == value:
return
self._progress = value
self.progressChanged.emit(self._progress)
def bar(self):
for i in range(100):
QMetaObject.invokeMethod(self, "updateProgress",
Qt.QueuedConnection,
Q_ARG(int, i))
QThread.msleep(1000)
然后,您可以通过 run_bar () 启动它,并通过 progress 属性显示它:
import QtQuick.Controls 1.4
import QtQuick.Layouts 1.0
import QtQuick 2.0
import Foo 1.0
ApplicationWindow{
width: 300
height: 300
visible: true
Text{
id:txt
text: "press me to start"
anchors.fill: parent
verticalAlignment: Text.AlignVCenter
horizontalAlignment: Text.AlignHCenter
}
Foo{
id: foo
onProgressChanged: txt.text= progress
}
MouseArea {
anchors.fill: parent
onClicked: foo.run_bar()
}
statusBar: StatusBar {
RowLayout {
anchors.fill: parent
ProgressBar {
value: foo.progress
maximumValue: 100
anchors.fill: parent
}
}
}
}
完整的示例可以在以下链接中找到