9 Commits

Author SHA1 Message Date
EillesWan
67ea8e1bf5 Merge branch 'master' of https://gitee.com/TriM-Organization/Linglun-Converter 2023-12-24 23:25:14 +08:00
EillesWan
5c2a56dd50 提供检查更新模块,非常人性化 2023-12-24 23:19:36 +08:00
Eilles Wan
0e36dd2418 临时修改测试用,一会更回去 update llc_win_wxPython.py.
Signed-off-by: Eilles Wan <w-yi_doctoryi@outlook.com>
2023-12-24 15:16:39 +00:00
Eilles Wan
50f1b08fc7 测试,等会改回去 update llc_win_wxPython.py.
Signed-off-by: Eilles Wan <w-yi_doctoryi@outlook.com>
2023-12-24 15:10:16 +00:00
Eilles Wan
68d93068bb update docs/download&start_EN/Windows.md.
ok呀,英文OK了

Signed-off-by: Eilles Wan <w-yi_doctoryi@outlook.com>
2023-11-10 11:49:15 +00:00
Eilles Wan
7d74b89649 update README.md.
Signed-off-by: Eilles Wan <w-yi_doctoryi@outlook.com>
2023-11-04 14:25:22 +00:00
EillesWan
7fad110d47 走你 2023-11-04 22:24:02 +08:00
EillesWan
36daf19407 删除冗余代码 2023-10-29 01:54:16 +08:00
EillesWan
a2c2afaede GUI版本更新 2023-10-29 00:52:56 +08:00
18 changed files with 2057 additions and 957 deletions

1
.gitignore vendored
View File

@@ -11,6 +11,7 @@ __pycache__/
*.bdx
*.json
/Musicreater
/TrimLog
/logs
# C extensions

View File

@@ -1,5 +1,11 @@
<h1 align="center">伶伦转换器</h1>
<p align="center">
<img width="128" height="128" src="https://gitee.com/TriM-Organization/Linglun-Converter/raw/master/resources/LLC_LOGO_OK_PLAIN_BANNER.png">
</img>
</p>
<h3 align="center">一个免费开源的《我的世界》数字音频转换器</h3>
<p align="center">
@@ -7,7 +13,7 @@
<p>
[![][Bilibili: 凌云金羿]](https://space.bilibili.com/397369002/)
[![][Bilibili: 金羿ELS]](https://space.bilibili.com/397369002/)
[![][Bilibili: 诸葛亮与八卦阵]](https://space.bilibili.com/604072474)
[![CodeStyle: black]](https://github.com/psf/black)
[![][python]](https://www.python.org/)
@@ -18,7 +24,7 @@
<!-- 简体中文 | [English](README_EN.md) -->
## 软件介绍🚀
## 介绍🚀
**伶伦** 是一款免费开源的 **《我的世界》** 数字音频工作站
@@ -40,20 +46,18 @@
[转换乐器对照参考表](https://gitee.com/TriM-Organization/Musicreater/blob/master/docs/%E8%BD%AC%E6%8D%A2%E4%B9%90%E5%99%A8%E5%AF%B9%E7%85%A7%E8%A1%A8.md)
## 致谢列表🙏
## 致谢🙏
> 感谢广大群友为此提供的测试和建议
>
> 若您对我们有所贡献但您的名字没有出现在此列表中,请联系我们!
> 感谢广大群友为此软件提供的测试和建议
## 联系我们📞
## 联系📞
QQ群 [861684859](https://jq.qq.com/?_wv=1027&k=hpeRxrYr)
电邮 [TriM-Organization@hotmail.com](mailto:TriM-Organization@hotmail.com)
[Bilibili: 凌云金羿]: https://img.shields.io/badge/Bilibili-%E5%87%8C%E4%BA%91%E9%87%91%E7%BE%BF-00A1E7?style=for-the-badge
[Bilibili: 金羿ELS]: https://img.shields.io/badge/Bilibili-%E9%87%91%E7%BE%BFELS-00A1E7?style=for-the-badge
[Bilibili: 诸葛亮与八卦阵]: https://img.shields.io/badge/Bilibili-%E8%AF%B8%E8%91%9B%E4%BA%AE%E4%B8%8E%E5%85%AB%E5%8D%A6%E9%98%B5-00A1E7?style=for-the-badge
[CodeStyle: black]: https://img.shields.io/badge/code%20style-black-121110.svg?style=for-the-badge
[python]: https://img.shields.io/badge/python-3.6-AB70FF?style=for-the-badge

View File

@@ -1,13 +1,38 @@
## 一、运行环境安装
## 运行打包文件(推荐)
### (一)安装 Python3.8+
### 一、下载打包好的应用程序
1. 在[代码仓库发布页](https://gitee.com/TriM-Organization/Linglun-Converter/releases)找到最新的下载。
<img src=https://foruda.gitee.com/images/1699105959996885442/be19644e_9911226.png>
请注意选择对您合适的构建版本
### 二、开始使用
2. 将下载得到的文件解压缩后,直接运行其中的 `llc_win_wxPython.exe`
<img src=https://foruda.gitee.com/images/1699106497816902973/48dc0942_9911226.png>
<img src=https://foruda.gitee.com/images/1699106542397488143/5638cd3e_9911226.png>
<img src=https://foruda.gitee.com/images/1699106557013748881/70f90fac_9911226.png>
3. 即可开始使用
<img src=https://foruda.gitee.com/images/1699106948861444130/30c156bc_9911226.png>
## 从代码运行(最新功能)
### 一、运行环境安装
#### (一)安装 Python3.8+
1. 首先需要下载Python的安装包最好是 *Python3.10*,因为作者就用的是这个版本
注意此程序现已不支持Python3.6。请更新到至少Python3.8但是我们对于Python3.8的支持也即将停止为了更好的兼容避免不必要的麻烦我们强烈建议您更新到Python3.10。(这意味着我们即将放弃对Windows7的支持)
> [下载64位Python安装包](https://www.python.org/ftp/python/3.8.10/python-3.8.10-amd64.exe)
> [下载32位Python安装包](https://www.python.org/ftp/python/3.8.10/python-3.8.10.exe)
> [下载64位Python3.10安装包](https://www.python.org/ftp/python/3.10.11/python-3.10.11-amd64.exe)
> [下载32位Python3.10安装包](https://www.python.org/ftp/python/3.10.11/python-3.10.11.exe)
2. 在安装时,最好需要勾选 `Add Python 3.X to PATH`,如下图所示:
@@ -19,12 +44,12 @@
<img src=https://foruda.gitee.com/images/1662736621235871190/2ac3d98f_9911226.png>
4. 安装结束之后可以在*终端*(命令行/PowerShell/Bash/etc)中输入:`python` 试试是否安装成功成功安装之后在终端中输入python会显示诸如如下图片的提示
4. 安装结束之后可以在*终端*(命令行/PowerShell/Bash/etc)中输入:`python -V` 试试是否安装成功成功安装之后在终端中输入python会显示诸如如下图片的提示
<img src=https://foruda.gitee.com/images/1659972669907359295/cmd.png>
<img src=https://foruda.gitee.com/images/1699107336707287940/1837e2f6_9911226.png>
### (二)安装依赖
#### (二)安装依赖
1. 请以管理员模式打开您的*终端*(命令行/PowerShell/Bash/etc)
@@ -42,7 +67,7 @@
<img src="https://foruda.gitee.com/images/1662737676719454287/f61a70f7_9911226.png">
## 二、本工具的下载与使用
### 二、本工具的下载与使用
0. 下载本代码库以及演示程序
@@ -66,13 +91,13 @@
<img src=https://foruda.gitee.com/images/1659974437388532868/输入.png>
<img src=https://foruda.gitee.com/images/1659974754378201859/输入c.png>
使用以下指令:
在打开的终端中使用以下指令:
```bash
python ./llc_cli.py
```
## 三、安装时错误的补充说明
### 三、安装时错误的补充说明
1. Microsoft Visual C++ Redistributable 环境出错
@@ -84,4 +109,4 @@
> [下载64位VCREDIST安装包](https://aka.ms/vs/17/release/vc_redist.x64.exe)
> [下载32位VCREDIST安装包](https://aka.ms/vs/17/release/vc_redist.x86.exe)
感谢群友*Mono*帮我们发现这个问题。
感谢群友 *Mono* 帮我们发现这个问题。

View File

@@ -1,8 +1,32 @@
## Install Runtime Environment
## Use a Pre-Built Release (Recommended)
### Install Python 3.6+
### 一、Download the Packed Program
1. First of all, you need to install the runtime environment of this library, *Python*. And a Installation Pack maybe the best choice:
1. Fne the newest release in [The Release Page](https://gitee.com/TriM-Organization/Linglun-Converter/releases)
<img src=https://foruda.gitee.com/images/1699105959996885442/be19644e_9911226.png>
Please consider a proper build release that satisfy your device.
### 二、Start Using
2. After downloading, extract the files and then double click `llc_win_wxPython.exe` to run.
<img src=https://foruda.gitee.com/images/1699106497816902973/48dc0942_9911226.png>
<img src=https://foruda.gitee.com/images/1699106542397488143/5638cd3e_9911226.png>
<img src=https://foruda.gitee.com/images/1699106557013748881/70f90fac_9911226.png>
3. Start to use now.
<img src=https://foruda.gitee.com/images/1699106948861444130/30c156bc_9911226.png>
## Run From Source Code (latest functions provided)
### Install Runtime Environment
#### Install Python 3.8+
1. First of all, you need to install the runtime environment of this library, *Python*. But please ensure that the version is above 3.8. Maybe a Installation Pack could be one of the choices:
> [Downloading Python 64-bit](https://www.python.org/ftp/python/3.8.10/python-3.8.10-amd64.exe)
> [Downloading Python 32-bit](https://www.python.org/ftp/python/3.8.10/python-3.8.10.exe)
@@ -22,7 +46,7 @@
<img src=https://foruda.gitee.com/images/1659972669907359295/cmd.png>
### Installing Requirements
#### Installing Requirements
1. It's better to open your *Terminal*(CMD/PowerShell/Bash/etc) under Administrator Mode.
@@ -42,14 +66,14 @@
<img src="https://foruda.gitee.com/images/1662737676719454287/f61a70f7_9911226.png">
## Downloading & Using of this tool
### Downloading & Using of this tool
1. Download This Package and Demo(s)
1. Download This Package
- If you use Git, you can clone this lib via the following commands:
```bash
git clone -b pkgver https://github.com/TriM-Organization/Musicreater.git
git clone https://github.com/TriM-Organization/Linglun-Converter.git
```
- If Git is not installed, you can download the zip package from the code page(from [GitHub](https://github.com/TriM-Organization/Musicreater.git) or [Gitee](https://gitee.com/TriM-Organization/Musicreater.git)). Or you are a Chinese fan having a QQ account, you can [Join the QQ Group 861684859](https://jq.qq.com/?_wv=1027&k=hpeRxrYr) and get it from our GroupFiles.
@@ -59,9 +83,9 @@
But it should be noticed that you're clear instructed to choose the branch "pkgver" first instead of downloading it directly from the "master" branch, the master branch is now under developing and has no practical use.
2. Start Using Demo(s)
2. Start Using
You can directly double click `magicDemo.py` to run the demo, or follow instructions below using Terminal APP to run it.
You can directly double click `llc_xxx(_xxx).py` to run the demo, or follow instructions below using Terminal APP to run it.
Open your terminal in the directory of this, taking CMD, for example, just enter the directory and enter `cmd` in the path box:
@@ -71,11 +95,11 @@
And enter the commands below:
```bash
python ./magicDemo.py
python ./llc_win_wxPython.py
```
## Addition for Error(s) Using or Installing
### Addition for Error(s) Using or Installing
1. Environment Error of Microsoft Visual C++ Redistributable

View File

@@ -28,7 +28,7 @@
- 速度倍率正小数其值不可为0用以表示游戏中播放此音乐的速度倍数。
- 进度条:是否启用自动生成进度条。
- 启用进度条:是否启用自动生成进度条。
- 若是,则可选是否自定义进度条

View File

View File

@@ -1,190 +0,0 @@
# -*- coding:utf-8 -*-
"""对于伶伦的语言支持兼语言文件编辑器"""
"""
Copyright © 2023 all the developers of LinglunStudio
"""
DEFAULTLANGUAGE = "ZH-CN"
LANGUAGELIST = {
# 第一个是语言的中文名称和地区
# 第二个是语言的英文名称和地区
# 第三个是语言的本地名称和地区
"ZH-CN": (
"简体中文 中国大陆",
"Simplified Chinese - China Mainland",
"简体中文 中国大陆",
),
"ZH-TW": (
"繁体中文 中国台湾",
"Traditional Chinese - Taiwan Province, China",
"正體中文,中国台灣省",
),
# 'ZH-HK': (
# "繁体中文 香港",
# "Traditional Chinese - Hong Kong SAR",
# "繁體中文,香港特別行政區",
# ),
# 'ZH-MO': (
# "繁体中文 澳门",
# "Traditional Chinese - Macau SAR",
# "繁體中文,澳門特別行政區",
# ),
"EN-GB": (
"英语 英国",
"British English - the United Kingdom",
"British English - the United Kingdom",
),
"ZH-ME": ("喵喵文 中国大陆", "Meow Catsnese - China Mainland" "喵喵喵~ 种花家~"),
}
languages = {
"ZH-CN": {
"MSCT": "音·创",
"ChooseLang": "选择语言",
"LangChd": "当前语言已经切换为",
"ZH-CN": "简体中文",
"ZH-TW": "繁体中文(台湾)",
"EN-GB": "英语(英国)",
"EN-US": "英语(美国)",
":": "",
",": "",
".": "",
"ChoosePath": "请输入MIDI路径或所在文件夹",
"ChooseFileFormat": "请输入输出格式[BDX(1) 或 MCPACK(0)]",
"EnterMethod": "请输入转换算法[{}~{}]",
"MethodRangeErr": "输入的转换算法应为 [{},{}](首尾皆含)之间的一个整数。",
"ChoosePlayer": "请选择播放方式[红石(2) 或 计分板(1) 或 延迟(0)]",
"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": "请按下回车键退出。",
}
}
class Lang:
def __init__(self, lang: str = "ZH-CN", debug: bool = False) -> None:
self.local = lang
self.debug = debug
def __load_language(self, language_file_name: str):
global logger
with open(language_file_name, "r", encoding="utf-8") as languageFile:
_text = {}
for line in languageFile:
if line.startswith("#"):
continue
line = line.split(" ", 1)
_text[line[0]] = line[1].replace("\n", "")
langkeys = _text.keys()
with open(
language_file_name.replace(language_file_name[-10:-5], "ZH-CN"),
"r",
encoding="utf-8",
) as defaultLangFile:
for line in defaultLangFile:
if line.startswith("#"):
continue
line = line.split(" ", 1)
if not line[0] in langkeys:
_text[line[0]] = line[1].replace("\n", "")
logger.warning(
f"丢失对于 {line[0]} 的本地化文本",
)
langkeys = _text.keys()
# print(_text)
return _text
# 这个函数是不被加载的
def passbt():
from utils.io import logger
try:
from utils.io import requests
except ImportError:
pass
def __loadLanguage(languageFilename: str):
with open(languageFilename, "r", encoding="utf-8") as languageFile:
_text = {}
for line in languageFile:
if line.startswith("#"):
continue
line = line.split(" ", 1)
_text[line[0]] = line[1].replace("\n", "")
langkeys = _text.keys()
with open(
languageFilename.replace(languageFilename[-10:-5], "ZH-CN"),
"r",
encoding="utf-8",
) as defaultLangFile:
for line in defaultLangFile:
if line.startswith("#"):
continue
line = line.split(" ", 1)
if not line[0] in langkeys:
_text[line[0]] = line[1].replace("\n", "")
logger.warning(
f"丢失对于 {line[0]} 的本地化文本",
)
langkeys = _text.keys()
# print(_text)
return _text
if DEFAULTLANGUAGE in LANGUAGELIST.keys():
_TEXT = __loadLanguage("./languages/" + DEFAULTLANGUAGE + ".lang")
else:
logger.error(f"无法打开当前本地化文本{DEFAULTLANGUAGE}")
raise KeyError(f"无法打开默认语言{DEFAULTLANGUAGE}")
def wordTranslate(singleWord: str, debug: bool = False):
try:
return (
requests.post(
"https://fanyi.baidu.com/sug", data={"kw": f"{singleWord}"}
)
.json()["data"][0]["v"]
.split("; ")[0]
)
except:
logger.warning(
f"无法翻译文本{singleWord}",
)
return None
def _(text: str, debug: bool = False):
try:
return _TEXT[text]
except:
if debug:
raise KeyError(f"无法找到本地化文本{text}")
else:
logger.warning(
f"无法找到本地化文本{text}",
)
return ""

View File

@@ -1,302 +0,0 @@
# -*- coding: utf-8 -*-
# 伶伦 开发交流群 861684859
"""
伶伦转换器
Linglun Converter
版权所有 © 2023 金羿 & 睿穆开发组
Copyright © 2023 EillesWan & TriM Org.
开源相关声明请见 ./License.md
Terms & Conditions: ./Lisense.md
"""
__version__ = "0.0.1"
import datetime
import os
import random
import sys
import Musicreater
from Musicreater.constants import DEFAULT_PROGRESSBAR_STYLE
from Musicreater.plugin import ConvertConfig
from Musicreater.plugin.addonpack import (
to_addon_pack_in_delay,
to_addon_pack_in_repeater,
to_addon_pack_in_score,
)
from Musicreater.plugin.bdxfile import to_BDX_file_in_delay, to_BDX_file_in_score
from utils.io import *
# from Musicreater.plugin.mcstructure import commands_to_structure, commands_to_redstone_delay_structure
MainConsole.print(
"[#121110 on #F0F2F4] ",
style="#121110 on #F0F2F4",
justify="center",
)
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.rule(title="[bold #AB70FF]欢迎使用伶伦独立转换器", characters="=", style="#26E2FF")
# MainConsole.rule(title="[bold #AB70FF]Welcome to Linglun Converter", characters="-")
MainConsole.rule(
title="[#AB70FF]版本{} | 音·创内核版本{}".format(__version__, Musicreater.__version__),
characters="-",
style="#26E2FF",
)
nowYang = datetime.datetime.now()
if nowYang.month == 8 and nowYang.day == 6:
# 诸葛八卦生日
MainConsole.print(
"[#7DB5F0 on #121110]今天可不是催更的日子!\n诸葛亮与八卦阵{}岁生日快乐!".format(nowYang.year - 2008),
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",
)
else:
# 显示箴言部分
MainConsole.print(
"[#121110 on #F0F2F4]{}".format(random.choice(myWords)),
style="#121110 on #F0F2F4",
justify="center",
)
# prt(f"{_('LangChd')}{_(':')}{_(currentLang)}")
def format_ipt(
notice: str,
fun,
err_note: str = "输入内容有误,请重新输入。",
*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:
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
# 获取输出地址
while True:
out_path = ipt(f"文件生成输出地址:")
try:
if not os.path.exists(out_path):
prt("该地址不存在,或无法访问该地址,请重新输入。")
continue
except PermissionError:
prt("无法访问该地址,请检查是否给予程序相关文件的访问权限。")
continue
break
# 选择输出格式
def is_in_bdx_mcpack(sth: str):
return isin(sth, {1: ("bdx", "1", "币帝查", "币帝·艾克斯"), 0: ("mcpack", "0", "唉姆西·派克")})
def is_in_player(sth: str):
return isin(
sth,
{
0: ("delay", "0", "延迟", "帝蕾"),
1: ("score", "1", "计分板", "积分", "积分板", "计分", "斯阔尔"),
2: ("repeater", "2", "中继器", "瑞皮特"),
},
)
output_file_format = format_ipt(
"输出文件类型 (mcpack/0|bdx/1)",
is_in_bdx_mcpack,
"输入内容有误,请重新输入。",
)[1]
if output_file_format == 0:
player_format = format_ipt(
"播放器类型 (延迟/0|计分板/1|中继器/2)",
is_in_player,
"输入内容有误,请重新输入。",
)[1]
else:
player_format = format_ipt(
"播放器类型 (延迟/0|计分板/1)",
is_in_player,
"输入内容有误,请重新输入。",
)[1]
old_exe_enabled = not format_ipt(
"启用新版代执行指令 (否/0|是/1)", bool_str, "输入内容格式错误,应为 是/1/真/t/y 或 否/0/假/f/n"
)[1]
if os.path.exists("./demo_config.json"):
import json
prompts = json.load(open("./demo_config.json", "r", encoding="utf-8"))
prompts = prompts[:-1]
else:
prompts = []
# 提示语 检测函数 错误提示语
for args in [
(
"音量大小 (0,1]",
float_str,
),
(
"速度倍率 (0,+∞)",
float_str,
),
(
"自动生成进度条 (否/0|是/1)",
bool_str,
),
(
"计分板名称:",
str,
)
if player_format == 1
else (
"受播放玩家的选择器:",
str,
),
(
"自动重置计分板 (否/0|是/1)",
bool_str,
)
if player_format == 1
else (),
(
"BDX作者署名",
str,
)
if output_file_format == 1
else (),
(
"结构生成最大高度 (0,+∞)",
int,
)
if player_format == 0
else (),
]:
if args:
prompts.append(
format_ipt(*args, err_note="输入内容格式错误,应符合 {}".format(args[1]))[1]
)
if prompts[2]:
costom_pgb_enabled = format_ipt(
"自定义进度条样式 (否/0|是/1)", bool_str, "输入内容格式错误,应为 是/1/真/t/y 或 否/0/假/f/n"
)[1]
if costom_pgb_enabled:
style = ipt("基本样式组 (回车默认)")
if not style:
style = DEFAULT_PROGRESSBAR_STYLE[0]
yet_part = ipt("未播放样式 (回车默认)")
if not yet_part:
yet_part = DEFAULT_PROGRESSBAR_STYLE[1][1]
done_part = ipt("已播放样式 (回车默认)")
if not done_part:
done_part = DEFAULT_PROGRESSBAR_STYLE[1][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
for singleMidi in midis:
prt("\n" f"正在处理 {singleMidi}")
cvt_mid = Musicreater.MidiConvert.from_midi_file(
singleMidi, old_exe_format=old_exe_enabled
)
cvt_cfg = ConvertConfig(out_path, *prompts[:2], progressbar=((style, (done_part, yet_part)) if costom_pgb_enabled else True) if prompts[2] else False) # type: ignore
conversion_result = (
(cvt_method(cvt_mid, cvt_cfg, *prompts[3:])) # type: ignore
if output_file_format == 0
else (
to_BDX_file_in_score(cvt_mid, cvt_cfg, *prompts[3:])
if player_format == 1
else to_BDX_file_in_delay(cvt_mid, cvt_cfg, *prompts[3:])
)
)
prt(
f" 指令总长:{conversion_result[0]},播放刻数:{conversion_result[1]}{f''',结构大小:{conversion_result[2]},末点坐标:{conversion_result[3]}''' if output_file_format == 1 else ''}" # type: ignore
)
exitSth = ipt("结束。换行以退出程序。")
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")

View File

@@ -4,8 +4,8 @@
"""
伶伦转换器
Linglun Converter
伶伦转换器 命令行
Linglun Converter CLI
版权所有 © 2023 金羿 & 睿穆开发组
Copyright © 2023 EillesWan & TriM Org.

View File

@@ -1,19 +0,0 @@
# -*- coding: utf-8 -*-
# 伶伦 开发交流群 861684859
"""
伶伦转换器安装器(视窗操作系统)
Linglun Converter Installer for Windows
版权所有 © 2023 金羿 & 睿穆开发组
Copyright © 2023 EillesWan & TriM Org.
开源相关声明请见 ../License.md
Terms & Conditions: ../Lisense.md
"""
# 下面为正文

View File

@@ -1,110 +0,0 @@
# -*- coding: utf-8 -*-
# 伶伦 开发交流群 861684859
"""
伶伦转换器
Linglun Converter
版权所有 © 2023 金羿 & 睿穆开发组
Copyright © 2023 EillesWan & TriM Org.
开源相关声明请见 ./License.md
Terms & Conditions: ./Lisense.md
"""
__version__ = "0.0.1"
import datetime
import os
import random
import sys
import Musicreater
from Musicreater.plugin import ConvertConfig
from Musicreater.plugin.bdxfile import to_BDX_file_in_delay, to_BDX_file_in_score
from Musicreater.plugin.addonpack import (
to_addon_pack_in_delay,
to_addon_pack_in_repeater,
to_addon_pack_in_score,
)
from Musicreater.constants import DEFAULT_PROGRESSBAR_STYLE
from utils.io import *
osc.project_name = "伶伦转换器"
osc.version = __version__
def __main__():
MainConsole.print(
"[#121110 on #F0F2F4] ",
style="#121110 on #F0F2F4",
justify="center",
)
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.rule(title="[bold #AB70FF]欢迎使用伶伦独立转换器", characters="=", style="#26E2FF")
MainConsole.rule(title="[bold #AB70FF]Welcome to Linglun Converter", characters="-")
nowYang = datetime.datetime.now()
if nowYang.month == 8 and nowYang.day == 6:
# 诸葛八卦生日
MainConsole.print(
"[#7DB5F0 on #121110]今天可不是催更的日子!\n诸葛亮与八卦阵{}岁生日快乐!".format(
nowYang.year - 2008
),
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",
)
else:
# 显示箴言部分
MainConsole.print(
"[#121110 on #F0F2F4]{}".format(random.choice(myWords)),
style="#121110 on #F0F2F4",
justify="center",
)
prt("伶伦转换器简易版 正在启动……")
prt("更新执行位置...")
if sys.platform == "win32":
try:
os.chdir(
__file__[: len(__file__) - __file__[len(__file__) :: -1].index("\\")]
)
logger.info("Win32 更新执行位置,当前文件位置 {}".format(__file__))
except FileNotFoundError:
pass
else:
try:
os.chdir(
__file__[: len(__file__) - __file__[len(__file__) :: -1].index("/")]
)
except FileNotFoundError:
pass
log("其他平台:{} 更新执行位置,当前文件位置 {}".format(sys.platform, __file__))
prt("完成!")
prt("载入功能……")

View File

@@ -1,5 +1,17 @@
# -*- coding: utf-8 -*-
"""
伶伦转换器 WXGUI
Linglun Converter WxPython GUI
版权所有 © 2023 金羿 & 睿穆开发组
Copyright © 2023 EillesWan & TriM Org.
开源相关声明请见 ./License.md
Terms & Conditions: ./Lisense.md
"""
# 导入所需库
import datetime
import os
@@ -19,45 +31,35 @@ from Musicreater.plugin.addonpack import (
)
from Musicreater.plugin.bdxfile import to_BDX_file_in_delay, to_BDX_file_in_score
import TrimLog
import wx
from TrimLog import Console, object_constants
from utils.io import prt
from utils.io import myWords, osc, logger, object_constants, TrimLog, is_logging
from utils.update_check import check_update
is_logging: bool = True
osc = object_constants.ObjectStateConstant()
logger = TrimLog.Logger(
is_logging=is_logging,
printing=not osc.isRelease,
in_suffix=".llc",
)
WHITE = (242, 244, 246) # F2F4F6
BLACK = (18, 17, 16) # 121110
try:
myWords = (
urllib.request.urlopen(
"https://gitee.com/TriM-Organization/LinglunStudio/raw/master/resources/myWords.txt"
)
.read()
.decode("utf-8")
.strip("\n")
.split("\n")
)
except (ConnectionError, urllib.error.HTTPError) as E:
logger.warning(f"读取言·论信息发生 互联网连接 错误:\n{E}")
myWords = ["以梦想为驱使 创造属于自己的未来"]
# noinspection PyBroadException
except BaseException as E:
logger.warning(f"读取言·论信息发生 未知 错误:\n{E}")
myWords = ["灵光焕发 深艺献心"]
__appname__ = "伶伦转换器"
__version__ = "WXGUI 0.0.1"
__version__ = "WXGUI 0.0.3"
__zhver__ = "WX图形界面 预代预版第三次修订"
osc.project_name = __appname__
osc.version = __version__
# osc = object_constants.ObjectStateConstant(
# logging_project_name=__appname__,
# logging_project_version=__version__,
# logging_exit_exec=lambda x: None,
# )
yanlun_length = len(myWords)
logger.info("加载窗口布局……")
# 创建应用程序类
class LinglunConverterApp(wx.App):
@@ -78,9 +80,9 @@ class LingLunMainFrame(wx.Frame):
self,
parent,
id=wx.ID_ANY,
title="{} {}".format(__appname__, __version__),
title="{} {}".format(__appname__, __zhver__),
pos=wx.DefaultPosition,
size=wx.Size(651, 759),
size=wx.Size(660, 780),
style=wx.DEFAULT_FRAME_STYLE | wx.TAB_TRAVERSAL,
)
@@ -116,10 +118,11 @@ class LingLunMainFrame(wx.Frame):
wx.StaticBox(self, wx.ID_ANY, "言·论"), wx.VERTICAL
)
self.yanlun_now = random.randrange(0, yanlun_length)
self.m_LinglunWords_staticText1 = wx.StaticText(
s_yanLunbarSizer.GetStaticBox(),
wx.ID_ANY,
random.choice(myWords) + "\r",
myWords[self.yanlun_now] + "\r",
wx.DefaultPosition,
wx.DefaultSize,
wx.ALIGN_CENTER_HORIZONTAL | wx.ST_ELLIPSIZE_MIDDLE | wx.ST_NO_AUTORESIZE,
@@ -155,33 +158,36 @@ class LingLunMainFrame(wx.Frame):
s_midiChooseSizer = wx.BoxSizer(wx.HORIZONTAL)
self.m_ChooseMidiTips_staticText3 = wx.StaticText(
self, wx.ID_ANY, "选择MIDI文件", wx.DefaultPosition, wx.DefaultSize, 0
self, wx.ID_ANY, "选择MIDI文件\n(双击移除)", wx.DefaultPosition, wx.DefaultSize, 0
)
self.m_ChooseMidiTips_staticText3.Wrap(-1)
s_midiChooseSizer.Add(self.m_ChooseMidiTips_staticText3, 0, wx.ALL, 5)
s_midiChooseSizer.Add(
self.m_ChooseMidiTips_staticText3, 0, wx.ALL | wx.ALIGN_CENTER_VERTICAL, 5
)
ss_MidiChooserSizer_bSizer9 = wx.BoxSizer(wx.VERTICAL)
self.m_ChooseMIDI_filePicker1 = wx.FilePickerCtrl(
self.midiFilesList = set()
self.m_midiFilesList_listBox2 = wx.ListBox(
self,
wx.ID_ANY,
"./",
"选择MIDI文件",
"MIDI 序列 (*.mid;*.midi)|*.mid;*.midi",
wx.DefaultPosition,
wx.DefaultSize,
wx.FLP_DEFAULT_STYLE
| wx.FLP_FILE_MUST_EXIST
| wx.FLP_OPEN
| wx.FLP_USE_TEXTCTRL,
)
ss_MidiChooserSizer_bSizer9.Add(
self.m_ChooseMIDI_filePicker1, 0, wx.ALL | wx.EXPAND, 5
[],
wx.LB_HSCROLL | wx.LB_SORT,
)
ss_MidiChooserSizer_bSizer9.Add(self.m_midiFilesList_listBox2, 0, wx.EXPAND, 0)
s_midiChooseSizer.Add(ss_MidiChooserSizer_bSizer9, 1, wx.EXPAND, 5)
self.m_midiBroseButton_button21 = wx.Button(
self, wx.ID_ANY, "打开…", wx.DefaultPosition, wx.DefaultSize, 0
)
s_midiChooseSizer.Add(
self.m_midiBroseButton_button21, 0, wx.ALL | wx.ALIGN_CENTER_VERTICAL, 5
)
m_mainBoxSizer.Add(s_midiChooseSizer, 1, wx.EXPAND, 5)
s_formatChooseSizer = wx.BoxSizer(wx.HORIZONTAL)
@@ -347,10 +353,10 @@ class LingLunMainFrame(wx.Frame):
self.m_BasicProgressBarStyleEntering_textCtrl4 = wx.TextCtrl(
ssss_basicProgressStylePattle_sbSizer9.GetStaticBox(),
wx.ID_ANY,
r"%%N [ %%s/%^s %%% __________ %%t|%^t ]",
DEFAULT_PROGRESSBAR_STYLE[0],
wx.DefaultPosition,
wx.DefaultSize,
wx.TE_LEFT | wx.TE_NO_VSCROLL | wx.TE_RICH,
wx.TE_LEFT | wx.TE_NO_VSCROLL,
)
ssss_basicProgressStylePattle_sbSizer9.Add(
self.m_BasicProgressBarStyleEntering_textCtrl4, 0, wx.ALL | wx.EXPAND, 5
@@ -370,7 +376,7 @@ class LingLunMainFrame(wx.Frame):
self.m_unplayedProgressbarStyleEntering_textCtrl5 = wx.TextCtrl(
ssss_UnplayedPartProgressbarPattle_sbSizer10.GetStaticBox(),
wx.ID_ANY,
"§7=§r",
DEFAULT_PROGRESSBAR_STYLE[1][1],
wx.DefaultPosition,
wx.DefaultSize,
wx.TE_LEFT | wx.TE_NO_VSCROLL,
@@ -393,7 +399,7 @@ class LingLunMainFrame(wx.Frame):
self.m_playedProgressbarStyleEntering_textCtrl5 = wx.TextCtrl(
ssss_PlayedPartProgressbarPattle_sbSizer11.GetStaticBox(),
wx.ID_ANY,
"§e=§r",
DEFAULT_PROGRESSBAR_STYLE[1][0],
wx.DefaultPosition,
wx.DefaultSize,
wx.TE_LEFT | wx.TE_NO_VSCROLL,
@@ -610,7 +616,7 @@ class LingLunMainFrame(wx.Frame):
self.m_button2 = wx.Button(
s_StartSizer_sbSizer18.GetStaticBox(),
wx.ID_ANY,
"转换,启动!",
"开始转换",
wx.DefaultPosition,
wx.DefaultSize,
0,
@@ -636,7 +642,13 @@ class LingLunMainFrame(wx.Frame):
self.m_LinglunWords_staticText1.Bind(wx.EVT_LEFT_DCLICK, self.onYanlunDClicked)
self.m_LinglunWords_staticText1.Bind(wx.EVT_MOUSEWHEEL, self.onYanlunWheeled)
self.m_ChooseMidiTips_staticText3.Bind(wx.EVT_LEFT_DCLICK, self.MidiEasterEgg)
self.m_ChooseMIDI_filePicker1.Bind(wx.EVT_FILEPICKER_CHANGED, self.onFileChosen)
self.m_midiFilesList_listBox2.Bind(wx.EVT_LISTBOX, self.onFileListUpdated)
self.m_midiFilesList_listBox2.Bind(
wx.EVT_LISTBOX_DCLICK, self.onFileDoubleClicked
)
self.m_midiBroseButton_button21.Bind(wx.EVT_BUTTON, self.openFile)
self.m_outformatChoice_choice1.Bind(wx.EVT_CHOICE, self.onOutputFormatChosen)
self.m_playerChoice_choice2.Bind(wx.EVT_CHOICE, self.onPlayerChosen)
self.m_volumn_slider.Bind(wx.EVT_SCROLL, self.onVolumeScrolling)
@@ -698,20 +710,52 @@ class LingLunMainFrame(wx.Frame):
self.Destroy()
def onYanlunDClicked(self, event):
self.m_LinglunWords_staticText1.SetLabelText(random.choice(myWords) + "\r")
self.yanlun_now = random.randrange(0, yanlun_length)
self.m_LinglunWords_staticText1.SetLabelText(myWords[self.yanlun_now] + "\r")
def onYanlunWheeled(self, event):
self.m_LinglunWords_staticText1.SetLabelText(random.choice(myWords) + "\r")
if event.GetWheelRotation() < 0:
self.yanlun_now += 1
else:
self.yanlun_now -= 1
self.yanlun_now += (
-yanlun_length
if self.yanlun_now >= yanlun_length
else (yanlun_length if self.yanlun_now < 0 else 0)
)
self.m_LinglunWords_staticText1.SetLabelText(myWords[self.yanlun_now] + "\r")
def MidiEasterEgg(self, event):
self.m_ChooseMIDI_filePicker1.GetTextCtrl().SetValue("诸葛亮与八卦阵-山水千年.mid")
self.Update()
if "诸葛亮与八卦阵-山水千年" not in self.midiFilesList:
self.midiFilesList.add("诸葛亮与八卦阵-山水千年")
self.m_midiFilesList_listBox2.Append("诸葛亮与八卦阵-山水千年")
def onFileChosen(self, event):
def onFileListUpdated(self, event):
# prt(self.m_ChooseMIDI_filePicker1.GetTextCtrl().Value)
# self.midi_cvt = Musicreater.MidiConvert.from_midi_file(self.m_ChooseMIDI_filePicker1.GetLabel(),self.m_oldExeFormatChecker_checkBox3.GetValue())
event.Skip()
def onFileDoubleClicked(self, event):
# print(self.m_midiFilesList_listBox2.Selection)
self.midiFilesList.remove(self.m_midiFilesList_listBox2.GetStringSelection())
# print(self.midiFilesList)
self.m_midiFilesList_listBox2.Delete(self.m_midiFilesList_listBox2.Selection)
def openFile(self, event):
fileDialog = wx.FileDialog(
None,
message="选择MIDI文件",
defaultDir="./",
wildcard="MIDI 序列 (*.mid;*.midi)|*.mid;*.midi",
style=wx.FD_OPEN | wx.FD_MULTIPLE | wx.FD_FILE_MUST_EXIST,
)
dialogResult = fileDialog.ShowModal()
if dialogResult == wx.ID_OK:
self.midiFilesList.update(fileDialog.GetPaths())
self.m_midiFilesList_listBox2.Set(list(self.midiFilesList))
fileDialog.Destroy()
def onOutputFormatChosen(self, event):
# 0: 附加包
# 1: BDX
@@ -815,14 +859,19 @@ class LingLunMainFrame(wx.Frame):
event.Skip()
def onStartButtonPressed(self, event):
if self.m_ChooseMIDI_filePicker1.GetTextCtrl().Value != "诸葛亮与八卦阵-山水千年.mid":
mid_cvt = Musicreater.MidiConvert.from_midi_file(
self.m_ChooseMIDI_filePicker1.GetTextCtrl().Value,
self.m_oldExeFormatChecker_checkBox3.GetValue(),
)
for file_name in self.m_midiFilesList_listBox2.GetStrings():
if file_name == "诸葛亮与八卦阵-山水千年":
mid_cvt = Musicreater.MidiConvert(
None, "山水千年", self.m_oldExeFormatChecker_checkBox3.GetValue()
)
else:
mid_cvt = Musicreater.MidiConvert.from_midi_file(
file_name,
self.m_oldExeFormatChecker_checkBox3.GetValue(),
)
cvt_cfg = ConvertConfig(
os.path.split(self.m_ChooseMIDI_filePicker1.GetTextCtrl().Value)[0],
os.path.split(file_name)[0],
self.m_volumn_spinCtrlDouble1.GetValue() / 100,
self.m_speed_spinCtrlDouble.GetValue(),
progressbar=(
@@ -896,233 +945,22 @@ class LingLunMainFrame(wx.Frame):
return
wx.MessageDialog(
None,
"完成!\n指令数量:{}\n延迟总长:{}\n结构大小:{}\n终点坐标:{}".format(
cmd_num, total_delay, size, final_pos
"{}\n\n完成!\n指令数量:{}\n延迟总长:{}\n结构大小:{}\n终点坐标:{}".format(
file_name, cmd_num, total_delay, size, final_pos
),
"转换成功",
wx.YES_DEFAULT | wx.ICON_INFORMATION,
).ShowModal()
# # 创建主窗口类
# class LinglunConverterFrame(wx.Frame):
# def __init__(self, *args, **kwargs):
# super(LinglunConverterFrame, self).__init__(*args, **kwargs)
# # 设置窗口属性
# self.SetSize((750, 300))
# self.Center()
# self.panel = wx.Panel(self)
# self.main_sizer = wx.BoxSizer(wx.VERTICAL)
# self.yanlun_sizer = wx.BoxSizer(wx.HORIZONTAL)
# now_word = random.choice(myWords).split("\t——")
# self.yanlun_label_a = wx.StaticText(
# self.panel, label=now_word[0], style=wx.ALIGN_LEFT
# )
# self.yanlun_label_a.SetForegroundColour(WHITE)
# self.yanlun_label_a.SetBackgroundColour(BLACK)
# self.yanlun_sizer.Add(self.yanlun_label_a, flag=wx.ALIGN_TOP)
# if len(now_word) > 1:
# self.yanlun_label_b = wx.StaticText(
# self.panel, label=now_word[1], style=wx.ALIGN_RIGHT
# )
# self.yanlun_label_b.SetForegroundColour(WHITE)
# self.yanlun_label_b.SetBackgroundColour(BLACK)
# self.yanlun_sizer.Add(self.yanlun_label_b, flag=wx.ALIGN_TOP)
# self.main_sizer.Add(self.yanlun_sizer, flag=wx.EXPAND)
# self.file_picker_frame = wx.StaticBoxSizer()
# # 文件选择控件
# self.file_picker = wx.FilePickerCtrl(
# self.panel,
# message="选择MIDI文件",
# wildcard="MIDI files (*.mid;*.midi)|*.mid;*.midi",
# style=wx.FLP_USE_TEXTCTRL,
# )
# self.main_sizer.Add(self.file_picker, flag=wx.EXPAND)
# # 输出结果类型下拉选择控件
# self.output_type_dropdown = wx.Choice(
# self.panel, choices=["MCPACK", "BDX", "MCSTRUCTURE"]
# )
# self.main_sizer.Add(wx.StaticText(self.panel, label="输出结果类型:"))
# self.main_sizer.Add(self.output_type_dropdown, flag=wx.EXPAND)
# # 播放器类型下拉选择控件
# self.player_type_dropdown = wx.Choice(self.panel, choices=["计分板", "延迟"])
# self.main_sizer.Add(wx.StaticText(self.panel, label="播放器类型:"))
# self.main_sizer.Add(self.player_type_dropdown, flag=wx.EXPAND)
# # 音量大小滑动条与数字输入框
# self.volume_slider = wx.Slider(
# self.panel, value=100, minValue=1, maxValue=100, style=wx.SL_HORIZONTAL
# )
# self.volume_text = wx.TextCtrl(self.panel, value="1.0")
# self.main_sizer.Add(wx.StaticText(self.panel, label="音量大小:"))
# self.main_sizer.Add(self.volume_slider, flag=wx.EXPAND)
# self.main_sizer.Add(self.volume_text, flag=wx.EXPAND)
# # 速度倍率滑动条与数字输入框
# self.speed_slider = wx.Slider(
# self.panel, value=100, minValue=1, maxValue=100, style=wx.SL_HORIZONTAL
# )
# self.speed_text = wx.TextCtrl(self.panel, value="1.0")
# self.main_sizer.Add(wx.StaticText(self.panel, label="速度倍率:"))
# self.main_sizer.Add(self.speed_slider, flag=wx.EXPAND)
# self.main_sizer.Add(self.speed_text, flag=wx.EXPAND)
# # 自动生成进度条勾选框
# self.progressbar_checkbox = wx.CheckBox(self.panel, label="是否自动生成进度条")
# self.main_sizer.Add(self.progressbar_checkbox, flag=wx.EXPAND)
# # 进度条相关控件
# self.whole_style_text = wx.TextCtrl(
# self.panel, value=" %%N [ %%s/%^s %%% __________ %%t|%^t]"
# )
# self.non_played_text = wx.TextCtrl(self.panel, value="§7=§r")
# self.has_played_text = wx.TextCtrl(self.panel, value="§e=§r")
# self.main_sizer.Add(wx.StaticText(self.panel, label="整体样式:"))
# self.main_sizer.Add(self.whole_style_text, flag=wx.EXPAND)
# self.main_sizer.Add(wx.StaticText(self.panel, label="动条(未播放):"))
# self.main_sizer.Add(self.non_played_text, flag=wx.EXPAND)
# self.main_sizer.Add(wx.StaticText(self.panel, label="动条(已播放):"))
# self.main_sizer.Add(self.has_played_text, flag=wx.EXPAND)
# # 计分板相关控件
# self.scoreboard_name_text = wx.TextCtrl(self.panel, value="mscply")
# self.auto_reset_checkbox = wx.CheckBox(self.panel, label="是否结束后重置计分板")
# # 延迟相关控件
# self.selecter_text = wx.TextCtrl(self.panel, value="@a")
# self.max_struct_height_slider = wx.Slider(
# self.panel, value=63, minValue=4, maxValue=256, style=wx.SL_HORIZONTAL
# )
# self.max_struct_height_text = wx.TextCtrl(self.panel, value="63")
# # BDX相关控件
# self.author_name_text = wx.TextCtrl(self.panel, value="Unfamous")
# # 导出结果按钮
# self.export_button = wx.Button(self.panel, label="导出结果")
# # 添加控件到主sizer中
# self.main_sizer.Add(self.scoreboard_name_text, flag=wx.EXPAND)
# self.main_sizer.Add(self.auto_reset_checkbox, flag=wx.EXPAND)
# self.main_sizer.Add(self.selecter_text, flag=wx.EXPAND)
# self.main_sizer.Add(self.max_struct_height_slider, flag=wx.EXPAND)
# self.main_sizer.Add(self.max_struct_height_text, flag=wx.EXPAND)
# self.main_sizer.Add(self.author_name_text, flag=wx.EXPAND)
# self.main_sizer.Add(self.export_button, flag=wx.EXPAND)
# self.panel.SetSizer(self.main_sizer)
# self.panel.Layout()
# # 绑定事件处理程序
# self.file_picker.Bind(wx.EVT_FILEPICKER_CHANGED, self.on_file_selected)
# self.output_type_dropdown.Bind(wx.EVT_CHOICE, self.on_output_type_selected)
# self.player_type_dropdown.Bind(wx.EVT_CHOICE, self.on_player_type_selected)
# self.volume_slider.Bind(wx.EVT_SCROLL_CHANGED, self.on_volume_slider_changed)
# self.speed_slider.Bind(wx.EVT_SCROLL_CHANGED, self.on_speed_slider_changed)
# self.progressbar_checkbox.Bind(
# wx.EVT_CHECKBOX, self.on_progressbar_checkbox_changed
# )
# self.max_struct_height_slider.Bind(
# wx.EVT_SCROLL_CHANGED, self.on_max_struct_height_slider_changed
# )
# self.export_button.Bind(wx.EVT_BUTTON, self.on_export_button_clicked)
# self.HideUnnecessaryControls()
# self.Layout()
# def HideUnnecessaryControls(self):
# # 隐藏所有不需要显示的控件
# self.scoreboard_name_text.Hide()
# self.auto_reset_checkbox.Hide()
# self.selecter_text.Hide()
# self.max_struct_height_slider.Hide()
# self.max_struct_height_text.Hide()
# self.author_name_text.Hide()
# self.whole_style_text.Hide()
# self.non_played_text.Hide()
# self.has_played_text.Hide()
# self.Layout()
# def on_file_selected(self, event):
# file_to_convert = self.file_picker.GetPath()
# # 处理选择的文件
# def on_output_type_selected(self, event):
# output_type = self.output_type_dropdown.GetStringSelection()
# # 处理选择的输出结果类型
# def on_player_type_selected(self, event):
# player_type = self.player_type_dropdown.GetStringSelection()
# if player_type == "计分板":
# # 显示计分板相关控件
# self.scoreboard_name_text.Show()
# self.auto_reset_checkbox.Show()
# # 隐藏延迟相关控件
# self.selecter_text.Hide()
# self.max_struct_height_slider.Hide()
# self.max_struct_height_text.Hide()
# elif player_type == "延迟":
# # 显示延迟相关控件
# self.selecter_text.Show()
# self.max_struct_height_slider.Show()
# self.max_struct_height_text.Show()
# # 隐藏计分板相关控件
# self.scoreboard_name_text.Hide()
# self.auto_reset_checkbox.Hide()
# self.Layout()
# def on_volume_slider_changed(self, event):
# volume = self.volume_slider.GetValue()
# self.volume_text.SetValue(str(volume / 100)) # 更新音量大小输入框的值
# def on_speed_slider_changed(self, event):
# speed = self.speed_slider.GetValue()
# self.speed_text.SetValue(str(speed / 100)) # 更新速度倍率输入框的值
# def on_progressbar_checkbox_changed(self, event):
# is_progressbar_selected = self.progressbar_checkbox.GetValue()
# if is_progressbar_selected:
# # 显示进度条相关控件
# self.whole_style_text.Show()
# self.non_played_text.Show()
# self.has_played_text.Show()
# else:
# # 隐藏进度条相关控件
# self.whole_style_text.Hide()
# self.non_played_text.Hide()
# self.has_played_text.Hide()
# self.Layout()
# def on_max_struct_height_slider_changed(self, event):
# max_struct_height = self.max_struct_height_slider.GetValue()
# self.max_struct_height_text.SetValue(str(max_struct_height)) # 更新结构最大高度输入框的值
# def on_export_button_clicked(self, event):
# output_path = wx.DirSelector(message="选择一个文件夹") # 选择文件夹并保存至output_path变量
# # 导出结果的操作
# class TextCtrlRedirector:
# def __init__(self, text_ctrl):
# self.text_ctrl = text_ctrl
# def write(self, text):
# wx.CallAfter(self.text_ctrl.WriteText, text)
# def flush(self):
# pass
logger.info("执行应用。")
# 启动应用程序
if __name__ == "__main__":
app = LinglunConverterApp()
check_update(__appname__,"https://gitee.com/TriM-Organization/Linglun-Converter/raw/master/llc_win_wxPython.py",__version__,lambda text:wx.MessageDialog(None,text,"软件更新",wx.ICON_INFORMATION | wx.YES_DEFAULT,).ShowModal(),logger,__zhver__)
app.MainLoop()
# input("按下回车退出……")

Binary file not shown.

After

Width:  |  Height:  |  Size: 51 KiB

File diff suppressed because one or more lines are too long

Binary file not shown.

After

Width:  |  Height:  |  Size: 535 KiB

View File

@@ -1,6 +1,20 @@
# -*- coding: utf-8 -*-
"""
伶伦转换器 命令行组件
Linglun Converter Command Line IO Component
版权所有 © 2023 金羿 & 睿穆开发组
Copyright © 2023 EillesWan & TriM Org.
开源相关声明请见 ./License.md
Terms & Conditions: ./Lisense.md
"""
import urllib.error
import urllib.request
from typing import Any, Callable, Dict, List, Literal, Optional, Set, TextIO, Tuple
from typing import Any, Callable, Dict, List, Literal, Optional, Set, TextIO, Tuple, Iterable, Sequence
import TrimLog
from TrimLog import Console, object_constants
@@ -11,7 +25,7 @@ MainConsole = Console()
osc = object_constants.ObjectStateConstant()
logger = TrimLog.Logger(
is_logging=is_logging,
printing=not osc.isRelease,
# printing=not osc.is_release,
in_suffix=".llc",
)

118
utils/update_check.py Normal file
View File

@@ -0,0 +1,118 @@
# -*- coding: utf-8 -*-
"""
伶伦转换器 版本检查组件
Linglun Converter Version Checking Component
版权所有 © 2023 金羿 & 睿穆开发组
Copyright © 2023 EillesWan & TriM Org.
开源相关声明请见 ./License.md
Terms & Conditions: ./Lisense.md
"""
from .io import TrimLog, urllib, Sequence, Iterable, Callable, Optional
def is_ver_char(text: str) -> bool:
return text.isnumeric() or text == "."
def cut_ver_str(text: str) -> str:
text += " "
len_of_text = len(text) - 1
i = 0
while i <= len_of_text:
if is_ver_char(text[i]) and (text[i + 1] if i < len_of_text else False):
j = i
while is_ver_char(text[j]) and j < len_of_text:
j += 1
temp_str = text[i:j].strip()
if ("." in temp_str) and (temp_str[0] != ".") and (temp_str[-1] != "."):
return temp_str
i = j
i += 1
return ""
def get_ver_str(text: str) -> Iterable[str]:
text += " "
all_ver_str = []
len_of_text = len(text) - 1
i = 0
while i <= len_of_text:
if is_ver_char(text[i]) and (text[i + 1] if i < len_of_text else False):
j = i
while is_ver_char(text[j]) and j < len_of_text:
j += 1
temp_str = text[i:j].strip()
if ("." in temp_str) and (temp_str[0] != ".") and (temp_str[-1] != "."):
all_ver_str.append(temp_str)
i = j
i += 1
return all_ver_str
def is_ver_bigger(ver_1: Sequence[int], ver_2: Sequence[int]) -> bool:
len_v1 = len(ver_1)
len_v2 = len(ver_2)
for i in range(min(len_v1, len_v2)):
if ver_1[i] == ver_2[i]:
continue
else:
return ver_1[i] > ver_2[i]
return len_v1 > len_v2
def check_update(
appname: str,
get_text_url: str,
version_now: str,
message_show_fun: Callable,
logger: TrimLog.Logger,
version_disp: Optional[str] = None,
):
if not version_disp:
version_disp = version_now
logger.info("当前版本信息:{}".format(version_now))
try:
code_content: str = urllib.request.urlopen(get_text_url).read().decode("utf-8")
except Exception as E: # noinspection PyBroadException
logger.warning("无法获取更新版本信息:{}".format(E))
return
code_content = code_content[code_content.find("__version__") :]
code_content = code_content[code_content.find('"') + 1 :]
version_content = code_content[: code_content.find('"')]
logger.info("已获取更新版本信息:{}".format(version_content))
if is_ver_bigger(
[int(v) for v in cut_ver_str(version_content).split(".")],
[int(v) for v in cut_ver_str(version_now).split(".")],
):
if "__zhver__" in code_content:
code_content = code_content[code_content.find("__zhver__") :]
code_content = code_content[code_content.find('"') + 1 :]
version_content = code_content[: code_content.find('"')]
message_show_fun(
"!有新版本!\n最新的 {app} 已经是 {latest} 版本,当前您正在使用的仍是 {current} 版本,您可以前往下载地址更新。".format(
app=appname, latest=version_content, current=version_disp
)
)
# code_content = code_content[code_content.find('"')+1:]
# version_content = code_content[:code_content.find('"')]
# version_content_len = len(version_content)
# for i in range(version_content_len):
# if is_ver_char(version_content[i]) and (version_content[i+1] if i < version_content.__len__() else False):
# j = i
# while is_ver_char(version_content[j]):j+=1
# return version_content[i:j]
# "".join([version_content[i] for i in range(version_content.__len__()) if is_ver_char(version_content[i]) and ((version_content[i-1] if i > 0 else False) or (version_content[i+1] if i < version_content.__len__() else False))]).split('.')

View File

@@ -47,7 +47,7 @@
<property name="minimum_size"></property>
<property name="name">LingLunMainFrame</property>
<property name="pos"></property>
<property name="size">659,778</property>
<property name="size">660,780</property>
<property name="style">wxDEFAULT_FRAME_STYLE</property>
<property name="subclass">; ; forward_declare</property>
<property name="title"></property>
@@ -126,11 +126,11 @@
<property name="name">m_mainBoxSizer</property>
<property name="orient">wxVERTICAL</property>
<property name="permission">none</property>
<object class="sizeritem" expanded="0">
<object class="sizeritem" expanded="1">
<property name="border">2</property>
<property name="flag">wxALL|wxEXPAND|wxRESERVE_SPACE_EVEN_IF_HIDDEN|wxTOP</property>
<property name="proportion">1</property>
<object class="wxStaticBoxSizer" expanded="0">
<object class="wxStaticBoxSizer" expanded="1">
<property name="id">wxID_ANY</property>
<property name="label">言·论</property>
<property name="minimum_size">-1,-1</property>
@@ -214,7 +214,7 @@
<property name="permission">none</property>
<object class="sizeritem" expanded="0">
<property name="border">5</property>
<property name="flag">wxALL</property>
<property name="flag">wxALL|wxALIGN_CENTER_VERTICAL</property>
<property name="proportion">0</property>
<object class="wxStaticText" expanded="0">
<property name="BottomDockable">1</property>
@@ -244,7 +244,7 @@
<property name="gripper">0</property>
<property name="hidden">0</property>
<property name="id">wxID_ANY</property>
<property name="label">选择MIDI文件</property>
<property name="label">选择MIDI文件&#x0A;(双击移除)</property>
<property name="markup">0</property>
<property name="max_size"></property>
<property name="maximize_button">0</property>
@@ -283,11 +283,11 @@
<property name="name">ss_MidiChooserSizer_bSizer9</property>
<property name="orient">wxVERTICAL</property>
<property name="permission">none</property>
<object class="sizeritem" expanded="0">
<property name="border">5</property>
<property name="flag">wxALL|wxEXPAND</property>
<object class="sizeritem" expanded="1">
<property name="border">0</property>
<property name="flag">wxEXPAND</property>
<property name="proportion">0</property>
<object class="wxFilePickerCtrl" expanded="0">
<object class="wxListBox" expanded="1">
<property name="BottomDockable">1</property>
<property name="LeftDockable">1</property>
<property name="RightDockable">1</property>
@@ -301,6 +301,7 @@
<property name="caption"></property>
<property name="caption_visible">1</property>
<property name="center_pane">0</property>
<property name="choices"></property>
<property name="close_button">1</property>
<property name="context_help"></property>
<property name="context_menu">1</property>
@@ -318,22 +319,21 @@
<property name="max_size"></property>
<property name="maximize_button">0</property>
<property name="maximum_size"></property>
<property name="message">选择MIDI文件</property>
<property name="min_size"></property>
<property name="minimize_button">0</property>
<property name="minimum_size">-1,-1</property>
<property name="minimum_size"></property>
<property name="moveable">1</property>
<property name="name">m_ChooseMIDI_filePicker1</property>
<property name="name">m_midiFilesList_listBox2</property>
<property name="pane_border">1</property>
<property name="pane_position"></property>
<property name="pane_size"></property>
<property name="permission">protected</property>
<property name="pin_button">1</property>
<property name="pos"></property>
<property name="resize">Fixed</property>
<property name="resize">Resizable</property>
<property name="show">1</property>
<property name="size"></property>
<property name="style">wxFLP_DEFAULT_STYLE|wxFLP_FILE_MUST_EXIST|wxFLP_OPEN|wxFLP_USE_TEXTCTRL</property>
<property name="style">wxLB_HSCROLL|wxLB_SORT</property>
<property name="subclass">; ; forward_declare</property>
<property name="toolbar_pane">0</property>
<property name="tooltip"></property>
@@ -341,16 +341,89 @@
<property name="validator_style">wxFILTER_NONE</property>
<property name="validator_type">wxDefaultValidator</property>
<property name="validator_variable"></property>
<property name="value">./</property>
<property name="wildcard">MIDI 序列 (*.mid;*.midi)|*.mid;*.midi</property>
<property name="window_extra_style"></property>
<property name="window_name"></property>
<property name="window_style"></property>
<event name="OnFileChanged">onFileChosen</event>
<event name="OnListBox">onFileListUpdated</event>
<event name="OnListBoxDClick">onFileDoubleClicked</event>
</object>
</object>
</object>
</object>
<object class="sizeritem" expanded="1">
<property name="border">5</property>
<property name="flag">wxALL|wxALIGN_CENTER_VERTICAL</property>
<property name="proportion">0</property>
<object class="wxButton" expanded="1">
<property name="BottomDockable">1</property>
<property name="LeftDockable">1</property>
<property name="RightDockable">1</property>
<property name="TopDockable">1</property>
<property name="aui_layer"></property>
<property name="aui_name"></property>
<property name="aui_position"></property>
<property name="aui_row"></property>
<property name="auth_needed">0</property>
<property name="best_size"></property>
<property name="bg"></property>
<property name="bitmap"></property>
<property name="caption"></property>
<property name="caption_visible">1</property>
<property name="center_pane">0</property>
<property name="close_button">1</property>
<property name="context_help"></property>
<property name="context_menu">1</property>
<property name="current"></property>
<property name="default">0</property>
<property name="default_pane">0</property>
<property name="disabled"></property>
<property name="dock">Dock</property>
<property name="dock_fixed">0</property>
<property name="docking">Left</property>
<property name="enabled">1</property>
<property name="fg"></property>
<property name="floatable">1</property>
<property name="focus"></property>
<property name="font"></property>
<property name="gripper">0</property>
<property name="hidden">0</property>
<property name="id">wxID_ANY</property>
<property name="label">打开…</property>
<property name="margins"></property>
<property name="markup">0</property>
<property name="max_size"></property>
<property name="maximize_button">0</property>
<property name="maximum_size"></property>
<property name="min_size"></property>
<property name="minimize_button">0</property>
<property name="minimum_size"></property>
<property name="moveable">1</property>
<property name="name">m_midiBroseButton_button21</property>
<property name="pane_border">1</property>
<property name="pane_position"></property>
<property name="pane_size"></property>
<property name="permission">protected</property>
<property name="pin_button">1</property>
<property name="pos"></property>
<property name="position"></property>
<property name="pressed"></property>
<property name="resize">Resizable</property>
<property name="show">1</property>
<property name="size"></property>
<property name="style"></property>
<property name="subclass">; ; forward_declare</property>
<property name="toolbar_pane">0</property>
<property name="tooltip"></property>
<property name="validator_data_type"></property>
<property name="validator_style">wxFILTER_NONE</property>
<property name="validator_type">wxDefaultValidator</property>
<property name="validator_variable"></property>
<property name="window_extra_style"></property>
<property name="window_name"></property>
<property name="window_style"></property>
<event name="OnButtonClick">openFile</event>
</object>
</object>
</object>
</object>
<object class="sizeritem" expanded="1">
@@ -1048,7 +1121,7 @@
<property name="resize">Fixed</property>
<property name="show">1</property>
<property name="size"></property>
<property name="style">wxTE_LEFT|wxTE_NO_VSCROLL|wxTE_RICH</property>
<property name="style">wxTE_LEFT|wxTE_NO_VSCROLL</property>
<property name="subclass">; ; forward_declare</property>
<property name="toolbar_pane">0</property>
<property name="tooltip"></property>
@@ -2083,7 +2156,7 @@
<property name="gripper">0</property>
<property name="hidden">0</property>
<property name="id">wxID_ANY</property>
<property name="label">转换,启动!</property>
<property name="label">开始转换</property>
<property name="margins"></property>
<property name="markup">0</property>
<property name="max_size"></property>
@@ -2093,7 +2166,7 @@
<property name="minimize_button">0</property>
<property name="minimum_size"></property>
<property name="moveable">1</property>
<property name="name">m_button2</property>
<property name="name">m_start_button2</property>
<property name="pane_border">1</property>
<property name="pane_position"></property>
<property name="pane_size"></property>