1
0
mirror of https://gitee.com/sui-feng-cb/AzurLaneAutoScript1 synced 2026-03-12 06:18:21 +08:00
Files
AzurLaneAutoScript/module/island/transport.py
2025-11-24 14:19:17 +08:00

157 lines
5.3 KiB
Python

from datetime import datetime, timedelta
from module.base.utils import area_offset
from module.island.assets import *
from module.island.ui import IslandUI
from module.logger import logger
from module.map.map_grids import SelectedGrids
from module.ocr.ocr import Duration
class IslandTransport:
# index of transport commission
index: int
# If success to parse transport commission
valid: bool
# Duration to run this transport commission
duration: timedelta
# Status of transport commission
# Value: finished, running, pending, unknown
status: str
def __init__(self, main, index):
self.index = index
self.valid = True
self.duration = None
self.can_start = True
self.parse_transport(main)
if not self.valid:
self.can_start = False
self.create_time = datetime.now()
def parse_transport(self, main):
offset = (-20, -20, 20, 20)
delta = 176
self.offset = area_offset(offset, (0, delta * self.index))
# commission locked
lock_offset = area_offset(offset, (0, delta * (self.index - 1)))
if self.index >= 1 and main.appear(TRANSPORT_LOCKED, lock_offset):
self.valid = False
return
self.status = self.get_transport_status(main)
if self.status == 'unknown':
self.valid = False
return
elif self.status == 'pending':
button = OCR_TRANSPORT_TIME.move((0, self.offset[1] + 20))
ocr = Duration(button, lang='cnocr', letter=(207, 207, 207), name='OCR_TRANSPORT_TIME')
self.duration = ocr.ocr(main.device.image)
if not self.duration.total_seconds():
self.valid = False
return
if not main.match_template_color(TRANSPORT_START, offset=self.offset):
self.can_start = False
return
elif self.status == 'running':
self.can_start = False
button = OCR_TRANSPORT_TIME_REMAIN.move((0, self.offset[1] + 20))
ocr = Duration(button, name='OCR_TRANSPORT_TIME')
self.duration = ocr.ocr(main.device.image)
if not self.duration.total_seconds():
self.valid = False
return
elif self.status == 'finished':
self.can_start = False
def get_transport_status(self, main):
if main.appear(TRANSPORT_STATUS_PENDING, offset=self.offset):
return 'pending'
elif main.appear(TRANSPORT_STATUS_RUNNING, offset=self.offset):
return 'running'
elif main.appear(TRANSPORT_RECEIVE, offset=self.offset):
return 'finished'
else:
return 'unknown'
def convert_to_running(self):
if self.valid:
self.status = 'running'
self.create_time = datetime.now()
@property
def finish_time(self):
if self.valid:
return (self.create_time + self.duration).replace(microsecond=0)
else:
return None
def __str__(self):
if not self.valid:
return f'Index: {self.index} (Invalid)'
info = {'Index': self.index, 'Status': self.status}
if self.duration:
info['Duration'] = self.duration
info['can_start'] = self.can_start
info = ', '.join([f'{k}: {v}' for k, v in info.items()])
return info
class IslandTransportRun(IslandUI):
def _transport_detect(self):
"""
Get all commissions from self.device.image.
Returns:
SelectedGrids:
"""
logger.hr('Transport Commission detect')
commission = []
for index in range(3):
comm = IslandTransport(main=self, index=index)
logger.attr(f'Transport Commission', comm)
commission.append(comm)
return SelectedGrids(commission)
def transport_detect(self, trial=1, skip_first_screenshot=True):
"""
Get all transport missions from self.device.image.
Args:
trial (int): Retry if has one invalid commission.
skip_first_screenshot (bool):
Returns:
SelectedGrids:
"""
commissions = SelectedGrids([])
for _ in range(trial):
if skip_first_screenshot:
skip_first_screenshot = False
else:
self.device.screenshot()
commissions = self._transport_detect()
if commissions.count >= 2 and commissions.select(valid=False).count == 1:
logger.warning('Found 1 invalid commission, retry commission detect')
continue
else:
return commissions.select(valid=True)
logger.info('trials of transport commission detect exhausted, stop')
return commissions.select(valid=True)
def island_transport_run(self):
logger.hr('Island Transport Run', level=1)
future_finish = []
commissions = self.transport_detect(trial=2)
future_finish = sorted([f for f in commissions.select(status='running').get('finish_time') if f is not None])
logger.info(f'Transport finish: {[str(f) for f in future_finish]}')
if not len(future_finish):
logger.info('No island transport running')
return future_finish