[英]pyqt5 - closing/terminating application
我正在通过此处找到的 pyqt5 教程工作Zetcode,PyQt5
作为我自己的练习,我正在尝试扩展一个示例,以便无论使用何种方法关闭应用程序,我都会看到相同的对话框消息框:
对话框消息框在closeEvent
方法中实现,最后提供了完整的脚本。
我有两个问题:
1. 单击“关闭”按钮时,我想调用包括消息框对话框在内的closeEvent
方法,而不是退出。
我已经为“关闭”按钮替换了一行示例代码:
btn.clicked.connect(QCoreApplication.instance().quit)
相反,我试图调用已经实现了我想要的对话框的closeEvent
方法:
btn.clicked.connect(self.closeEvent)
但是,当我运行脚本并单击“关闭”按钮和 select 对话框中生成的“关闭”选项时,我得到以下信息:
Traceback (most recent call last):
File "5-terminator.py", line 41, in closeEvent
event.accept()
AttributeError: 'bool' object has no attribute 'accept'
Aborted
谁能告诉我我做错了什么以及这里需要做什么?
2. 当以某种方式按下转义键时,会出现消息框对话框并且工作正常。
好的,它很好用,但我想知道在CloseEvent
方法中定义的消息框功能是如何以及为什么在keyPressEvent
方法中调用的。
完整脚本如下:
import sys
from PyQt5.QtWidgets import (
QApplication, QWidget, QToolTip, QPushButton, QMessageBox)
from PyQt5.QtCore import QCoreApplication, Qt
class Window(QWidget):
def __init__(self):
super().__init__()
self.initUI()
def initUI(self):
btn = QPushButton("Close", self)
btn.setToolTip("Close Application")
# btn.clicked.connect(QCoreApplication.instance().quit)
# instead of above button signal, try to call closeEvent method below
btn.clicked.connect(self.closeEvent)
btn.resize(btn.sizeHint())
btn.move(410, 118)
self.setGeometry(30, 450, 500, 150)
self.setWindowTitle("Terminator")
self.show()
def closeEvent(self, event):
"""Generate 'question' dialog on clicking 'X' button in title bar.
Reimplement the closeEvent() event handler to include a 'Question'
dialog with options on how to proceed - Save, Close, Cancel buttons
"""
reply = QMessageBox.question(
self, "Message",
"Are you sure you want to quit? Any unsaved work will be lost.",
QMessageBox.Save | QMessageBox.Close | QMessageBox.Cancel,
QMessageBox.Save)
if reply == QMessageBox.Close:
event.accept()
else:
event.ignore()
def keyPressEvent(self, event):
"""Close application from escape key.
results in QMessageBox dialog from closeEvent, good but how/why?
"""
if event.key() == Qt.Key_Escape:
self.close()
if __name__ == '__main__':
app = QApplication(sys.argv)
w = Window()
sys.exit(app.exec_())
希望有人能抽空赐教。
你的第二个问题回答了第一个问题。
重新实现的keyPressEvent
方法调用close()
,它将QCloseEvent
发送到窗口小部件。 随后,将使用该事件作为其参数调用窗口小部件的closeEvent
。
所以你只需要将按钮连接到widget的close()
插槽,一切都会按预期工作:
btn.clicked.connect(self.close)
与X
按钮不同,您的自定义按钮似乎没有通过一个close event
只是一个bool
。 这就是为什么这个练习应该适用于X
按钮但不适用于普通按钮。 在任何情况下,对于您的第一个问题,您可以使用destroy()
并pass
( accept
和ignore
),如下所示:
import sys
from PyQt5.QtWidgets import (
QApplication, QWidget, QToolTip, QPushButton, QMessageBox)
from PyQt5.QtCore import QCoreApplication, Qt
class Window(QWidget):
def __init__(self):
super().__init__()
self.initUI()
def initUI(self):
btn = QPushButton("Close", self)
btn.setToolTip("Close Application")
# btn.clicked.connect(QCoreApplication.instance().quit)
# instead of above button signal, try to call closeEvent method below
btn.clicked.connect(self.closeEvent)
btn.resize(btn.sizeHint())
btn.move(410, 118)
self.setGeometry(30, 450, 500, 150)
self.setWindowTitle("Terminator")
self.show()
def closeEvent(self, event):
"""Generate 'question' dialog on clicking 'X' button in title bar.
Reimplement the closeEvent() event handler to include a 'Question'
dialog with options on how to proceed - Save, Close, Cancel buttons
"""
reply = QMessageBox.question(
self, "Message",
"Are you sure you want to quit? Any unsaved work will be lost.",
QMessageBox.Save | QMessageBox.Close | QMessageBox.Cancel,
QMessageBox.Save)
if reply == QMessageBox.Close:
app.quit()
else:
pass
def keyPressEvent(self, event):
"""Close application from escape key.
results in QMessageBox dialog from closeEvent, good but how/why?
"""
if event.key() == Qt.Key_Escape:
self.close()
if __name__ == '__main__':
app = QApplication(sys.argv)
w = Window()
sys.exit(app.exec_())
对于你的第二个问题,Qt具有默认行为,具体取决于Widget(对话框可能有另一个,当你的Message Dialog打开时,尝试按Esc
键只是为了查看)。 当您确实需要覆盖Esc
行为时,您可以尝试这样做:
def keyPressEvent(self, event):
if event.key() == QtCore.Qt.Key_Escape:
print("esc")
正如你最终会在ZetCode中看到的那样。
关闭事件()
在右上方的勾号代码中,请检查 closeEvent 它是相同的,否则浪费时间进行研究。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.