问题描述
在窗口中,有两个事件:
1.编辑label后,点击空白位置会自动触发参数修改函数,如果参数有问题会弹出一个弹窗警告
2.button按钮,点击按钮会触发点击事件(与1的功能不一样)
错误修改label的值后,直接点击button按钮:
弹出一个弹窗警告,然后button的点击事件没有触发
原因分析
弹窗警告使用了:
QMessageBox.warning(self, 'warning','warning message')
QMessageBox.warning 默认使用了 Qt.ApplicationModal,应用程序模态会阻止和任何其他窗口交互,然后导致事件被吞了
解决方案
第一种,可以想到去重载 QWidget 或者是 QMessageBox 的父类 QDialog,但是比较麻烦要设置字体、label位置大小,button位置大小、弹窗大小等,实现完觉得太丑了我把代码删了。
第二种,由于第一种太麻烦,我只需要有个窗口和Q MessageBox.warning 的功能、外观一样,并且不会阻塞界面的事件,于是我翻了 QMessageBox 的文档,并实践证明可以直接new(实例化)一个 QMessageBox。
阻塞问题跟 WindowModality 有关
参数 | 解释 |
---|---|
Qt.NonModal | 非模态,不影响程序的其他窗口交互 |
Qt.WindowModa | 窗口模态,在未处理完当前对话框时,阻止对话框的父进程的交互 |
Qt.ApplicationModal | 应用程序模态,默认,阻止和任何其他窗口交互 |
最开始我是写:
try:
# 抛异常
raise
except:
# 实例化
self.msg = QMessageBox()
# 设置非模态
self.msg.setWindowModality(Qt.NonModal)
# 弹窗警告
self.msg.QMessageBox.warning(self, '警告','弹窗警告')
... # pass
上面这段代码等价于直接写了个:QMessageBox.warning(self, ‘警告’,‘弹窗警告’)
因为QMessageBox() 和 QMessageBox.warning() 实现的功能不一样,QMessageBox.warning 更像是执行了一个静态成员函数,修改 QMessageBox 的参数没有对 warning 产生影响。
正确写法是:
try:
# 抛异常
raise
except:
# 实例化
self.msg = QMessageBox()
# 设置非模态
self.msg.setWindowModality(Qt.NonModal)
# 设置弹窗标题和内容
self.msg.setWindowTitle('警告')
self.msg.setText('弹窗警告')
# 设置弹窗的按钮为OK,StandardButtons采用位标记,可以用与运算添加其他想要的按钮
self.msg.setStandardButtons(QMessageBox.Ok)
# 显示窗口
self.msg.show()
... # pass
QMessageBox 的实例化一定要赋给 self.xx 的内部成员,没有self它将会变成形参(临时变量),在我的电脑上我感觉弹窗没有弹出,实际上弹窗是一闪而过,加了 self 就会让弹窗保留在屏幕上