7

我对 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_screen)作为主窗口的中央小部件?loginButtonlogin_screen

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

4

2 回答 2

17

您可以使用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方法应该在哪里,这取决于。可能您可以为您的主窗口定义一个简单的界面来添加/删除/显示特定的中央小部件,并loginLoginScreen. 这样,LoginScreen类就不必知道实现细节,例如中央小部件是否实际上是 aQStackedWidget或者这件事是以其他方式完成的。

于 2012-11-25T12:30:48.220 回答
-3

您可以使用 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_())
于 2012-11-30T19:53:50.430 回答