mirror of
https://github.com/TriM-Organization/Musicreater.git
synced 2024-11-11 01:27:35 +08:00
461 lines
15 KiB
Python
461 lines
15 KiB
Python
# -*- coding: utf-8 -*-
|
||
|
||
|
||
# 音·创 开发交流群 861684859
|
||
# Email EillesWan2006@163.com W-YI_DoctorYI@outlook.com EillesWan@outlook.com
|
||
# 版权所有 金羿("Eilles Wan") & 诸葛亮与八卦阵("bgArray") & 鸣凤鸽子("MingFengPigeon")
|
||
# 若需转载或借鉴 许可声明请查看仓库目录下的 Lisence.md
|
||
|
||
|
||
"""
|
||
音·创 库版 MIDI转换示例程序
|
||
Musicreater Package Version : Demo for Midi Conversion
|
||
|
||
Copyright 2023 all the developers of Musicreater
|
||
|
||
开源相关声明请见 ./Lisence.md
|
||
Terms & Conditions: ./Lisence.md
|
||
"""
|
||
|
||
languages = {
|
||
"ZH_CN": {
|
||
"MSCT": "音·创",
|
||
"ChooseLang": "选择语言",
|
||
"LangChd": "当前语言已经切换为",
|
||
"ZH_CN": "简体中文",
|
||
"ZH_TW": "繁体中文(台湾)",
|
||
"EN_GB": "英语(英国)",
|
||
"EN_US": "英语(美国)",
|
||
":": ":",
|
||
",": ",",
|
||
".": "。",
|
||
"ChooseFileFormat": "请输入输出格式[BDX(1)或MCPACK(0)]",
|
||
"ChoosePlayer": "请选择播放方式[计分板(1) 或 延迟(0)]",
|
||
"ChoosePath": "请输入MIDI路径或所在文件夹",
|
||
"WhetherArgEntering": "是否为文件夹内文件的转换统一参数[是(1) 或 否(0)]",
|
||
"EnterArgs": "请输入转换参数",
|
||
"noteofArgs": "注:文件夹内的全部midi将统一以此参数转换",
|
||
"EnterVolume": "请输入音量大小(0~1)",
|
||
"EnterSpeed": "请输入速度倍率",
|
||
"WhetherPgb": "是否自动生成进度条[是(1) 或 否(0)]",
|
||
"WhetherCstmProgressBar": "是否自定义进度条[是(1) 或 否(0)]",
|
||
"EnterProgressBarStyle": "请输入进度条样式",
|
||
"EnterSbName": "请输入计分板名称",
|
||
"EnterSelecter": "请输入播放者选择器",
|
||
"WhetherSbReset": "是否自动重置计分板[是(1) 或 否(0)]",
|
||
"EnterAuthor": "请输入作者",
|
||
"EnterMaxHeight": "请输入指令结构最大生成高度",
|
||
"ErrEnter": "输入错误",
|
||
"Re-Enter": "请重新输入",
|
||
"Dealing": "正在处理",
|
||
"FileNotFound": "文件(夹)不存在",
|
||
"ChooseOutPath": "请输入结果输出路径",
|
||
"Saying": "言·论",
|
||
"Failed": "失败",
|
||
"CmdLength": "指令数量",
|
||
"MaxDelay": "曲目时间(游戏刻)",
|
||
"PlaceSize": "结构占用大小",
|
||
"LastPos": "最末方块坐标",
|
||
"PressEnterExit": "请按下回车键退出。",
|
||
}
|
||
}
|
||
|
||
|
||
import sys
|
||
|
||
if sys.argv.__len__() > 0:
|
||
currentLang = sys.argv[0]
|
||
if not currentLang in languages.keys():
|
||
currentLang = "ZH_CN"
|
||
else:
|
||
currentLang = "ZH_CN"
|
||
|
||
|
||
def _(__):
|
||
'''
|
||
`languages`
|
||
'''
|
||
return languages[currentLang][__]
|
||
|
||
|
||
import os
|
||
import random
|
||
import datetime
|
||
|
||
from msctPkgver.main import *
|
||
|
||
try:
|
||
from rich.console import Console
|
||
except ModuleNotFoundError as E:
|
||
if input("您需要安装 Rich 模块才能使用这个样例\n请问是否安装?(y/n)").lower() in ('y', '1'):
|
||
os.system("pip install Rich -i https://mirrors.aliyun.com/pypi/")
|
||
from rich.console import Console
|
||
else:
|
||
raise E
|
||
|
||
try:
|
||
import zhdate
|
||
except ModuleNotFoundError as E:
|
||
if input("您需要安装 zhdate 模块才能使用这个样例\n请问是否安装?(y/n)").lower() in ('y', '1'):
|
||
os.system("pip install zhdate -i https://mirrors.aliyun.com/pypi/")
|
||
import zhdate
|
||
else:
|
||
raise E
|
||
|
||
|
||
try:
|
||
import requests
|
||
except ModuleNotFoundError as E:
|
||
if input("您需要安装 requests 模块才能使用这个样例\n请问是否安装?(y/n)").lower() in ('y', '1'):
|
||
os.system("pip install requests -i https://mirrors.aliyun.com/pypi/")
|
||
import requests
|
||
else:
|
||
raise E
|
||
|
||
|
||
MainConsole = Console()
|
||
|
||
MainConsole.print(
|
||
"[#121110 on #F0F2F4] ",
|
||
style="#121110 on #F0F2F4",
|
||
justify="center",
|
||
)
|
||
|
||
|
||
# 显示大标题
|
||
MainConsole.rule(title="[bold #AB70FF]欢迎使用音·创独立转换器", characters="=", style="#26E2FF")
|
||
MainConsole.rule(
|
||
title="[bold #AB70FF]Welcome to Independent Musicreater Convernter", characters="-"
|
||
)
|
||
|
||
nowYang = datetime.datetime.now()
|
||
nowYin = zhdate.ZhDate.from_datetime(nowYang)
|
||
|
||
if nowYang.month == 8 and nowYang.day == 6:
|
||
# 诸葛八卦生日
|
||
MainConsole.print(
|
||
"[#7DB5F0 on #121110]今天可不是催更的日子!\n诸葛亮与八卦阵{}岁生日快乐!".format(nowYang.year - 2009),
|
||
style="#7DB5F0 on #121110",
|
||
justify="center",
|
||
)
|
||
elif nowYang.month == 4 and nowYang.day == 3:
|
||
# 金羿生日快乐
|
||
MainConsole.print(
|
||
"[#0089F2 on #F0F2F4]今天就不要催更啦!\n金羿{}岁生日快乐!".format(nowYang.year - 2006),
|
||
style="#0089F2 on #F0F2F4",
|
||
justify="center",
|
||
)
|
||
elif nowYin.lunar_month == 12 and nowYin.lunar_day == 30:
|
||
MainConsole.print(
|
||
"[#FF3432 on #121110]除夕到了,你是否与家人共处,融融其乐?",
|
||
style="#FF3432 on #121110",
|
||
justify="center",
|
||
)
|
||
elif nowYin.leap_month == 1 and nowYin.lunar_day in range(1, 9):
|
||
MainConsole.print(
|
||
"[#FFF642 on #FF3432]春节快乐!\n在你使用音·创的时候,是不是也要去感受一下喜庆的氛围呢?",
|
||
style="#FFF642 on #FF3432",
|
||
justify="center",
|
||
)
|
||
else:
|
||
# 显示箴言部分
|
||
MainConsole.print(
|
||
"[#121110 on #F0F2F4]{}".format(
|
||
random.choice(
|
||
requests.get(
|
||
'https://gitee.com/EillesWan/Musicreater/raw/master/resources/myWords.txt'
|
||
)
|
||
.text.strip('\r\n')
|
||
.split('\r\n')
|
||
)
|
||
),
|
||
style="#121110 on #F0F2F4",
|
||
justify="center",
|
||
)
|
||
|
||
|
||
from typing import Any, Literal, Optional, TextIO
|
||
|
||
JustifyMethod = Literal["default", "left", "center", "right", "full"]
|
||
OverflowMethod = Literal["fold", "crop", "ellipsis", "ignore"]
|
||
|
||
# 高级的打印函数
|
||
def prt(
|
||
*objects: Any,
|
||
sep: str = " ",
|
||
end: str = "\n",
|
||
justify: Optional[JustifyMethod] = None,
|
||
overflow: Optional[OverflowMethod] = None,
|
||
no_wrap: Optional[bool] = None,
|
||
emoji: Optional[bool] = None,
|
||
markup: Optional[bool] = None,
|
||
highlight: Optional[bool] = None,
|
||
width: Optional[int] = None,
|
||
height: Optional[int] = None,
|
||
crop: bool = True,
|
||
soft_wrap: Optional[bool] = None,
|
||
new_line_start: bool = False,
|
||
) -> None:
|
||
"""打印到控制台。
|
||
|
||
Args:
|
||
objects (位置性的args): 要记录到终端的对象。
|
||
sep (str, 可选): 要在打印数据之间写入的字符串。默认为""。
|
||
end (str, optio可选nal): 在打印数据结束时写入的字符串。默认值为"\\\\n"。
|
||
style (Union[str, Style], 可选): 应用于输出的样式。默认为`None`。
|
||
justify (str, 可选): 校正位置,可为"default", "left", "right", "center" 或 "full". 默认为`None`。
|
||
overflow (str, 可选): 控制溢出:"ignore"忽略, "crop"裁剪, "fold"折叠, "ellipsis"省略号。默认为`None`。
|
||
no_wrap (Optional[bool], 可选): 禁用文字包装。默认为`None`。
|
||
emoji (Optional[bool], 可选): 启用表情符号代码,或使用控制台默认的`None`。默认为`None`。
|
||
markup (Optional[bool], 可选): 启用标记,或`None`使用控制台默认值。默认为`None`。
|
||
highlight (Optional[bool], 可选): 启用自动高亮,或`None`使用控制台默认值。默认为`None`。
|
||
width (Optional[int], 可选): 输出的宽度,或`None`自动检测。默认为`None`。
|
||
crop (Optional[bool], 可选): 裁剪输出到终端的宽度。默认为`True`。
|
||
soft_wrap (bool, 可选): 启用软包装模式,禁止文字包装和裁剪,或`None``用于 控制台默认值。默认为`None`。
|
||
new_line_start (bool, False): 如果输出包含多行,在开始时插入一个新行。默认值为`False`。
|
||
"""
|
||
MainConsole.print(
|
||
*objects,
|
||
sep=sep,
|
||
end=end,
|
||
style="#F0F2F4 on #121110",
|
||
justify=justify,
|
||
overflow=overflow,
|
||
no_wrap=no_wrap,
|
||
emoji=emoji,
|
||
markup=markup,
|
||
highlight=highlight,
|
||
width=width,
|
||
height=height,
|
||
crop=crop,
|
||
soft_wrap=soft_wrap,
|
||
new_line_start=new_line_start,
|
||
)
|
||
|
||
|
||
prt(f"{_('LangChd')}{_(':')}{_(currentLang)}")
|
||
|
||
# 高级的输入函数
|
||
def ipt(
|
||
*objects: Any,
|
||
sep: str = " ",
|
||
justify: Optional[JustifyMethod] = None,
|
||
overflow: Optional[OverflowMethod] = None,
|
||
no_wrap: Optional[bool] = None,
|
||
emoji: Optional[bool] = None,
|
||
markup: Optional[bool] = None,
|
||
highlight: Optional[bool] = None,
|
||
width: Optional[int] = None,
|
||
height: Optional[int] = None,
|
||
crop: bool = True,
|
||
soft_wrap: Optional[bool] = None,
|
||
new_line_start: bool = False,
|
||
password: bool = False,
|
||
stream: Optional[TextIO] = None,
|
||
) -> str:
|
||
"""显示一个提示并等待用户的输入。
|
||
|
||
它的工作方式与Python内建的 :func:`input` 函数相同,如果Python内建的 :mod:`readline` 模块先前已经加载,则提供详细的行编辑和历史功能。
|
||
|
||
Args:
|
||
objects (位置性的args): 要记录到终端的对象。
|
||
sep (str, 可选): 要在打印数据之间写入的字符串。默认为""。
|
||
end (str, optio可选nal): 在打印数据结束时写入的字符串。默认值为"\\\\n"。
|
||
style (Union[str, Style], 可选): 应用于输出的样式。默认为`None`。
|
||
justify (str, 可选): 校正位置,可为"default", "left", "right", "center" 或 "full". 默认为`None`。
|
||
overflow (str, 可选): 控制溢出:"ignore"忽略, "crop"裁剪, "fold"折叠, "ellipsis"省略号。默认为`None`。
|
||
no_wrap (Optional[bool], 可选): 禁用文字包装。默认为`None`。
|
||
emoji (Optional[bool], 可选): 启用表情符号代码,或使用控制台默认的`None`。默认为`None`。
|
||
markup (Optional[bool], 可选): 启用标记,或`None`使用控制台默认值。默认为`None`。
|
||
highlight (Optional[bool], 可选): 启用自动高亮,或`None`使用控制台默认值。默认为`None`。
|
||
width (Optional[int], 可选): 输出的宽度,或`None`自动检测。默认为`None`。
|
||
crop (Optional[bool], 可选): 裁剪输出到终端的宽度。默认为`True`。
|
||
soft_wrap (bool, 可选): 启用软包装模式,禁止文字包装和裁剪,或`None``用于 控制台默认值。默认为`None`。
|
||
new_line_start (bool, False): 如果输出包含多行,在开始时插入一个新行。默认值为`False`。
|
||
password (bool, 可选): 隐藏已经输入的文案,默认值为`False`。
|
||
stream (TextIO, 可选): 可选从文件中读取(而非控制台),默认为 `None`。
|
||
|
||
Returns:
|
||
str: 从stdin读取的字符串
|
||
"""
|
||
MainConsole.print(
|
||
*objects,
|
||
sep=sep,
|
||
end="",
|
||
style="#F0F2F4 on #121110",
|
||
justify=justify,
|
||
overflow=overflow,
|
||
no_wrap=no_wrap,
|
||
emoji=emoji,
|
||
markup=markup,
|
||
highlight=highlight,
|
||
width=width,
|
||
height=height,
|
||
crop=crop,
|
||
soft_wrap=soft_wrap,
|
||
new_line_start=new_line_start,
|
||
)
|
||
|
||
return MainConsole.input("", password=password, stream=stream)
|
||
|
||
|
||
def formatipt(
|
||
notice: str,
|
||
fun,
|
||
errnote: str = f"{_('ErrEnter')}{_(',')}{_('Re-Enter')}{_('.')}",
|
||
*extraArg,
|
||
):
|
||
'''循环输入,以某种格式
|
||
notice: 输入时的提示
|
||
fun: 格式函数
|
||
errnote: 输入不符格式时的提示
|
||
*extraArg: 对于函数的其他参数'''
|
||
while True:
|
||
result = ipt(notice)
|
||
try:
|
||
funresult = fun(result, *extraArg)
|
||
break
|
||
except:
|
||
prt(errnote)
|
||
continue
|
||
return result, funresult
|
||
|
||
|
||
# 获取midi列表
|
||
while True:
|
||
midipath = ipt(f"{_('ChoosePath')}{_(':')}").lower()
|
||
if os.path.exists(midipath):
|
||
if os.path.isfile(midipath):
|
||
midis = (midipath,)
|
||
elif os.path.isdir(midipath):
|
||
midis = tuple(
|
||
(
|
||
os.path.join(midipath, i)
|
||
for i in os.listdir(midipath)
|
||
if i.lower().endswith('.mid') or i.lower().endswith('.midi')
|
||
)
|
||
)
|
||
else:
|
||
prt(f"{_('ErrEnter')}{_(',')}{_('Re-Enter')}{_('.')}")
|
||
continue
|
||
else:
|
||
prt(f"{_('FileNotFound')}{_(',')}{_('Re-Enter')}{_('.')}")
|
||
continue
|
||
break
|
||
|
||
# 获取输出地址
|
||
outpath = formatipt(
|
||
f"{_('ChooseOutPath')}{_(':')}",
|
||
os.path.exists,
|
||
f"{_('FileNotFound')}{_(',')}{_('Re-Enter')}{_('.')}",
|
||
)[0].lower()
|
||
|
||
|
||
# 选择输出格式
|
||
while True:
|
||
fileFormat = ipt(f"{_('ChooseFileFormat')}{_(':')}").lower()
|
||
if fileFormat in ('0', 'mcpack'):
|
||
fileFormat = 0
|
||
playerFormat = 1
|
||
break
|
||
|
||
elif fileFormat in ('1', 'bdx'):
|
||
fileFormat = 1
|
||
while True:
|
||
playerFormat = ipt(f"{_('ChoosePlayer')}{_(':')}").lower()
|
||
if playerFormat in ('0', '延迟', 'delay'):
|
||
playerFormat = 0
|
||
elif playerFormat in ('1', '计分板', 'scoreboard'):
|
||
playerFormat = 1
|
||
else:
|
||
prt(f"{_('ErrEnter')}{_(',')}{_('Re-Enter')}{_('.')}")
|
||
continue
|
||
break
|
||
else:
|
||
prt(f"{_('ErrEnter')}{_(',')}{_('Re-Enter')}{_('.')}")
|
||
continue
|
||
break
|
||
|
||
|
||
# 真假字符串判断
|
||
def boolstr(sth: str) -> bool:
|
||
try:
|
||
return bool(int(sth))
|
||
except:
|
||
if str(sth).lower() == 'true':
|
||
return True
|
||
elif str(sth).lower() == 'false':
|
||
return False
|
||
else:
|
||
raise "布尔字符串啊?"
|
||
|
||
|
||
prompts = []
|
||
# 提示语 检测函数 错误提示语
|
||
for args in [
|
||
(
|
||
f'{_("EnterVolume")}{_(":")}',
|
||
float,
|
||
),
|
||
(
|
||
f'{_("EnterSpeed")}{_(":")}',
|
||
float,
|
||
),
|
||
(
|
||
f'{_("WhetherPgb")}{_(":")}',
|
||
boolstr,
|
||
),
|
||
(
|
||
f'{_("EnterSbName")}{_(":")}',
|
||
str,
|
||
)
|
||
if playerFormat == 1
|
||
else (
|
||
f'{_("EnterSelecter")}{_(":")}',
|
||
str,
|
||
),
|
||
(
|
||
f'{_("WhetherSbReset")}{_(":")}',
|
||
boolstr,
|
||
)
|
||
if playerFormat == 1
|
||
else (),
|
||
(
|
||
f'{_("EnterAuthor")}{_(":")}',
|
||
str,
|
||
)
|
||
if fileFormat == 1
|
||
else (),
|
||
(
|
||
f'{_("EnterMaxHeight")}{_(":")}',
|
||
int,
|
||
)
|
||
if fileFormat == 1
|
||
else (),
|
||
]:
|
||
if args:
|
||
prompts.append(formatipt(*args)[1])
|
||
|
||
|
||
newLine = '\n'
|
||
conversion = midiConvert()
|
||
for singleMidi in midis:
|
||
conversion.convert(singleMidi, outpath)
|
||
conversion_result = (
|
||
conversion.tomcpack(2, *prompts)
|
||
if fileFormat == 0
|
||
else (
|
||
conversion.toBDXfile(2, *prompts)
|
||
if playerFormat == 1
|
||
else conversion.toBDXfile_withDelay(1, *prompts)
|
||
)
|
||
)
|
||
if conversion_result[0]:
|
||
prt(
|
||
f"{newLine}{_('Dealing')} {singleMidi} {_(':')}{newLine} {_('CmdLength')}{_(':')}{conversion_result[1]}{_(',')}{_('MaxDelay')}{_(':')}{conversion_result[2]}{f'''{_(',')}{_('PlaceSize')}{_(':')}{conversion_result[3]}{_(',')}{_('LastPos')}{_(':')}{conversion_result[4]}''' if fileFormat == 1 else ''}"
|
||
)
|
||
else:
|
||
prt(f"{_('Failed')}")
|
||
|
||
ipt(_("PressEnterExit"))
|