繁体   English   中英

pyqt5 - 关闭/终止申请

[英]pyqt5 - closing/terminating application

我正在通过此处找到的 pyqt5 教程工作Zetcode,PyQt5

作为我自己的练习,我正在尝试扩展一个示例,以便无论使用何种方法关闭应用程序,我都会看到相同的对话框消息框:

  • 单击标题栏中的“X”按钮(按预期工作)
  • 单击“关闭”按钮(产生属性错误)
  • 按“退出”键(有效但不确定如何/为什么)

对话框消息框在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()passacceptignore ),如下所示:

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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM