mirror of
https://github.com/TriM-Organization/Linglun-Converter.git
synced 2024-11-28 16:15:25 +08:00
226 lines
6.6 KiB
Python
226 lines
6.6 KiB
Python
|
# -*- coding: utf-8 -*-
|
|||
|
|
|||
|
# 伶伦 开发交流群 861684859
|
|||
|
|
|||
|
|
|||
|
"""
|
|||
|
伶伦转换器暨模组特用奇巧
|
|||
|
Linglun Converter for Special Use with addons
|
|||
|
|
|||
|
版权所有 © 2023 金羿 & 睿穆开发组
|
|||
|
Copyright © 2023 EillesWan & TriM Org.
|
|||
|
|
|||
|
开源相关声明请见 ./License.md
|
|||
|
Terms & Conditions: ./Lisense.md
|
|||
|
"""
|
|||
|
|
|||
|
__version__ = "0.0.1"
|
|||
|
|
|||
|
import json
|
|||
|
import os
|
|||
|
import sys
|
|||
|
|
|||
|
from rich.table import Table
|
|||
|
|
|||
|
import Musicreater
|
|||
|
from Musicreater.constants import MIDI_PITCH_NAME_TABLE, PERCUSSION_INSTRUMENT_LIST
|
|||
|
from utils.io import *
|
|||
|
|
|||
|
osc.project_name = "伶伦暨模组特用奇巧"
|
|||
|
osc.version = __version__
|
|||
|
|
|||
|
|
|||
|
if len(sys.argv) > 0:
|
|||
|
|
|||
|
def go_for_args(debugMode: str = "False", logfile: str = "True"):
|
|||
|
global logger
|
|||
|
osc.isRelease = False if debugMode.lower() in ("true", "1") else True
|
|||
|
logger.printing = not osc.isRelease
|
|||
|
logger.writing = True if logfile.lower() in ("true", "1") else False
|
|||
|
|
|||
|
go_for_args(*sys.argv)
|
|||
|
|
|||
|
|
|||
|
# 显示大标题
|
|||
|
|
|||
|
MainConsole.print(
|
|||
|
"[#121110 on #F0F2F4] ",
|
|||
|
style="#121110 on #F0F2F4",
|
|||
|
justify="center",
|
|||
|
)
|
|||
|
MainConsole.rule(
|
|||
|
title="[bold #AB70FF]欢迎使用{}".format(osc.project_name),
|
|||
|
characters="=",
|
|||
|
style="#26E2FF",
|
|||
|
)
|
|||
|
# MainConsole.rule(title="[bold #AB70FF]Welcome to Linglun Converter", characters="-")
|
|||
|
MainConsole.rule(
|
|||
|
title="[#AB70FF]版本{} | 音·创内核版本{}".format(__version__, Musicreater.__version__),
|
|||
|
characters="-",
|
|||
|
style="#26E2FF",
|
|||
|
)
|
|||
|
MainConsole.print(
|
|||
|
"[#121110 on #F0F2F4] ",
|
|||
|
style="#121110 on #F0F2F4",
|
|||
|
justify="center",
|
|||
|
)
|
|||
|
|
|||
|
|
|||
|
# 获取midi列表
|
|||
|
while True:
|
|||
|
midi_path = ipt(f"MIDI地址或所在目录地址:")
|
|||
|
try:
|
|||
|
if os.path.exists(midi_path):
|
|||
|
if os.path.isfile(midi_path):
|
|||
|
midis = (midi_path,)
|
|||
|
elif os.path.isdir(midi_path):
|
|||
|
midis = (
|
|||
|
os.path.join(midi_path, i)
|
|||
|
for i in os.listdir(midi_path)
|
|||
|
if i.lower().endswith(".mid") or i.lower().endswith(".midi")
|
|||
|
)
|
|||
|
else:
|
|||
|
prt("输入内容有误,请重新输入。")
|
|||
|
continue
|
|||
|
else:
|
|||
|
prt("该地址不存在,或无法访问该地址,请重新输入。")
|
|||
|
continue
|
|||
|
except PermissionError:
|
|||
|
prt("无法访问该地址,请检查是否给予程序相关文件的访问权限。")
|
|||
|
continue
|
|||
|
break
|
|||
|
|
|||
|
|
|||
|
# percussion_only = format_ipt(
|
|||
|
# "仅处理打击乐器 (否/0|是/1):", bool_str, "输入内容格式错误,应为 是/1/真/t/y 或 否/0/假/f/n"
|
|||
|
# )[1]
|
|||
|
|
|||
|
speed: float = format_ipt("播放速度", float_str, "错误,需要浮点型数据;请重新输入。")[1]
|
|||
|
|
|||
|
|
|||
|
final_result: Dict[str, Dict[int, List[Tuple[str, int, str]]]] = {}
|
|||
|
|
|||
|
for single_midi in midis:
|
|||
|
fn = os.path.splitext(os.path.split(single_midi)[-1])[0]
|
|||
|
midi_cvt = Musicreater.MidiConvert.from_midi_file(
|
|||
|
single_midi,
|
|||
|
)
|
|||
|
midi_cvt.to_music_channels()
|
|||
|
|
|||
|
max_delay = 0
|
|||
|
instrument_ID = -1
|
|||
|
total_track = {}
|
|||
|
instriments = {}
|
|||
|
|
|||
|
# 此处 我们把通道视为音轨
|
|||
|
for i in midi_cvt.channels.keys():
|
|||
|
# 如果当前通道为空 则跳过
|
|||
|
if not midi_cvt.channels[i]:
|
|||
|
continue
|
|||
|
|
|||
|
# 第十通道是打击乐通道
|
|||
|
SpecialBits = True if i == 9 else False
|
|||
|
|
|||
|
for track_no, track in midi_cvt.channels[i].items():
|
|||
|
for msg in track:
|
|||
|
if msg[0] == "PgmC":
|
|||
|
instrument_ID = msg[1]
|
|||
|
|
|||
|
if msg[0] == "NoteS":
|
|||
|
soundID, _X = (
|
|||
|
midi_cvt.perc_inst_to_soundID_withX(msg[1])
|
|||
|
if SpecialBits
|
|||
|
else midi_cvt.inst_to_souldID_withX(instrument_ID)
|
|||
|
)
|
|||
|
score_now = round(msg[-1] / float(speed) / 50)
|
|||
|
max_delay = max(max_delay, score_now)
|
|||
|
mc_pitch = "" if SpecialBits else 2 ** ((msg[1] - 60 - _X) / 12)
|
|||
|
try:
|
|||
|
total_track[score_now].append(
|
|||
|
(MIDI_PITCH_NAME_TABLE[msg[1]], mc_pitch, soundID),
|
|||
|
)
|
|||
|
except (IndexError, KeyError):
|
|||
|
total_track[score_now] = [
|
|||
|
(MIDI_PITCH_NAME_TABLE[msg[1]], mc_pitch, soundID),
|
|||
|
]
|
|||
|
|
|||
|
try:
|
|||
|
instriments[soundID] += 1
|
|||
|
except (IndexError, KeyError):
|
|||
|
instriments[soundID] = 1
|
|||
|
|
|||
|
del midi_cvt
|
|||
|
|
|||
|
table = Table(
|
|||
|
*instriments.keys(),
|
|||
|
title="[bold #AB70FF on #121110]{} 乐器统计".format(fn),
|
|||
|
title_style="#26E2FF on #121110",
|
|||
|
)
|
|||
|
table.add_row(*[str(i) for i in instriments.values()])
|
|||
|
|
|||
|
MainConsole.print(
|
|||
|
"[bold #8B50DF on #F0F2F4]-={}=-".format(fn),
|
|||
|
style="#AB70FF on #F0F2F4",
|
|||
|
justify="center",
|
|||
|
)
|
|||
|
|
|||
|
prt(table)
|
|||
|
|
|||
|
inst_selected: Set[str] = format_ipt(
|
|||
|
"""请选择需要保留的乐器;以空格作分割;以 percussion 表示全部打击乐器|以 pitched 表示全部乐音乐器|以 all 表示所有乐器\n:""",
|
|||
|
lambda x: set(
|
|||
|
[
|
|||
|
i.lower()
|
|||
|
for i in x.split(" ")
|
|||
|
if isin(
|
|||
|
i,
|
|||
|
{
|
|||
|
True: (
|
|||
|
list(instriments.keys()) + ["all", "pitched", "percussion"]
|
|||
|
)
|
|||
|
},
|
|||
|
)
|
|||
|
]
|
|||
|
),
|
|||
|
"输入错误,需要在已有的乐器范围之内。",
|
|||
|
strict_mode=True,
|
|||
|
)[1]
|
|||
|
|
|||
|
if "all" in inst_selected:
|
|||
|
inst_selected.remove("all")
|
|||
|
for i in instriments.keys():
|
|||
|
inst_selected.add(i)
|
|||
|
|
|||
|
if "percussion" in inst_selected:
|
|||
|
inst_selected.remove("percussion")
|
|||
|
for i in instriments.keys():
|
|||
|
if i in PERCUSSION_INSTRUMENT_LIST:
|
|||
|
inst_selected.add(i)
|
|||
|
|
|||
|
if "pitched" in inst_selected:
|
|||
|
inst_selected.remove("pitched")
|
|||
|
for i in instriments.keys():
|
|||
|
if i not in PERCUSSION_INSTRUMENT_LIST:
|
|||
|
inst_selected.add(i)
|
|||
|
|
|||
|
result_piece = []
|
|||
|
for i, ele in total_track.items():
|
|||
|
this_ele = [k for k in ele if k[-1] in inst_selected]
|
|||
|
if this_ele:
|
|||
|
result_piece.append((i, this_ele))
|
|||
|
|
|||
|
result_piece.sort(key=lambda x: x[0])
|
|||
|
|
|||
|
final_result[fn] = {}
|
|||
|
for ind, ele in result_piece:
|
|||
|
final_result[fn][ind - result_piece[0][0]] = ele
|
|||
|
|
|||
|
with open("result.json", "w", encoding="utf-8") as f:
|
|||
|
json.dump(
|
|||
|
final_result,
|
|||
|
f,
|
|||
|
ensure_ascii=False,
|
|||
|
indent=3,
|
|||
|
sort_keys=True,
|
|||
|
)
|