使用lambda函数时,弹出窗口不再关闭

时间:2018-02-09 20:03:09

标签: python python-3.x lambda pyqt pyqt4

使用这个问题的答案: Python: PyQt Popup Window 我能够用一个带弹出窗口的按钮制作一个gui。

我现在要做的是按下弹出窗口中的按钮并将命令传递给MyPopup类中的函数。这很容易使用lambda函数完成,但是,当您按下主窗口中的按钮时,弹出窗口不再关闭,并且会创建一个新的弹出窗口实例,从而产生两个弹出窗口。根据我的理解,这是由于lambda函数产生的信号。有没有办法清除这个lambda函数,以便在按下主按钮时关闭旧实例并加载弹出窗口的新实例?

如果使用lambda无法做到这一点,是否有另一种方法可以将变量传递给函数以获得我想要的结果?

以下是一些示例屏幕截图,可以更好地说明我的问题:

在弹出窗口中运行没有lambda的脚本

在弹出窗口中使用lambda运行脚本

以下是上一个问题的修改后的弹出式代码:

import sys
from PyQt4.Qt import *

class MyPopup(QWidget):
    def __init__(self):
        QWidget.__init__(self)
        self.btn_popup = QPushButton("broken", self)
        self.btn_popup.clicked.connect(lambda state, x='lambda prevents refresh': self.function(x))

    def function(self, word):
        print('Now I dont close',word)

    def paintEvent(self, e):
        dc = QPainter(self)
        dc.drawLine(0, 0, 100, 100)
        dc.drawLine(100, 0, 0, 100)

class MainWindow(QMainWindow):
    def __init__(self, *args):
        QMainWindow.__init__(self, *args)
        self.cw = QWidget(self)
        self.setCentralWidget(self.cw)
        self.btn1 = QPushButton("Click me", self.cw)
        self.btn1.setGeometry(QRect(0, 0, 100, 30))
        self.connect(self.btn1, SIGNAL("clicked()"), self.doit)
        self.w = None

def doit(self):
    print ("Opening a new popup window...")
    self.w = MyPopup()
    self.w.setGeometry(QRect(100, 100, 400, 200))
    self.w.show()

class App(QApplication):
    def __init__(self, *args):
        QApplication.__init__(self, *args)
        self.main = MainWindow()
        self.main.show()

def main(args):
    global app
    app = App(args)
    app.exec_()

if __name__ == "__main__":
    main(sys.argv)

1 个答案:

答案 0 :(得分:0)

似乎如果lambda函数不存在弹出窗口被破坏,请通过添加以下内容来验证:

class MyPopup(QWidget):
    def __init__(self, i):
        [..]
        self.destroyed.connect(lambda: print("destroyed"))

在没有lambda的情况下,打印消息,而在另一种情况下则不打印。所以解决方案是使用deleteLater()方法手动销毁它:

def doit(self):
    print ("Opening a new popup window...")
    if self.w:
        self.w.deleteLater()
    self.w = MyPopup()
    self.w.setGeometry(QRect(100, 100, 400, 200))
    self.w.show()