1

我创建了一个自定义QDialog类并覆盖了closeEvent只是隐藏对话框,因为它是另一个小部件的子级。我的对话框只能在其父级关闭时关闭,而不是在它被接受、拒绝或用户单击关闭按钮时关闭。

这一切都很好,但现在我需要打开一个到数据库的连接,并且只有在对话框被破坏时才关闭它,而不仅仅是在它关闭时。

我的代码:

from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys

def Log_Closed():
    print "Bye bye"

class My_dlg(QDialog):
    def __init__(self, parent=None):
        QDialog.__init__( self, parent )

        #self.conn = open_connection()
        print "Connection Opened"

        close_btn  = QPushButton("Actually Close")
        QVBoxLayout(self).addWidget(close_btn)

        close_btn.clicked.connect(self.Actually_Close)
        self.destroyed.connect(Log_Closed)

    def Actually_Close(self):
        print "Actually Close"
        self.parent().close()

    def closeEvent(self, event):
        if event.type() == QEvent.Close:
            event.ignore()
            self.hide()
            print "hidden"

    # And I guess I need something like
    def destroyEvent(self, event):
        #self.conn.close()
        print "Connection Closed"
        event.accept()

if __name__ == "__main__":
    app = QApplication(sys.argv)
    main= QMainWindow()
    tsd = My_dlg(main)
    tsd.show()
    sys.exit(app.exec_())

有任何想法吗?

4

1 回答 1

8

要在删除 a 时接收通知QObject,请连接到其destroyed(QObject*)信号。

但是,在 Python 中,对象删除的可预测性不如 C++,因为对象是垃圾回收的。例如,请参阅Python 文档__del__()中与相关的所有警告。程序退出时它们可能不会被删除,这可能是您没有收到信号的原因。

您可以显式管理数据库连接,而不是依赖删除对话框。在这个简单的示例中,您甚至可以使用上下文管理器来编写更好的代码。


无论哪种方式,这是您的代码版本,其行为与您预期的一样:https ://gist.github.com/3827718

我所做的更改是:

  • 设置app.setQuitOnLastWindowClosed以确保在对话框关闭时应用程序不会退出。我认为这是您想要的行为,因为否则这个问题没有意义。

  • 在对话框上设置Qt.WA_DeleteOnCloseFalse以防止它在关闭时自行删除。这比覆盖更可取closeEvent

  • Actually_Close()中,对话框会自行删除(这也会关闭它)。这会触发destroyed信号。

使用此代码,当您单击按钮时,输出与您预期的一样:

Connection Opened
Actually Close
Bye bye
于 2012-10-02T15:34:00.820 回答