替换MainWindow中的CentralWidget

时间:2012-11-25 10:16:23

标签: python user-interface pyqt pyside

我是PySide的新手。我有一个主窗口对象,一次显示一个小部件。我一直在尝试更改QMainWindow类的中央窗口小部件,以便在按下按钮时替换窗口中的可见窗口小部件。问题是按下的按钮是在Widget类中,而不是在主窗口类中。

说...

class App(QtGui.QMainWindow):

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

        self.initUI()

    def initUI(self):

        self.statusBar().showMessage('Listo.') #Status Bar
        self.login_screen = LoginScreen()
        self.logged_in_screen = LoggedInScreen()

        self.setCentralWidget(self.login_screen)

        self.setGeometry(300, 300, 450, 600) #Window Size
        self.setWindowTitle('PyTransactio - Client') #Window Title
        self.setWindowIcon(QtGui.QIcon('icon.png')) #App Icon
        self.show()

按下的按钮位于login_screen实例中。单击按钮时调用的方法位于LoginScreen类:

def login(self):
        """ Send login data to the server in order to log in """

        #Process

        self.setParent(None)

将父窗口小部件设置为None会从主窗口中删除窗口小部件(login_screen)。当按下logged_in_screenloginButton窗口小部件内部)时,我应该怎么做才能将另一个窗口小部件(例如login_screen)作为主窗口的中央窗口小部件?

也许登录方法应该在主窗口类中?如果是这样,我如何将login_screen中按下的按钮与主窗口的方法连接?

2 个答案:

答案 0 :(得分:12)

您可以使用QStackedWidget作为中央窗口小部件,并在其中添加登录屏幕和“登录”屏幕。

示例用法:

from PyQt4 import QtCore, QtGui


class MainWindow(QtGui.QMainWindow):
    def __init__(self, parent=None):
        super(MainWindow, self).__init__(parent)
        self.central_widget = QtGui.QStackedWidget()
        self.setCentralWidget(self.central_widget)
        login_widget = LoginWidget(self)
        login_widget.button.clicked.connect(self.login)
        self.central_widget.addWidget(login_widget)
    def login(self):
        logged_in_widget = LoggedWidget(self)
        self.central_widget.addWidget(logged_in_widget)
        self.central_widget.setCurrentWidget(logged_in_widget)


class LoginWidget(QtGui.QWidget):
    def __init__(self, parent=None):
        super(LoginWidget, self).__init__(parent)
        layout = QtGui.QHBoxLayout()
        self.button = QtGui.QPushButton('Login')
        layout.addWidget(self.button)
        self.setLayout(layout)
        # you might want to do self.button.click.connect(self.parent().login) here


class LoggedWidget(QtGui.QWidget):
    def __init__(self, parent=None):
        super(LoggedWidget, self).__init__(parent)
        layout = QtGui.QHBoxLayout()
        self.label = QtGui.QLabel('logged in!')
        layout.addWidget(self.label)
        self.setLayout(layout)



if __name__ == '__main__':
    app = QtGui.QApplication([])
    window = MainWindow()
    window.show()
    app.exec_()

如果您不想使用此小部件,那么我认为每次更改中央小部件时都必须调用QMainWindow.setCentralWidget

至于login方法应该在哪里,取决于。您可以为主窗口定义一个简单的界面来添加/删除/显示特定的中央窗口小部件,并使用login LoginScreen方法调用它。通过这种方式,LoginScreen类不必了解实现细节,例如中央窗口小部件实际上是QStackedWidget还是以其他方式完成此事。

答案 1 :(得分:-2)

您可以使用QMainWindow.setCentralWidget执行此操作(重复):

#! /usr/bin/env python3

from PySide import QtGui
from PySide import QtCore

import sys 

app = QtGui.QApplication(sys.argv)
mw = QtGui.QMainWindow()
w2 = QtGui.QWidget()
pb = QtGui.QPushButton('push me', w2) 

l1 = QtGui.QLabel('orig')
l2 = QtGui.QLabel('changed')
mw.setCentralWidget(l1)
pb.clicked.connect(lambda: mw.setCentralWidget(l2))
mw.show()
w2.show()
sys.exit(app.exec_())