python制作视频播放器_Python应用03 使用PyQT制作视频播放器
最近研究了Python的兩個(gè)GUI包,Tkinter和PyQT。這兩個(gè)GUI包的底層分別是Tcl/Tk和QT。相比之下,我覺(jué)得PyQT使用起來(lái)更加方便,功能也相對(duì)豐富。這一篇用PyQT實(shí)現(xiàn)一個(gè)視頻播放器,并借此來(lái)說(shuō)明PyQT的基本用法。
視頻播放器
先把已經(jīng)完成的代碼放出來(lái)。代碼基于Python 3.5:
import timeimport sysfrom PyQt4 import QtGui, QtCorefrom PyQt4.phonon import Phononclass 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: returnclass 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)用,用來(lái)播放視頻文件。視頻播放利用了PyQT中的Phonon模塊。此外,還有一個(gè)進(jìn)程每隔一秒發(fā)出一個(gè)信號(hào)。窗口在接收到信號(hào)后,更新視頻播放的時(shí)間。這個(gè)應(yīng)用的效果如下:
測(cè)試運(yùn)行環(huán)境為Mac OSX El Capitan。
視圖部分
寫(xiě)完這個(gè)代碼之后,我發(fā)現(xiàn)這個(gè)代碼雖然簡(jiǎn)單,但涉及了幾個(gè)重要機(jī)制,可以用PyQT的練習(xí)題。下面對(duì)代碼進(jìn)行一些簡(jiǎn)要的說(shuō)明,首先是主程序部分:app = QtGui.QApplication(sys.argv)...window = Window()window.show()sys.exit(app.exec_())
在PyQT程序中,QApplication是最上層的對(duì)象,指代整個(gè)GUI應(yīng)用。我們?cè)诔绦虻囊婚_(kāi)始創(chuàng)建了一個(gè)應(yīng)用對(duì)象,在程序最后調(diào)用exec_()來(lái)運(yùn)行這個(gè)應(yīng)用。sys.exit()用來(lái)要求應(yīng)用的主循環(huán)結(jié)束后干凈地退出程序。PyQT程序的開(kāi)始和結(jié)尾都是類(lèi)似的固定套路。關(guān)鍵就在于其間定義的QWidget對(duì)象。
我們自定義的Window類(lèi)繼承自QWidget。其實(shí)QWidget是所有用戶(hù)界面對(duì)象的基類(lèi),并不單單指代一個(gè)窗口。表格、輸入框、按鈕都繼承自QWidget。在一個(gè)Window對(duì)象中,我們還組合有QPushButton和QLabel這樣的對(duì)象,分別代表一個(gè)按鈕和一個(gè)文本框。它們通過(guò)QGridLayout的方式,布局在Window的界面上,即下面一部分代碼:# layoutlayout = QtGui.QGridLayout(self)...layout.addWidget(self.info, 4, 1, 1, 3)layout.addWidget(self.button, 5, 1, 1, 3)
QGridLayout把界面分成網(wǎng)格,并把某個(gè)視圖對(duì)象附著在特定的網(wǎng)格位置。比如說(shuō),addWidget()(self.info, 4, 1, 1, 3)表示把一個(gè)文本框?qū)ο蠓旁诘?排、第1列的位置。該文本框縱向?qū)⒄紦?jù)1排,橫向占據(jù)3列。這樣,上下層視圖的位置關(guān)系就通過(guò)布局確定了下來(lái)。除了網(wǎng)格式的布局,PyQT還支持其他形式的布局,如橫向堆砌、縱向堆砌等等,可以進(jìn)一步了解。
除了QWidget,PyQT還提供了常用的對(duì)話(huà)框,如:self.path = QtGui.QFileDialog.getOpenFileName(self, self.button.text())
這里的QFileDialog對(duì)話(huà)框用于選擇文件。對(duì)話(huà)框?qū)⒃L問(wèn)所選文件的路徑。除了文件選擇,對(duì)話(huà)框還有確認(rèn)對(duì)話(huà)框、文件輸入對(duì)話(huà)框、色彩對(duì)話(huà)框。這些對(duì)話(huà)框?qū)崿F(xiàn)了不少常用的GUI輸入功能。通過(guò)利用這些對(duì)話(huà)框,可以減少程序員從頭開(kāi)發(fā)的工作量。
多線程
GUI界面的主線程通常留給應(yīng)用做主循環(huán)。其他的很多工作要通過(guò)其他的線程來(lái)完成。PyQT多線程編程很簡(jiǎn)單,只需要重寫(xiě)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() # 啟動(dòng)線程
...self.thread.terminate() # 終止線程
信號(hào)與槽
GUI經(jīng)常要用到異步處理。比如說(shuō)點(diǎn)擊某個(gè)按鈕,然后調(diào)用相應(yīng)的回調(diào)函數(shù)。QT的“信號(hào)與槽”(signal-slot)機(jī)制就是為了解決異步處理問(wèn)題。我們?cè)诰€程中創(chuàng)建了信號(hào),并通過(guò)emit()方法來(lái)發(fā)出信號(hào):
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
有了信號(hào),我們就可以給該信號(hào)連接到一個(gè)“槽”,其實(shí)就是對(duì)應(yīng)于該信號(hào)的回調(diào)函數(shù):self.thread.update.connect(self.update)
每當(dāng)信號(hào)被發(fā)出時(shí),“槽”就會(huì)被調(diào)用。在這個(gè)例子中,就是更新視頻播放時(shí)間。QT中的“信號(hào)與槽”是普遍存在的機(jī)制。一些組建如按鍵,預(yù)設(shè)了“點(diǎn)擊”這樣的信號(hào),可以直接對(duì)應(yīng)到“槽”。如代碼中的:self.button.clicked.connect(self.handleButton)
此外,Phonon是一個(gè)很好用的多媒體模塊,使用方法也很簡(jiǎn)單,可以參考代碼本身,這里不再贅述。
歡迎繼續(xù)閱讀“Python快速教程”
總結(jié)
以上是生活随笔為你收集整理的python制作视频播放器_Python应用03 使用PyQT制作视频播放器的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 如何把python可视化到前端_pyth
- 下一篇: union和union all有什么区别