Python應(yīng)用03 使用PyQT制作視頻播放器實(shí)例
最近研究了Python的兩個(gè)GUI包,Tkinter和PyQT。這兩個(gè)GUI包的底層分別是Tcl/Tk和QT。相比之下,我覺得PyQT使用起來更加方便,功能也相對豐富。這一篇用PyQT實(shí)現(xiàn)一個(gè)視頻播放器,并借此來說明PyQT的基本用法。
視頻播放器
先把已經(jīng)完成的代碼放出來。代碼基于Python 3.5:
import time
import sys
from PyQt4 import QtGui, QtCore
from PyQt4.phonon import Phonon
class PollTimeThread(QtCore.QThread):
"""
This thread works as a timer.
"""
update = QtCore.pyqtSignal()
def __init__(self, parent):
super(PollTimeThread, self).__init__(parent)
def run(self):
while True:
time.sleep(1)
if self.isRunning():
# emit signal
self.update.emit()
else:
return
class Window(QtGui.QWidget):
def __init__(self):
QtGui.QWidget.__init__(self)
# media
self.media = Phonon.MediaObject(self)
self.media.stateChanged.connect(self.handleStateChanged)
self.video = Phonon.VideoWidget(self)
self.video.setMinimumSize(200, 200)
self.audio = Phonon.AudioOutput(Phonon.VideoCategory, self)
Phonon.createPath(self.media, self.audio)
Phonon.createPath(self.media, self.video)
# control button
self.button = QtGui.QPushButton('選擇文件', self)
self.button.clicked.connect(self.handleButton)
# for display of time lapse
self.info = QtGui.QLabel(self)
# layout
layout = QtGui.QGridLayout(self)
layout.addWidget(self.video, 1, 1, 3, 3)
layout.addWidget(self.info, 4, 1, 1, 3)
layout.addWidget(self.button, 5, 1, 1, 3)
# signal-slot, for time lapse
self.thread = PollTimeThread(self)
self.thread.update.connect(self.update)
def update(self):
# slot
lapse = self.media.currentTime()/1000.0
self.info.setText("%4.2f 秒" % lapse)
def startPlay(self):
if self.path:
self.media.setCurrentSource(Phonon.MediaSource(self.path))
# use a thread as a timer
self.thread = PollTimeThread(self)
self.thread.update.connect(self.update)
self.thread.start()
self.media.play()
def handleButton(self):
if self.media.state() == Phonon.PlayingState:
self.media.stop()
self.thread.terminate()
else:
self.path = QtGui.QFileDialog.getOpenFileName(self, self.button.text())
self.startPlay()
def handleStateChanged(self, newstate, oldstate):
if newstate == Phonon.PlayingState:
self.button.setText('停止')
elif (newstate != Phonon.LoadingState and
newstate != Phonon.BufferingState):
self.button.setText('選擇文件')
if newstate == Phonon.ErrorState:
source = self.media.currentSource().fileName()
print ('錯(cuò)誤:不能播放:', source.toLocal8Bit().data())
print (' %s' % self.media.errorString().toLocal8Bit().data())
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
app.setApplicationName('視頻播放')
window = Window()
window.show()
sys.exit(app.exec_())
代碼實(shí)現(xiàn)了一個(gè)有GUI窗口的應(yīng)用,用來播放視頻文件。視頻播放利用了PyQT中的Phonon模塊。此外,還有一個(gè)進(jìn)程每隔一秒發(fā)出一個(gè)信號。窗口在接收到信號后,更新視頻播放的時(shí)間。這個(gè)應(yīng)用的效果如下:
測試運(yùn)行環(huán)境為Mac OSX El Capitan。

視圖部分
寫完這個(gè)代碼之后,我發(fā)現(xiàn)這個(gè)代碼雖然簡單,但涉及了幾個(gè)重要機(jī)制,可以用PyQT的練習(xí)題。下面對代碼進(jìn)行一些簡要的說明,首先是主程序部分:
app = QtGui.QApplication(sys.argv) ... window = Window() window.show() sys.exit(app.exec_())
在PyQT程序中,QApplication是最上層的對象,指代整個(gè)GUI應(yīng)用。我們在程序的一開始創(chuàng)建了一個(gè)應(yīng)用對象,在程序最后調(diào)用exec_()來運(yùn)行這個(gè)應(yīng)用。sys.exit()用來要求應(yīng)用的主循環(huán)結(jié)束后干凈地退出程序。PyQT程序的開始和結(jié)尾都是類似的固定套路。關(guān)鍵就在于其間定義的QWidget對象。
我們自定義的Window類繼承自QWidget。其實(shí)QWidget是所有用戶界面對象的基類,并不單單指代一個(gè)窗口。表格、輸入框、按鈕都繼承自QWidget。在一個(gè)Window對象中,我們還組合有QPushButton和QLabel這樣的對象,分別代表一個(gè)按鈕和一個(gè)文本框。它們通過QGridLayout的方式,布局在Window的界面上,即下面一部分代碼:
# layout layout = QtGui.QGridLayout(self) ... layout.addWidget(self.info, 4, 1, 1, 3) layout.addWidget(self.button, 5, 1, 1, 3)
QGridLayout把界面分成網(wǎng)格,并把某個(gè)視圖對象附著在特定的網(wǎng)格位置。比如說,addWidget()(self.info, 4, 1, 1, 3)表示把一個(gè)文本框?qū)ο蠓旁诘?排、第1列的位置。該文本框縱向?qū)⒄紦?jù)1排,橫向占據(jù)3列。這樣,上下層視圖的位置關(guān)系就通過布局確定了下來。除了網(wǎng)格式的布局,PyQT還支持其他形式的布局,如橫向堆砌、縱向堆砌等等,可以進(jìn)一步了解。
除了QWidget,PyQT還提供了常用的對話框,如:
self.path = QtGui.QFileDialog.getOpenFileName(self, self.button.text())
這里的QFileDialog對話框用于選擇文件。對話框?qū)⒃L問所選文件的路徑。除了文件選擇,對話框還有確認(rèn)對話框、文件輸入對話框、色彩對話框。這些對話框?qū)崿F(xiàn)了不少常用的GUI輸入功能。通過利用這些對話框,可以減少程序員從頭開發(fā)的工作量。
多線程
GUI界面的主線程通常留給應(yīng)用做主循環(huán)。其他的很多工作要通過其他的線程來完成。PyQT多線程編程很簡單,只需要重寫QThread的run()方法就可以了:
class PollTimeThread(QtCore.QThread):
def __init__(self, parent):
super(PollTimeThread, self).__init__(parent)
def run(self):
...
創(chuàng)建線程后,只需要調(diào)用start()方法,就可以運(yùn)行:
self.thread = PollTimeThread() ... self.thread.start() # 啟動線程 ... self.thread.terminate() # 終止線程
信號與槽
GUI經(jīng)常要用到異步處理。比如說點(diǎn)擊某個(gè)按鈕,然后調(diào)用相應(yīng)的回調(diào)函數(shù)。QT的“信號與槽”(signal-slot)機(jī)制就是為了解決異步處理問題。我們在線程中創(chuàng)建了信號,并通過emit()方法來發(fā)出信號:
class PollTimeThread(QtCore.QThread):
"""
This thread works as a timer.
"""
update = QtCore.pyqtSignal()
def __init__(self, parent):
super(PollTimeThread, self).__init__(parent)
def run(self):
while True:
time.sleep(1)
if self.isRunning():
# emit signal
self.update.emit()
else:
return
有了信號,我們就可以給該信號連接到一個(gè)“槽”,其實(shí)就是對應(yīng)于該信號的回調(diào)函數(shù):
self.thread.update.connect(self.update)
每當(dāng)信號被發(fā)出時(shí),“槽”就會被調(diào)用。在這個(gè)例子中,就是更新視頻播放時(shí)間。QT中的“信號與槽”是普遍存在的機(jī)制。一些組建如按鍵,預(yù)設(shè)了“點(diǎn)擊”這樣的信號,可以直接對應(yīng)到“槽”。如代碼中的:
self.button.clicked.connect(self.handleButton)
此外,Phonon是一個(gè)很好用的多媒體模塊,使用方法也很簡單,可以參考代碼本身,這里不再贅述。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python xmltodict模塊安裝及代碼實(shí)例
這篇文章主要介紹了Python xmltodict模塊安裝及代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-10-10
Python分支結(jié)構(gòu)和循環(huán)結(jié)構(gòu)示例代碼
在Python中,分支結(jié)構(gòu)通過if、elif和else關(guān)鍵字來實(shí)現(xiàn)條件判斷,在使用if語句時(shí),程序會根據(jù)條件表達(dá)式的真假執(zhí)行相應(yīng)的代碼塊,這篇文章主要介紹了Python分支結(jié)構(gòu)和循環(huán)結(jié)構(gòu),需要的朋友可以參考下2024-03-03
python index() 與 rindex() 方法的使用示例詳解
這篇文章主要介紹了python index() 與 rindex() 方法的使用,需要的朋友可以參考下2022-12-12
python實(shí)現(xiàn)ssh及sftp功能(實(shí)例代碼)
這篇文章主要介紹了python實(shí)現(xiàn)ssh及sftp功能 ,本文分步驟通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
Python中range、np.arange和np.linspace的區(qū)別
本文主要介紹了Python中range、np.arange和np.linspace的區(qū)別,文中根據(jù)實(shí)例編碼詳細(xì)介紹的十分詳盡,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
python如何處理matlab的mat數(shù)據(jù)
這篇文章主要介紹了python如何處理matlab的mat數(shù)據(jù),具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05

