この記事はN高等学校(1) Advent Calendar 2020 3日目の記事です。
去年に引き続き、今年も参加させていただきます。
私は現在3年生で、通学コースに通っています。今年も去年と同じくプログラミングの時間を使ってプログラムを組んで遊んでいました。
来年はS高等学校とのPV数・いいね数対決とかするんですかね?見ものです。
この記事では、タイトル通りPythonとVLCで作って改善した音楽クイズの話をします。去年よりちゃんとアドカレっぽい
すでに作ってみたとか改善してみたとかの記事は投稿しているのですが、今回はより詳細に思い出しつつ書いてみようと思います。
音楽クイズを作るきっかけ
2020年4月、世は緊急事態宣言の真っ最中。
ゲームセンターも多くが営業を停止してしまい、音ゲーマーとしてはなかなか辛い時期でした。
そんな中娯楽を探していると、ふと音ゲーのサウンドトラックが目に留まりました。
**このCDを使って、イントロクイズをやったら面白いのでは?**と考えたのです。
最初は親に頼み込んで、スマホでランダムに音楽を再生してもらいました。
可能ではありましたが、第三者がいなければプレイできないため、やりたいときにやれないという問題がありました。
そこで、プログラムを組んでそこを自動化してしまえば良いのではないか?と考えたのが始まりでした。
どうやって作るか考える
プログラミング言語
これに関しては、最初からPython一択でした。
個人的にPythonが一番使い慣れた言語だったためです。
音楽を再生する方法
適当に検索してみただけでもそこそこ方法がありました。
その中で、私はVLCを使うという手段を取りました。
なぜVLCなのか?
私はいつもCDからflacで取り込んでいるのですが、それ以外にもGoogle Play Musicやレコチョク等で購入したmp3やwav、m4aなどがありました。
wavやmp3はともかく、flacやm4aが再生できるライブラリは多くはなさそうに思えたのです。
その点VLCなら対応しているコーデックも豊富で、再生できなくなるということはないだろうと判断しました。
今考えればクイズ用の音楽はmp3に変換しておくとかしたほうが楽だった気もします。
仕様を決め、作ってみる
最初は以下の通りに作ることにしました。
- 同じディレクトリから音楽ファイルを1つランダムに選ぶ
- 10秒間再生したあと、曲名を入力させる
- 3文字以上合っていたら正解と見なす
- 完全に合っていたらパーフェクトにする
- ギブアップコマンドを用意しておく
この通りに作ったソースコードはこちらのGistに載せてあります。
実際に使ってみて、改善する
友人に協力してもらい、Discordで通話を繋ぎながらイントロクイズに参加してもらいました。
なかなか盛り上がった一方で、こちらの記事にある通り、改善すべき点もいくつか浮かび上がってきました。
- リプレイ不可能
- 大文字/小文字が区別される
- 正解表示後一瞬で始まってしまう
- イントロの無音区間が長い曲がある
- 何度かやると覚えてしまう
- 判定バグがある
- タイトルが長い曲でも3文字で正解になってしまう
これらを改善するため、ギブアップやリプレイといった機能をコマンドとしてまとめたり、判定を3文字から1/3にしたりといった工夫を重ねました。
また、4番目の問題を解決するため、イントロクイズをやめて曲のランダムな部分を再生するようにしました。
これらが解決されたコードはこちらのGistに置いてあります。
さらに改善する
曲名直接取りたい問題
上で紹介した記事で、「曲名を直接取得したい」と述べました。
そもそも音楽ファイルには曲名やアーティスト名といったパラメータが存在しています。
例えばWindowsのエクスプローラーでは以下のように表示されます。
さて、「TiamaT:F minor」という楽曲が存在します。しかしファイル名に:
は使えません。
上の画像にもある通り、ファイル名は「TiamaT-F minor.flac」となってしまっています。
ファイル名から曲名を取得するという仕様では、この曲をクイズに含めることはできません。
さて、この曲名はどうやって取得すれば良いのでしょうか?
Windowsのプロパティから(失敗)
ファイルを右クリックして、メニューから「プロパティ」を選び「詳細」タブを開くとこのような表示が出ます。
ここには曲名やアーティスト名も記載されています。
これをなんとか取得できればよいのでは?と考え、取得する方法を探しました。
結果としては、こちらのようにPowershell経由でプロパティにアクセスする方法はあるようでしたが、Pythonのsubprocessモジュールを用いてみたものの、どこを間違えたのか結局動作せず、挫折してしまいました。
mutagenモジュールを使う(成功)
詳しい友人に泣きついたところ、「mutagenモジュールを使えばいいのでは」という回答が返ってきました。
mutagenは、オーディオメタデータを扱うPythonモジュールです(参考: https://mutagen.readthedocs.io/en/latest/ )。
It supports ASF, FLAC, MP4, Monkey’s Audio, MP3, Musepack, Ogg Opus, Ogg FLAC, Ogg Speex, Ogg Theora, Ogg Vorbis, True Audio, WavPack, OptimFROG, and AIFF audio files.
サポートされているファイルの種類も豊富です。できれば最初からこれを知りたかった……。
扱うファイルは数種類あるので、こちらを参考にして実装しました。
こちらで種類を判別しなくてもmutagen.File()
に食わせておけばいい感じに扱ってくれます。
ちなみにこの方法でアーティスト名も取得できるので、ヒントの一つに加えることにしました。
不正解のときにどれくらい合ってるか知りたい問題
「Lethaeus」という曲があります。
遊んでいるときにもこの曲は何度か登場したのですが、綴りが覚えづらいのかやたらみんな間違えます。
しかも、1文字違うだけなのに「じゃあ全く違う曲か」と勘違いして答えから遠ざかってしまうこともしばしばありました。
そこで、入力した答えと正解がどれくらい違うのかがわかるようにしたら良いのではないかと考えました。
探してみると、difflibというまさに探していたものが標準モジュールにありました。
さすがはBattery includedという思想を標榜しているだけあって、素晴らしい充実度だと思います。
文字種多すぎ問題
曲名に記号が含まれていると厄介なことになります。
「♡」と「♥」は別物ですが、さすがにその違いをいちいち指摘していては楽しめなくなるのでは?と思いました。
実装としては、replaceメソッドを鎖のように繋げて数種類の文字を消したり変換したりしています。
他にも、友人の指摘を受けて正解判定を1/3から1/2にするなどのマイナーアップデートを行いました。
上に挙げた数点の改善を施したソースコードはこちらのGistに載せています。
作ってみての感想
最初は自分が楽しむために作ったものでしたが、友人と楽しんでフィードバックをもらっていくと、至らない点もだんだん見えてくるようになりました。
プログラミングをする人たちは皆口を揃えて「アウトプットが大事」と言いますが、本当にその通りだと実感しました。
制作したものは、何らかの形で発表してみるべきでしょう。
12月も始まったばかりです。まだ空いてるアドベントカレンダーに滑り込むチャンスはありますよ!
こういう機会に、ぜひアウトプットしてみてください。
それではまた2日後、N高等学校(2) Advent Calendar 2020でお会いしましょう。
明日は @Souken521 さんの「HomeKit対応機器+HomeBridge+Nature Remoで家を快適にする話」です。スマートホーム、憧れますよね。