PyQT4 - 替换主窗口上的小部件

时间:2011-11-24 20:11:59

标签: python pyqt qmainwindow

我是PyQT4的新手,我遇到了下面的问题。

某些应用程序必须在一个屏幕上收集用户的数据 点击按钮显示下一个屏幕。

main.py

app = QtGui.QApplication(sys.argv)
#here I create main window with inherited class
w = SAN_MainWindow()
#and apply some basic geometry
w.SetUI()
#here first screen rendered and button event applyed
w.ApplyWidget( SAN_Intro() )
w.show()
sys.exit(app.exec_())

正确渲染第一个小部件。

SAN_MainWindow.py:

class SAN_MainWindow(QtGui.QMainWindow):
    def __init__(self ):
        super(SAN_MainWindow, self).__init__()

    def SetUI(self):
        self.setObjectName(_fromUtf8("MainWindow"))
        self.resize(789, 602)
        self.setWindowTitle(QtGui.QApplication.translate("MainWindow", "MainWindow", None, QtGui.QApplication.UnicodeUTF8)) 
        self.central_widget = QtGui.QWidget(self)   
        self.central_widget.setObjectName(_fromUtf8("central_widget"))
        self.setCentralWidget(self.central_widget)

    def ApplyWidget( self, widget ):
        self.active_widget = widget
        self.active_widget.Setup(self.central_widget)
        self.active_widget.SetControls( self )

    def RemoveActiveWidget( self ):
        self.active_widget.widget().setParent(None)

    def ShowInstruction( self ):
        self.RemoveActiveWidget()
        self.ApplyWidget( SAN_Instruction() )
        #self.centralWidget().update() <- Problem

SAN_Intro.py:

class SAN_Intro(object):
    def __init__(self):
        super(SAN_Intro, self).__init__()

    def Setup(self, widget):
        self.gridLayoutWidget = QtGui.QWidget(widget)
        self.gridLayoutWidget.setGeometry(QtCore.QRect(80, 30, 638, 451))
        self.gridLayoutWidget.setObjectName(_fromUtf8("gridLayoutWidget"))
        self.gridLayout = QtGui.QGridLayout(self.gridLayoutWidget)
        self.gridLayout.setContentsMargins(-1, 16, 16, -1)
        self.gridLayout.setObjectName(_fromUtf8("gridLayout"))
        #a lot of form inputs instructions....
    def SetControls( self, main_window ):
        QtCore.QObject.connect(self.pushButton, QtCore.SIGNAL("clicked()"), main_window.ShowInstruction)

    def widget(self):
        return self.gridLayoutWidget

SAN_Instruction 与SAN_Intro几乎相同,但在安装程序中使用不同的命令:

class SAN_Instruction(object):

    def __init__(self):
        super(SAN_Instruction, self).__init__()

    def Setup(self, widget):
        self.verticalLayoutWidget = QtGui.QWidget(widget)
        self.verticalLayoutWidget.setGeometry(QtCore.QRect(40, 20, 591, 521))
        self.verticalLayoutWidget.setObjectName(_fromUtf8("verticalLayoutWidget"))
        self.verticalLayout = QtGui.QVBoxLayout(self.verticalLayoutWidget)
        self.verticalLayout.setMargin(0)
        self.verticalLayout.setObjectName(_fromUtf8("verticalLayout"))
        #a lot of form inputs instructions.... 

    def SetControls( self, main_window ):
        pass

    def widget(self):
        return self.verticalLayoutWidget

这里是问题: 我该如何显示第二个(使用SAN_Instruction生成)小部件? 我试图更新/ show / repaint mainwindow / centralwidget / parent。 但是我点击按钮的所有内容都是self.RemoveActiveWidget()。

也许我的观念不正确?

提前完成

2 个答案:

答案 0 :(得分:2)

好吧,你的代码不完整,但我建议你利用hide()方法。 您可以在按钮功能中定义它,以便隐藏所有不想显示的小部件,并在同一个函数上调用要在其位置显示的小部件。我这样做是因为它使我的项目更有条理,更容易跟踪和调试。此外,这还允许您在需要时使用show()方法再次显示小部件。

答案 1 :(得分:0)

您可以使用“QtGui.QStackedWidget()”。您将窗口拆分为多个帧,将每个帧添加到QStackedWidget()并使用函数setCurrentWidget()来选择要显示的帧(在帧之间切换)。

self.central_widget.addWidget(self.mainFrame)
self.central_widget.setCurrentWidget(self.mainFrame)