0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

冥鳴ひまりのMP3プレイヤーを作った

Posted at

この間作ったラジオ風のMP3プレイヤーを進化させました。
https://qiita.com/butaushi/items/fe8817107601f405f730

tkinterも少しずつわかってきたので背景に画像を設定。
せっかくなので、冥鳴ひまりの立ち絵をお借りして。
なんとか動きをつけたいなぁ、と悩んだ結果、ハートをピコピコさせるだけにしました。
ちなみに、ハートのリズムは、MP3の曲調(BPM)に合わせて変わります。
Pythonでlibrosaというライブラリでできました。
image.png

ということでまた全文投下。
コアなロジックは前回からあんまり変わっていません。
(フェードアウトのタイミングだけ微調整しました)

他にも、以下機能追加。
・プレイリスト機能追加
・ひまりがたまにしゃべる

お借りした素材はこちらです。
・MP3プレイヤーのフレーム
【1280×720☆背景素材】冥鳴ひまりフレーム③
https://commons.nicovideo.jp/works/nc364749

・立ち絵素材
冥鳴ひまり立ち絵素材【とらっかぁ】
https://seiga.nicovideo.jp/seiga/im10916868

himariradio.py
#!/home/shu/python/radio/radio_venv/bin/python
"""
 機能:MP3をラジオっぽく再生する
	  見た目が「冥鳴ひまり」になりました
 注意:フォルダ内にあるmp3をランダムで再生
"""
import os
import glob
import threading
import subprocess
import time
import random
import atexit
import librosa		# pip install librosa

import tkinter as tk
from tkinter import filedialog
import pygame
from mutagen.mp3 import MP3

# 定数
FOLDER_PATH = "/home/shu/music/"		# 初期フォルダ
BG_COLOR = "#dfdbda"		# 背景フレームに合わせた色。擬似的に透過。
FILE_DIALOG_COLOR = "#8b008b"
FILE_DIALOG_COLOR_HOVAR = "#4b0082"


# ひまりの声 絶対パスでないとダメっぽい
JINGLE_VOICE = [ "/home/shu/python/radio/sound/voice/himari/next.wav"
				,"/home/shu/python/radio/sound/voice/himari/love.wav"
				,"/home/shu/python/radio/sound/voice/himari/pman.wav"
				,"/home/shu/python/radio/sound/voice/himari/niceday.wav"
				,"/home/shu/python/radio/sound/voice/himari/journey.wav"
				,"/home/shu/python/radio/sound/voice/himari/sky.wav"
				]

global t1
global nowplaying		# 0:再生していない、1:再生中
global playlist			# フォルダ内のMP3リスト
global listnum			# 再生中のリスト番号
global musictitle		# ファイル名(表示用)
global playtime			# 再生時間
global gtempo			# BPM

global process


"""
 main関数
"""
class HimariRadioPlayer:
	"""
	初期化
	"""
	def __init__(self, root):
		
		global t1, nowplaying, playlist, listnum, musictitle, playtime, gtempo
		
		t1 = 0
		nowplaying = 0
		playtime = 0
		listnum = 0
		musictitle = FOLDER_PATH		# 初期フォルダを表示。変なコメント出すよりも有効。
		self.img = [None, None, None]			# 画像用リスト
		self.j = 0
		gtempo = 60			# ゼロ割になるので60=1s更新で
		
		# ウィンドウ作成
		self.root = root
		self.root.title("ひまりMP3プレイヤー")
		self.root.geometry("800x450")  # ウィンドウサイズ
		
		# 画像ファイルの読み込み
		self.img_frame = tk.PhotoImage(file="/home/shu/picture/himari/himari_frame_m.png")
		self.img[0] = tk.PhotoImage(file="/home/shu/picture/himari/normal_c.png")
		self.img[1] = tk.PhotoImage(file="/home/shu/picture/himari/himari_heart.png")
		self.img[2] = tk.PhotoImage(file="/home/shu/picture/himari/himari_heart2.png")
		
		# 背景(フレーム)の配置
		self.label1 = tk.Label(root, image=self.img_frame)
		self.label1.place(x = -1, y = -1)		# ウィンドウ枠を見えなくするために-1で設定する。

		# ひまり画像配置
		self.label2 = tk.Label(root, image=self.img[self.j])
		self.label2.configure(bg = BG_COLOR, width=320, height=364)
		self.label2.place(x = 460, y = 70)

		# 再生ボタン
		self.playbutton = tk.Label(root, text = "▶️")
		self.playbutton.configure(bg = BG_COLOR, foreground = "#32cd32", font=("",20))
		self.playbutton.bind("<Button-1>", self.mouse_click)
		self.playbutton.bind("<Motion>", self.mouse_on)
		self.playbutton.bind("<Leave>", self.mouse_leave)
		self.playbutton.place(x = 731, y = 400)
		
		# 今流れているのは
		self.titleis = tk.Label(root, text = "")
		self.titleis.configure(bg = BG_COLOR, width = 25, font=("",14))
		self.titleis.place(x = 30, y = 180)
		
		# 曲名表示
		self.label = tk.Label(root, text = musictitle)
		self.label.configure(bg = BG_COLOR, width = 43, font=("",14))
		self.label.place(x = 30, y = 250)
		
		# です!
		self.desu = tk.Label(root, text = "")
		self.desu.configure(bg = BG_COLOR, width = 15, font=("",14))
		self.desu.place(x = 330, y = 320)
		
		# ファイル指定ボタン
		self.filebutton = tk.Label(root, text = "")
		self.filebutton.configure(bg = BG_COLOR, foreground = FILE_DIALOG_COLOR, font=("",16))
		self.filebutton.bind("<Button-1>", self.on_filebutton_click)
		self.filebutton.bind("<Motion>", self.mouse_on_file)
		self.filebutton.bind("<Leave>", self.mouse_leave_file)
		self.filebutton.place(x = 731, y = 365)

		# フォルダ指定ボタン
		self.stopbutton = tk.Label(root, text = "")
		self.stopbutton.configure(bg = BG_COLOR, foreground = FILE_DIALOG_COLOR, font=("",16))
		self.stopbutton.bind("<Button-1>", self.select_folder)
		self.stopbutton.bind("<Motion>", self.mouse_on_f)
		self.stopbutton.bind("<Leave>", self.mouse_leave_f)
		self.stopbutton.place(x = 730, y = 325)

		# フォルダパスからプレイリスト作成
		make_playlist( FOLDER_PATH )

		# 更新処理(ループ 1sec)
		self.update_music()
		
		# ひまり画像用ループ
		self.update_disp()


	"""
	ひまり画像用ループ
	"""
	def update_disp(self):
		
		global nowplaying, gtempo
		
		# 再生中のみ画像更新
		if nowplaying == 1:
			# ひまり画像更新
			self.j = self.j + 1
			if self.j > 2:
				self.j = 1
			self.label2.configure(image=self.img[self.j])
			self.titleis.configure(text="今流れている曲は…") 
			self.desu.configure(text="です!") 
		else:
			self.label2.configure(image=self.img[0])
			self.titleis.configure(text="")
			self.desu.configure(text="") 

		# 再度コールでループ
		# 曲のBPMに応じて更新ペースを変える
		updatetiming = 1000*60/gtempo
		# print(round(updatetiming))	# デバッグ用
		self.root.after(round(updatetiming), self.update_disp)


	"""
	ループ処理 ある意味メイン処理
	"""
	def update_music(self):
		
		global t1, nowplaying, playlist, listnum, musictitle, playtime
		
		# 指定時間経過で次の曲再生
		if (nowplaying == 1) and (time.time() - t1 > playtime):
			# 次の曲
			listnum = listnum + 1
			
			# 全曲再生したら停止
			if listnum >= len(playlist):
				nowplaying = 0
				musictitle = "全部流したよ"
				self.playbutton.configure(text = "▶️")
			else:
				# ひまりの一言
				thread_v = threading.Thread(target=playjingle).start()
								
				# 次の音楽作成
				musicpath = make_music()
				
				# 再生後もGUIは操作したいのでスレッドを分ける
				thread = threading.Thread(target=playmusic,args=(musicpath, playtime)).start()
				
				# 現在時刻取得
				t1 = time.time()

		# タイトルラベル更新
		self.label.config(text = musictitle)
		
		# 1秒後に再度コール
		self.root.after(1000, self.update_music)

	"""
	 機能:ファイル選択ボタン
	"""
	# マウスオーバー
	def mouse_on_file(self, e):
		self.filebutton.configure(foreground = FILE_DIALOG_COLOR_HOVAR)

	# マウスリリース
	def mouse_leave_file(self, e):
		self.filebutton.configure(foreground = FILE_DIALOG_COLOR)

	"""
	 機能:ファイル(プレイリスト)選択ダイアログ
	"""
	def on_filebutton_click(self, e):
		
		global playlist, musictitle
		
		# ファイル拡張子はtxt限定
		# フォルダダイアログ表示時の初期フォルダは、ユーザのホームフォルダ
		file_name = tk.filedialog.askopenfilename(filetypes=[("Play List", ".txt")], initialdir="~/")
		# ファイルが選択されたらプレイリスト更新
		# 一旦正しく書かれている前提
		if file_name:
			with open(file_name, "r", encoding="utf-8") as f:
				playlist = [line.strip() for line in f.readlines()]  # 改行を削除
				musictitle = file_name


	"""
	 機能:フォルダ選択ボタン
	"""
	# マウスオーバー
	def mouse_on_f(self, e):
		self.stopbutton.configure(foreground = FILE_DIALOG_COLOR_HOVAR)

	# マウスリリース
	def mouse_leave_f(self, e):
		self.stopbutton.configure(foreground = FILE_DIALOG_COLOR)


	"""
	 機能:フォルダ選択
	"""
	def select_folder(self, e):
		
		global musictitle
		
		# 停止中のみ変更可
		if nowplaying == 0:
			# ダイアログ表示
			folder_path = filedialog.askdirectory(initialdir="~/")
			
			if folder_path:
				print("選択されたフォルダ:", folder_path)
				musictitle = folder_path
				make_playlist(folder_path)


	"""
	 ボタン表示処理
	"""
	# マウスオーバー
	def mouse_on(self, e):
		self.playbutton.configure(foreground = "#808000")

	# マウスリリース
	def mouse_leave(self, e):
		if nowplaying == 0:
			self.playbutton.configure(foreground = "#32cd32")
		else:
			self.playbutton.configure(foreground = "#ff4500")

	# クリック
	def mouse_click(self, e):
		
		global musictitle
		
		if nowplaying == 0:
			if len(playlist) > 0:
				# 再生していないなら流す。ボタンもストップに変える
				self.playbutton.configure(text = "⏸️")
				on_button_click(e)
			else:
				musictitle = "再生できるMP3ファイルがありません"
		else:
			# 再生中なら止める。ボタンも再生に変える。
			self.playbutton.configure(text = "▶️")
			on_button2_click(e)



"""
 機能:ひまりの一言再生
"""
def playjingle():
	
	# 発言タイミング調整
	time.sleep(2.5)

	# 毎回言わなくてもいいとも思うので3回に1回くらい
	# 曲の切り替わりに一言
	if random.randint(0, 2) > 1:
		process_v = subprocess.Popen(["paplay", random.choice(JINGLE_VOICE)])


"""
 機能:プレイリスト作成
 備考:フォルダ内のMP3ファイルを検索してリスト化
"""
def make_playlist(folderpath):
	
	global playlist

	# フォルダ内を再帰検索してmp3のリストを作る
	searchpath = folderpath + "/**/*.mp3"
	playlist = glob.glob(searchpath, recursive=True)
	
	# 取得したMP3一覧をテキストファイルに書き込み
	with open("playlist.txt", "w", encoding="utf-8") as f:
		f.writelines(line + "\n" for line in playlist)

	# 一応作っておく
	with open("playlist.txt", "r", encoding="utf-8") as f:
		playlist = [line.strip() for line in f.readlines()]  # 改行を削除


"""
 機能:mp3再生
 備考:プロセスをわけているのは、開発当初、subprocess.runだった名残
"""
def playmusic(filepath, playtime):
	
	global process
	
	process = subprocess.Popen([
		"/home/shu/python/radio/radio_venv/bin/python",
		"/home/shu/python/radio/src/mcore.py",
		filepath,
		str(playtime)
	])



"""
 機能:再生する音楽情報の生成
 引数:なし
 戻り値:mp3ファイルパス
 備考:結局グローバル変数使いまくっているので直したい
"""
def make_music():

	global playlist, listnum, musictitle, playtime, gtempo

	# リスト番号からファイルパス取得
	musicpath = playlist[listnum]
	print(musicpath)
	
	# ファイルパス → ファイル名取得(表示用)
	musictitle, ext = os.path.splitext(os.path.basename(musicpath))
	
	# 曲の長さを取得して、2分〜曲の長さ - 10秒で再生時間確定
	# 10秒はフェードアウトを考慮(適当)
	audio = MP3(musicpath)
	playtime = random.randint(120, (int(audio.info.length) - 10))
	
	# MP3の解析
	y, sr = librosa.load(musicpath)

	# テンポ(BPM)を取得
	tempo, _ = librosa.beat.beat_track(y=y, sr=sr)
	gtempo = tempo[0]

	print(f'Tempo: {tempo[0]} BPM')
	
	return musicpath


"""
 機能:再生ボタンクリックイベント
"""
def on_button_click(e):

	global t1, nowplaying, playtime, listnum, musictitle

	# リストをシャッフルすることでランダム再生
	random.shuffle(playlist)
	
	# リストの1曲目から
	listnum = 0
	
	# 1曲目再生するときに時間かかるのでしゃべってもらう
	# subprocess.run(["paplay", "/home/shu/python/nowboot.wav"])

	# 音楽情報取得
	musicpath = make_music()
	
	# 再生後もGUIは操作したいのでスレッドを分ける
	thread = threading.Thread(target=playmusic,args=(musicpath, playtime)).start()
	
	# 再生中に設定
	nowplaying = 1
	
	# 現在時刻取得
	t1 = time.time()


"""
 機能:停止ボタンクリックイベント
"""
def on_button2_click(e):
	
	kill_music()


"""
 機能:停止処理
"""
def kill_music():
	
	global nowplaying, musictitle
	
	# 停止に設定
	nowplaying = 0
	musictitle = "何も流してないよ"
	process.kill()
	process.wait()  # ちゃんと終わるまで待つ


"""
 終了処理
"""
# プログラム終了時の処理
atexit.register(kill_music)


"""
 main関数呼び出し
"""
if __name__ == "__main__":
	root = tk.Tk()
	app = HimariRadioPlayer(root)
	root.mainloop()

0
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?