PyQt5每天必學(xué)之事件與信號
這一部分我們將探索 PyQt5 的事件和信號是如何在應(yīng)用程序中實現(xiàn)的。
Events事件
所有的GUI應(yīng)用程序都是事件驅(qū)動的。應(yīng)用程序事件主要產(chǎn)生自用戶,但它們也可通過其他方法來產(chǎn)生,例如一個互聯(lián)網(wǎng)連接,一個窗口管理器,或計時器。當我們調(diào)用應(yīng)用程序的exec_()方法,應(yīng)用程序進入主循環(huán)。主循環(huán)檢測各種事件,并把它們發(fā)送到事件對象。
在事件模型中,有三個參與者:
- event source(事件源)
- event object(事件對象)
- event target(事件目標)
事件源是對象的狀態(tài)改變而產(chǎn)生事件。事件對象(事件)是封裝在事件源中狀態(tài)變化的對象。事件目標是希望被通知的對象。事件源對象代表處理一個事件到事件目標的任務(wù)。
PyQt5使用獨特的信號和槽機制來處理事件。信號和槽用于對象之間的通信,當一個特定的事件發(fā)生時,信號被發(fā)射。槽可以是任意的Python調(diào)用。信號發(fā)射時連接到槽被調(diào)用。
Signals & slots信號和槽
這是一個簡單的例子演示PyQt5的信號和槽。
#!/usr/bin/python3 # -*- coding: utf-8 -*- """ PyQt5 教程 這個例子中,我們將QSlider的滑動信號連接到QLCDNumber中。 作者:我的世界你曾經(jīng)來過 博客:http://blog.csdn.net/weiaitaowang 最后編輯:2016年8月1日 """ import sys from PyQt5.QtWidgets import (QApplication, QWidget, QSlider, QLCDNumber, QVBoxLayout) from PyQt5.QtCore import Qt class Example(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): lcd = QLCDNumber(self) sld = QSlider(Qt.Horizontal, self) vbox = QVBoxLayout() vbox.addWidget(lcd) vbox.addWidget(sld) self.setLayout(vbox) sld.valueChanged.connect(lcd.display) self.setGeometry(300, 300, 250, 150) self.setWindowTitle('信號/槽') self.show() if __name__ == '__main__': app = QApplication(sys.argv) ex = Example() sys.exit(app.exec_())
在我們的例子中,將用到QtGui.QLCDNumber和QtGui.QSlider。我們通過拖動滑塊改變LCD數(shù)字。
sld.valueChanged.connect(lcd.display)
在這里,滑塊的 valueChanged 信號連接到 lcd 的顯示(display)槽。
發(fā)送器是對象發(fā)送信號。接收器是接收信號的對象。槽的是反饋給信號的方法。
程序執(zhí)行后
覆寫系統(tǒng)事件處理程序
事件在PyQt5中的處理往往通過重寫事件來處理程序。
#!/usr/bin/python3 # -*- coding: utf-8 -*- """ PyQt5 教程 在這個例子中,我們執(zhí)行事件處理程序。 作者:我的世界你曾經(jīng)來過 博客:http://blog.csdn.net/weiaitaowang 最后編輯:2016年8月1日 """ import sys from PyQt5.QtWidgets import QApplication, QWidget from PyQt5.QtCore import Qt class Example(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): self.setGeometry(300, 300, 250, 150) self.setWindowTitle('事件處理') self.show() def keyPressEvent(self, e): if e.key() == Qt.Key_Escape: self.close() if __name__ == '__main__': app = QApplication(sys.argv) ex = Example() sys.exit(app.exec_())
在我們的例子中,我們重新實現(xiàn) keyPressEvent() 事件處理程序。
def keyPressEvent(self, e): if e.key() == Qt.Key_Escape: self.close()
如果我們按下鍵盤上的 Esc 鍵,應(yīng)用程序終止。
Event sender事件發(fā)送
為了方便區(qū)分多個連接到同一事件目標的事件源,在PyQt5中可以使用sender()方法。
#!/usr/bin/python3 # -*- coding: utf-8 -*- """ PyQt5 教程 在這個例子中,我們確定事件發(fā)送對象。 作者:我的世界你曾經(jīng)來過 博客:http://blog.csdn.net/weiaitaowang 最后編輯:2016年8月1日 """ import sys from PyQt5.QtWidgets import QApplication, QMainWindow, QPushButton class Example(QMainWindow): def __init__(self): super().__init__() self.initUI() def initUI(self): btn1 = QPushButton('按鈕一', self) btn1.move(30, 50) btn2 = QPushButton('按鈕二', self) btn2.move(150, 50) btn1.clicked.connect(self.buttonClicked) btn2.clicked.connect(self.buttonClicked) self.statusBar() self.setGeometry(300, 300, 300, 150) self.setWindowTitle('事件發(fā)送') self.show() def buttonClicked(self): sender = self.sender() self.statusBar().showMessage(sender.text() + ' 被按下') if __name__ == '__main__': app = QApplication(sys.argv) ex = Example() sys.exit(app.exec_())
在我們的例子中有兩個按鈕。兩個按鈕都連接 buttonClicked() 方法,我們通過調(diào)用 sender() 方法響應(yīng)單擊的按鈕。
btn1.clicked.connect(self.buttonClicked) btn2.clicked.connect(self.buttonClicked)
兩個按鈕連接到同一個槽。
def buttonClicked(self): sender = self.sender() self.statusBar().showMessage(sender.text() + ' 被按下')
我們通過調(diào)用 sender() 方法確定信號源。在應(yīng)用程序的狀態(tài)欄,顯示被按下按鈕的標簽。
程序執(zhí)行后
定制發(fā)射信號
從一個QObject 創(chuàng)建的對象可以發(fā)出信號。在下面的例子中,我們將看看我們?nèi)绾文軌蚨ㄖ瓢l(fā)出信號。
#!/usr/bin/python3 # -*- coding: utf-8 -*- """ PyQt5 教程 在這個例子中,我們顯示了如何以發(fā)射信號。 作者:我的世界你曾經(jīng)來過 博客:http://blog.csdn.net/weiaitaowang 最后編輯:2016年8月1日 """ import sys from PyQt5.QtWidgets import QApplication, QMainWindow from PyQt5.QtCore import pyqtSignal, QObject class Communicate(QObject): closeApp = pyqtSignal() class Example(QMainWindow): def __init__(self): super().__init__() self.initUI() def initUI(self): self.c = Communicate() self.c.closeApp.connect(self.close) self.setGeometry(300, 300, 300, 150) self.setWindowTitle('發(fā)射信號') self.show() def mousePressEvent(self, event): self.c.closeApp.emit() if __name__ == '__main__': app = QApplication(sys.argv) ex = Example() sys.exit(app.exec_())
我們創(chuàng)建一個名為closeApp新的信號。這個信號是發(fā)射按下鼠標事件。該信號被連接到QMainWindow中的close()槽。
class Communicate(QObject): closeApp = pyqtSignal()
創(chuàng)建繼承自 QObject 的 Communicate 類,該類中有一個 pyqtSignal() 類的屬性。
self.c = Communicate() self.c.closeApp.connect(self.close)
將我們自定義的 closeApp 信號連接到QMainWindow中的close()槽。
def mousePressEvent(self, event): self.c.closeApp.emit()
當我們鼠標在程序窗口出現(xiàn)點擊動作時,closeApp信號被發(fā)射(emit)。應(yīng)用程序終止。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
詳解將Pandas中的DataFrame類型轉(zhuǎn)換成Numpy中array類型的三種方法
這篇文章主要介紹了詳解將Pandas中的DataFrame類型轉(zhuǎn)換成Numpy中array類型的三種方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友們下面隨著小編來一起學(xué)習學(xué)習吧2019-07-07Python數(shù)據(jù)結(jié)構(gòu)之單鏈表詳解
這篇文章主要為大家詳細介紹了Python數(shù)據(jù)結(jié)構(gòu)之單鏈表的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-09-09Python中l(wèi)ogging日志記錄到文件及自動分割的操作代碼
這篇文章主要介紹了Python中l(wèi)ogging日志記錄到文件及自動分割,本文給大家介紹的非常詳細,對大家的學(xué)習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-08-08python中的匿名函數(shù)及編寫無參數(shù)decorator詳解
這篇文章主要介紹了python中的匿名函數(shù)及編寫無參數(shù)decorator詳解,高階函數(shù)可以接收函數(shù)做參數(shù),有些時候,我們不需要顯式地定義函數(shù),直接傳入匿名函數(shù)更方便,需要的朋友可以參考下2023-12-12flask框架使用orm連接數(shù)據(jù)庫的方法示例
這篇文章主要介紹了flask框架使用orm連接數(shù)據(jù)庫的方法,結(jié)合實例形式分析了flask框架使用flask_sqlalchemy包進行mysql數(shù)據(jù)庫連接操作的具體步驟與相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2018-07-07