2022-07-22 16:24:57 +08:00
|
|
|
from datetime import datetime
|
2022-07-20 22:18:00 +08:00
|
|
|
|
2022-07-17 01:17:27 +08:00
|
|
|
from module.base.button import ButtonGrid
|
2022-07-18 12:16:34 +08:00
|
|
|
from module.base.decorator import cached_property, Config
|
2022-08-23 00:39:38 +08:00
|
|
|
from module.base.utils import get_color
|
2023-02-04 17:36:52 +08:00
|
|
|
from module.exception import GameBugError
|
2022-07-16 19:13:58 +08:00
|
|
|
from module.logger import logger
|
2022-07-17 02:56:08 +08:00
|
|
|
from module.ocr.ocr import Duration
|
2022-07-17 22:58:29 +08:00
|
|
|
from module.research.assets import *
|
|
|
|
from module.research.ui import ResearchUI
|
2022-07-17 02:56:08 +08:00
|
|
|
|
|
|
|
OCR_QUEUE_REMAIN = Duration(QUEUE_REMAIN, letter=(255, 255, 255), threshold=128, name='OCR_QUEUE_REMAIN')
|
2022-07-16 19:13:58 +08:00
|
|
|
|
|
|
|
|
2022-07-17 22:58:29 +08:00
|
|
|
class ResearchQueue(ResearchUI):
|
2022-07-16 19:13:58 +08:00
|
|
|
def research_queue_add(self, skip_first_screenshot=True):
|
|
|
|
"""
|
2022-08-23 00:39:38 +08:00
|
|
|
Returns:
|
|
|
|
bool: True if success to add to queue,
|
|
|
|
False if project requirements not satisfied, can't be added to queue
|
|
|
|
|
2022-07-16 19:13:58 +08:00
|
|
|
Pages:
|
|
|
|
in: RESEARCH_QUEUE_ADD (is_in_research, DETAIL_NEXT)
|
|
|
|
out: is_in_research and stabled
|
|
|
|
"""
|
2022-07-18 01:48:51 +08:00
|
|
|
logger.hr('Research queue add')
|
2022-07-16 19:13:58 +08:00
|
|
|
# POPUP_CONFIRM has just been clicked in research_project_start()
|
2022-07-17 01:17:27 +08:00
|
|
|
self.popup_interval_clear()
|
2022-07-17 23:46:41 +08:00
|
|
|
self.interval_clear([RESEARCH_QUEUE_ADD])
|
2022-07-16 19:13:58 +08:00
|
|
|
while 1:
|
|
|
|
if skip_first_screenshot:
|
|
|
|
skip_first_screenshot = False
|
|
|
|
else:
|
|
|
|
self.device.screenshot()
|
|
|
|
|
2022-07-17 23:46:41 +08:00
|
|
|
# End
|
2022-08-23 00:39:38 +08:00
|
|
|
if self.is_research_stabled():
|
2022-07-17 23:46:41 +08:00
|
|
|
break
|
|
|
|
|
2022-08-23 00:39:38 +08:00
|
|
|
if self.appear(RESEARCH_QUEUE_ADD, offset=(20, 20), interval=5):
|
|
|
|
if self._research_queue_add_available():
|
|
|
|
self.device.click(RESEARCH_QUEUE_ADD)
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
logger.info('Project requirements not satisfied, cancel it')
|
|
|
|
self.research_detail_cancel()
|
|
|
|
return False
|
|
|
|
|
2022-07-22 13:43:39 +08:00
|
|
|
if self.handle_popup_confirm('RESEARCH_QUEUE'):
|
|
|
|
self.interval_reset(RESEARCH_QUEUE_ADD)
|
|
|
|
continue
|
|
|
|
|
2022-07-17 22:58:29 +08:00
|
|
|
self.ensure_research_center_stable()
|
2022-08-23 00:39:38 +08:00
|
|
|
return True
|
|
|
|
|
|
|
|
def _research_queue_add_available(self):
|
|
|
|
"""
|
|
|
|
Returns:
|
|
|
|
bool: True if able add to queue,
|
|
|
|
False if project requirements not satisfied, can't be added to queue
|
|
|
|
"""
|
|
|
|
# RESEARCH_QUEUE_ADD.area is the letter `Queue`
|
|
|
|
# RESEARCH_QUEUE_ADD.button is the entire clickable area of button
|
|
|
|
# Available: (90, 142, 203)
|
|
|
|
# Unavailable: (153, 160, 170)
|
|
|
|
r, g, b = get_color(self.device.image, RESEARCH_QUEUE_ADD.button)
|
|
|
|
if b - min(r, g) > 60:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
2022-07-17 01:17:27 +08:00
|
|
|
|
|
|
|
@cached_property
|
2022-07-18 12:16:34 +08:00
|
|
|
@Config.when(SERVER='en')
|
|
|
|
def queue_status_grids(self):
|
|
|
|
"""
|
|
|
|
Status icons on the left
|
|
|
|
"""
|
|
|
|
return ButtonGrid(
|
|
|
|
origin=(8, 259), delta=(0, 40.5), button_shape=(25, 25), grid_shape=(1, 5), name='QUEUE_STATUS')
|
|
|
|
|
2022-07-20 10:56:05 +08:00
|
|
|
@cached_property
|
|
|
|
@Config.when(SERVER='jp')
|
|
|
|
def queue_status_grids(self):
|
|
|
|
"""
|
|
|
|
Status icons on the left
|
|
|
|
"""
|
|
|
|
return ButtonGrid(
|
2022-07-22 14:18:10 +08:00
|
|
|
origin=(18, 259), delta=(0, 40.5), button_shape=(25, 25), grid_shape=(1, 5), name='QUEUE_STATUS')
|
2022-07-20 10:56:05 +08:00
|
|
|
|
2022-07-26 18:24:04 +08:00
|
|
|
@cached_property
|
|
|
|
@Config.when(SERVER='tw')
|
|
|
|
def queue_status_grids(self):
|
|
|
|
"""
|
|
|
|
Status icons on the left
|
|
|
|
"""
|
|
|
|
return ButtonGrid(
|
2022-07-29 02:00:31 +08:00
|
|
|
origin=(8, 259), delta=(0, 40.5), button_shape=(25, 25), grid_shape=(1, 5), name='QUEUE_STATUS')
|
2022-07-26 18:24:04 +08:00
|
|
|
|
2022-07-18 12:16:34 +08:00
|
|
|
@cached_property
|
|
|
|
@Config.when(SERVER=None)
|
2022-07-17 01:17:27 +08:00
|
|
|
def queue_status_grids(self):
|
|
|
|
"""
|
|
|
|
Status icons on the left
|
|
|
|
"""
|
|
|
|
return ButtonGrid(
|
2022-07-17 22:58:29 +08:00
|
|
|
origin=(18, 259), delta=(0, 40.5), button_shape=(25, 25), grid_shape=(1, 5), name='QUEUE_STATUS')
|
2022-07-17 01:17:27 +08:00
|
|
|
|
|
|
|
def _queue_status_detect(self, button):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
button: Button of status icon
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
str:
|
|
|
|
'finished': Orange ✓ surrounded by orange border
|
|
|
|
'running': Black ✓ surrounded by research progress, gray and blue
|
|
|
|
'waiting': Gray … surrounded by gray border
|
|
|
|
'empty': Black … surrounded by black border or just nothing
|
|
|
|
"""
|
|
|
|
center = button.crop((7, 7, 21, 21))
|
|
|
|
if self.image_color_count(center, color=(255, 158, 57), threshold=180, count=20):
|
|
|
|
return 'finished'
|
2022-12-29 22:17:17 +08:00
|
|
|
if self.image_color_count(center, color=(90, 97, 132), threshold=221, count=10):
|
2022-07-17 01:17:27 +08:00
|
|
|
return 'waiting'
|
|
|
|
if self.image_color_count(center, color=(24, 24, 41), threshold=221, count=10):
|
|
|
|
below = button.crop((7, 14, 21, 21))
|
|
|
|
if self.image_color_count(below, color=(24, 24, 41), threshold=221, count=10):
|
|
|
|
return 'running'
|
|
|
|
else:
|
|
|
|
return 'empty'
|
|
|
|
logger.warning(f'Unknown queue status from {button}, assume running')
|
|
|
|
return 'running'
|
|
|
|
|
2022-07-17 02:56:08 +08:00
|
|
|
def get_queue_slot(self):
|
2022-07-17 01:17:27 +08:00
|
|
|
"""
|
|
|
|
Returns:
|
|
|
|
int: Number of empty slots in queue
|
|
|
|
|
|
|
|
Pages:
|
|
|
|
in: is_in_queue
|
|
|
|
"""
|
|
|
|
status = [self._queue_status_detect(button) for button in self.queue_status_grids.buttons]
|
|
|
|
logger.info(f'Research queue: {status}')
|
2022-08-15 00:05:07 +08:00
|
|
|
status = status[::-1]
|
|
|
|
for index, s in enumerate(status):
|
|
|
|
if s != 'empty':
|
|
|
|
logger.attr('Research queue slot', index)
|
|
|
|
return index
|
|
|
|
index = len(status)
|
|
|
|
logger.attr('Research queue slot', index)
|
|
|
|
return index
|
2022-07-17 02:56:08 +08:00
|
|
|
|
2022-07-20 22:18:00 +08:00
|
|
|
def get_research_ended(self):
|
2022-07-17 02:56:08 +08:00
|
|
|
"""
|
|
|
|
Returns:
|
2022-07-20 22:18:00 +08:00
|
|
|
datetime: Time of the end of the first research in the queue.
|
2022-07-17 02:56:08 +08:00
|
|
|
|
|
|
|
Pages:
|
|
|
|
in: is_in_queue
|
2023-02-04 17:36:52 +08:00
|
|
|
|
|
|
|
Raises:
|
|
|
|
GameBugError:
|
2022-07-17 02:56:08 +08:00
|
|
|
"""
|
2023-02-04 17:36:52 +08:00
|
|
|
if self.image_color_count(QUEUE_REMAIN, color=(123, 125, 123), threshold=235, count=100):
|
|
|
|
logger.error('The first research of queue is not running,'
|
|
|
|
'probably a game bug from AL,'
|
|
|
|
'restart the game should fix it.')
|
|
|
|
raise GameBugError
|
2022-07-17 02:56:08 +08:00
|
|
|
if not self.image_color_count(QUEUE_REMAIN, color=(255, 255, 255), threshold=221, count=100):
|
|
|
|
logger.info('Research queue empty')
|
2022-07-25 20:55:25 +08:00
|
|
|
return datetime.now()
|
2022-07-17 02:56:08 +08:00
|
|
|
|
2022-07-22 16:24:57 +08:00
|
|
|
end_time = datetime.now() + OCR_QUEUE_REMAIN.ocr(self.device.image)
|
2022-07-20 22:18:00 +08:00
|
|
|
logger.info(f'The first research ended at: {end_time}')
|
|
|
|
return end_time
|