AzurLaneAutoScript/dev_tools/button_extract.py
2024-06-28 03:23:28 +08:00

227 lines
7.8 KiB
Python

import os
import imageio
import numpy as np
from tqdm.contrib.concurrent import process_map
from module.base.utils import get_bbox, get_color, image_size, load_image
from module.config.config_manual import ManualConfig as AzurLaneConfig
from module.config.server import VALID_SERVER
from module.logger import logger
MODULE_FOLDER = './module'
BUTTON_FILE = 'assets.py'
IMPORT_EXP = """
from module.base.button import Button
from module.base.template import Template
# This file was automatically generated by dev_tools/button_extract.py.
# Don't modify it manually.
"""
IMPORT_EXP = IMPORT_EXP.strip().split('\n') + ['']
class ImageExtractor:
def __init__(self, module, file):
"""
Args:
module(str):
file(str): xxx.png or xxx.gif
"""
self.module = module
self.name, self.ext = os.path.splitext(file)
self.area, self.color, self.button, self.file = {}, {}, {}, {}
for server in VALID_SERVER:
self.load(server)
def get_file(self, genre='', server='cn'):
for ext in ['.png', '.gif']:
file = f'{self.name}.{genre}{ext}' if genre else f'{self.name}{ext}'
file = os.path.join(AzurLaneConfig.ASSETS_FOLDER, server, self.module, file).replace('\\', '/')
if os.path.exists(file):
return file
ext = '.png'
file = f'{self.name}.{genre}{ext}' if genre else f'{self.name}{ext}'
file = os.path.join(AzurLaneConfig.ASSETS_FOLDER, server, self.module, file).replace('\\', '/')
return file
def extract(self, file):
if os.path.splitext(file)[1] == '.gif':
# In a gif Button, use the first image.
bbox = None
mean = None
for image in imageio.mimread(file):
image = image[:, :, :3] if len(image.shape) == 3 else image
new_bbox, new_mean = self._extract(image, file)
if bbox is None:
bbox = new_bbox
elif bbox != new_bbox:
logger.warning(f'{file} has multiple different bbox, this will cause unexpected behaviour')
if mean is None:
mean = new_mean
return bbox, mean
else:
image = load_image(file)
return self._extract(image, file)
@staticmethod
def _extract(image, file):
size = image_size(image)
if size != (1280, 720):
logger.warning(f'{file} has wrong resolution: {size}')
bbox = get_bbox(image)
mean = get_color(image=image, area=bbox)
mean = tuple(np.rint(mean).astype(int))
return bbox, mean
def load(self, server='cn'):
file = self.get_file(server=server)
if os.path.exists(file):
area, color = self.extract(file)
button = area
override = self.get_file('AREA', server=server)
if os.path.exists(override):
area, _ = self.extract(override)
override = self.get_file('COLOR', server=server)
if os.path.exists(override):
_, color = self.extract(override)
override = self.get_file('BUTTON', server=server)
if os.path.exists(override):
button, _ = self.extract(override)
self.area[server] = area
self.color[server] = color
self.button[server] = button
self.file[server] = file
else:
logger.attr(server, f'{self.name} not found, use cn server assets')
self.area[server] = self.area['cn']
self.color[server] = self.color['cn']
self.button[server] = self.button['cn']
self.file[server] = self.file['cn']
@property
def expression(self):
return '%s = Button(area=%s, color=%s, button=%s, file=%s)' % (
self.name, self.area, self.color, self.button, self.file)
class TemplateExtractor(ImageExtractor):
# def __init__(self, module, file, config):
# """
# Args:
# module(str):
# file(str): xxx.png
# config(AzurLaneConfig):
# """
# self.module = module
# self.file = file
# self.config = config
@staticmethod
def extract(file):
image = load_image(file)
bbox = get_bbox(image)
mean = get_color(image=image, area=bbox)
mean = tuple(np.rint(mean).astype(int))
return bbox, mean
@property
def expression(self):
return '%s = Template(file=%s)' % (
self.name, self.file)
# return '%s = Template(area=%s, color=%s, button=%s, file=\'%s\')' % (
# self.name, self.area, self.color, self.button,
# self.config.ASSETS_FOLDER + '/' + self.module + '/' + self.name + '.png')
# class OcrExtractor(ImageExtractor):
# @property
# def expression(self):
# return '%s = OcrArea(area=%s, color=%s, button=%s, file=\'%s\')' % (
# self.name, self.area, self.color, self.button,
# self.config.ASSETS_FOLDER + '/' + self.module + '/' + self.name + '.png')
class ModuleExtractor:
def __init__(self, name):
self.name = name
self.folder = os.path.join(AzurLaneConfig.ASSETS_FOLDER, 'cn', name)
@staticmethod
def split(file):
name, ext = os.path.splitext(file)
name, sub = os.path.splitext(name)
return name, sub, ext
def is_base_image(self, file):
_, sub, _ = self.split(file)
return sub == ''
@property
def expression(self):
exp = []
for file in os.listdir(self.folder):
if file[0].isdigit():
continue
if file.startswith('TEMPLATE_'):
exp.append(TemplateExtractor(module=self.name, file=file).expression)
continue
# if file.startswith('OCR_'):
# exp.append(OcrExtractor(module=self.name, file=file, config=self.config).expression)
# continue
if self.is_base_image(file):
exp.append(ImageExtractor(module=self.name, file=file).expression)
continue
exp.sort()
logger.info('Module: %s(%s)' % (self.name, len(exp)))
exp = IMPORT_EXP + exp
return exp
def write(self):
folder = os.path.join(MODULE_FOLDER, self.name)
if not os.path.exists(folder):
os.mkdir(folder)
with open(os.path.join(folder, BUTTON_FILE), 'w', newline='') as f:
for text in self.expression:
f.write(text + '\n')
def worker(module):
me = ModuleExtractor(name=module)
me.write()
class AssetExtractor:
"""
Extract Asset to asset.py.
All the filename of assets should be in uppercase.
Asset name starts with digit will be ignore.
E.g. 2020XXXX.png.
Asset name starts with 'TEMPLATE_' will treat as template.
E.g. TEMPLATE_AMBUSH_EVADE_SUCCESS.png
> TEMPLATE_AMBUSH_EVADE_SUCCESS = Template(file='./assets/handler/TEMPLATE_AMBUSH_EVADE_SUCCESS.png')
Asset name starts other will treat as button.
E.g. GET_MISSION.png
> Button(area=(553, 482, 727, 539), color=(93, 142, 203), button=(553, 482, 727, 539), name='GET_MISSION')
Asset name like XXX.AREA.png, XXX.COLOR.png, XXX.BUTTON.png, will overwrite the attribute of XXX.png.
E.g. BATTLE_STATUS_S.BUTTON.png overwrites the attribute 'button' of BATTLE_STATUS_S
Asset name starts with 'OCR_' will be treat as button.
E.g. OCR_EXERCISE_TIMES.png.
"""
def __init__(self):
logger.info('Assets extract')
modules = [m for m in os.listdir(AzurLaneConfig.ASSETS_FOLDER + '/cn')
if os.path.isdir(os.path.join(AzurLaneConfig.ASSETS_FOLDER + '/cn', m))]
process_map(worker, modules)
if __name__ == '__main__':
ae = AssetExtractor()