1

我可以QStatusBar通过输入单个字符串来显示消息,例如:

self.statusBar().showMessage("My message here, also show it for 1 sec", 1000)

在我的事件循环中,上述消息将在状态栏上显示 1000 毫秒。但是说"My message here",我有一个字符串列表,而不是一个字符串,我想当时显示一个。我可以通过给它一个延迟来通过一个循环来做到这一点time.sleep(1)- 但不幸的是,这会阻塞 gui,直到循环结束,我不希望那样。我可以将主事件循环和状态栏更新的过程分开吗?如果可以,如何?

下面的示例代码有一个按钮,当按下该按钮时,会生成三个不同的消息,存储在一个列表中。然后它们会显示在状态栏中,一次一个,并且在循环结束之前无法按下按钮。我想要的行为是该按钮是可点击的(gui 未被阻止),如果在显示以前的消息时单击它,则显示被中断并显示新消息。

---下面的示例代码---

import sys
import time
from PyQt5 import QtWidgets

class Window(QtWidgets.QMainWindow):
    """Main Window."""
    MSG_ID = 1
    def __init__(self, parent=None):
        """Initializer."""
        super().__init__(parent)
        #stuff
        self.messages = []
        self.setWindowTitle('Main Window')
        self.setStatusBar(QtWidgets.QStatusBar())
        self.statusBar().showMessage("My message first time")
        self.button = QtWidgets.QPushButton("Test",self)
        self.button.clicked.connect(self.handleButton)
        
    def handleButton(self):
        self.messages = [f"message_num {msg_id}" for msg_id in range(Window.MSG_ID,Window.MSG_ID+3)]
        Window.MSG_ID+=3
        self.updateStatus()
        print(self.messages)
        
    def updateStatus(self):
        self.statusBar().clearMessage()
        for item in self.messages:
            self.statusBar().showMessage(item,1000)
            time.sleep(1.2)
           
if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    win = Window()
    win.show()
    sys.exit(app.exec_())

上面的代码创建了一个消息列表

4

1 回答 1

2

无需为此使用计时器或睡眠,因为每次临时消息更改(包括删除时)时,状态栏都会发送messageChanged 信号。这可用于创建一个简单的回调循环来执行您想要的操作:

class Window(QtWidgets.QMainWindow):
    ...
    def __init__(self, parent=None):
        ...
        self.statusBar().messageChanged.connect(self.updateStatus)

    def handleButton(self):
        self.messages = [f"message_num {msg_id}" for msg_id in range(Window.MSG_ID,Window.MSG_ID+3)]
        Window.MSG_ID+=3
        self.updateStatus()

    def updateStatus(self, message=None):
        print('update-status:', (message, self.messages))
        if not message and self.messages:
            self.statusBar().showMessage(self.messages.pop(0), 1000)

这不会阻止 gui,并允许通过单击按钮随时重新启动消息序列。

于 2022-01-19T18:47:12.493 回答