StarRailCopilot/tasks/base/main_page.py

207 lines
6.9 KiB
Python
Raw Normal View History

import re
2023-10-22 17:49:48 +00:00
import module.config.server as server
from module.config.server import VALID_LANG
from module.exception import RequestHumanTakeover, ScriptError
from module.logger import logger
2023-11-10 13:36:53 +00:00
from module.ocr.ocr import OcrWhiteLetterOnComplexBackground
from tasks.base.assets.assets_base_main_page import OCR_MAP_NAME, ROGUE_LEAVE_FOR_NOW
from tasks.base.assets.assets_base_page import CLOSE, MAP_EXIT
from tasks.base.page import Page, page_gacha, page_main
from tasks.base.popup import PopupHandler
from tasks.daily.assets.assets_daily_trial import START_TRIAL
from tasks.map.keywords import KEYWORDS_MAP_PLANE, MapPlane
2023-10-22 17:49:48 +00:00
class OcrPlaneName(OcrWhiteLetterOnComplexBackground):
def after_process(self, result):
# RobotSettlement1
result = re.sub(r'-[Ii1]$', '', result)
2023-09-27 15:49:56 +00:00
result = re.sub(r'I$', '', result)
result = re.sub(r'\d+$', '', result)
2023-09-25 04:24:29 +00:00
# Herta's OfficeY/
result = re.sub(r'Y/?$', '', result)
2023-09-27 15:49:56 +00:00
# Stargazer Navatia -> Stargazer Navalia
result = result.replace('avatia', 'avalia')
2023-10-02 09:15:03 +00:00
# DomainiRespite
result = result.replace('omaini', 'omain')
# Domain=Combat
result = result.replace('=', '')
2023-10-04 17:58:14 +00:00
# Domain--Occunrence
# Domain'--Occurence
# Domain-Qccurrence
result = result.replace('cunr', 'cur').replace('uren', 'urren').replace('Qcc', 'Occ')
2023-10-07 17:15:15 +00:00
# Domain-Elit
2023-10-18 13:01:48 +00:00
# Domain--Etite
2023-10-07 17:15:15 +00:00
result = re.sub(r'[Ee]lit$', 'Elite', result)
2023-10-18 13:01:48 +00:00
result = result.replace('tite', 'lite')
2023-10-04 17:58:14 +00:00
# 区域-战
result = re.sub(r'区域.*战$', '区域战斗', result)
2023-11-17 16:21:23 +00:00
# 区域-事
result = re.sub(r'区域.*事$', '区域事件', result)
2024-03-27 11:40:55 +00:00
# 区域-战
result = re.sub(r'区域.*交$', '区域交易', result)
2023-10-04 17:58:14 +00:00
# 区域-事伴, 区域-事祥
2023-10-20 18:06:23 +00:00
result = re.sub(r'事[伴祥]', '事件', result)
2023-10-04 17:58:14 +00:00
# 医域-战斗
result = result.replace('医域', '区域')
2023-10-20 18:06:23 +00:00
# 区域-战半, 区域-战头, 区域-战头书
2023-11-28 16:31:04 +00:00
result = re.sub(r'战[半头卒三]', '战斗', result)
2023-10-20 18:06:23 +00:00
# 区域一战斗
result = re.sub(r'区域[\-—-一=]', '区域-', result)
# 累塔的办公室
result = result.replace('累塔', '黑塔')
2023-09-27 15:49:56 +00:00
if '星港' in result:
result = '迴星港'
result = result.replace('太司', '太卜司')
2023-10-02 09:15:03 +00:00
result = result.replace(' ', '')
return super().after_process(result)
class MainPage(PopupHandler):
# Same as BigmapPlane class
# Current plane
plane: MapPlane = KEYWORDS_MAP_PLANE.Herta_ParlorCar
_lang_checked = False
2023-12-08 14:00:11 +00:00
_lang_check_success = True
2023-10-22 17:49:48 +00:00
def update_plane(self, lang=None) -> MapPlane | None:
"""
Pages:
in: page_main
"""
if lang is None:
lang = server.lang
ocr = OcrPlaneName(OCR_MAP_NAME, lang=lang)
result = ocr.ocr_single_line(self.device.image)
# Try to match
keyword = ocr._match_result(result, keyword_classes=MapPlane, lang=lang)
if keyword is not None:
self.plane = keyword
logger.attr('CurrentPlane', keyword)
return keyword
# Try to remove suffix
for suffix in range(1, 5):
keyword = ocr._match_result(result[:-suffix], keyword_classes=MapPlane, lang=lang)
if keyword is not None:
self.plane = keyword
logger.attr('CurrentPlane', keyword)
return keyword
return None
2023-10-22 17:49:48 +00:00
def check_lang_from_map_plane(self) -> str | None:
logger.info('check_lang_from_map_plane')
lang_unknown = self.config.Emulator_GameLanguage == 'auto'
if lang_unknown:
lang_list = VALID_LANG
else:
# Try current lang first
lang_list = [server.lang] + [lang for lang in VALID_LANG if lang != server.lang]
for lang in lang_list:
logger.info(f'Try ocr in lang {lang}')
2023-10-22 17:49:48 +00:00
keyword = self.update_plane(lang)
if keyword is not None:
logger.info(f'check_lang_from_map_plane matched lang: {lang}')
if lang_unknown or lang != server.lang:
self.config.Emulator_GameLanguage = lang
server.set_lang(lang)
2023-12-08 14:00:11 +00:00
MainPage._lang_checked = True
MainPage._lang_check_success = True
return lang
if lang_unknown:
logger.critical('Cannot detect in-game text language, please set it to 简体中文 or English')
raise RequestHumanTakeover
else:
logger.warning(f'Cannot detect in-game text language, assume current lang={server.lang} is correct')
2023-12-08 14:00:11 +00:00
MainPage._lang_checked = True
MainPage._lang_check_success = False
return server.lang
def handle_lang_check(self, page: Page):
"""
Args:
page:
Returns:
bool: If checked
"""
if MainPage._lang_checked:
return False
if page != page_main:
return False
self.check_lang_from_map_plane()
return True
def acquire_lang_checked(self):
"""
Returns:
bool: If checked
"""
if MainPage._lang_checked:
return False
logger.info('acquire_lang_checked')
try:
self.ui_goto(page_main)
except AttributeError:
logger.critical('Method ui_goto() not found, class MainPage must be inherited by class UI')
raise ScriptError
self.handle_lang_check(page=page_main)
return True
def ui_leave_special(self):
"""
Leave from:
- Rogue domains
- Character trials
Returns:
bool: If left a special plane
Pages:
in: Any
out: page_main
"""
if not self.appear(MAP_EXIT):
return False
logger.info('UI leave special')
skip_first_screenshot = True
clicked = False
while 1:
if skip_first_screenshot:
skip_first_screenshot = False
else:
self.device.screenshot()
# End
if clicked:
if self.appear(page_main.check_button):
logger.info(f'Leave to {page_main}')
break
if self.appear_then_click(MAP_EXIT, interval=2):
continue
if self.handle_popup_confirm():
continue
if self.match_template_color(START_TRIAL, interval=2):
logger.info(f'{START_TRIAL} -> {CLOSE}')
self.device.click(CLOSE)
clicked = True
continue
if self.handle_ui_close(page_gacha.check_button, interval=2):
continue
if self.appear_then_click(ROGUE_LEAVE_FOR_NOW, interval=2):
clicked = True
continue