天天看點

第9天 | 28天學會PyQt5,資料傳遞事件

在開發桌面應用程式時,經常需要在兩個界面之間做資料的傳遞,在PyQt5中,你可以用pyqtSignal進行實作。通過pyqtSignal可以聲明不帶參數和帶參數的執行個體對象,帶參數的執行個體對象可以整型、字元型、清單和字典等資料類型。

pyqtSignal對象用connect()方法綁定多個處理函數,在emit()方法發送資料時,綁定的函數都會收到。了解過設計模式的朋友應該知道,這是觀察者模式的應用。記得将connect()方法放在視窗顯示方法show()之前,否則不起作用。

讓打開的視窗處于模态,需要在show()之前,用setWindowModality(Qt.ApplicationModal)方法進行設定。

視窗控件的布局,用了move(x,y)方法進行x、y坐标絕對定位的方式,這種方式在固定視窗,即大小不變的應用程式中沒多大問題,如果想要用在自動适配視窗大小方面,則用布局管理更加合适。

程式清單:data_event.py

import sys
from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel, QLineEdit,
  \QMessageBox, QDesktopWidget, QPushButton, QDialog
from PyQt5.QtCore import Qt, pyqtSignal
from PyQt5.QtGui import QPalette


# 激活視窗
class ActivateWindow(QDialog):
    # 聲明pyqtSignal
    signal = pyqtSignal(int)

    def __init__(self):
        super().__init__()
        self.init_ui()

    def init_ui(self):
        self.title = QLabel(self)
        self.title.move(55, 50)
        palette = QPalette()
        palette.setColor(QPalette.WindowText, Qt.red)
        self.title.setPalette(palette)
        info = QLabel(self)
        info.move(30, 120)
        # 激活碼和操作按鈕
        info.setText("激活碼:")
        self.code = QLineEdit(self)
        self.code.setFixedSize(160, 22)
        self.code.setPlaceholderText("請輸入激活碼")
        self.code.move(80, 118)
        submit_btn = QPushButton(self)
        submit_btn.setText("激活")
        submit_btn.move(50, 200)
        close_btn = QPushButton(self)
        close_btn.setText("退出")
        close_btn.move(150, 200)
        # 控件綁定方法
        submit_btn.clicked.connect(self.activate)
        close_btn.clicked.connect(self.close_win)
        self.setWindowFlags(Qt.CustomizeWindowHint |
                            Qt.MSWindowsFixedSizeDialogHint | 
                            Qt.WindowStaysOnTopHint)
        self.setWindowTitle("系統激活")
        self.resize(280, 380)

    # 接受從父視窗發送過來的資料
    def get_data(self, num):
        self.title.setText("%d個月的試用期已到,請激活" % num)

    # 激活操作
    def activate(self):
        if self.code.text().strip() == "":
            QMessageBox.information(self, "退出提醒", "激活碼不可為空",
                                    QMessageBox.Ok)
        else:
            # 發送到主視窗
            self.signal.emit(100)
            self.close()

    # 退出視窗
    def close_win(self):
        self.close()


# 繼承QWidget
class MainWindow(QMainWindow):
    # 聲明pyqtSignal
    signal = pyqtSignal(int)

    def __init__(self):
        super().__init__()
        self.init_ui()

    def init_ui(self):
        # 建立狀态欄
        self.status = self.statusBar()
        self.status.showMessage("等待訓示")
        # 建立按鈕
        activate_btn = QPushButton(self)
        activate_btn.setFixedSize(160, 80)
        activate_btn.setText("軟體試用期已過\n請點選進行激活")
        palette = QPalette()
        palette.setColor(QPalette.ButtonText, Qt.red)
        activate_btn.setPalette(palette)
        activate_btn.setAutoFillBackground(True)
        activate_btn.move(100, 100)
        activate_btn.clicked.connect(self.openWindow)
        # 調整視窗大小
        self.resize(900, 500)
        # 視窗居中
        self.center()
        # 視窗标題
        self.setWindowTitle("管理視窗")
        # 顯示視窗
        self.show()

    # 操作結果
    def result(self, flag):
        self.status.showMessage("恭喜你,激活成功了!")

    def update(self, num):
        self.status.showMessage("發送到子視窗的資料是:%d" % num)

    # 打開新視窗
    def openWindow(self):
        activate_win = ActivateWindow()
        # 綁定要通知的方法(目前方法和子視窗),支援多個
        self.signal.connect(self.update)
        self.signal.connect(activate_win.get_data)
        # 發送資料
        self.signal.emit(3)
        # 子視窗的signal綁定本視窗的方法
        activate_win.signal.connect(self.result)
        # 設定模态視窗
        activate_win.setWindowModality(Qt.ApplicationModal)
        activate_win.show()
        activate_win.exec_()

    # 關閉視窗
    def closeEvent(self, e):
        result = QMessageBox.information(self, "退出提醒", 
                                         "你确定要退出系統?",
                                         QMessageBox.Yes | 
                                         QMessageBox.No)
        if result == QMessageBox.Yes:
            print("執行退出動作")
            self.close()

    # 實作居中
    def center(self):
        f = self.frameGeometry()
        c = QDesktopWidget().availableGeometry().center()
        f.moveCenter(c)
        self.move(f.topLeft())


if __name__ == "__main__":
    app = QApplication(sys.argv)
    w = MainWindow()
    sys.exit(app.exec_())
           

運作程式之後,彈出的視窗如下:

第9天 | 28天學會PyQt5,資料傳遞事件

好了,資料傳遞事件的内容就說到這了,關注我,下一節更精彩。

碼字不易,你的關注和轉發是對我最大的鼓勵,謝謝!

一個當了10年技術總監的老家夥,分享多年的程式設計經驗。想學程式設計的朋友,關注我,你就賺到了。我正在分享Python、前端、Java和App方面的幹貨了。趕緊來圍觀啊!!!