mirror of
https://github.com/HIllya51/LunaTranslator.git
synced 2025-01-15 08:53:53 +08:00
1266 lines
43 KiB
Python
1266 lines
43 KiB
Python
import time, uuid
|
||
import os, threading, re, winreg
|
||
from qtsymbols import *
|
||
from traceback import print_exc
|
||
from myutils.config import (
|
||
globalconfig,
|
||
savehook_new_list,
|
||
findgameuidofpath,
|
||
savehook_new_data,
|
||
static_data,
|
||
getlanguse,
|
||
_TR,
|
||
get_platform,
|
||
isascii,
|
||
)
|
||
from myutils.utils import (
|
||
minmaxmoveobservefunc,
|
||
parsemayberegexreplace,
|
||
find_or_create_uid,
|
||
checkisusingwine,
|
||
checkpostusing,
|
||
stringfyerror,
|
||
targetmod,
|
||
translate_exits,
|
||
safe_escape,
|
||
)
|
||
from myutils.wrapper import threader, tryprint
|
||
from gui.showword import searchwordW
|
||
from myutils.hwnd import getpidexe, getExeIcon, getcurrexe
|
||
from textsource.copyboard import copyboard
|
||
from textsource.texthook import texthook
|
||
from textsource.ocrtext import ocrtext
|
||
from textsource.textsourcebase import basetext
|
||
from textsource.filetrans import filetrans
|
||
from gui.selecthook import hookselect
|
||
from gui.translatorUI import TranslatorWindow
|
||
import functools, gobject
|
||
from gui.transhist import transhist
|
||
from gui.edittext import edittext
|
||
import importlib, qtawesome
|
||
from functools import partial
|
||
from gui.setting import Setting
|
||
from gui.attachprocessdialog import AttachProcessDialog
|
||
import windows
|
||
import winsharedutils
|
||
from winsharedutils import collect_running_pids
|
||
from myutils.post import POSTSOLVE
|
||
from myutils.utils import nowisdark, dynamicapiname
|
||
from myutils.traceplaytime import playtimemanager
|
||
from myutils.audioplayer import series_audioplayer
|
||
from gui.dynalang import LAction, LMenu
|
||
from gui.setting_textinput_ocr import showocrimage
|
||
|
||
|
||
class MAINUI:
|
||
def __init__(self) -> None:
|
||
super().__init__()
|
||
self.update_avalable = False
|
||
self.translators = {}
|
||
self.cishus = {}
|
||
self.specialreaders = {}
|
||
self.textsource_p = None
|
||
self.currenttext = ""
|
||
self.currenttranslate = ""
|
||
self.currentread = ""
|
||
self.refresh_on_get_trans_signature = 0
|
||
self.currentsignature = None
|
||
self.isrunning = True
|
||
self.solvegottextlock = threading.Lock()
|
||
self.gettranslatelock = threading.Lock()
|
||
self.outputers = {}
|
||
self.processmethods = []
|
||
self.AttachProcessDialog = None
|
||
self.edittextui = None
|
||
self.edittextui_cached = None
|
||
self.edittextui_sync = True
|
||
self.notifyonce = set()
|
||
self.audioplayer = series_audioplayer(playovercallback=self.ttsautoforward)
|
||
self._internal_reader = None
|
||
self.reader_uid = None
|
||
self.__hwnd = None
|
||
self.gameuid = 0
|
||
self.showocrimage = None
|
||
self.showocrimage_cached = None
|
||
self.showocrimage_cached2 = None
|
||
self.autoswitchgameuid = True
|
||
self.istriggertoupdate = False
|
||
self.thishastranslated = True
|
||
|
||
@threader
|
||
def ttsautoforward(self):
|
||
if not globalconfig["ttsautoforward"]:
|
||
return
|
||
windows.SetForegroundWindow(self.hwnd)
|
||
time.sleep(0.001)
|
||
windows.keybd_event(windows.VK_RETURN, 0, 0, 0)
|
||
time.sleep(0.001)
|
||
windows.keybd_event(windows.VK_RETURN, 0, windows.KEYEVENTF_KEYUP, 0)
|
||
|
||
def maybesetimage(self, pair):
|
||
if self.showocrimage:
|
||
try:
|
||
self.showocrimage.setimage.emit(pair)
|
||
except:
|
||
print_exc()
|
||
self.showocrimage_cached = pair
|
||
|
||
def maybesetocrresult(self, pair):
|
||
if self.showocrimage:
|
||
try:
|
||
self.showocrimage.setresult.emit(pair)
|
||
except:
|
||
print_exc()
|
||
self.showocrimage_cached2 = pair
|
||
|
||
def createshowocrimage(self):
|
||
try:
|
||
self.showocrimage = showocrimage(
|
||
self.settin_ui, self.showocrimage_cached, self.showocrimage_cached2
|
||
)
|
||
if self.showocrimage:
|
||
self.showocrimage.show()
|
||
except:
|
||
print_exc()
|
||
|
||
@property
|
||
def reader(self):
|
||
return self._internal_reader
|
||
|
||
@reader.setter
|
||
def reader(self, _):
|
||
if _ is None:
|
||
self._internal_reader = None
|
||
self.reader_uid = None
|
||
self.settin_ui.voicelistsignal.emit(None)
|
||
else:
|
||
if self.reader_uid != _.uid:
|
||
return
|
||
self._internal_reader = _
|
||
self.settin_ui.voicelistsignal.emit(_)
|
||
|
||
@property
|
||
def textsource(self) -> basetext:
|
||
return self.textsource_p
|
||
|
||
@property
|
||
def hwnd(self):
|
||
return self.__hwnd
|
||
|
||
@hwnd.setter
|
||
def hwnd(self, __hwnd):
|
||
self.__hwnd = __hwnd
|
||
if not __hwnd:
|
||
self.translation_ui.processismuteed = False
|
||
self.translation_ui.isbindedwindow = False
|
||
self.translation_ui.refreshtooliconsignal.emit()
|
||
self.translation_ui.thistimenotsetop = False
|
||
if self.autoswitchgameuid:
|
||
self.gameuid = 0
|
||
else:
|
||
_pid = windows.GetWindowThreadProcessId(__hwnd)
|
||
if _pid:
|
||
_mute = winsharedutils.GetProcessMute(_pid)
|
||
self.translation_ui.processismuteed = _mute
|
||
self.translation_ui.isbindedwindow = True
|
||
self.translation_ui.refreshtooliconsignal.emit()
|
||
try:
|
||
if self.autoswitchgameuid:
|
||
gameuid, _ = findgameuidofpath(getpidexe(_pid))
|
||
if gameuid:
|
||
self.gameuid = gameuid
|
||
except:
|
||
print_exc()
|
||
else:
|
||
if self.autoswitchgameuid:
|
||
self.gameuid = 0
|
||
if globalconfig["keepontop"]:
|
||
self.translation_ui.settop()
|
||
|
||
@textsource.setter
|
||
def textsource(self, _):
|
||
if _ is None:
|
||
if self.textsource_p:
|
||
try:
|
||
self.textsource_p.endX()
|
||
except:
|
||
print_exc()
|
||
self.autoswitchgameuid = True
|
||
self.hwnd = None
|
||
self.gameuid = 0
|
||
self.textsource_p = _
|
||
|
||
@threader
|
||
def safeloadprocessmodels(self):
|
||
for item in static_data["transoptimi"]:
|
||
name = item["name"]
|
||
try:
|
||
checkpath = "./LunaTranslator/transoptimi/" + name + ".py"
|
||
if os.path.exists(checkpath) == False:
|
||
continue
|
||
mm = "transoptimi." + name
|
||
|
||
Process = importlib.import_module(mm).Process
|
||
|
||
def __(kls, _name):
|
||
class klass(kls):
|
||
@property
|
||
def using(self):
|
||
if "using_X" in dir(self):
|
||
try:
|
||
return self.using_X
|
||
except:
|
||
return False
|
||
return checkpostusing(_name)
|
||
|
||
return klass()
|
||
|
||
object = __(Process, name)
|
||
self.processmethods.append({"name": name, "object": object})
|
||
except:
|
||
print_exc()
|
||
|
||
def solvebeforetrans(self, content):
|
||
contexts = []
|
||
for method in self.processmethods:
|
||
context = None
|
||
try:
|
||
if method["object"].using:
|
||
if "process_before" in dir(method["object"]):
|
||
content, context = method["object"].process_before(content)
|
||
except:
|
||
print_exc()
|
||
contexts.append(context)
|
||
return content, contexts
|
||
|
||
def solveaftertrans(self, res, mp):
|
||
for i, method in enumerate(self.processmethods):
|
||
|
||
context = mp[i]
|
||
try:
|
||
if method["object"].using:
|
||
if "process_after" in dir(method["object"]):
|
||
res = method["object"].process_after(res, context)
|
||
except:
|
||
print_exc()
|
||
return res
|
||
|
||
def parsehira(self, text):
|
||
try:
|
||
if self.hira_:
|
||
return self.hira_.safeparse(text)
|
||
else:
|
||
return []
|
||
except:
|
||
return []
|
||
|
||
def displayinfomessage(self, text, infotype):
|
||
if infotype == "<notrans>":
|
||
self.translation_ui.displayres.emit(
|
||
dict(
|
||
color=globalconfig["rawtextcolor"],
|
||
res=text,
|
||
clear=True,
|
||
)
|
||
)
|
||
self.currenttext = text
|
||
self.currenttranslate = text
|
||
self.currentread = text
|
||
return
|
||
else:
|
||
msgs = [
|
||
("<msg_info_refresh>", False, True),
|
||
("<msg_error_not_refresh>", True, False),
|
||
("<msg_error_refresh>", True, True),
|
||
]
|
||
for msg, color, refresh in msgs:
|
||
if infotype == msg:
|
||
self.translation_ui.displaystatus.emit(text, color, refresh)
|
||
return
|
||
|
||
def maybeneedtranslateshowhidetranslate(self):
|
||
if globalconfig["showfanyi"]:
|
||
if not self.thishastranslated:
|
||
self.textgetmethod(self.currenttext, is_auto_run=False)
|
||
self.translation_ui.translate_text.textbrowser.showhidetranslate(True)
|
||
else:
|
||
self.translation_ui.translate_text.textbrowser.showhidetranslate(False)
|
||
|
||
def textgetmethod(
|
||
self,
|
||
text,
|
||
is_auto_run=True,
|
||
waitforresultcallback=None,
|
||
waitforresultcallbackengine=None,
|
||
donttrans=False,
|
||
):
|
||
with self.solvegottextlock:
|
||
succ = self.textgetmethod_1(
|
||
text,
|
||
is_auto_run,
|
||
waitforresultcallback,
|
||
waitforresultcallbackengine,
|
||
donttrans,
|
||
)
|
||
if waitforresultcallback and not succ:
|
||
waitforresultcallback("")
|
||
|
||
def textgetmethod_1(
|
||
self,
|
||
text,
|
||
is_auto_run=True,
|
||
waitforresultcallback=None,
|
||
waitforresultcallbackengine=None,
|
||
donttrans=False,
|
||
):
|
||
if not text:
|
||
return
|
||
if is_auto_run and text == self.currenttext:
|
||
return
|
||
origin = text
|
||
currentsignature = uuid.uuid4()
|
||
if not waitforresultcallback:
|
||
self.currentsignature = currentsignature
|
||
# 内嵌&文件翻译不要进行文本预处理
|
||
try:
|
||
text = POSTSOLVE(text)
|
||
self.settin_ui.showandsolvesig.emit(origin, text)
|
||
if not text:
|
||
return
|
||
except Exception as e:
|
||
self.translation_ui.displaystatus.emit(stringfyerror(e), True, True)
|
||
return
|
||
|
||
if is_auto_run and (
|
||
len(text) < globalconfig["minlength"]
|
||
or len(text) > globalconfig["maxlength"]
|
||
):
|
||
|
||
if len(text) > globalconfig["maxlength"]:
|
||
text = text[: globalconfig["maxlength"]] + "……"
|
||
|
||
self.translation_ui.displayraw1.emit(
|
||
dict(text=text, color=globalconfig["rawtextcolor"])
|
||
)
|
||
return
|
||
|
||
try:
|
||
self.textsource.sqlqueueput((text, origin))
|
||
except:
|
||
pass
|
||
if donttrans:
|
||
return
|
||
_showrawfunction_unsafe = None
|
||
if not waitforresultcallback:
|
||
self.currenttext = text
|
||
self.currenttranslate = ""
|
||
if globalconfig["read_raw"]:
|
||
self.currentread = text
|
||
which = self.__usewhich()
|
||
if which.get(
|
||
"tts_repair_use_at_translate",
|
||
globalconfig["ttscommon"]["tts_repair_use_at_translate"],
|
||
):
|
||
text = self.readcurrent(needresult=True)
|
||
else:
|
||
self.readcurrent()
|
||
if globalconfig["textoutput_origin"]:
|
||
self.dispatchoutputer(text)
|
||
|
||
_showrawfunction_unsafe = functools.partial(
|
||
self.translation_ui.displayraw1.emit,
|
||
dict(text=text, color=globalconfig["rawtextcolor"]),
|
||
)
|
||
self.thishastranslated = globalconfig["showfanyi"]
|
||
_showrawfunction = lambda: (
|
||
_showrawfunction_unsafe() if _showrawfunction_unsafe else None
|
||
)
|
||
|
||
self.transhis.getnewsentencesignal.emit(text)
|
||
self.maybesetedittext(text)
|
||
|
||
if not waitforresultcallback and not globalconfig["showfanyi"]:
|
||
return _showrawfunction()
|
||
|
||
text_solved, optimization_params = self.solvebeforetrans(text)
|
||
|
||
maybehaspremt = {}
|
||
skip_other_on_success = False
|
||
fix_rank = globalconfig["fix_translate_rank_rank"].copy()
|
||
if "rengong" in self.translators:
|
||
contentraw = self.analyzecontent(text_solved, optimization_params)
|
||
try:
|
||
res = self.translators["rengong"].translate(contentraw)
|
||
except:
|
||
print_exc()
|
||
res = None
|
||
maybehaspremt["rengong"] = res
|
||
skip_other_on_success = (
|
||
res and self.translators["rengong"].config["skip_other_on_success"]
|
||
)
|
||
|
||
if (not skip_other_on_success) and ("premt" in self.translators):
|
||
contentraw = self.analyzecontent(text_solved, optimization_params)
|
||
try:
|
||
maybehaspremt = self.translators["premt"].translate(contentraw)
|
||
except:
|
||
print_exc()
|
||
other = list(set(maybehaspremt.keys()) - set(fix_rank))
|
||
idx = fix_rank.index("premt")
|
||
fix_rank = fix_rank[:idx] + other + fix_rank[idx + 1 :]
|
||
|
||
real_fix_rank = []
|
||
if skip_other_on_success:
|
||
real_fix_rank.append("rengong")
|
||
else:
|
||
for engine in fix_rank:
|
||
if (engine not in self.translators) and (engine not in maybehaspremt):
|
||
continue
|
||
real_fix_rank.append(engine)
|
||
|
||
if len(real_fix_rank) == 0:
|
||
return _showrawfunction()
|
||
|
||
if waitforresultcallbackengine:
|
||
if waitforresultcallbackengine in real_fix_rank:
|
||
real_fix_rank = [waitforresultcallbackengine]
|
||
else:
|
||
waitforresultcallbackengine = None
|
||
|
||
usefultranslators = real_fix_rank.copy()
|
||
if globalconfig["fix_translate_rank"] and (not waitforresultcallback):
|
||
_showrawfunction = functools.partial(
|
||
self._delaypreparefixrank, _showrawfunction, real_fix_rank
|
||
)
|
||
if not globalconfig["refresh_on_get_trans"]:
|
||
_showrawfunction()
|
||
_showrawfunction = None
|
||
for engine in real_fix_rank:
|
||
if engine in globalconfig["fanyi"]:
|
||
_colork = engine
|
||
else:
|
||
_colork = "premt"
|
||
self.create_translate_task(
|
||
currentsignature,
|
||
usefultranslators,
|
||
_colork,
|
||
optimization_params,
|
||
_showrawfunction,
|
||
text,
|
||
text_solved,
|
||
waitforresultcallback,
|
||
is_auto_run,
|
||
result=maybehaspremt.get(engine),
|
||
)
|
||
return True
|
||
|
||
def analyzecontent(self, text_solved, optimization_params):
|
||
for _ in optimization_params:
|
||
if isinstance(_, dict):
|
||
_gpt_dict = _.get("gpt_dict", None)
|
||
if _gpt_dict is None:
|
||
continue
|
||
return _.get("gpt_dict_origin")
|
||
return text_solved
|
||
|
||
def _delaypreparefixrank(self, _showrawfunction, real_fix_rank):
|
||
_showrawfunction()
|
||
for engine in real_fix_rank:
|
||
colorx = globalconfig["fanyi"].get(engine, globalconfig["fanyi"]["premt"])
|
||
displayreskwargs = dict(
|
||
name="",
|
||
color=colorx["color"],
|
||
res="",
|
||
iter_context=(1, engine),
|
||
)
|
||
self.translation_ui.displayres.emit(displayreskwargs)
|
||
|
||
def create_translate_task(
|
||
self,
|
||
currentsignature,
|
||
usefultranslators,
|
||
engine,
|
||
optimization_params,
|
||
_showrawfunction,
|
||
text,
|
||
text_solved,
|
||
waitforresultcallback,
|
||
is_auto_run,
|
||
result=None,
|
||
):
|
||
callback = partial(
|
||
self.GetTranslationCallback,
|
||
usefultranslators,
|
||
waitforresultcallback,
|
||
engine,
|
||
currentsignature,
|
||
optimization_params,
|
||
_showrawfunction,
|
||
text,
|
||
)
|
||
task = (
|
||
callback,
|
||
text_solved,
|
||
waitforresultcallback,
|
||
is_auto_run,
|
||
optimization_params,
|
||
)
|
||
if result:
|
||
# 预翻译
|
||
callback(result, 1)
|
||
callback(result, 2)
|
||
else:
|
||
|
||
self.translators[engine].gettask(task)
|
||
|
||
def GetTranslationCallback(
|
||
self,
|
||
usefultranslators,
|
||
waitforresultcallback,
|
||
classname,
|
||
currentsignature,
|
||
optimization_params,
|
||
_showrawfunction,
|
||
contentraw,
|
||
res,
|
||
iter_res_status,
|
||
iserror=False,
|
||
):
|
||
with self.gettranslatelock:
|
||
if classname in usefultranslators:
|
||
usefultranslators.remove(classname)
|
||
if (
|
||
waitforresultcallback is None
|
||
and currentsignature != self.currentsignature
|
||
):
|
||
return
|
||
|
||
safe_callback = (
|
||
waitforresultcallback if waitforresultcallback else lambda _: 1
|
||
)
|
||
apiname = _TR(dynamicapiname(classname))
|
||
if iserror:
|
||
if currentsignature == self.currentsignature:
|
||
self.translation_ui.displaystatus.emit(
|
||
apiname + " " + res,
|
||
True,
|
||
False,
|
||
)
|
||
if len(usefultranslators) == 0:
|
||
safe_callback("")
|
||
return
|
||
|
||
res = self.solveaftertrans(res, optimization_params)
|
||
if not res:
|
||
if len(usefultranslators) == 0:
|
||
safe_callback("")
|
||
return
|
||
needshowraw = (
|
||
_showrawfunction
|
||
and self.refresh_on_get_trans_signature != _showrawfunction
|
||
)
|
||
if needshowraw:
|
||
self.refresh_on_get_trans_signature = _showrawfunction
|
||
_showrawfunction()
|
||
if (currentsignature == self.currentsignature) and (
|
||
iter_res_status in (0, 1)
|
||
):
|
||
displayreskwargs = dict(
|
||
name=apiname,
|
||
color=globalconfig["fanyi"][classname]["color"],
|
||
res=res,
|
||
iter_context=(iter_res_status, classname),
|
||
)
|
||
self.translation_ui.displayres.emit(displayreskwargs)
|
||
if iter_res_status in (0, 2): # 0为普通,1为iter,2为iter终止
|
||
|
||
self.transhis.getnewtranssignal.emit(apiname, res)
|
||
if not waitforresultcallback:
|
||
if (
|
||
globalconfig["read_trans"]
|
||
and globalconfig["read_translator2"] == classname
|
||
):
|
||
self.currentread = res
|
||
self.readcurrent()
|
||
|
||
if globalconfig["textoutput_trans"]:
|
||
self.dispatchoutputer(res)
|
||
try:
|
||
self.textsource.sqlqueueput((contentraw, classname, res))
|
||
except:
|
||
pass
|
||
try:
|
||
gobject.edittrans.dispatch.emit(classname, res)
|
||
except:
|
||
pass
|
||
if len(self.currenttranslate):
|
||
self.currenttranslate += "\n"
|
||
self.currenttranslate += res
|
||
safe_callback(res)
|
||
|
||
def __usewhich(self):
|
||
|
||
try:
|
||
for _ in (0,):
|
||
gameuid = self.gameuid
|
||
if not gameuid:
|
||
break
|
||
if savehook_new_data[gameuid]["tts_follow_default"]:
|
||
break
|
||
return savehook_new_data[gameuid]
|
||
except:
|
||
pass
|
||
return globalconfig["ttscommon"]
|
||
|
||
def ttsrepair(self, text, usedict):
|
||
if usedict["tts_repair"]:
|
||
text = parsemayberegexreplace(usedict["tts_repair_regex"], text)
|
||
return text
|
||
|
||
def matchwhich(self, dic: dict, res: str):
|
||
|
||
for item in dic:
|
||
if item["regex"]:
|
||
retext = safe_escape(item["key"])
|
||
if item["condition"] == 1:
|
||
if re.search(retext, res):
|
||
return item
|
||
elif item["condition"] == 0:
|
||
if re.match(retext, res) or re.search(retext + "$", res):
|
||
# 用^xxx|xxx$有可能有点危险
|
||
return item
|
||
else:
|
||
if item["condition"] == 1:
|
||
if (
|
||
isascii(res)
|
||
and isascii(item["key"])
|
||
and (" " not in item["key"])
|
||
): # 目标可能有空格
|
||
resx = res.split(" ")
|
||
for i in range(len(resx)):
|
||
if resx[i] == item["key"]:
|
||
return item
|
||
else:
|
||
if item["key"] in res:
|
||
return item
|
||
elif item["condition"] == 0:
|
||
if res.startswith(item["key"]) or res.endswith(item["key"]):
|
||
return item
|
||
return None
|
||
|
||
def ttsskip(self, text, usedict) -> dict:
|
||
if usedict["tts_skip"]:
|
||
return self.matchwhich(usedict["tts_skip_regex"], text)
|
||
return None
|
||
|
||
@threader
|
||
def __readcurrent(self, text1, text2, force=False):
|
||
if (not force) and (not globalconfig["autoread"]):
|
||
return
|
||
matchitme = self.ttsskip(text1, self.__usewhich())
|
||
reader = None
|
||
if matchitme is None:
|
||
reader = self.reader
|
||
else:
|
||
target = matchitme.get("target", "default")
|
||
if target == "default":
|
||
reader = self.reader
|
||
elif target == "skip":
|
||
if not force:
|
||
return
|
||
reader = self.reader
|
||
else:
|
||
engine, voice, _ = target
|
||
key = str((engine, voice)) # voice可能是list,无法hash
|
||
reader = self.specialreaders.get(key, None)
|
||
if reader == -1:
|
||
reader = self.reader
|
||
elif reader is None:
|
||
try:
|
||
reader = self.loadreader(engine, privateconfig={"voice": voice})
|
||
self.specialreaders[key] = reader
|
||
except:
|
||
reader = self.reader
|
||
self.specialreaders[key] = -1
|
||
if reader is None:
|
||
return
|
||
if text2 is None:
|
||
text2 = self.ttsrepair(text1, self.__usewhich())
|
||
self.audioplayer.timestamp = uuid.uuid4()
|
||
reader.read(text2, force, self.audioplayer.timestamp)
|
||
|
||
@tryprint
|
||
def read_text(self, text):
|
||
if not self.reader:
|
||
return
|
||
self.audioplayer.timestamp = uuid.uuid4()
|
||
self.reader.read(text, True, self.audioplayer.timestamp)
|
||
|
||
def readcurrent(self, force=False, needresult=False):
|
||
if needresult:
|
||
text = self.ttsrepair(self.currentread, self.__usewhich())
|
||
self.__readcurrent(self.currentread, text, force)
|
||
return text
|
||
else:
|
||
self.__readcurrent(self.currentread, None, force)
|
||
|
||
def loadreader(self, use, privateconfig=None, init=True, uid=None):
|
||
aclass = importlib.import_module("tts." + use).TTS
|
||
if uid is None:
|
||
uid = uuid.uuid4()
|
||
obj = aclass(use, self.audioplayer.play, privateconfig, init, uid)
|
||
return obj
|
||
|
||
def __reader_usewhich(self):
|
||
|
||
for key in globalconfig["reader"]:
|
||
if globalconfig["reader"][key]["use"] and os.path.exists(
|
||
("./LunaTranslator/tts/" + key + ".py")
|
||
):
|
||
return key
|
||
return None
|
||
|
||
@threader
|
||
def startreader(self, use=None, checked=True, setting=False):
|
||
if setting:
|
||
if use != self.__reader_usewhich():
|
||
return
|
||
self.reader = None
|
||
self.reader_uid = uuid.uuid4()
|
||
self.reader = self.loadreader(use, uid=self.reader_uid)
|
||
return
|
||
if not checked:
|
||
self.reader = None
|
||
return
|
||
if use is None:
|
||
use = self.__reader_usewhich()
|
||
if not use:
|
||
self.reader = None
|
||
return
|
||
self.reader = None
|
||
self.reader_uid = uuid.uuid4()
|
||
self.reader = self.loadreader(use, uid=self.reader_uid)
|
||
|
||
@tryprint
|
||
def selectprocess(self, selectedp, title):
|
||
pids, pexe, hwnd = selectedp
|
||
if len(collect_running_pids(pids)) == 0:
|
||
return
|
||
if not title:
|
||
title = windows.GetWindowText(hwnd)
|
||
|
||
if not globalconfig["sourcestatus2"]["texthook"]["use"]:
|
||
return
|
||
gameuid, reflist = findgameuidofpath(pexe)
|
||
if gameuid:
|
||
if globalconfig["startgamenototop"] == False:
|
||
idx = reflist.index(gameuid)
|
||
reflist.insert(0, reflist.pop(idx))
|
||
else:
|
||
gameuid = find_or_create_uid(savehook_new_list, pexe, title)
|
||
savehook_new_list.insert(0, gameuid)
|
||
self.textsource.start(hwnd, pids, pexe, gameuid, autostart=False)
|
||
|
||
def starttextsource(self, use=None, checked=True):
|
||
self.translation_ui.showhidestate = False
|
||
self.translation_ui.refreshtooliconsignal.emit()
|
||
|
||
self.translation_ui.set_color_transparency()
|
||
self.translation_ui.adjustbuttons()
|
||
try:
|
||
self.settin_ui.selectbutton.setEnabled(
|
||
globalconfig["sourcestatus2"]["texthook"]["use"]
|
||
)
|
||
self.settin_ui.selecthookbutton.setEnabled(
|
||
globalconfig["sourcestatus2"]["texthook"]["use"]
|
||
)
|
||
except:
|
||
pass
|
||
self.textsource = None
|
||
if checked:
|
||
classes = {
|
||
"ocr": ocrtext,
|
||
"copy": copyboard,
|
||
"texthook": texthook,
|
||
"filetrans": filetrans,
|
||
}
|
||
if use is None:
|
||
use = list(
|
||
filter(
|
||
lambda _: globalconfig["sourcestatus2"][_]["use"],
|
||
classes.keys(),
|
||
)
|
||
)
|
||
use = None if len(use) == 0 else use[0]
|
||
if use is None:
|
||
return
|
||
else:
|
||
self.textsource = classes[use]()
|
||
|
||
@threader
|
||
def starthira(self, use=None, checked=True):
|
||
if checked:
|
||
hirasettingbase = globalconfig["hirasetting"]
|
||
_hira = None
|
||
for name in hirasettingbase:
|
||
if hirasettingbase[name]["use"]:
|
||
if (
|
||
os.path.exists("./LunaTranslator/hiraparse/" + name + ".py")
|
||
== False
|
||
):
|
||
continue
|
||
_hira = importlib.import_module("hiraparse." + name)
|
||
_hira = getattr(_hira, name)
|
||
break
|
||
|
||
try:
|
||
if _hira:
|
||
self.hira_ = _hira(name)
|
||
else:
|
||
self.hira_ = None
|
||
except:
|
||
print_exc()
|
||
self.hira_ = None
|
||
else:
|
||
self.hira_ = None
|
||
|
||
@threader
|
||
def startoutputer_re(self, klass):
|
||
self.outputers[klass].init()
|
||
|
||
@threader
|
||
def startoutputer(self):
|
||
for classname in globalconfig["textoutputer"]:
|
||
if not os.path.exists("./LunaTranslator/textoutput/" + classname + ".py"):
|
||
continue
|
||
aclass = importlib.import_module("textoutput." + classname).Outputer
|
||
self.outputers[classname] = aclass(classname)
|
||
|
||
def dispatchoutputer(self, text):
|
||
for _, kls in self.outputers.items():
|
||
if kls.config["use"]:
|
||
kls.puttask(text)
|
||
|
||
def fanyiinitmethod(self, classname):
|
||
try:
|
||
which = translate_exits(classname, which=True)
|
||
if which is None:
|
||
return None
|
||
if which == 0:
|
||
aclass = importlib.import_module("translator." + classname).TS
|
||
elif which == 1:
|
||
aclass = importlib.import_module("userconfig.copyed." + classname).TS
|
||
return aclass(classname)
|
||
except Exception as e:
|
||
self.displayinfomessage(
|
||
dynamicapiname(classname) + " import failed : " + str(stringfyerror(e)),
|
||
"<msg_error_not_refresh>",
|
||
)
|
||
raise e
|
||
|
||
@threader
|
||
def prepare(self, now=None, _=None):
|
||
self.commonloader("fanyi", self.translators, self.fanyiinitmethod, now)
|
||
|
||
def commonloader(self, fanyiorcishu, dictobject, initmethod, _type=None):
|
||
if _type:
|
||
self.commonloader_warp(fanyiorcishu, dictobject, initmethod, _type)
|
||
else:
|
||
for key in globalconfig[fanyiorcishu]:
|
||
self.commonloader_warp(fanyiorcishu, dictobject, initmethod, key)
|
||
|
||
@threader
|
||
def commonloader_warp(self, fanyiorcishu, dictobject, initmethod, _type):
|
||
try:
|
||
if _type in dictobject:
|
||
try:
|
||
dictobject[_type].notifyqueuforend()
|
||
except:
|
||
pass
|
||
dictobject.pop(_type)
|
||
use = globalconfig[fanyiorcishu][_type]["use"]
|
||
rankkeys = {"cishu": "cishuvisrank", "fanyi": "fix_translate_rank_rank"}
|
||
rankkey = rankkeys.get(fanyiorcishu)
|
||
if use:
|
||
if _type not in globalconfig[rankkey]:
|
||
globalconfig[rankkey].append(_type)
|
||
else:
|
||
if _type in globalconfig[rankkey]:
|
||
globalconfig[rankkey].remove(_type)
|
||
return
|
||
item = initmethod(_type)
|
||
if item:
|
||
dictobject[_type] = item
|
||
except:
|
||
if _type in globalconfig[rankkey]:
|
||
globalconfig[rankkey].remove(_type)
|
||
print_exc()
|
||
|
||
@threader
|
||
def startxiaoxueguan(self, type_=None, _=None):
|
||
self.commonloader("cishu", self.cishus, self.cishuinitmethod, type_)
|
||
|
||
def cishuinitmethod(self, type_):
|
||
|
||
aclass = importlib.import_module("cishu." + type_)
|
||
aclass = getattr(aclass, type_)
|
||
return aclass(type_)
|
||
|
||
def maybesetedittext(self, text):
|
||
if self.edittextui:
|
||
try:
|
||
self.edittextui.getnewsentencesignal.emit(text)
|
||
except:
|
||
print_exc()
|
||
self.edittextui_cached = text
|
||
|
||
def createedittextui(self):
|
||
try:
|
||
self.edittextui = edittext(self.commonstylebase, self.edittextui_cached)
|
||
if self.edittextui:
|
||
self.edittextui.show()
|
||
except:
|
||
print_exc()
|
||
|
||
def safecloseattachprocess(self):
|
||
|
||
if self.AttachProcessDialog:
|
||
try:
|
||
self.AttachProcessDialog.close()
|
||
except:
|
||
pass
|
||
|
||
def createattachprocess(self):
|
||
try:
|
||
self.AttachProcessDialog = AttachProcessDialog(
|
||
self.commonstylebase, self.selectprocess, self.hookselectdialog
|
||
)
|
||
if self.AttachProcessDialog:
|
||
self.AttachProcessDialog.show()
|
||
except:
|
||
print_exc()
|
||
|
||
def setdarkandbackdrop(self, widget, dark):
|
||
if self.__dontshowintaborsetbackdrop(widget):
|
||
return
|
||
winsharedutils.SetTheme(
|
||
int(widget.winId()), dark, globalconfig["WindowBackdrop"]
|
||
)
|
||
|
||
@threader
|
||
def clickwordcallback(self, word, append=False):
|
||
if globalconfig["usewordorigin"] == False:
|
||
word = word["orig"]
|
||
else:
|
||
word = word.get("origorig", word["orig"])
|
||
|
||
if globalconfig["usecopyword"]:
|
||
winsharedutils.clipboard_set(
|
||
(winsharedutils.clipboard_get() + word) if append else word
|
||
)
|
||
if globalconfig["usesearchword"]:
|
||
self.searchwordW.search_word.emit(word, append)
|
||
|
||
def __dontshowintaborsetbackdrop(self, widget):
|
||
window_flags = widget.windowFlags()
|
||
if (
|
||
Qt.WindowType.FramelessWindowHint & window_flags
|
||
== Qt.WindowType.FramelessWindowHint
|
||
):
|
||
return True
|
||
return False
|
||
|
||
def setshowintab_checked(self, widget):
|
||
try:
|
||
self.translation_ui
|
||
except:
|
||
return
|
||
if widget == self.translation_ui:
|
||
winsharedutils.showintab(
|
||
int(widget.winId()), globalconfig["showintab"], True
|
||
)
|
||
return
|
||
if self.__dontshowintaborsetbackdrop(widget):
|
||
return
|
||
if isinstance(widget, (QMenu, QFrame)):
|
||
return
|
||
if (
|
||
isinstance(widget, QWidget)
|
||
and widget.parent() is None
|
||
and len(widget.children()) == 0
|
||
):
|
||
# combobox的下拉框,然后这个widget会迅速销毁,会导致任务栏闪一下。没别的办法了姑且这样过滤一下
|
||
return
|
||
winsharedutils.showintab(
|
||
int(widget.winId()), globalconfig["showintab_sub"], False
|
||
)
|
||
|
||
def inittray(self):
|
||
self.tray = QSystemTrayIcon()
|
||
self.tray.setIcon(getExeIcon(getcurrexe()))
|
||
trayMenu = LMenu(self.commonstylebase)
|
||
showAction = LAction("显示", trayMenu)
|
||
showAction.triggered.connect(self.translation_ui.show_)
|
||
settingAction = LAction(qtawesome.icon("fa.gear"), "设置", trayMenu)
|
||
settingAction.triggered.connect(self.settin_ui.showsignal)
|
||
quitAction = LAction(qtawesome.icon("fa.times"), "退出", trayMenu)
|
||
quitAction.triggered.connect(self.translation_ui.close)
|
||
trayMenu.addAction(showAction)
|
||
trayMenu.addAction(settingAction)
|
||
trayMenu.addSeparator()
|
||
trayMenu.addAction(quitAction)
|
||
trayMenu.addAction(showAction)
|
||
trayMenu.addAction(settingAction)
|
||
trayMenu.addSeparator()
|
||
trayMenu.addAction(quitAction)
|
||
self.tray.setContextMenu(trayMenu)
|
||
self.tray.activated.connect(self.leftclicktray)
|
||
self.tray.messageClicked.connect(self.triggertoupdate)
|
||
self.tray.show()
|
||
|
||
def triggertoupdate(self):
|
||
self.istriggertoupdate = True
|
||
winsharedutils.dispatchcloseevent()
|
||
|
||
def leftclicktray(self, reason):
|
||
if reason == QSystemTrayIcon.ActivationReason.Trigger:
|
||
self.translation_ui.showhideui()
|
||
|
||
def showtraymessage(self, title, message):
|
||
self.tray.showMessage(title, message, getExeIcon(getcurrexe()))
|
||
|
||
def destroytray(self):
|
||
self.tray.hide()
|
||
self.tray = None
|
||
|
||
def setshowintab(self):
|
||
for widget in QApplication.topLevelWidgets():
|
||
self.setshowintab_checked(widget)
|
||
|
||
def setcommonstylesheet(self):
|
||
|
||
dark = nowisdark()
|
||
darklight = ["light", "dark"][dark]
|
||
|
||
self.currentisdark = dark
|
||
|
||
for widget in QApplication.topLevelWidgets():
|
||
self.setdarkandbackdrop(widget, dark)
|
||
style = ""
|
||
for _ in (0,):
|
||
try:
|
||
idx = globalconfig[darklight + "theme"] - int(not dark)
|
||
if idx == -1:
|
||
break
|
||
_fn = static_data["themes"][darklight][idx]["file"]
|
||
|
||
if _fn.endswith(".py"):
|
||
style = importlib.import_module(
|
||
"files.LunaTranslator_qss." + _fn[:-3].replace("/", ".")
|
||
).stylesheet()
|
||
elif _fn.endswith(".qss"):
|
||
with open(
|
||
"./files/LunaTranslator_qss/{}".format(_fn),
|
||
"r",
|
||
) as ff:
|
||
style = ff.read()
|
||
except:
|
||
print_exc()
|
||
style += (
|
||
"*{font: %spt '" % (globalconfig["settingfontsize"])
|
||
+ (globalconfig["settingfonttype"])
|
||
+ "' ; }"
|
||
)
|
||
style += """
|
||
QListWidget {{
|
||
font:{fontsize}pt {fonttype}; }}
|
||
""".format(
|
||
fontsize=globalconfig["settingfontsize"] + 4,
|
||
fonttype=globalconfig["settingfonttype"],
|
||
)
|
||
self.commonstylebase.setStyleSheet(style)
|
||
font = QFont()
|
||
font.setFamily(globalconfig["settingfonttype"])
|
||
font.setPointSizeF(globalconfig["settingfontsize"])
|
||
QApplication.instance().setFont(font)
|
||
|
||
def get_font_default(self, lang: str, issetting: bool) -> str:
|
||
# global font_default_used
|
||
# if lang in font_default_used.keys():
|
||
# return font_default_used[lang]
|
||
|
||
t = "setting_font_type_default" if issetting else "font_type_default"
|
||
l = lang if lang in static_data[t].keys() else "default"
|
||
|
||
font_default = ""
|
||
|
||
if isinstance(static_data[t][l], list):
|
||
fontlist = static_data[t][l]
|
||
elif isinstance(static_data[t][l], dict):
|
||
if get_platform() == "xp":
|
||
target = "xp"
|
||
else:
|
||
target = "normal"
|
||
fontlist = static_data[t][l].get(target, [])
|
||
else:
|
||
fontlist = []
|
||
is_font_installed = lambda font: QFont(font).exactMatch()
|
||
for font in fontlist:
|
||
if is_font_installed(font):
|
||
font_default = font
|
||
break
|
||
if font_default == "":
|
||
font_default = QFontDatabase.systemFont(
|
||
QFontDatabase.SystemFont.GeneralFont
|
||
).family()
|
||
|
||
# font_default_used["lang"] = font_default
|
||
return font_default
|
||
|
||
def set_font_default(self, lang: str, fonttype: str) -> None:
|
||
globalconfig[fonttype] = self.get_font_default(
|
||
lang, True if fonttype == "settingfonttype" else False
|
||
)
|
||
|
||
def parsedefaultfont(self):
|
||
for k in ["fonttype", "fonttype2", "settingfonttype"]:
|
||
if globalconfig[k] == "":
|
||
l = "ja" if k == "fonttype" else getlanguse()
|
||
self.set_font_default(l, k)
|
||
# globalconfig[k] = QFontDatabase.systemFont(
|
||
# QFontDatabase.SystemFont.GeneralFont
|
||
# ).family()
|
||
|
||
def loadui(self):
|
||
self.installeventfillter()
|
||
self.parsedefaultfont()
|
||
self.loadmetadatas()
|
||
self.translation_ui = TranslatorWindow()
|
||
winsharedutils.showintab(
|
||
int(self.translation_ui.winId()), globalconfig["showintab"], True
|
||
)
|
||
self.translation_ui.show()
|
||
self.translation_ui.aftershowdosomething()
|
||
self.mainuiloadafter()
|
||
|
||
def mainuiloadafter(self):
|
||
self.__checkmutethread()
|
||
self.safeloadprocessmodels()
|
||
self.prepare()
|
||
self.startxiaoxueguan()
|
||
self.starthira()
|
||
self.startoutputer()
|
||
|
||
class commonstylebase(QWidget):
|
||
setstylesheetsignal = pyqtSignal()
|
||
|
||
def __init__(__, parent) -> None:
|
||
super().__init__(parent)
|
||
__.setstylesheetsignal.connect(self.setcommonstylesheet)
|
||
|
||
self.commonstylebase = commonstylebase(self.translation_ui)
|
||
self.setcommonstylesheet()
|
||
self.settin_ui = Setting(self.commonstylebase)
|
||
self.transhis = transhist(self.commonstylebase)
|
||
self.startreader()
|
||
self.searchwordW = searchwordW(self.commonstylebase)
|
||
self.hookselectdialog = hookselect(self.commonstylebase)
|
||
self.starttextsource()
|
||
threading.Thread(
|
||
target=minmaxmoveobservefunc, args=(self.translation_ui,)
|
||
).start()
|
||
self.messagecallback__ = winsharedutils.globalmessagelistener_cb(
|
||
self.messagecallback
|
||
)
|
||
winsharedutils.globalmessagelistener(self.messagecallback__)
|
||
self.inittray()
|
||
self.playtimemanager = playtimemanager()
|
||
self.__count = 0
|
||
|
||
@threader
|
||
def __checkmutethread(self):
|
||
while True:
|
||
time.sleep(0.5)
|
||
if not self.hwnd:
|
||
continue
|
||
pid = windows.GetWindowThreadProcessId(self.hwnd)
|
||
if not pid:
|
||
continue
|
||
_mute = winsharedutils.GetProcessMute(pid)
|
||
if self.translation_ui.processismuteed != _mute:
|
||
self.translation_ui.processismuteed = _mute
|
||
self.translation_ui.refreshtooliconsignal.emit()
|
||
|
||
def openlink(self, file):
|
||
if file.startswith("http") and checkisusingwine():
|
||
self.translation_ui.displaylink.emit(file)
|
||
return
|
||
return os.startfile(file)
|
||
|
||
def messagecallback(self, msg, param):
|
||
if msg == 0:
|
||
if globalconfig["darklight2"] == 0:
|
||
if self.__count % 2:
|
||
self.commonstylebase.setstylesheetsignal.emit()
|
||
self.__count += 1
|
||
elif msg == 1:
|
||
if bool(param):
|
||
self.translation_ui.settop()
|
||
else:
|
||
if not globalconfig["keepontop"]:
|
||
self.translation_ui.canceltop()
|
||
elif msg == 2:
|
||
self.translation_ui.closesignal.emit()
|
||
|
||
def _dowhenwndcreate(self, obj):
|
||
hwnd = obj.winId()
|
||
if not hwnd: # window create/destroy,when destroy winId is None
|
||
return
|
||
windows.SetProp(
|
||
int(obj.winId()),
|
||
"Magpie.ToolWindow",
|
||
windows.HANDLE(1),
|
||
)
|
||
winsharedutils.setdwmextendframe(int(hwnd))
|
||
if self.currentisdark is not None:
|
||
self.setdarkandbackdrop(obj, self.currentisdark)
|
||
self.setshowintab_checked(obj)
|
||
|
||
def installeventfillter(self):
|
||
class WindowEventFilter(QObject):
|
||
def eventFilter(_, obj: QObject, event: QEvent):
|
||
if event.type() == QEvent.Type.LanguageChange:
|
||
if "updatelangtext" in dir(obj):
|
||
obj.updatelangtext()
|
||
elif event.type() == QEvent.Type.WinIdChange:
|
||
self._dowhenwndcreate(obj)
|
||
|
||
return False
|
||
|
||
self.currentisdark = None
|
||
self.__filter = WindowEventFilter() # keep ref
|
||
QApplication.instance().installEventFilter(self.__filter)
|
||
|
||
def loadmetadatas(self):
|
||
|
||
for k in globalconfig["metadata"]:
|
||
try:
|
||
targetmod[k] = importlib.import_module("metadata." + k).searcher(k)
|
||
except:
|
||
print_exc()
|
||
|
||
@tryprint
|
||
def urlprotocol(self):
|
||
|
||
key = winreg.CreateKey(
|
||
winreg.HKEY_CURRENT_USER, r"Software\Classes\lunatranslator"
|
||
)
|
||
winreg.SetValue(key, None, winreg.REG_SZ, "URL:lunatranslator")
|
||
winreg.SetValueEx(key, r"URL Protocol", 0, winreg.REG_SZ, "")
|
||
keysub = winreg.CreateKey(
|
||
winreg.HKEY_CURRENT_USER,
|
||
r"Software\Classes\lunatranslator\shell\open\command",
|
||
)
|
||
command = '"{}" --URLProtocol "%1"'.format(getcurrexe())
|
||
winreg.SetValue(keysub, r"", winreg.REG_SZ, command)
|