2021-09-08 23:38:34 +08:00
|
|
|
|
# -------------------------------
|
|
|
|
|
# Difficult Rocket
|
2022-06-27 16:51:14 +08:00
|
|
|
|
# Copyright © 2021-2022 by shenjackyuanjie 3695888@qq.com
|
2021-09-08 23:38:34 +08:00
|
|
|
|
# All rights reserved
|
|
|
|
|
# -------------------------------
|
|
|
|
|
|
2021-09-02 22:47:10 +08:00
|
|
|
|
"""
|
2022-11-26 21:48:55 +08:00
|
|
|
|
writen by shenjackyuanjie <3695888@qq.com>
|
2021-09-02 22:47:10 +08:00
|
|
|
|
mail: 3695888@qq.com
|
|
|
|
|
github: @shenjackyuanjie
|
|
|
|
|
gitee: @shenjackyuanjie
|
|
|
|
|
"""
|
|
|
|
|
|
2022-07-01 13:59:08 +08:00
|
|
|
|
import inspect
|
|
|
|
|
|
2023-01-03 16:24:06 +08:00
|
|
|
|
from dataclasses import dataclass
|
|
|
|
|
from typing import Union, Tuple, Any, Type, List, Dict, Hashable, Optional
|
2021-09-22 06:21:48 +08:00
|
|
|
|
|
2022-08-12 21:07:36 +08:00
|
|
|
|
from Difficult_Rocket import DR_runtime, DR_option
|
2022-04-26 22:05:58 +08:00
|
|
|
|
from Difficult_Rocket.utils import tools
|
2022-07-04 10:36:19 +08:00
|
|
|
|
from Difficult_Rocket.exception.language import *
|
2021-09-22 06:21:48 +08:00
|
|
|
|
|
2022-11-26 22:45:30 +08:00
|
|
|
|
|
2023-01-03 16:24:06 +08:00
|
|
|
|
@dataclass
|
|
|
|
|
class TranslateConfig:
|
2023-01-16 18:34:17 +08:00
|
|
|
|
raise_error: bool = False # 引用错误时抛出错误
|
|
|
|
|
crack_normal: bool = False # 出现错误引用后 将引用到的正确内容替换为引用路径
|
|
|
|
|
insert_crack: bool = True # 加入引用的错误内容
|
|
|
|
|
is_final: bool = False # 是否为最终内容
|
|
|
|
|
keep_get: bool = True # 引用错误后是否继续引用
|
|
|
|
|
always_copy: bool = False # 是否一直新建 Translate (为 True 会降低性能)
|
2023-01-03 16:24:06 +08:00
|
|
|
|
|
|
|
|
|
def set(self, item: str, value: bool) -> 'TranslateConfig':
|
|
|
|
|
assert getattr(self, item, None) is not None, f'Config {item} is not in TranslateConfig'
|
|
|
|
|
assert type(value) is bool
|
|
|
|
|
setattr(self, item, value)
|
|
|
|
|
return self
|
|
|
|
|
|
2023-01-03 22:28:26 +08:00
|
|
|
|
def __copy__(self) -> 'TranslateConfig':
|
|
|
|
|
return TranslateConfig(raise_error=self.raise_error,
|
|
|
|
|
crack_normal=self.crack_normal,
|
|
|
|
|
insert_crack=self.insert_crack,
|
|
|
|
|
is_final=self.is_final,
|
2023-01-16 18:34:17 +08:00
|
|
|
|
keep_get=self.keep_get,
|
|
|
|
|
always_copy=self.always_copy)
|
2023-01-03 22:28:26 +08:00
|
|
|
|
|
|
|
|
|
def copy(self) -> 'TranslateConfig':
|
|
|
|
|
return self.__copy__()
|
|
|
|
|
|
2023-01-03 16:24:06 +08:00
|
|
|
|
|
2023-01-16 18:34:17 +08:00
|
|
|
|
key_type = Union[str, int, Hashable]
|
|
|
|
|
|
|
|
|
|
|
2022-11-26 23:35:49 +08:00
|
|
|
|
class Translates:
|
2023-01-06 20:01:34 +08:00
|
|
|
|
name = 'Translate'
|
|
|
|
|
|
2022-11-27 13:45:59 +08:00
|
|
|
|
def __init__(self,
|
|
|
|
|
value: Union[Dict[str, Any], list, tuple, str],
|
2023-01-03 16:24:06 +08:00
|
|
|
|
config: Optional[TranslateConfig] = None,
|
2023-01-05 21:36:47 +08:00
|
|
|
|
get_list: List[Tuple[bool, str]] = None):
|
2022-11-27 13:45:59 +08:00
|
|
|
|
"""
|
2022-12-25 23:15:49 +08:00
|
|
|
|
一个用于翻译的东西
|
2022-11-27 13:45:59 +08:00
|
|
|
|
:param value: 翻译键节点
|
2023-01-03 22:28:26 +08:00
|
|
|
|
:param config: 配置
|
|
|
|
|
:param get_list: 获取列表
|
2022-11-27 13:45:59 +08:00
|
|
|
|
"""
|
2022-11-26 22:45:30 +08:00
|
|
|
|
self.value: Union[Dict[str, Any], list, tuple] = value
|
2023-01-03 16:24:06 +08:00
|
|
|
|
self.config = config or TranslateConfig()
|
2022-11-26 22:45:30 +08:00
|
|
|
|
self.get_list = get_list or []
|
2023-01-03 16:24:06 +08:00
|
|
|
|
|
2023-01-16 18:34:17 +08:00
|
|
|
|
def set_conf_(self, option: Union[str, TranslateConfig],
|
|
|
|
|
value: Optional[Union[bool, List[str]]] = None) -> 'Translates':
|
|
|
|
|
"""
|
|
|
|
|
设置翻译设置
|
|
|
|
|
:param option: 设置名称 / 新设置
|
|
|
|
|
:param value:
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2023-01-03 16:24:06 +08:00
|
|
|
|
assert type(option) is str or isinstance(option, TranslateConfig)
|
|
|
|
|
if isinstance(option, TranslateConfig):
|
|
|
|
|
self.config = option
|
|
|
|
|
return self
|
|
|
|
|
self.config.set(option, value)
|
2023-01-16 18:34:17 +08:00
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
def _raise_no_value(self, e: Exception, item: key_type):
|
|
|
|
|
if self.config.raise_error:
|
|
|
|
|
raise TranslateKeyNotFound(self.value, [x[1] for x in self.get_list]) from None
|
|
|
|
|
elif DR_option.report_translate_no_found:
|
|
|
|
|
frame = inspect.currentframe()
|
|
|
|
|
if frame is not None:
|
|
|
|
|
frame = frame.f_back
|
|
|
|
|
if frame.f_back.f_code is not self.__getattr__.__code__:
|
|
|
|
|
frame = frame.f_back
|
|
|
|
|
frame = f'call at {frame.f_code.co_filename}:{frame.f_lineno}'
|
|
|
|
|
else:
|
|
|
|
|
frame = 'but No Frame environment'
|
|
|
|
|
raise_info = f"{self.name} Cause a error when getting {item} {frame}"
|
|
|
|
|
print(raise_info)
|
2022-11-26 22:45:30 +08:00
|
|
|
|
|
2023-01-16 18:34:17 +08:00
|
|
|
|
def __getitem__(self, item: key_type) -> Union["Translates"]:
|
2022-12-08 09:53:22 +08:00
|
|
|
|
"""
|
|
|
|
|
:param item: 取用的内容/小天才
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2022-11-26 22:45:30 +08:00
|
|
|
|
try:
|
2023-01-16 18:34:17 +08:00
|
|
|
|
cache_value = self.value[item]
|
|
|
|
|
self.get_list.append((True, item))
|
|
|
|
|
if self.config.always_copy:
|
|
|
|
|
return self.copy
|
|
|
|
|
except (KeyError, TypeError, AttributeError) as e:
|
|
|
|
|
self.get_list.append((False, item))
|
|
|
|
|
self._raise_no_value(e, item)
|
|
|
|
|
|
|
|
|
|
# except (KeyError, TypeError):s
|
|
|
|
|
# cache_get_list.append((False, item))
|
|
|
|
|
# # 出现问题
|
|
|
|
|
# if DR_option.report_translate_no_found:
|
|
|
|
|
# frame = inspect.currentframe()
|
|
|
|
|
# last_frame = frame.f_back
|
|
|
|
|
# if last_frame.f_code == self.__getattr__.__code__:
|
|
|
|
|
# last_frame = last_frame.f_back
|
|
|
|
|
# call_info = f'{self.name} Not Found at {last_frame.f_code.co_name} by ' \
|
|
|
|
|
# f'{".".join([x[1] for x in cache_get_list])} at:' \
|
|
|
|
|
# f'{last_frame.f_code.co_filename}:{last_frame.f_lineno}'
|
|
|
|
|
# print(call_info)
|
|
|
|
|
# # 如果不抛出错误
|
|
|
|
|
# if self.config.raise_error:
|
|
|
|
|
# raise TranslateKeyNotFound(item_names=cache_get_list) from None
|
|
|
|
|
# if self.final: # 如果已经是翻译结果
|
|
|
|
|
# return Translates(value='.'.join(cache_get_list))
|
|
|
|
|
# else:
|
|
|
|
|
# if self.final:
|
|
|
|
|
# return self
|
|
|
|
|
# else:
|
|
|
|
|
# return Translates(value=cache, get_list=cache_get_list)
|
|
|
|
|
|
|
|
|
|
def __copy__(self) -> 'Translates':
|
|
|
|
|
return Translates(value=self.value,
|
|
|
|
|
config=self.config,
|
|
|
|
|
get_list=self.get_lists)
|
2022-11-26 21:48:55 +08:00
|
|
|
|
|
2022-11-26 23:35:49 +08:00
|
|
|
|
def __getattr__(self, item: Union[str, Hashable]) -> Union["Translates"]:
|
2023-01-05 21:36:47 +08:00
|
|
|
|
# 实际上我这里完全不需要处理正常需求,因为 __getattribute__ 已经帮我处理过了
|
2022-11-26 21:48:55 +08:00
|
|
|
|
return self.__getitem__(item)
|
|
|
|
|
|
|
|
|
|
def __str__(self):
|
2022-12-25 23:15:49 +08:00
|
|
|
|
if self.final: # 如果是字符串
|
2022-12-22 10:54:28 +08:00
|
|
|
|
return f'{self.value}.{".".join(self.get_list)}'
|
2022-11-26 21:48:55 +08:00
|
|
|
|
return str(self.value)
|
|
|
|
|
|
2021-09-02 22:47:10 +08:00
|
|
|
|
|
2022-07-01 13:59:08 +08:00
|
|
|
|
class Tr:
|
|
|
|
|
"""
|
2023-01-16 18:34:17 +08:00
|
|
|
|
我不装了,我就抄了tr(实际上没啥关系)
|
2022-08-16 13:25:44 +08:00
|
|
|
|
GOOD
|
2022-07-01 13:59:08 +08:00
|
|
|
|
"""
|
2023-01-03 16:24:06 +08:00
|
|
|
|
|
2023-01-03 22:28:26 +08:00
|
|
|
|
def __init__(self, language: str = None, config: Optional[TranslateConfig] = None):
|
2022-12-08 09:53:22 +08:00
|
|
|
|
"""
|
|
|
|
|
诶嘿,我抄的MCDR
|
|
|
|
|
:param language: Tr 所使用的的语言
|
2023-01-03 22:28:26 +08:00
|
|
|
|
:param config: 配置
|
2022-12-08 09:53:22 +08:00
|
|
|
|
"""
|
2022-11-27 13:45:59 +08:00
|
|
|
|
self.language_name = language or DR_runtime.language
|
|
|
|
|
self.translates: Dict = tools.load_file(f'configs/lang/{self.language_name}.toml')
|
2022-11-26 21:48:55 +08:00
|
|
|
|
self.default_translate: Dict = tools.load_file(f'configs/lang/{DR_runtime.default_language}.toml')
|
2023-01-03 22:28:26 +08:00
|
|
|
|
self.default_config = config or TranslateConfig()
|
|
|
|
|
self.translates_cache = Translates(value=self.translates, config=TranslateConfig().copy())
|
2023-01-03 16:24:06 +08:00
|
|
|
|
|
|
|
|
|
# def __call__(self, ):
|
|
|
|
|
# ...
|
2022-12-08 09:53:22 +08:00
|
|
|
|
|
2023-01-05 21:36:47 +08:00
|
|
|
|
def __getattr__(self, item) -> Translates:
|
|
|
|
|
...
|
|
|
|
|
|
|
|
|
|
def __getitem__(self, item: Union[str, int]):
|
|
|
|
|
return self.__getattr__(item)
|
2022-07-04 10:36:19 +08:00
|
|
|
|
|
2022-07-01 13:59:08 +08:00
|
|
|
|
|
2021-09-02 22:47:10 +08:00
|
|
|
|
class Lang:
|
|
|
|
|
"""
|
|
|
|
|
用于创建一个对应语言的翻译类
|
|
|
|
|
感谢Fallen的MCDR提供idea
|
|
|
|
|
https://github.com/Fallen-Breath/MCDReforged
|
2021-09-22 06:21:48 +08:00
|
|
|
|
可以用
|
|
|
|
|
lang['language'] = 'abc' 或
|
|
|
|
|
lang['lang'] = 'abc'
|
|
|
|
|
的方式直接更改并刷新翻译
|
2021-10-04 20:41:41 +08:00
|
|
|
|
用
|
|
|
|
|
lang.lang(xxx, xxx)来获取翻译过的值
|
2021-09-02 22:47:10 +08:00
|
|
|
|
"""
|
2021-09-05 00:50:05 +08:00
|
|
|
|
|
2022-08-12 21:07:36 +08:00
|
|
|
|
def __init__(self) -> None:
|
2022-11-26 21:48:55 +08:00
|
|
|
|
self.translates = tools.load_file(f'configs/lang/{DR_runtime.language}.toml')
|
|
|
|
|
self.default_translates = tools.load_file('configs/lang/zh-CN.toml')
|
2022-06-14 20:52:03 +08:00
|
|
|
|
self.直接返回原始数据 = True
|
2021-09-02 22:47:10 +08:00
|
|
|
|
|
2021-09-22 06:21:48 +08:00
|
|
|
|
def __str__(self) -> str:
|
2022-08-12 21:07:36 +08:00
|
|
|
|
return DR_option.language
|
2021-09-02 22:47:10 +08:00
|
|
|
|
|
2021-10-22 06:36:06 +08:00
|
|
|
|
def __getitem__(self, item) -> Union[int, str, list, dict]:
|
2021-09-22 06:21:48 +08:00
|
|
|
|
try:
|
2022-11-26 21:48:55 +08:00
|
|
|
|
return self.translates[item]
|
2021-09-22 06:21:48 +08:00
|
|
|
|
except KeyError:
|
2021-09-28 22:47:19 +08:00
|
|
|
|
try:
|
2022-11-26 21:48:55 +08:00
|
|
|
|
return self.default_translates[item]
|
2021-09-28 22:47:19 +08:00
|
|
|
|
except KeyError:
|
2023-01-03 16:24:06 +08:00
|
|
|
|
raise TranslateKeyNotFound
|
|
|
|
|
# raise TranslateKeyNotFound(f'there\'s no key {item} in both {DR_option.language} and zh-CN')
|
2021-09-22 06:21:48 +08:00
|
|
|
|
|
2022-12-08 09:53:22 +08:00
|
|
|
|
def lang(self, *args) -> Union[int, str, list, dict, tuple]:
|
2022-07-04 10:36:19 +08:00
|
|
|
|
# frame = inspect.currentframe()
|
|
|
|
|
# # print("调用当前log的文件名:", frame.f_back.f_code.co_filename)
|
|
|
|
|
# objprint.objprint(frame.f_back.f_code,
|
|
|
|
|
# honor_existing=False,
|
|
|
|
|
# depth=2)
|
2021-09-28 22:47:19 +08:00
|
|
|
|
try:
|
2022-11-26 21:48:55 +08:00
|
|
|
|
result = self.translates
|
|
|
|
|
for option in args:
|
|
|
|
|
result = result[option]
|
|
|
|
|
return result
|
2021-09-28 22:47:19 +08:00
|
|
|
|
except KeyError:
|
|
|
|
|
try:
|
2022-11-26 21:48:55 +08:00
|
|
|
|
result = self.default_translates
|
|
|
|
|
for option in args:
|
|
|
|
|
result = result[option]
|
|
|
|
|
return result
|
|
|
|
|
except KeyError as e:
|
2022-06-14 20:52:03 +08:00
|
|
|
|
if self.直接返回原始数据:
|
|
|
|
|
return args
|
2023-01-03 16:24:06 +08:00
|
|
|
|
raise TranslateKeyNotFound from e
|
|
|
|
|
# raise TranslateKeyNotFound(f'there\'s no key {args} in both {DR_option.language} and zh-CN') from e
|
2021-09-28 22:47:19 +08:00
|
|
|
|
|
2021-10-22 06:36:06 +08:00
|
|
|
|
def 翻译(self, *args) -> Union[int, str, list, dict]:
|
|
|
|
|
return self.lang(args)
|
2021-10-04 20:41:41 +08:00
|
|
|
|
|
2022-08-12 21:07:36 +08:00
|
|
|
|
def _update_lang(self) -> str:
|
|
|
|
|
"""
|
|
|
|
|
用于更新语言(内部调用)
|
2022-11-26 21:48:55 +08:00
|
|
|
|
:return: 设置完成后的语言
|
2022-08-12 21:07:36 +08:00
|
|
|
|
"""
|
2022-11-26 21:48:55 +08:00
|
|
|
|
self.translates = tools.load_file(f'configs/lang/{DR_option.language}.toml')
|
2022-08-12 21:07:36 +08:00
|
|
|
|
return DR_option.language
|
|
|
|
|
|
2021-09-22 06:21:48 +08:00
|
|
|
|
|
2022-08-12 21:07:36 +08:00
|
|
|
|
if not __name__ == '__main__':
|
|
|
|
|
tr = Lang()
|
2022-11-27 13:45:59 +08:00
|
|
|
|
else:
|
|
|
|
|
tr_ = Tr()
|