2023-08-29 14:35:13 +00:00
|
|
|
import time
|
2023-05-14 07:48:34 +00:00
|
|
|
from typing import Dict
|
|
|
|
|
|
|
|
from module.config.utils import *
|
|
|
|
from module.webui.fake import list_mod
|
2023-08-29 14:35:13 +00:00
|
|
|
from module.webui.setting import State
|
2023-05-14 07:48:34 +00:00
|
|
|
|
|
|
|
LANG = "zh-CN"
|
|
|
|
TRANSLATE_MODE = False
|
|
|
|
|
|
|
|
|
|
|
|
def set_language(s: str, refresh=False):
|
|
|
|
global LANG
|
|
|
|
for i, lang in enumerate(LANGUAGES):
|
|
|
|
# pywebio.session.info.user_language return `zh-CN` or `zh-cn`, depends on browser
|
|
|
|
if lang.lower() == s.lower():
|
|
|
|
LANG = LANGUAGES[i]
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
LANG = "en-US"
|
|
|
|
|
|
|
|
State.deploy_config.Language = LANG
|
|
|
|
|
|
|
|
if refresh:
|
|
|
|
from pywebio.session import run_js
|
|
|
|
|
|
|
|
run_js("location.reload();")
|
|
|
|
|
|
|
|
|
|
|
|
def t(s, *args, **kwargs):
|
|
|
|
"""
|
|
|
|
Get translation.
|
|
|
|
other args, kwargs pass to .format()
|
|
|
|
"""
|
|
|
|
if TRANSLATE_MODE:
|
|
|
|
return s
|
|
|
|
return _t(s, LANG).format(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
def _t(s, lang=None):
|
|
|
|
"""
|
|
|
|
Get translation, ignore TRANSLATE_MODE
|
|
|
|
"""
|
|
|
|
if not lang:
|
|
|
|
lang = LANG
|
|
|
|
try:
|
|
|
|
return dic_lang[lang][s]
|
|
|
|
except KeyError:
|
|
|
|
print(f"Language key ({s}) not found")
|
|
|
|
return s
|
|
|
|
|
|
|
|
|
|
|
|
dic_lang: Dict[str, Dict[str, str]] = {}
|
|
|
|
|
|
|
|
|
|
|
|
def reload():
|
|
|
|
for lang in LANGUAGES:
|
|
|
|
if lang not in dic_lang:
|
|
|
|
dic_lang[lang] = {}
|
|
|
|
|
|
|
|
for mod_name, dir_name in list_mod():
|
|
|
|
for path, v in deep_iter(read_file(filepath_i18n(lang, mod_name)), depth=3):
|
|
|
|
dic_lang[lang][".".join(path)] = v
|
|
|
|
|
|
|
|
for path, v in deep_iter(read_file(filepath_i18n(lang)), depth=3):
|
|
|
|
dic_lang[lang][".".join(path)] = v
|
|
|
|
|
|
|
|
for key in dic_lang["ja-JP"].keys():
|
|
|
|
if dic_lang["ja-JP"][key] == key:
|
|
|
|
dic_lang["ja-JP"][key] = dic_lang["en-US"][key]
|
2023-08-29 14:35:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
def readable_time(before: str) -> str:
|
|
|
|
"""
|
|
|
|
Convert "2023-08-29 12:30:53" to "3 Minutes Ago"
|
|
|
|
"""
|
|
|
|
if not before:
|
|
|
|
return t("Gui.Dashboard.NoData")
|
|
|
|
try:
|
|
|
|
ti = datetime.fromisoformat(before)
|
|
|
|
except ValueError:
|
|
|
|
return t("Gui.Dashboard.TimeError")
|
|
|
|
if ti == DEFAULT_TIME:
|
|
|
|
return t("Gui.Dashboard.NoData")
|
|
|
|
|
|
|
|
diff = time.time() - ti.timestamp()
|
|
|
|
if diff < -1:
|
|
|
|
return t("Gui.Dashboard.TimeError")
|
|
|
|
elif diff < 60:
|
|
|
|
# < 1 min
|
|
|
|
return t("Gui.Dashboard.JustNow")
|
2023-11-02 02:35:13 +00:00
|
|
|
elif diff < 5400:
|
|
|
|
# < 90 min
|
2023-08-29 14:35:13 +00:00
|
|
|
return t("Gui.Dashboard.MinutesAgo", time=int(diff // 60))
|
2023-11-02 02:35:13 +00:00
|
|
|
elif diff < 129600:
|
|
|
|
# < 36 hours
|
2023-08-29 14:35:13 +00:00
|
|
|
return t("Gui.Dashboard.HoursAgo", time=int(diff // 3600))
|
2023-09-30 17:56:03 +00:00
|
|
|
elif diff < 1296000:
|
2023-11-02 02:35:13 +00:00
|
|
|
# < 15 days
|
2023-08-29 14:35:13 +00:00
|
|
|
return t("Gui.Dashboard.DaysAgo", time=int(diff // 86400))
|
|
|
|
else:
|
2023-11-02 02:35:13 +00:00
|
|
|
# >= 15 days
|
2023-08-29 14:35:13 +00:00
|
|
|
return t("Gui.Dashboard.LongTimeAgo")
|