mirror of
https://gitee.com/sui-feng-cb/AzurLaneAutoScript1
synced 2026-03-12 06:18:21 +08:00
157 lines
5.3 KiB
Python
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
|