PyQt4:创建返回参数的自定义对话框

Posted

技术标签:

【中文标题】PyQt4:创建返回参数的自定义对话框【英文标题】:PyQt4: Create a custom dialog that returns parameters 【发布时间】:2011-04-22 22:20:08 【问题描述】:

我正在尝试向我当前的 GUI 添加一个自定义对话框,可以启动该对话框供用户设置一些参数。理想情况下,我想使用 QtDesigner 创建自定义对话框。下面是pyuic4从QtDesigner ui代码中为对话框生成的代码。

from PyQt4 import QtCore, QtGui

class Ui_Dialog(object):
    def setupUi(self, Dialog):
        Dialog.setObjectName("Dialog")
        Dialog.resize(508, 300)
        self.buttonBox = QtGui.QDialogButtonBox(Dialog)
        self.buttonBox.setGeometry(QtCore.QRect(150, 250, 341, 32))
        self.buttonBox.setOrientation(QtCore.Qt.Horizontal)
        self.buttonBox.setStandardButtons(QtGui.QDialogButtonBox.Cancel|QtGui.QDialogButtonBox.Ok)
        self.buttonBox.setObjectName("buttonBox")
        self.label = QtGui.QLabel(Dialog)
        self.label.setGeometry(QtCore.QRect(10, 120, 181, 31))
        font = QtGui.QFont()
        font.setPointSize(16)
        self.label.setFont(font)
        self.label.setObjectName("label")
        self.sl_value = QtGui.QSlider(Dialog)
        self.sl_value.setGeometry(QtCore.QRect(220, 120, 161, 31))
        self.sl_value.setOrientation(QtCore.Qt.Horizontal)
        self.sl_value.setObjectName("sl_value")
        self.ed_value = QtGui.QLineEdit(Dialog)
        self.ed_value.setGeometry(QtCore.QRect(400, 120, 41, 31))
        font = QtGui.QFont()
        font.setPointSize(16)
        self.ed_value.setFont(font)
        self.ed_value.setObjectName("ed_value")
        self.retranslateUi(Dialog)
        QtCore.QObject.connect(self.buttonBox, QtCore.SIGNAL("accepted()"), Dialog.accept)
        QtCore.QObject.connect(self.buttonBox, QtCore.SIGNAL("rejected()"), Dialog.reject)
        QtCore.QMetaObject.connectSlotsByName(Dialog)


    def retranslateUi(self, Dialog):
        Dialog.setWindowTitle(QtGui.QApplication.translate("Dialog", "Dialog", None, QtGui.QApplication.UnicodeUTF8))
        self.label.setText(QtGui.QApplication.translate("Dialog", "Set example value:", None, QtGui.QApplication.UnicodeUTF8))

这是保存在Sub2.py 然后,在我的主要 python 文件中,添加

from Sub2 import Ui_Dialog

我使用以下代码创建了一个名为 StartSub2 的新类

class StartSub2(QtGui.QDialog):
    def __init__(self,parent=None):
        QtGui.QDialog.__init__(self,parent)
        self.ui = Ui_Dialog
        self.ui.setupUi(self)

最后,在我的主 GUI 中,有一个带有以下代码的函数应该启动对话框

def exampleSubGui(self):
    dialog = StartSub2(self)
    result = dialog.exec_()

请注意,对话框尚未完成。一旦我解决了如何启动它,我将为滑块和编辑框添加信号/插槽连接。另外,如果我理解正确的话,我需要重载accept() 方法来返回用户的输入。

我遇到的第一个问题是StartSub2__init__ 方法。我收到以下错误:

TypeError: unbound method setupUi() must be called with Ui_Dialog instance as
first argument (got StartSub2 instance instead)

我正在尝试采用与主 GUI 相同的方法,即使用以下代码

class StartQT4(QtGui.QMainWindow):
    def __init__(self, parent=None):
        QtGui.QWidget.__init__(self, parent)
        self.ui = Ui_MainWindow()
        self.ui.setupUi(self)

但这并没有抱怨setupUi() 得到一个StartQT4 实例而不是Ui_MainWindow 实例。谁能解释完成我想做的事情的正确方法?或者有人可以指出一个明确的例子或参考吗?如果您需要更多信息或说明,请告诉我。

【问题讨论】:

【参考方案1】:
class StartSub2(QtGui.QDialog, Ui_Dialog):
    def __init__(self,parent=None):
        QtGui.QDialog.__init__(self,parent)
        self.setupUi(self)

应该可以解决您让对话框初始化的第一个问题。

为了获取信息,我通常会在StartSub2 中添加一个名为getValues 的方法,即

def getValues(self):
    return somethingUseful

然后做

dlg = StartSub2()
if dlg.exec_():
    values = dlg.getValues()
    # Do stuff with values

【讨论】:

非常感谢。这正是我想要的。 值得注意的是,如果你 self.close() 对话框(因为从 .exec() 返回的值为 0),这将不起作用,你必须 self.accept()。起初,我从问题中的旧式 SIGNAL 语法中并不完全清楚。【参考方案2】:

只是想为设置带有返回值的自定义对话框提供我自己的答案(不回答代码特定的问题,Whatang 已经这样做了)。

我发现用class method 构造一个简单的对话框类会更简洁一些,它可以根据需要返回不同的内容。我最近一直在做这些!这个想法是类方法将构造对话框类的实例,并从实例中返回对象(在本例中为布尔ok),这或多或少是factory method(无论如何,据我所知,仍然对 OOP 有点陌生)。

这是一个非常简化的示例对话框。在对话框类中将其扩展到您需要的任何内容应该相对容易:

class OkDialog(QtGui.QDialog):
    def __init__(self, parent):
        super(OkDialog, self).__init__(parent)

        self.ok = False

        self.btn_ok = QtGui.QPushButton("Ok", self)
        self.btn_ok.clicked.connect(self.button_press)
        self.btn_cancel = QtGui.QPushButton("Cancel", self)
        self.btn_cancel.clicked.connect(self.button_press)
        self.btn_cancel.move(80, 0)

    def button_press(self):
        if self.sender() == self.btn_ok:
            self.ok = True
        self.close()

    @classmethod
    def isOkay(cls, parent):
        dialog = cls(parent)
        dialog.exec_()
        return dialog.ok

美妙之处在于,当需要构建此对话框时,您只需一行 OkDialog.isOkay(parent) 即可完成。将其组合成一个完整的工作示例:

import sys
from PyQt4 import QtCore, QtGui

class OkDialog(QtGui.QDialog):
    def __init__(self, parent):
        super(OkDialog, self).__init__(parent)

        self.ok = False

        self.btn_ok = QtGui.QPushButton("Ok", self)
        self.btn_ok.clicked.connect(self.button_press)
        self.btn_cancel = QtGui.QPushButton("Cancel", self)
        self.btn_cancel.clicked.connect(self.button_press)
        self.btn_cancel.move(80, 0)

    def button_press(self):
        if self.sender() == self.btn_ok:
            self.ok = True
        self.close()

    @classmethod
    def isOkay(cls, parent):
        dialog = cls(parent)
        dialog.exec_()
        return dialog.ok

class Ui_Dialog(QtGui.QDialog):
    def __init__(self):
        super(Ui_Dialog, self).__init__()

        button = QtGui.QPushButton("Launch custom dialog", self)
        button.pressed.connect(self.launch_dialog)

    def launch_dialog(self):
        print OkDialog.isOkay(self)

def main():
    app = QtGui.QApplication(sys.argv)
    ex = Ui_Dialog()

    ex.show()
    sys.exit(app.exec_())

if __name__ == "__main__":
    main()

【讨论】:

以上是关于PyQt4:创建返回参数的自定义对话框的主要内容,如果未能解决你的问题,请参考以下文章

根据模型预测,使用丢失的自定义函数的返回值

自定义类中的自定义对话框返回布尔值

引导表服务器端分页的自定义参数

将参数传递给 AsyncTask,并返回结果

是否有从 Windows 窗体中的自定义对话框返回值的标准方法?

MYSQL的自定义函数