Python使用PyQt5/PySide2編寫(xiě)一個(gè)極簡(jiǎn)的音樂(lè)播放器功能
疫情肆虐,憋在家實(shí)在無(wú)聊,索性寫(xiě)點(diǎn)東西,于是就有了這個(gè)極極極極極簡(jiǎn)的音樂(lè)播放器。
這個(gè)極極極簡(jiǎn)的音樂(lè)播放器類(lèi)似于“閱后即焚”的軟件,播放器可以隨機(jī)播放歌曲,獲取下一首歌曲,不能重新播放上一首歌曲,不能獲取歌曲的名稱(chēng)和演唱者。聽(tīng)過(guò)的歌曲,就像過(guò)眼云煙,放完即散。
下面來(lái)看看如何用Python實(shí)現(xiàn)這個(gè)音樂(lè)播放器軟件吧!
一、創(chuàng)建UI界面
首先,我們來(lái)創(chuàng)建這個(gè)播放器的UI界面。這個(gè)播放器一共有6個(gè)控件:
- 左上角的程序關(guān)閉按鈕;
- 左側(cè)的播放狀態(tài)標(biāo)簽;
- 頂部的slogan;
- 播放/暫停按鈕;
- 下一首按鈕;
- 播放進(jìn)度條;
基于以上控件,我們用一個(gè)網(wǎng)格布局來(lái)排列;
class Music(QtWidgets.QMainWindow): def __init__(self): super().__init__() self.setFixedSize(400,200) self.setWindowTitle("州的先生") self.init_ui() self.custom_style() self.playing = False # 播放狀態(tài)初始化為否 self.player = QMediaPlayer(self) # 設(shè)置樣式 def custom_style(self): self.setStyleSheet(''' #main_widget{ border-radius:5px; } #play_btn,#pervious_btn,#next_btn{ border:none; } #play_btn:hover,#pervious_btn:hover,#next_btn:hover{ background:gray; border-radius:5px; cursor:pointer; } ''') self.close_btn.setStyleSheet(''' QPushButton{ background:#F76677; border-radius:5px; } QPushButton:hover{ background:red; }''') self.status_label.setStyleSheet(''' QLabel{ background:#F7D674; border-radius:5px; } ''') # 初始化UI界面 def init_ui(self): # 窗口布局 self.main_widget = QtWidgets.QWidget() self.main_widget.setObjectName("main_widget") self.main_layout = QtWidgets.QGridLayout() self.main_widget.setLayout(self.main_layout) # 標(biāo)題 self.title_lable = QtWidgets.QLabel("聽(tīng)者,如過(guò)眼云煙……") # 關(guān)閉按鈕 self.close_btn = QtWidgets.QPushButton("") # 關(guān)閉按鈕 self.close_btn.setFixedSize(15,15) # 音樂(lè)狀態(tài)按鈕 self.status_label = QtWidgets.QLabel("") self.status_label.setFixedSize(15,15) # 播放按鈕 play_icon = qta.icon("fa.play-circle",) self.play_btn = QtWidgets.QPushButton(play_icon,"") self.play_btn.setIconSize(QtCore.QSize(80, 80)) self.play_btn.setFixedSize(82,82) self.play_btn.setObjectName("play_btn") # 下一首按鈕 next_icon = qta.icon("fa.play-circle-o") self.next_btn = QtWidgets.QPushButton(next_icon,"") self.next_btn.setIconSize(QtCore.QSize(80,80)) self.next_btn.setFixedSize(82,82) self.next_btn.setObjectName("next_btn") # 進(jìn)度條 self.process_bar = QtWidgets.QProgressBar() self.process_value = 0 self.process_bar.setValue(self.process_value) self.process_bar.setFixedHeight(5) self.process_bar.setTextVisible(False) self.main_layout.addWidget(self.close_btn,0,0,1,1) self.main_layout.addWidget(self.title_lable,0,1,1,1) self.main_layout.addWidget(self.status_label,1,0,1,1) self.main_layout.addWidget(self.play_btn, 1, 1, 1, 1) self.main_layout.addWidget(self.next_btn, 1, 2, 1, 1) self.main_layout.addWidget(self.process_bar,2,0,1,3) self.setCentralWidget(self.main_widget)
在上面的代碼中,我們通過(guò)Qt內(nèi)置的按鈕控件QPushButton、標(biāo)簽控件QLabel、進(jìn)度條控件QProgressBar等把播放器的基礎(chǔ)界面構(gòu)建好了,然后通過(guò)setStyleSheet()方法設(shè)置各個(gè)控件的樣式,最終得到下面的界面:
頂部的邊框有點(diǎn)丑丑的,我們通過(guò)setWindowFlag()將其隱藏起來(lái):
self.setWindowFlag(QtCore.Qt.FramelessWindowHint) # 隱藏邊框
這樣,完整且好看的界面就完成了,如下圖所示:
二、獲取網(wǎng)絡(luò)歌曲
既然是音樂(lè)播放器,當(dāng)然要能夠播放音樂(lè)。我們創(chuàng)建的這個(gè)音樂(lè)播放器不能添加本地的音樂(lè),那就只能播放網(wǎng)絡(luò)的音樂(lè)了。州的先生(https://zmister.com)在網(wǎng)上找了很久,終于找到了一個(gè)免費(fèi)穩(wěn)定的音樂(lè)接口。請(qǐng)求這個(gè)接口會(huì)隨機(jī)返回一個(gè)音樂(lè)的播放地址。正好符合我們的播放器的設(shè)定,就它了:
def run(self): reps = requests.post("https://api.uomg.com/api/rand.music?format=json") # print(reps.json()) file_url = reps.json()['data']['url'] self.finished_signal.emit(file_url)
我們將其用Qt的線(xiàn)程模塊QThread包裝起來(lái),作為一個(gè)異步的子線(xiàn)程調(diào)用:
有了UI界面和網(wǎng)絡(luò)音樂(lè),下面我們UI界面上的控件與音樂(lè)的播放通過(guò)信號(hào)槽(事件響應(yīng))連接起來(lái)。
三、創(chuàng)建和鏈接信號(hào)槽
因?yàn)槲覀冸[藏的UI界面自帶的邊框,所以沒(méi)有了關(guān)閉按鈕。我們之前創(chuàng)建了左側(cè)小紅點(diǎn)作為播放器關(guān)閉的按鈕,接著還需要一個(gè)響應(yīng)函數(shù)來(lái)關(guān)閉播放器:
# 關(guān)閉程序 def close_btn_event(self): self.close() self.close_btn.clicked.connect(self.close_btn_event)
這樣,點(diǎn)擊小紅點(diǎn),播放器就可以關(guān)閉退出了。下面來(lái)處理音樂(lè)的播放。
在這里,我們借助Qt提供的QtMultimedia子模塊中的QMediaPlayer來(lái)實(shí)現(xiàn)MP3音樂(lè)的播放,QtMultimedia的詳細(xì)使用可見(jiàn)Qt的官方文檔:https://doc.qt.io/qtforpython/PySide2/QtMultimedia/index.html#module-PySide2.QtMultimedia:
這個(gè)方法,接受一個(gè)MP3音樂(lè)的網(wǎng)絡(luò)地址參數(shù),然后通過(guò)配置,對(duì)其進(jìn)行播放。
我們之前創(chuàng)建了一個(gè)繼承于QThread的網(wǎng)絡(luò)歌曲獲取類(lèi)GetMusicThread,通過(guò)創(chuàng)建另一個(gè)方法對(duì)其進(jìn)行調(diào)用,并將完成信號(hào)(finished_signal)連接到init_player()方法即可實(shí)現(xiàn)網(wǎng)絡(luò)歌曲的獲取和播放:
在這里,next_music()方法調(diào)用GetMusicThread隨機(jī)獲取一首歌曲,然后將歌曲的URL地址返回給init_player()方法,以實(shí)現(xiàn)播放。
next_music()方法是播放歌曲的主要方法,播放按鈕、下一首按鈕、一首歌曲播放完之后自動(dòng)播放下一首歌曲都可以調(diào)用它。
播放音樂(lè)按鈕需要控制和判斷音樂(lè)的狀態(tài),可通過(guò)QMediaPlayer的mediaStatus()方法獲取,并以此來(lái)進(jìn)行判斷:
為了能夠在播放完一首歌曲后自動(dòng)獲取和播放下一首歌曲,我們需要?jiǎng)?chuàng)建一個(gè)計(jì)時(shí)器,每隔一秒獲取當(dāng)前播放器的狀態(tài),判斷其是否已經(jīng)播放完了音樂(lè),如果播放完了,就調(diào)用next_music()方法:
self.timer = QtCore.QTimer() self.timer.setInterval(1000) self.timer.start() self.timer.timeout.connect(self.check_music_status) def check_music_status(self): player_status = self.player.mediaStatus() player_duration = self.player.duration() # print("音樂(lè)時(shí)間:",player_duration) # print("當(dāng)前播放器狀態(tài)",player_status) if player_status == 7: self.next_music()
再經(jīng)過(guò)一些細(xì)節(jié)調(diào)整,我們就完成了這個(gè)“聽(tīng)后即焚”的網(wǎng)絡(luò)音樂(lè)播放器:
完整的代碼可以前往州的先生在Gitee上的倉(cāng)庫(kù)(https://gitee.com/zmister/yunyan_music )
總結(jié)
以上所述是小編給大家介紹的使用PyQt5/PySide2編寫(xiě)一個(gè)極簡(jiǎn)的音樂(lè)播放器功能,希望對(duì)大家有所幫助!
相關(guān)文章
django 實(shí)現(xiàn)將本地圖片存入數(shù)據(jù)庫(kù),并能顯示在web上的示例
今天小編就為大家分享一篇django 實(shí)現(xiàn)將本地圖片存入數(shù)據(jù)庫(kù),并能顯示在web上的示例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-08-08numpy如何取出對(duì)角線(xiàn)元素、計(jì)算對(duì)角線(xiàn)元素和np.diagonal
這篇文章主要介紹了numpy如何取出對(duì)角線(xiàn)元素、計(jì)算對(duì)角線(xiàn)元素和np.diagonal問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06Python實(shí)現(xiàn)批量合并Excel文件的第二張合并Excel
在數(shù)據(jù)處理和分析中,經(jīng)常需要對(duì)多個(gè)Excel文件進(jìn)行批量操作,特別是當(dāng)這些文件具有相似的結(jié)構(gòu)時(shí),下面我們就來(lái)看看Python如何實(shí)現(xiàn)批量合并文件夾下所有Excel文件的第二張表吧2024-03-03Python輕松獲取網(wǎng)絡(luò)時(shí)間和本地時(shí)間技巧揭秘
這篇文章主要為大家介紹了Python輕松獲取網(wǎng)絡(luò)時(shí)間和本地時(shí)間技巧揭秘,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01python實(shí)現(xiàn)識(shí)別手寫(xiě)數(shù)字 python圖像識(shí)別算法
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)識(shí)別手寫(xiě)數(shù)字,python圖像識(shí)別算法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01Django ORM 自定義 char 類(lèi)型字段解析
這篇文章主要介紹了Django ORM 自定義 char 類(lèi)型字段解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08如何構(gòu)建第二個(gè)Django的應(yīng)用程序
這篇文章主要介紹了如何構(gòu)建第二個(gè)Django的應(yīng)用程序問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2025-03-03