歡迎光臨
每天分享高質量文章

Python使用PyQT製作視頻播放器

最近研究了Python的兩個GUI包,Tkinter和PyQT。這兩個GUI包的底層分別是Tcl/Tk和QT。相比之下,我覺得PyQT使用起來更加方便,功能也相對豐富。這一篇用PyQT實現一個視頻播放器,並藉此來說明PyQT的基本用法。

視頻播放器

先把已經完成的代碼放出來。代碼基於Python 3.5:

  1. import time

  2. import sys

  3. from PyQt4 import QtGui, QtCore

  4. from PyQt4.phonon import Phonon

  5. class PollTimeThread(QtCore.QThread):

  6.    """

  7.    This thread works as a timer.

  8.    """

  9.    update = QtCore.pyqtSignal()

  10.    def __init__(self, parent):

  11.        super(PollTimeThread, self).__init__(parent)

  12.    def run(self):

  13.        while True:

  14.            time.sleep(1)

  15.            if self.isRunning():

  16.                # emit signal

  17.                self.update.emit()

  18.            else:

  19.                return

  20. class Window(QtGui.QWidget):

  21.    def __init__(self):

  22.        QtGui.QWidget.__init__(self)

  23.        # media

  24.        self.media = Phonon.MediaObject(self)

  25.        self.media.stateChanged.connect(self.handleStateChanged)

  26.        self.video = Phonon.VideoWidget(self)

  27.        self.video.setMinimumSize(200, 200)

  28.        self.audio = Phonon.AudioOutput(Phonon.VideoCategory, self)

  29.        Phonon.createPath(self.media, self.audio)

  30.        Phonon.createPath(self.media, self.video)

  31.        # control button

  32.        self.button = QtGui.QPushButton('選擇檔案', self)

  33.        self.button.clicked.connect(self.handleButton)

  34.        # for display of time lapse

  35.        self.info = QtGui.QLabel(self)

  36.        # layout

  37.        layout = QtGui.QGridLayout(self)

  38.        layout.addWidget(self.video, 1, 1, 3, 3)

  39.        layout.addWidget(self.info, 4, 1, 1, 3)

  40.        layout.addWidget(self.button, 5, 1, 1, 3)

  41.        # signal-slot, for time lapse

  42.        self.thread = PollTimeThread(self)

  43.        self.thread.update.connect(self.update)

  44.    def update(self):

  45.        # slot

  46.        lapse = self.media.currentTime()/1000.0

  47.        self.info.setText("%4.2f 秒" % lapse)

  48.    def startPlay(self):

  49.        if self.path:

  50.            self.media.setCurrentSource(Phonon.MediaSource(self.path))

  51.            # use a thread as a timer

  52.            self.thread = PollTimeThread(self)

  53.            self.thread.update.connect(self.update)

  54.            self.thread.start()

  55.            self.media.play()

  56.    def handleButton(self):

  57.        if self.media.state() == Phonon.PlayingState:

  58.            self.media.stop()

  59.            self.thread.terminate()

  60.        else:

  61.            self.path = QtGui.QFileDialog.getOpenFileName(self, self.button.text())

  62.            self.startPlay()

  63.    def handleStateChanged(self, newstate, oldstate):

  64.        if newstate == Phonon.PlayingState:

  65.            self.button.setText('停止')

  66.        elif (newstate != Phonon.LoadingState and

  67.              newstate != Phonon.BufferingState):

  68.            self.button.setText('選擇檔案')

  69.            if newstate == Phonon.ErrorState:

  70.                source = self.media.currentSource().fileName()

  71.                print ('錯誤:不能播放:', source.toLocal8Bit().data())

  72.                print ('  %s' % self.media.errorString().toLocal8Bit().data())

  73. if __name__ == '__main__':

  74.    app = QtGui.QApplication(sys.argv)

  75.    app.setApplicationName('視頻播放')

  76.    window = Window()

  77.    window.show()

  78.    sys.exit(app.exec_())


代碼實現了一個有GUI視窗的應用,用來播放視頻檔案。視頻播放利用了PyQT中的Phonon模塊。此外,還有一個行程每隔一秒發出一個信號。視窗在接收到信號後,更新視頻播放的時間。這個應用的效果如下:

測試運行環境為Mac OSX El Capitan。

視圖部分

寫完這個代碼之後,我發現這個代碼雖然簡單,但涉及了幾個重要機制,可以用PyQT的練習題。下麵對代碼進行一些簡要的說明,首先是主程式部分:

  1. app = QtGui.QApplication(sys.argv)

  2. ...

  3. window = Window()

  4. window.show()

  5. sys.exit(app.exec_())

在PyQT程式中,QApplication是最上層的物件,指代整個GUI應用。我們在程式的一開始創建了一個應用物件,在程式最後呼叫exec_()來運行這個應用。sys.exit()用來要求應用的主迴圈結束後乾凈地退出程式。PyQT程式的開始和結尾都是類似的固定套路。關鍵就在於其間定義的QWidget物件。

 我們自定義的Window類繼承自QWidget。其實QWidget是所有用戶界面物件的基類,並不單單指代一個視窗。表格、輸入框、按鈕都繼承自QWidget。在一個Window物件中,我們還組合有QPushButton和QLabel這樣的物件,分別代表一個按鈕和一個文本框。它們通過QGridLayout的方式,佈局在Window的界面上,即下麵一部分代碼:

  1. # layout

  2. layout = QtGui.QGridLayout(self)

  3. ...

  4. layout.addWidget(self.info, 4, 1, 1, 3)

  5. layout.addWidget(self.button, 5, 1, 1, 3)

QGridLayout把界面分成網格,並把某個視圖物件附著在特定的網格位置。比如說,addWidget()(self.info, 4, 1, 1, 3)表示把一個文本框物件放在第4排、第1列的位置。該文本框縱向將占據1排,橫向占據3列。這樣,上下層視圖的位置關係就通過佈局確定了下來。除了網格式的佈局,PyQT還支持其他形式的佈局,如橫向堆砌、縱向堆砌等等,可以進一步瞭解。

 

除了QWidget,PyQT還提供了常用的對話框,如:

self.path = QtGui.QFileDialog.getOpenFileName(self, self.button.text())

這裡的QFileDialog對話框用於選擇檔案。對話框將訪問所選檔案的路徑。除了檔案選擇,對話框還有確認對話框、檔案輸入對話框、色彩對話框。這些對話框實現了不少常用的GUI輸入功能。通過利用這些對話框,可以減少程式員從頭開發的工作量。

多執行緒

GUI界面的主執行緒通常留給應用做主迴圈。其他的很多工作要通過其他的執行緒來完成。PyQT多執行緒編程很簡單,只需要重寫QThread的run()方法就可以了:

  1. class PollTimeThread(QtCore.QThread):

  2.    def __init__(self, parent):

  3.        super(PollTimeThread, self).__init__(parent)

  4.    def run(self):

  5.        ...


創建執行緒後,只需要呼叫start()方法,就可以運行:


self.thread = PollTimeThread()
... self.thread.start()       # 啟動執行緒
... self.thread.terminate()   # 終止執行緒

信號與槽

GUI經常要用到異步處理。比如說點擊某個按鈕,然後呼叫相應的回呼函式。QT的“信號與槽”(signal-slot)機制就是為瞭解決異步處理問題。我們在執行緒中創建了信號,並通過emit()方法來發出信號:

  1. class PollTimeThread(QtCore.QThread):

  2.    """

  3.    This thread works as a timer.

  4.    """

  5.    update = QtCore.pyqtSignal()

  6.    def __init__(self, parent):

  7.        super(PollTimeThread, self).__init__(parent)

  8.    def run(self):

  9.        while True:

  10.            time.sleep(1)

  11.            if self.isRunning():

  12.                # emit signal

  13.                self.update.emit()

  14.            else:

  15.                return

有了信號,我們就可以給該信號連接到一個“槽”,其實就是對應於該信號的回呼函式:

self.thread.update.connect(self.update)

每當信號被髮出時,“槽”就會被呼叫。在這個例子中,就是更新視頻播放時間。QT中的“信號與槽”是普遍存在的機制。一些組建如按鍵,預設了“點擊”這樣的信號,可以直接對應到“槽”。如代碼中的:

self.button.clicked.connect(self.handleButton)

 

此外,Phonon是一個很好用的多媒體模塊,使用方法也很簡單,可以參考代碼本身,這裡不再贅述。

作者:Vamei

源自:

http://www.cnblogs.com/vamei/p/6139513.html

赞(0)

分享創造快樂

© 2021 知識星球   网站地图