Linglun-Converter/llc_cli.py

404 lines
11 KiB
Python
Raw Permalink Normal View History

2023-02-02 02:11:28 +08:00
# -*- coding: utf-8 -*-
# 伶伦 开发交流群 861684859
2023-02-02 02:11:28 +08:00
"""
伶伦转换器 命令行
Linglun Converter CLI
2023-02-02 02:11:28 +08:00
版权所有 © 2024 金羿
Copyright © 2024 EillesWan
2023-02-02 02:11:28 +08:00
伶伦转换器命令行版本本项目的协议颁发者为 金羿
The Licensor of _Linglun Converter CLI_("this project") is Eilles Wan.
本项目根据 第一版 汉钰律许可协议本协议授权
任何人皆可从以下地址获得本协议副本https://gitee.com/EillesWan/YulvLicenses
若非因法律要求或经过了特殊准许此作品在根据本协议原样提供的基础上不予提供任何形式的担保任何明示任何暗示或类似承诺也就是说用户将自行承担因此作品的质量或性能问题而产生的全部风险
详细的准许和限制条款请见原协议文本
2023-02-02 02:11:28 +08:00
"""
__version__ = "0.0.8"
2023-02-02 02:11:28 +08:00
import os
import random
2024-02-02 18:10:12 +08:00
# import sys
2023-02-02 02:11:28 +08:00
import Musicreater
from Musicreater import DEFAULT_PROGRESSBAR_STYLE
2023-09-24 02:14:04 +08:00
from Musicreater.plugin.addonpack import (
to_addon_pack_in_delay,
to_addon_pack_in_repeater,
to_addon_pack_in_score,
)
from Musicreater.plugin.mcstructfile import (
to_mcstructure_file_in_delay,
to_mcstructure_file_in_repeater,
to_mcstructure_file_in_score,
)
2023-10-02 18:22:40 +08:00
from Musicreater.plugin.bdxfile import to_BDX_file_in_delay, to_BDX_file_in_score
from utils.io import logger, int_str, prt, ipt, isin, bool_str, float_str
from utils.yanlun import yanlun_texts, solar_date
2023-09-24 02:14:04 +08:00
# from Musicreater.plugin.mcstructure import commands_to_structure, commands_to_redstone_delay_structure
2024-02-24 21:44:38 +08:00
logger.console.print(
2023-02-02 02:11:28 +08:00
"[#121110 on #F0F2F4] ",
style="#121110 on #F0F2F4",
justify="center",
)
2024-02-02 18:10:12 +08:00
# osc.project_name = "伶伦转换器"
# osc.version = __version__
2023-02-02 02:11:28 +08:00
2024-02-02 18:10:12 +08:00
# if len(sys.argv) > 0:
2023-02-02 02:11:28 +08:00
2024-02-02 18:10:12 +08:00
# 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
2023-02-11 23:23:05 +08:00
2024-02-02 18:10:12 +08:00
# go_for_args(*sys.argv)
2023-02-02 02:11:28 +08:00
# 显示大标题
2024-02-24 21:44:38 +08:00
logger.console.rule(
title="[bold #AB70FF]欢迎使用伶伦独立转换器", characters="=", style="#26E2FF"
)
2023-09-24 02:14:04 +08:00
# MainConsole.rule(title="[bold #AB70FF]Welcome to Linglun Converter", characters="-")
2024-02-24 21:44:38 +08:00
logger.console.rule(
title="[#AB70FF]版本{} | 音·创内核版本{}".format(
__version__, Musicreater.__version__
),
2023-09-24 02:14:04 +08:00
characters="-",
style="#26E2FF",
)
2023-02-02 02:11:28 +08:00
if solar_date == (4, 3):
2023-02-02 02:11:28 +08:00
# 诸葛八卦生日
style_ = "#7DB5F0 on #121110"
elif solar_date == (8, 6):
2023-02-02 02:11:28 +08:00
# 金羿生日快乐
style_ = "#0089F2 on #F0F2F4"
elif solar_date == (8, 16):
# 旧梦生日快乐
style_ = "#F0F2F4 on #0089F2" # TODO
2023-02-02 02:11:28 +08:00
else:
style_ = "#121110 on #F0F2F4"
# 显示箴言部分
logger.console.print(
"[{}]{}".format(style_, random.choice(yanlun_texts)),
style=style_,
justify="center",
)
del style_
2023-02-02 02:11:28 +08:00
2023-09-24 02:14:04 +08:00
# prt(f"{_('LangChd')}{_(':')}{_(currentLang)}")
2023-02-02 02:11:28 +08:00
def format_ipt(
notice: str,
fun,
2023-09-24 02:14:04 +08:00
err_note: str = "输入内容有误,请重新输入。",
2023-02-02 02:11:28 +08:00
*extraArg,
):
"""循环输入,以某种格式
notice: 输入时的提示
fun: 格式函数
err_note: 输入不符格式时的提示
*extraArg: 对于函数的其他参数"""
while True:
result = ipt(notice)
try:
fun_result = fun(result, *extraArg)
break
except ValueError:
prt(err_note)
continue
return result, fun_result
# 获取midi列表
while True:
2023-09-29 17:20:03 +08:00
midi_path = ipt(f"MIDI地址或所在目录地址")
2023-09-24 02:14:04 +08:00
try:
if os.path.exists(midi_path):
if os.path.isfile(midi_path):
midis = (midi_path,)
elif os.path.isdir(midi_path):
midis = (
2023-02-02 02:11:28 +08:00
os.path.join(midi_path, i)
for i in os.listdir(midi_path)
if i.lower().endswith(".mid") or i.lower().endswith(".midi")
)
2023-09-24 02:14:04 +08:00
else:
prt("输入内容有误,请重新输入。")
continue
2023-02-02 02:11:28 +08:00
else:
2023-09-24 02:14:04 +08:00
prt("该地址不存在,或无法访问该地址,请重新输入。")
2023-02-02 02:11:28 +08:00
continue
2023-09-24 02:14:04 +08:00
except PermissionError:
prt("无法访问该地址,请检查是否给予程序相关文件的访问权限。")
2023-02-02 02:11:28 +08:00
continue
break
# 获取输出地址
2023-09-24 02:14:04 +08:00
while True:
2023-09-29 17:20:03 +08:00
out_path = ipt(f"文件生成输出地址:")
2023-09-24 02:14:04 +08:00
try:
if not os.path.exists(out_path):
prt("该地址不存在,或无法访问该地址,请重新输入。")
continue
except PermissionError:
prt("无法访问该地址,请检查是否给予程序相关文件的访问权限。")
continue
break
2023-02-02 02:11:28 +08:00
# 选择输出格式
2023-04-29 16:58:22 +08:00
def is_in_bdx_mcpack(sth: str):
2023-09-24 02:14:04 +08:00
return isin(
sth.lower(),
2023-09-24 02:14:04 +08:00
{
1: ("bdx", "1", "币帝查", "币帝·艾克斯", "", ""),
0: ("mcpack", "0", "唉姆西·派克", "", ""),
2: ("MCSTRUCTURE", "2", "", "麦块结构", "MC结构", "", "我的世界结构"),
2023-09-24 02:14:04 +08:00
},
)
2023-04-29 16:58:22 +08:00
def is_in_player(sth: str, in_ok: tuple = (0, 1, 2)):
return isin(
sth.lower(),
dict(
[
(i, v)
for i, v in [
(
0,
("delay", "0", "延迟", "帝蕾"),
),
(
1,
("score", "1", "计分板", "积分", "积分板", "计分", "斯阔尔"),
),
(
2,
("repeater", "2", "中继器", "瑞皮特"),
),
]
if i in in_ok
]
),
)
2023-09-24 02:14:04 +08:00
output_file_format = format_ipt(
"输出文件类型 (mcpack/0|bdx/1|mcstructure/2)",
2023-04-29 16:58:22 +08:00
is_in_bdx_mcpack,
2023-09-24 02:14:04 +08:00
"输入内容有误,请重新输入。",
2023-04-29 16:58:22 +08:00
)[1]
if output_file_format == 1:
2023-09-24 02:14:04 +08:00
player_format = format_ipt(
"播放器类型 (延迟/0|计分板/1)",
lambda a: is_in_player(a, (0, 1)),
"输入内容有误,请重新输入。",
)[1]
else:
player_format = format_ipt(
"播放器类型 (延迟/0|计分板/1|中继器/2)",
is_in_player,
2023-09-24 02:14:04 +08:00
"输入内容有误,请重新输入。",
)[1]
old_exe_enabled = format_ipt(
"启用旧版代行指令 (否/0|是/1)",
bool_str,
"输入内容格式错误,应为 是/1/真/t/y 或 否/0/假/f/n",
2023-04-29 16:58:22 +08:00
)[1]
2023-02-02 02:11:28 +08:00
if os.path.exists("./demo_config.json"):
import json
prompts = json.load(open("./demo_config.json", "r", encoding="utf-8"))
2023-09-24 02:14:04 +08:00
2023-02-02 02:11:28 +08:00
prompts = prompts[:-1]
else:
prompts = []
# 提示语 检测函数 错误提示语
for args in [
(
"速度倍率 (0,+∞)",
2023-09-24 02:14:04 +08:00
float_str,
2023-02-02 02:11:28 +08:00
),
(
"最小音量 (0,1]",
2023-09-24 02:14:04 +08:00
float_str,
2023-02-02 02:11:28 +08:00
),
(
(
"自动生成进度条 (否/0|是/1)",
bool_str,
)
if output_file_format != 2
else ()
2023-02-02 02:11:28 +08:00
),
(
(
"计分板名称:",
str,
)
if player_format == 1
else (
"受播放玩家的选择器:",
str,
)
2023-02-02 02:11:28 +08:00
),
(
(
"自动重置计分板 (否/0|是/1)",
bool_str,
)
if player_format == 1
else ()
),
2023-02-02 02:11:28 +08:00
(
(
"BDX作者署名",
str,
)
if output_file_format == 1
else (
(
"结构延展方向 (x+|z+|x-|z-)",
lambda a: isin(
a,
{
"z+": ["z+", "Z+"],
"x+": ["X+", "x+"],
"z-": ["Z-", "z-"],
"x-": ["x-", "X-"],
},
),
)
if (player_format == 2 and output_file_format == 2)
else ()
)
),
2023-02-02 02:11:28 +08:00
(
(
"基础底部方块 (例concrete)",
str,
)
if (player_format == 2 and output_file_format == 2)
else (
(
"结构生成最大高度 (0,+∞)",
int_str,
)
if player_format == 0 or output_file_format != 0
else ()
)
),
2023-02-02 02:11:28 +08:00
]:
if args:
2023-09-24 02:14:04 +08:00
prompts.append(
format_ipt(
*args, err_note="输入内容格式错误,应符合 {}".format(args[1])
)[1]
2023-09-24 02:14:04 +08:00
)
if output_file_format != 2 and prompts[2]:
style = DEFAULT_PROGRESSBAR_STYLE
2023-09-24 02:14:04 +08:00
costom_pgb_enabled = format_ipt(
"自定义进度条样式 (否/0|是/1)",
bool_str,
"输入内容格式错误,应为 是/1/真/t/y 或 否/0/假/f/n",
2023-09-24 02:14:04 +08:00
)[1]
if costom_pgb_enabled:
base_style = ipt("基本样式组 (留空默认)")
if base_style:
style.set_base_style(base_style)
yet_part = ipt("未播放样式 (留空默认)")
if yet_part:
style.set_to_play_style(yet_part)
done_part = ipt("已播放样式 (留空默认)")
if done_part:
style.set_played_style(done_part)
else:
style = None
2023-09-24 02:14:04 +08:00
if output_file_format == 0:
if player_format == 1:
cvt_method = to_addon_pack_in_score
elif player_format == 0:
cvt_method = to_addon_pack_in_delay
elif player_format == 2:
cvt_method = to_addon_pack_in_repeater
elif output_file_format == 2:
if player_format == 1:
cvt_method = to_mcstructure_file_in_score
if player_format == 0:
cvt_method = to_mcstructure_file_in_delay
elif player_format == 2:
cvt_method = to_mcstructure_file_in_repeater
2023-09-10 19:26:19 +08:00
2023-02-02 02:11:28 +08:00
for singleMidi in midis:
2023-09-24 02:14:04 +08:00
prt("\n" f"正在处理 {singleMidi}")
cvt_mid = Musicreater.MidiConvert.from_midi_file(
midi_file_path=singleMidi,
play_speed=prompts[0],
old_exe_format=old_exe_enabled,
min_volume=prompts[1],
)
2023-09-24 02:14:04 +08:00
conversion_result = (
(cvt_method(cvt_mid, out_path, style, *prompts[3:])) # type: ignore
2023-09-24 02:14:04 +08:00
if output_file_format == 0
2023-02-02 02:11:28 +08:00
else (
(
to_BDX_file_in_score(cvt_mid, out_path, style, *prompts[3:])
if player_format == 1
else to_BDX_file_in_delay(cvt_mid, out_path, style, *prompts[3:])
)
if output_file_format == 1
else (cvt_method(cvt_mid, out_path, *prompts[2:]))
2023-09-24 02:14:04 +08:00
)
)
2023-02-02 02:11:28 +08:00
2023-05-27 20:05:34 +08:00
prt(
"\t{}{},播放刻数:{}{}".format(
"结构大小" if output_file_format == 2 else "指令总长",
conversion_result[0],
conversion_result[1],
(
",结构大小:{},末点坐标:{}".format(*conversion_result[2:])
if output_file_format == 1
else ""
),
)
2023-09-24 02:14:04 +08:00
)
2023-02-02 02:11:28 +08:00
2023-09-24 02:14:04 +08:00
exitSth = ipt("结束。换行以退出程序。")
2023-02-02 02:11:28 +08:00
if exitSth == "record":
import json
with open("./demo_config.json", "w", encoding="utf-8") as f:
json.dump(prompts, f)
elif exitSth == "delrec":
os.remove("./demo_config.json")