当我的应用程序做一些耗时的事情时,我想在应用程序忙碌时向用户显示一个消息框。我不想要任何按钮(例如OK
或Cancel
),也不能在消息框上调用exec_()
,因为那是阻止的。
我检查了许多qt
站点,我需要的代码似乎归结为:
message_box = QMessageBox()
message_box.setText(str('Reading Device, Please Wait...'))
message_box.show()
# do work here
message_box.close()
运行代码时,我会得到消息框,但没有文本。我在做什么错?
我在下面包括一个工作示例:
#!/usr/bin/env python3
import sys
import time
from PySide2.QtWidgets import (QLineEdit, QPushButton, QApplication,
QVBoxLayout, QDialog, QMessageBox)
class Form(QDialog):
def __init__(self, parent=None):
super(Form, self).__init__(parent)
self.button = QPushButton("Click Me")
layout = QVBoxLayout()
layout.addWidget(self.button)
self.setLayout(layout)
# Add button signal to dowork slot
self.button.clicked.connect(self.dowork)
def dowork(self):
message_box = QMessageBox()
message_box.setText(str('Reading Device, Please Wait...'))
message_box.show()
delay = 2.5
while delay:
sys.stdout.write('Working...n')
time.sleep(0.5) # do some time-consuming stuff...
delay -= 0.5
message_box.close()
if __name__ == '__main__':
app = QApplication(sys.argv)
form = Form()
print('starting app...')
form.show()
sys.exit(app.exec_())
如果单击按钮,则弹出消息框并在完成"工作"时显示。完成"工作"完成后,消息框再次消失 - 应该。但是没有文字在消息框中显示。
这里有一个类似的问题:qmessagebox-not-show-text-ther-ther-tall-how,但这没有回答我的问题。
您无法完成消耗大量时间(超过30毫秒)的任务,因为它阻止GUI Eventloop阻止QT正常工作,而是使用旁边的线程从另一个线程更新GUI的信号:
import sys
import threading
import time
from PySide2 import QtCore, QtWidgets
class Form(QtWidgets.QDialog):
started = QtCore.Signal()
finished = QtCore.Signal()
def __init__(self, parent=None):
super(Form, self).__init__(parent)
self.button = QtWidgets.QPushButton("Click Me")
layout = QtWidgets.QVBoxLayout(self)
layout.addWidget(self.button)
# Add button signal to dowork slot
self.button.clicked.connect(self.on_clicled)
self._message_box = QtWidgets.QMessageBox()
self._message_box.setText(str('Reading Device, Please Wait...'))
self._message_box.setStandardButtons(QtWidgets.QMessageBox.NoButton)
self.started.connect(self._message_box.show)
self.finished.connect(self._message_box.accept)
@QtCore.Slot()
def on_clicled(self):
thread = threading.Thread(target=self.dowork, daemon=True)
thread.start()
def dowork(self):
delay = 2.5
self.started.emit()
while delay:
sys.stdout.write('Working...n')
time.sleep(0.5) # do some time-consuming stuff...
delay -= 0.5
self.finished.emit()
if __name__ == '__main__':
app = QtWidgets.QApplication(sys.argv)
form = Form()
print('starting app...')
form.show()
sys.exit(app.exec_())
您几乎应该使用qprogressdialog。将"不确定"外观设置为最小值和最大值。
https://doc.qt.io/qt-5/qprogressdialog.html
尽管在您的setText()
之后强迫重新绘制可能会起作用:
self._message_box.repaint()
QtWidgets.QApplication.processEvents()