Pyqt5 - 如何从 Secondary Window 返回隐藏的 Main Window?

Pyqt5 - how to go back to hided Main Window from Secondary Window?

如果我从第二个 window 单击 Back,程序将退出。在这种情况下如何返回 mainwindow?我假设我需要在那个 clickMethodBack 函数中添加更多代码。

import os
import PyQt5
from PyQt5 import QtCore, QtWidgets
from PyQt5.QtWidgets import QMainWindow, QWidget, QLabel, QPushButton
import time
from PyQt5.QtCore import QSize

class GUI_Window():
 
    def __init__(self):
        self.main_window()
        return

    def main_window(self):
        app = PyQt5.QtWidgets.QApplication(sys.argv)
        self.MainWindow = MainWindow_()                                                  
        self.MainWindow.show()                                                                    
        app.exec_()                                                                      
        return
                  

class MainWindow_(QMainWindow):

    def __init__(self):
        QMainWindow.__init__(self)

        self.TestAButton = QPushButton("TestA", self)
        self.TestAButton.clicked.connect(self.TestA_clickMethod)
        self.TestAButton.move(20, 0)

        self.CloseButton = QPushButton("Close", self)
        self.CloseButton.clicked.connect(self.Close_clickMethod)
        self.CloseButton.move(20, 40)

        self.TestingA = TestA_MainWindow()     
    def TestA_clickMethod(self):
        self.TestAButton.setEnabled(False)
        time.sleep(0.2)
        self.TestingA.show()
        self.hide()   

        try:
            if self.TestingA.back == True:
                self.show()
        except:
            None

    def Close_clickMethod(self):
        self.Test_Choice = 'Exit'
        self.close()


class TestA_MainWindow(QWidget):
    def __init__(self):
        super().__init__()
        self.setMinimumSize(QSize(980,700))
        self.setWindowTitle("TestA")

        self.Back_Button = False

        self.closeButton = QPushButton("Close", self)
        self.closeButton.clicked.connect(self.clickMethodClose)

        self.returnButton = QPushButton("Back", self)
        self.returnButton.clicked.connect(self.clickMethodBack)
        self.returnButton.move(0,30)

    def clickMethodClose(self):
        self.Back_Button = False
        self.close()

    def clickMethodBack(self):
        self.returnButton.setEnabled(False)
        time.sleep(0.5)
        self.back = True
        self.close()






# Run if Script
if __name__ == "__main__":
    main = GUI_Window()                                                   # Initialize GUI

您的代码有两个非常重要的问题。

  1. 您正在使用拦截功能,time.sleep;与几乎所有 UI 工具包一样,Qt 是 事件驱动的 ,这意味着它必须能够 不断地 接收和处理事件(来自系统或在用户交互之后):当某些东西 阻塞 事件队列时,它会完全冻结整个程序,直到该块释放控制;
  2. 您检查变量的时间过早:即使假设 sleep 可以工作,您也无法知道 window 是否已关闭 after睡眠定时器已结束;

解决方法是使用signals and slots。由于您需要知道第二个 window 何时使用“后退”按钮关闭,因此为第二个 window 创建一个自定义信号,只要按钮调用的函数关闭,该信号就会发出.

from PyQt5 import QtCore, QtWidgets

class MainWindow(QtWidgets.QMainWindow):
    def __init__(self):
        super().__init__()
        central = QtWidgets.QWidget()
        layout = QtWidgets.QHBoxLayout(central)
        self.testButton = QtWidgets.QPushButton('Test A')
        self.closeButton = QtWidgets.QPushButton('Close')
        layout.addWidget(self.testButton)
        layout.addWidget(self.closeButton)
        self.setCentralWidget(central)

        self.testButton.clicked.connect(self.launchWindow)
        self.closeButton.clicked.connect(self.close)

    def launchWindow(self):
        self.test = TestA_MainWindow()
        self.test.backSignal.connect(self.show)
        self.hide()
        self.test.show()

class TestA_MainWindow(QtWidgets.QWidget):
    backSignal = QtCore.pyqtSignal()
    def __init__(self):
        super().__init__()
        layout = QtWidgets.QHBoxLayout(self)
        self.closeButton = QtWidgets.QPushButton('Close')
        self.backButton = QtWidgets.QPushButton('Back')
        layout.addWidget(self.closeButton)
        layout.addWidget(self.backButton)

        self.closeButton.clicked.connect(self.close)
        self.backButton.clicked.connect(self.goBack)

    def goBack(self):
        self.close()
        self.backSignal.emit()


def GUI_Window():
    import sys
    app = QtWidgets.QApplication(sys.argv)
    mainWindow = MainWindow()
    mainWindow.show()
    sys.exit(app.exec_())


if __name__ == '__main__':
    GUI_Window()

备注:

  • 我删除了 GUI_Window class 并创建了一个函数,因为使用 class 并不是很有用;
  • 你应该总是喜欢 layout managers 而不是手动设置几何图形;
  • 不应将小部件作为直接子级添加到 QMainWindow 中,应始终使用中央小部件(参见示例中 central 的创建和使用);在 documentation;
  • 阅读更多相关信息
  • 只有 classes 和常量应大写,而变量、属性和函数的名称应始终以小写字母开头;