欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python應(yīng)用03 使用PyQT制作視頻播放器實(shí)例

 更新時(shí)間:2016年12月07日 08:21:03   作者:Vamei  
本篇文章主要介紹了Python使用PyQT制作視頻播放器實(shí)例,具有一定的參考價(jià)值,有興趣的可以了解一下。

最近研究了Python的兩個(gè)GUI包,Tkinter和PyQT。這兩個(gè)GUI包的底層分別是Tcl/Tk和QT。相比之下,我覺得PyQT使用起來更加方便,功能也相對(duì)豐富。這一篇用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è)信號(hào)。窗口在接收到信號(hào)后,更新視頻播放的時(shí)間。這個(gè)應(yīng)用的效果如下:
測(cè)試運(yùn)行環(huán)境為Mac OSX El Capitan。

 

視圖部分

寫完這個(gè)代碼之后,我發(fā)現(xiàn)這個(gè)代碼雖然簡單,但涉及了幾個(gè)重要機(jī)制,可以用PyQT的練習(xí)題。下面對(duì)代碼進(jìn)行一些簡要的說明,首先是主程序部分:

app = QtGui.QApplication(sys.argv)
...
window = Window()
window.show()
sys.exit(app.exec_())

在PyQT程序中,QApplication是最上層的對(duì)象,指代整個(gè)GUI應(yīng)用。我們?cè)诔绦虻囊婚_始創(chuàng)建了一個(gè)應(yīng)用對(duì)象,在程序最后調(diào)用exec_()來運(yùn)行這個(gè)應(yīng)用。sys.exit()用來要求應(yīng)用的主循環(huán)結(jié)束后干凈地退出程序。PyQT程序的開始和結(jié)尾都是類似的固定套路。關(guān)鍵就在于其間定義的QWidget對(duì)象。

我們自定義的Window類繼承自QWidget。其實(shí)QWidget是所有用戶界面對(duì)象的基類,并不單單指代一個(gè)窗口。表格、輸入框、按鈕都繼承自QWidget。在一個(gè)Window對(duì)象中,我們還組合有QPushButton和QLabel這樣的對(duì)象,分別代表一個(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è)視圖對(duì)象附著在特定的網(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還提供了常用的對(duì)話框,如:

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

這里的QFileDialog對(duì)話框用于選擇文件。對(duì)話框?qū)⒃L問所選文件的路徑。除了文件選擇,對(duì)話框還有確認(rèn)對(duì)話框、文件輸入對(duì)話框、色彩對(duì)話框。這些對(duì)話框?qū)崿F(xiàn)了不少常用的GUI輸入功能。通過利用這些對(duì)話框,可以減少程序員從頭開發(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()    # 啟動(dòng)線程
...
self.thread.terminate()  # 終止線程

信號(hào)與槽

GUI經(jīng)常要用到異步處理。比如說點(diǎn)擊某個(gè)按鈕,然后調(diào)用相應(yīng)的回調(diào)函數(shù)。QT的“信號(hào)與槽”(signal-slot)機(jī)制就是為了解決異步處理問題。我們?cè)诰€程中創(chuàng)建了信號(hào),并通過emit()方法來發(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è)很好用的多媒體模塊,使用方法也很簡單,可以參考代碼本身,這里不再贅述。

以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Python xmltodict模塊安裝及代碼實(shí)例

    Python xmltodict模塊安裝及代碼實(shí)例

    這篇文章主要介紹了Python xmltodict模塊安裝及代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-10-10
  • Python分支結(jié)構(gòu)和循環(huán)結(jié)構(gòu)示例代碼

    Python分支結(jié)構(gòu)和循環(huán)結(jié)構(gòu)示例代碼

    在Python中,分支結(jié)構(gòu)通過if、elif和else關(guān)鍵字來實(shí)現(xiàn)條件判斷,在使用if語句時(shí),程序會(huì)根據(jù)條件表達(dá)式的真假執(zhí)行相應(yīng)的代碼塊,這篇文章主要介紹了Python分支結(jié)構(gòu)和循環(huán)結(jié)構(gòu),需要的朋友可以參考下
    2024-03-03
  • 一文帶大家了解python中的換行以及轉(zhuǎn)義

    一文帶大家了解python中的換行以及轉(zhuǎn)義

    這篇文章主要為大家詳細(xì)介紹了python中的換行以及轉(zhuǎn)義的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),對(duì)我們深入了解python有一定的幫助,需要的小伙伴可以了解下
    2023-11-11
  • python index() 與 rindex() 方法的使用示例詳解

    python index() 與 rindex() 方法的使用示例詳解

    這篇文章主要介紹了python index() 與 rindex() 方法的使用,需要的朋友可以參考下
    2022-12-12
  • python實(shí)現(xiàn)ssh及sftp功能(實(shí)例代碼)

    python實(shí)現(xiàn)ssh及sftp功能(實(shí)例代碼)

    這篇文章主要介紹了python實(shí)現(xiàn)ssh及sftp功能 ,本文分步驟通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-03-03
  • python使用MkDocs自動(dòng)生成文檔的操作方法

    python使用MkDocs自動(dòng)生成文檔的操作方法

    python代碼注釋風(fēng)格有很多,比較主流的有 reStructuredText風(fēng)格、numpy風(fēng)格、Google風(fēng)格,自動(dòng)生成文檔的工具也有很多,常見的有:Pydocs,Sphinx和MkDocs,本文給大家介紹了python使用MkDocs自動(dòng)生成文檔的操作方法,需要的朋友可以參考下
    2024-06-06
  • Python中range、np.arange和np.linspace的區(qū)別

    Python中range、np.arange和np.linspace的區(qū)別

    本文主要介紹了Python中range、np.arange和np.linspace的區(qū)別,文中根據(jù)實(shí)例編碼詳細(xì)介紹的十分詳盡,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • Python中的裝飾器用法詳解

    Python中的裝飾器用法詳解

    這篇文章主要介紹了Python中的裝飾器用法,以實(shí)例形式詳細(xì)的分析了Python中的裝飾器的使用技巧及相關(guān)注意事項(xiàng),具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-01-01
  • python如何處理matlab的mat數(shù)據(jù)

    python如何處理matlab的mat數(shù)據(jù)

    這篇文章主要介紹了python如何處理matlab的mat數(shù)據(jù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • python爬取新聞門戶網(wǎng)站的示例

    python爬取新聞門戶網(wǎng)站的示例

    短期目前旨在爬取所有新聞門戶網(wǎng)站的新聞,每個(gè)門戶網(wǎng)站爬蟲開箱即用,并自動(dòng)保存到同目錄下的 csv/excel 文件中,禁止將所得數(shù)據(jù)商用。
    2021-04-04

最新評(píng)論