StarRailCopilot/tasks/combat/prepare.py

137 lines
4.2 KiB
Python
Raw Normal View History

import re
2023-06-16 19:15:26 +00:00
from module.base.timer import Timer
2023-06-27 15:52:41 +00:00
from module.logger import logger
from module.ocr.ocr import Digit, DigitCounter
2023-06-15 14:12:47 +00:00
from tasks.base.ui import UI
from tasks.combat.assets.assets_combat_prepare import (
OCR_TRAILBLAZE_POWER,
2023-06-27 15:52:41 +00:00
OCR_WAVE_COST,
OCR_WAVE_COUNT,
WAVE_MINUS,
WAVE_PLUS
)
2023-06-15 14:12:47 +00:00
class TrailblazePowerOcr(DigitCounter):
def after_process(self, result):
result = super().after_process(result)
# The trailblaze power icon is recognized as 买
# OCR_TRAILBLAZE_POWER includes the icon because the length varies by value
result = re.sub(r'[买米装:()]', '', result)
return result
2023-06-15 14:12:47 +00:00
class CombatPrepare(UI):
2023-07-10 14:23:11 +00:00
# Current combat waves,
combat_waves = 1
# Limit combat runs, 0 means no limit.
combat_wave_limit = 0
combat_wave_done = 0
# E.g. 10, 30, 40
combat_wave_cost = 10
2023-06-27 15:52:41 +00:00
2023-06-15 14:12:47 +00:00
def combat_set_wave(self, count=6):
"""
Args:
count: 1 to 6
Pages:
in: COMBAT_PREPARE
2023-06-15 14:12:47 +00:00
"""
self.ui_ensure_index(
count, letter=Digit(OCR_WAVE_COUNT),
2023-06-15 14:12:47 +00:00
next_button=WAVE_PLUS, prev_button=WAVE_MINUS,
skip_first_screenshot=True
)
2023-06-16 19:15:26 +00:00
def combat_has_multi_wave(self) -> bool:
"""
2023-06-16 19:15:26 +00:00
If combat has waves to set.
Most dungeons can do 6 times at one time while bosses don't.
"""
return self.appear(WAVE_MINUS) or self.appear(WAVE_PLUS)
def combat_get_trailblaze_power(self, expect_reduce=False, skip_first_screenshot=True) -> int:
"""
Args:
expect_reduce: Current value is supposed to be lower than the previous.
skip_first_screenshot:
Pages:
in: COMBAT_PREPARE or COMBAT_REPEAT
"""
2023-06-16 19:15:26 +00:00
timeout = Timer(1, count=2).start()
while 1:
if skip_first_screenshot:
skip_first_screenshot = False
else:
self.device.screenshot()
current, _, total = TrailblazePowerOcr(OCR_TRAILBLAZE_POWER).ocr_single_line(self.device.image)
# Empty result
if total == 0:
continue
2023-06-16 19:15:26 +00:00
# Confirm if it is > 180, sometimes just OCR errors
if current > 240 and timeout.reached():
2023-06-16 19:15:26 +00:00
break
2023-08-27 16:41:49 +00:00
if expect_reduce and current >= self.config.stored.TrailblazePower.value:
2023-06-16 19:15:26 +00:00
continue
if current <= 240:
2023-06-16 19:15:26 +00:00
break
2023-08-27 16:41:49 +00:00
self.config.stored.TrailblazePower.value = current
return current
2023-06-27 15:52:41 +00:00
def combat_get_wave_cost(self, skip_first_screenshot=True):
2023-06-27 15:52:41 +00:00
"""
Get traiblaze power cost and set it to `combat_cost`
Returns:
2023-07-10 14:23:11 +00:00
int: E.g. 10, 30, 40
2023-06-27 15:52:41 +00:00
Pages:
in: COMBAT_PREPARE
"""
multi = self.combat_has_multi_wave()
logger.attr('CombatMultiWave', multi)
timeout = Timer(1.5, count=6).start()
while 1:
if skip_first_screenshot:
skip_first_screenshot = False
else:
self.device.screenshot()
2023-06-27 15:52:41 +00:00
cost = Digit(OCR_WAVE_COST).ocr_single_line(self.device.image)
if cost == 10:
if multi:
self.combat_wave_cost = cost
return cost
else:
logger.warning(f'Combat wave costs {cost} but does not has multiple waves')
self.combat_wave_cost = cost
return cost
elif cost in [30, 40]:
if multi:
logger.warning(f'Combat wave costs {cost} but has multiple waves, '
f'probably wave amount is preset')
self.combat_set_wave(1)
skip_first_screenshot = True
2023-06-27 15:52:41 +00:00
timeout.reset()
continue
else:
self.combat_wave_cost = cost
return cost
else:
logger.warning(f'Unexpected combat wave cost: {cost}')
continue
if multi:
cost = 10
else:
cost = 40
2023-06-27 15:52:41 +00:00
logger.warning(f'Get combat wave cost timeout, assume it costs {cost}')
self.combat_wave_cost = cost
return cost