2021-09-13 20:38:29 +08:00
|
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
|
2022-05-10 00:35:28 +08:00
|
|
|
|
import module.config.server as server
|
2021-09-13 20:38:29 +08:00
|
|
|
|
from module.base.decorator import Config
|
|
|
|
|
from module.base.filter import Filter
|
|
|
|
|
from module.base.utils import *
|
|
|
|
|
from module.commission.project_data import *
|
|
|
|
|
from module.logger import logger
|
2022-04-14 16:37:54 -03:00
|
|
|
|
from module.ocr.ocr import Duration, Ocr
|
2021-09-13 20:38:29 +08:00
|
|
|
|
from module.reward.assets import *
|
|
|
|
|
|
|
|
|
|
COMMISSION_FILTER = Filter(
|
|
|
|
|
regex=re.compile(
|
|
|
|
|
'(major|daily|extra|urgent|night)?'
|
|
|
|
|
'-?'
|
|
|
|
|
'(resource|chip|event|drill|part|cube|oil|book|retrofit|box|gem|ship)?'
|
|
|
|
|
'-?'
|
|
|
|
|
'(\d\d?:\d\d)?'
|
|
|
|
|
'(\d\d?.\d\d?|\d\d?)?'
|
|
|
|
|
),
|
|
|
|
|
attr=('category_str', 'genre_str', 'duration_hm', 'duration_hour'),
|
2021-12-09 18:24:52 +08:00
|
|
|
|
preset=('shortest',)
|
2021-09-13 20:38:29 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
2021-09-20 09:33:35 +08:00
|
|
|
|
class SuffixOcr(Ocr):
|
|
|
|
|
def pre_process(self, image):
|
|
|
|
|
image = super().pre_process(image)
|
|
|
|
|
|
|
|
|
|
left = np.where(np.min(image[5:-5, :], axis=0) < 85)[0]
|
2022-05-10 00:35:28 +08:00
|
|
|
|
# Look back several pixels
|
|
|
|
|
if server.server in ['jp']:
|
|
|
|
|
look_back = 21
|
|
|
|
|
else:
|
|
|
|
|
look_back = 18
|
2021-09-20 09:33:35 +08:00
|
|
|
|
if len(left):
|
2022-05-10 00:35:28 +08:00
|
|
|
|
image = image[:, left[-1] - look_back:]
|
2021-09-20 09:33:35 +08:00
|
|
|
|
|
2024-11-30 03:00:31 +08:00
|
|
|
|
if server.server in ['jp']:
|
|
|
|
|
# slice top and bottom part to get clearer roman digits
|
|
|
|
|
# will need to pad white background for better recognization
|
|
|
|
|
image = image[8:-10, :]
|
|
|
|
|
cv2.normalize(image, image, -55, 255, cv2.NORM_MINMAX)
|
2025-01-16 00:52:17 +08:00
|
|
|
|
image = (image > 128).astype(np.uint8) * 255
|
|
|
|
|
image = np.pad(image, ((4, 4), (0, 0)), mode='constant', constant_values=255)
|
2021-09-20 09:33:35 +08:00
|
|
|
|
return image
|
|
|
|
|
|
|
|
|
|
|
2022-10-02 18:25:07 +08:00
|
|
|
|
class TwOcr(Ocr):
|
|
|
|
|
def after_process(self, result):
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
result (str): '第二行'
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
str:
|
|
|
|
|
"""
|
|
|
|
|
# There no letter `艦` in training dataset
|
2023-03-16 00:56:00 +08:00
|
|
|
|
result = result.replace('鑑', '艦').replace('盤', '艦')
|
2024-05-03 16:49:11 +08:00
|
|
|
|
# 支援土蒙爾島
|
|
|
|
|
result = result.replace('土蒙爾', '土豪爾')
|
2022-10-02 18:25:07 +08:00
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
2021-09-13 20:38:29 +08:00
|
|
|
|
class Commission:
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# Button to enter commission start
|
2021-09-13 20:38:29 +08:00
|
|
|
|
button: Button
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# OCR result
|
2021-09-13 20:38:29 +08:00
|
|
|
|
name: str
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# If success to parse commission name
|
|
|
|
|
valid: bool
|
|
|
|
|
# Suffix in roman numerals
|
|
|
|
|
# May be wrong if commission does not have a suffix
|
|
|
|
|
# Value: ⅠⅡⅢⅤⅣⅥ
|
2021-09-20 09:33:35 +08:00
|
|
|
|
suffix: str
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# Genre name in project_data.py
|
|
|
|
|
# Value: major_comm, daily_resource, urgent_cube, ...
|
2021-09-13 20:38:29 +08:00
|
|
|
|
genre: str
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# Status of commission
|
|
|
|
|
# Value: finished, running, pending
|
2021-09-13 20:38:29 +08:00
|
|
|
|
status: str
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# Duration to run this commission
|
2021-09-13 20:38:29 +08:00
|
|
|
|
duration: timedelta
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# Expire, only in urgent commission, None in others
|
2021-09-13 20:38:29 +08:00
|
|
|
|
expire: timedelta
|
2021-09-24 14:44:39 +08:00
|
|
|
|
# Category for filter
|
|
|
|
|
# Value: major|daily|extra|urgent|night
|
|
|
|
|
category_str: str
|
|
|
|
|
# Genre for filter
|
|
|
|
|
# Value: resource|chip|event|drill|part|cube|oil|book|retrofit|box|gem|ship
|
|
|
|
|
genre_str: str
|
|
|
|
|
# Duration in hours
|
|
|
|
|
# Value: 0.5, 1, 1.16, 2.5, ...
|
|
|
|
|
duration_hour: str
|
|
|
|
|
# Duration in HH:MM
|
|
|
|
|
# Value: 1:30, 1:45, 2:00, 8:00, 12:00, ...
|
|
|
|
|
duration_hm: str
|
2021-09-13 20:38:29 +08:00
|
|
|
|
|
|
|
|
|
def __init__(self, image, y, config):
|
|
|
|
|
self.config = config
|
|
|
|
|
self.y = y
|
|
|
|
|
self.area = (188, y - 119, 1199, y)
|
|
|
|
|
self.image = image
|
|
|
|
|
self.valid = True
|
|
|
|
|
self.commission_parse()
|
|
|
|
|
|
|
|
|
|
if not self.duration.total_seconds():
|
|
|
|
|
self.valid = False
|
|
|
|
|
|
|
|
|
|
self.create_time = datetime.now()
|
2021-11-18 12:04:16 +08:00
|
|
|
|
self.repeat_count = 1
|
2021-09-13 20:38:29 +08:00
|
|
|
|
self.category_str = 'unknown'
|
|
|
|
|
self.genre_str = 'unknown'
|
|
|
|
|
self.duration_hour = 'unknown'
|
|
|
|
|
self.duration_hm = 'unknown'
|
|
|
|
|
if self.valid:
|
|
|
|
|
self.category_str, self.genre_str = self.genre.split('_', 1)
|
|
|
|
|
self.duration_hour = str(int(self.duration.total_seconds() / 36) / 100).strip('.0')
|
|
|
|
|
self.duration_hm = str(self.duration).rsplit(':', 1)[0]
|
|
|
|
|
|
|
|
|
|
@Config.when(SERVER='en')
|
|
|
|
|
def commission_parse(self):
|
|
|
|
|
# Name
|
|
|
|
|
# This is different from CN, EN has longer names
|
|
|
|
|
area = area_offset((176, 23, 420, 53), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='COMMISSION')
|
|
|
|
|
ocr = Ocr(button, lang='cnocr')
|
|
|
|
|
self.button = button
|
|
|
|
|
self.name = ocr.ocr(self.image)
|
|
|
|
|
self.genre = self.commission_name_parse(self.name.upper())
|
|
|
|
|
|
2021-09-20 09:33:35 +08:00
|
|
|
|
# Suffix
|
|
|
|
|
ocr = SuffixOcr(button, lang='azur_lane', letter=(255, 255, 255), threshold=128, alphabet='IV')
|
|
|
|
|
self.suffix = self.beautify_name(ocr.ocr(self.image))
|
|
|
|
|
|
2021-09-13 20:38:29 +08:00
|
|
|
|
# Duration time
|
|
|
|
|
area = area_offset((290, 68, 390, 95), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='DURATION')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.duration = ocr.ocr(self.image)
|
|
|
|
|
|
|
|
|
|
# Expire time
|
|
|
|
|
area = area_offset((-49, 68, -45, 84), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(189, 65, 66),
|
|
|
|
|
button=area, name='IS_URGENT')
|
2022-03-10 21:07:42 +08:00
|
|
|
|
if button.appear_on(self.image, threshold=30):
|
2021-09-13 20:38:29 +08:00
|
|
|
|
area = area_offset((-49, 67, 45, 94), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='EXPIRE')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.expire = ocr.ocr(self.image)
|
|
|
|
|
else:
|
|
|
|
|
self.expire = timedelta(seconds=0)
|
|
|
|
|
|
|
|
|
|
# Status
|
|
|
|
|
area = area_offset((179, 71, 187, 93), self.area[0:2])
|
|
|
|
|
dic = {
|
|
|
|
|
0: 'finished',
|
|
|
|
|
1: 'running',
|
|
|
|
|
2: 'pending'
|
|
|
|
|
}
|
2023-04-27 22:12:44 +08:00
|
|
|
|
color = np.array(get_color(self.image, area))
|
|
|
|
|
if self.genre == 'daily_event':
|
2021-09-13 20:38:29 +08:00
|
|
|
|
color -= [50, 30, 20]
|
|
|
|
|
self.status = dic[int(np.argmax(color))]
|
|
|
|
|
|
|
|
|
|
@Config.when(SERVER='jp')
|
|
|
|
|
def commission_parse(self):
|
|
|
|
|
# Name
|
|
|
|
|
area = area_offset((176, 23, 420, 53), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='COMMISSION')
|
2024-10-17 22:08:05 +08:00
|
|
|
|
ocr = Ocr(button, letter=(201, 201, 201), lang='jp')
|
2021-09-13 20:38:29 +08:00
|
|
|
|
self.button = button
|
|
|
|
|
self.name = ocr.ocr(self.image)
|
|
|
|
|
self.genre = self.commission_name_parse(self.name)
|
|
|
|
|
|
2021-09-20 09:33:35 +08:00
|
|
|
|
# Suffix
|
2024-11-30 03:00:31 +08:00
|
|
|
|
ocr = SuffixOcr(button, lang='azur_lane', letter=(201, 201, 201), threshold=128, alphabet='IV')
|
2021-09-20 09:33:35 +08:00
|
|
|
|
self.suffix = self.beautify_name(ocr.ocr(self.image))
|
|
|
|
|
|
2021-09-13 20:38:29 +08:00
|
|
|
|
# Duration time
|
|
|
|
|
area = area_offset((290, 68, 390, 95), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='DURATION')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.duration = ocr.ocr(self.image)
|
|
|
|
|
|
|
|
|
|
# Expire time
|
|
|
|
|
area = area_offset((-49, 68, -45, 84), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(189, 65, 66),
|
|
|
|
|
button=area, name='IS_URGENT')
|
2022-03-10 21:07:42 +08:00
|
|
|
|
if button.appear_on(self.image, threshold=30):
|
2021-09-13 20:38:29 +08:00
|
|
|
|
area = area_offset((-49, 67, 45, 94), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='EXPIRE')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.expire = ocr.ocr(self.image)
|
|
|
|
|
else:
|
|
|
|
|
self.expire = timedelta(seconds=0)
|
|
|
|
|
|
|
|
|
|
# Status
|
|
|
|
|
area = area_offset((179, 71, 187, 93), self.area[0:2])
|
|
|
|
|
dic = {
|
|
|
|
|
0: 'finished',
|
|
|
|
|
1: 'running',
|
|
|
|
|
2: 'pending'
|
|
|
|
|
}
|
2023-04-27 22:12:44 +08:00
|
|
|
|
color = np.array(get_color(self.image, area))
|
|
|
|
|
if self.genre == 'daily_event':
|
2021-09-13 20:38:29 +08:00
|
|
|
|
color -= [50, 30, 20]
|
|
|
|
|
self.status = dic[int(np.argmax(color))]
|
|
|
|
|
|
2021-11-14 13:00:50 +08:00
|
|
|
|
@Config.when(SERVER='tw')
|
|
|
|
|
def commission_parse(self):
|
|
|
|
|
# Name
|
|
|
|
|
area = area_offset((176, 23, 420, 53), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='COMMISSION')
|
2022-10-02 18:25:07 +08:00
|
|
|
|
ocr = TwOcr(button, lang='tw', threshold=256)
|
2021-11-14 13:00:50 +08:00
|
|
|
|
self.button = button
|
|
|
|
|
self.name = ocr.ocr(self.image)
|
|
|
|
|
self.genre = self.commission_name_parse(self.name)
|
|
|
|
|
|
|
|
|
|
# Suffix
|
|
|
|
|
ocr = SuffixOcr(button, lang='azur_lane', letter=(255, 255, 255), threshold=128, alphabet='IV')
|
|
|
|
|
self.suffix = self.beautify_name(ocr.ocr(self.image))
|
|
|
|
|
|
|
|
|
|
# Duration time
|
|
|
|
|
area = area_offset((290, 68, 390, 95), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='DURATION')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.duration = ocr.ocr(self.image)
|
|
|
|
|
|
|
|
|
|
# Expire time
|
|
|
|
|
area = area_offset((-49, 68, -45, 84), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(189, 65, 66),
|
|
|
|
|
button=area, name='IS_URGENT')
|
2022-03-10 21:07:42 +08:00
|
|
|
|
if button.appear_on(self.image, threshold=30):
|
2021-11-14 13:00:50 +08:00
|
|
|
|
area = area_offset((-49, 67, 45, 94), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='EXPIRE')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.expire = ocr.ocr(self.image)
|
|
|
|
|
else:
|
|
|
|
|
self.expire = timedelta(seconds=0)
|
|
|
|
|
|
|
|
|
|
# Status
|
|
|
|
|
area = area_offset((179, 71, 187, 93), self.area[0:2])
|
|
|
|
|
dic = {
|
|
|
|
|
0: 'finished',
|
|
|
|
|
1: 'running',
|
|
|
|
|
2: 'pending'
|
|
|
|
|
}
|
2023-04-27 22:12:44 +08:00
|
|
|
|
color = np.array(get_color(self.image, area))
|
|
|
|
|
if self.genre == 'daily_event':
|
2021-11-14 13:00:50 +08:00
|
|
|
|
color -= [50, 30, 20]
|
|
|
|
|
self.status = dic[int(np.argmax(color))]
|
|
|
|
|
|
|
|
|
|
@Config.when(SERVER=None)
|
2021-09-13 20:38:29 +08:00
|
|
|
|
def commission_parse(self):
|
|
|
|
|
# Name
|
|
|
|
|
area = area_offset((176, 23, 420, 53), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='COMMISSION')
|
|
|
|
|
ocr = Ocr(button, lang='cnocr', threshold=256)
|
|
|
|
|
self.button = button
|
|
|
|
|
self.name = ocr.ocr(self.image)
|
|
|
|
|
self.genre = self.commission_name_parse(self.name)
|
|
|
|
|
|
2021-09-20 09:33:35 +08:00
|
|
|
|
# Suffix
|
|
|
|
|
ocr = SuffixOcr(button, lang='azur_lane', letter=(255, 255, 255), threshold=128, alphabet='IV')
|
|
|
|
|
self.suffix = self.beautify_name(ocr.ocr(self.image))
|
|
|
|
|
|
2021-09-13 20:38:29 +08:00
|
|
|
|
# Duration time
|
|
|
|
|
area = area_offset((290, 68, 390, 95), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='DURATION')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.duration = ocr.ocr(self.image)
|
|
|
|
|
|
|
|
|
|
# Expire time
|
|
|
|
|
area = area_offset((-49, 68, -45, 84), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(189, 65, 66),
|
|
|
|
|
button=area, name='IS_URGENT')
|
2022-03-10 21:07:42 +08:00
|
|
|
|
if button.appear_on(self.image, threshold=30):
|
2021-09-13 20:38:29 +08:00
|
|
|
|
area = area_offset((-49, 67, 45, 94), self.area[0:2])
|
|
|
|
|
button = Button(area=area, color=(), button=area, name='EXPIRE')
|
|
|
|
|
ocr = Duration(button)
|
|
|
|
|
self.expire = ocr.ocr(self.image)
|
|
|
|
|
else:
|
|
|
|
|
self.expire = timedelta(seconds=0)
|
|
|
|
|
|
|
|
|
|
# Status
|
|
|
|
|
area = area_offset((179, 71, 187, 93), self.area[0:2])
|
|
|
|
|
dic = {
|
|
|
|
|
0: 'finished',
|
|
|
|
|
1: 'running',
|
|
|
|
|
2: 'pending'
|
|
|
|
|
}
|
2023-04-27 22:12:44 +08:00
|
|
|
|
color = np.array(get_color(self.image, area))
|
|
|
|
|
if self.genre == 'daily_event':
|
2021-09-13 20:38:29 +08:00
|
|
|
|
color -= [50, 30, 20]
|
|
|
|
|
self.status = dic[int(np.argmax(color))]
|
|
|
|
|
|
|
|
|
|
def __str__(self):
|
2021-11-18 12:04:16 +08:00
|
|
|
|
name = f'{self.name} | {self.suffix}'
|
|
|
|
|
if not self.valid:
|
|
|
|
|
return f'{name} (Invalid)'
|
|
|
|
|
info = {'Genre': self.genre, 'Status': self.status, 'Duration': self.duration}
|
|
|
|
|
if self.expire:
|
|
|
|
|
info['Expire'] = self.expire
|
|
|
|
|
if self.repeat_count > 1:
|
|
|
|
|
info['Repeat'] = self.repeat_count
|
|
|
|
|
info = ', '.join([f'{k}: {v}' for k, v in info.items()])
|
|
|
|
|
return f'{name} ({info})'
|
2021-09-13 20:38:29 +08:00
|
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
other (Commission):
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
bool:
|
|
|
|
|
"""
|
|
|
|
|
if not isinstance(other, Commission):
|
|
|
|
|
return False
|
|
|
|
|
threshold = timedelta(seconds=120)
|
|
|
|
|
if not self.valid or not other.valid:
|
|
|
|
|
return False
|
|
|
|
|
if self.genre != other.genre or self.status != other.status:
|
|
|
|
|
return False
|
2021-09-20 21:32:59 +08:00
|
|
|
|
if self.category_str == 'daily':
|
|
|
|
|
if self.suffix != other.suffix:
|
|
|
|
|
return False
|
2021-09-24 14:44:39 +08:00
|
|
|
|
if self.genre == 'urgent_box':
|
|
|
|
|
for tag in ['NYB', 'BIW']:
|
|
|
|
|
if tag in self.name.upper() and tag not in other.name.upper():
|
|
|
|
|
return False
|
|
|
|
|
if tag not in self.name.upper() and tag in other.name.upper():
|
|
|
|
|
return False
|
2021-09-13 20:38:29 +08:00
|
|
|
|
if (other.duration < self.duration - threshold) or (other.duration > self.duration + threshold):
|
|
|
|
|
return False
|
|
|
|
|
if (not self.expire and other.expire) or (self.expire and not other.expire):
|
|
|
|
|
return False
|
|
|
|
|
if self.expire and other.expire:
|
|
|
|
|
if (other.expire < self.expire - threshold) or (other.expire > self.expire + threshold):
|
|
|
|
|
return False
|
2021-11-18 12:04:16 +08:00
|
|
|
|
if self.repeat_count != other.repeat_count:
|
|
|
|
|
return False
|
2022-08-10 17:51:47 +08:00
|
|
|
|
if self.genre in ['extra_oil', 'night_oil'] and self.suffix != other.suffix:
|
2022-08-10 01:00:43 +08:00
|
|
|
|
return False
|
2021-09-13 20:38:29 +08:00
|
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
def __hash__(self):
|
|
|
|
|
return hash(f'{self.genre}_{self.name}')
|
|
|
|
|
|
|
|
|
|
def parse_time(self, string):
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
string (str): Such as 01:00:00, 05:47:10, 17:50:51.
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
timedelta: datetime.timedelta instance.
|
|
|
|
|
"""
|
|
|
|
|
string = string.replace('D', '0') # Poor OCR
|
|
|
|
|
result = re.search('(\d+):(\d+):(\d+)', string)
|
|
|
|
|
if not result:
|
|
|
|
|
logger.warning(f'Invalid time string: {string}')
|
|
|
|
|
self.valid = False
|
|
|
|
|
return None
|
|
|
|
|
else:
|
|
|
|
|
result = [int(s) for s in result.groups()]
|
|
|
|
|
return timedelta(hours=result[0], minutes=result[1], seconds=result[2])
|
|
|
|
|
|
|
|
|
|
@Config.when(SERVER='en')
|
|
|
|
|
def commission_name_parse(self, string):
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
string (str): Commission name, such as 'NYB要员护卫'.
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
str: Commission genre, such as 'urgent_gem'.
|
|
|
|
|
"""
|
|
|
|
|
# string = string.replace(' ', '').replace('-', '')
|
|
|
|
|
if self.is_event_commission():
|
|
|
|
|
return 'daily_event'
|
|
|
|
|
for key, value in dictionary_en.items():
|
|
|
|
|
for keyword in value:
|
|
|
|
|
if keyword in string:
|
|
|
|
|
return key
|
|
|
|
|
|
|
|
|
|
logger.warning(f'Name with unknown genre: {string}')
|
|
|
|
|
self.valid = False
|
|
|
|
|
return ''
|
|
|
|
|
|
|
|
|
|
@Config.when(SERVER='jp')
|
|
|
|
|
def commission_name_parse(self, string):
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
string (str): Commission name, such as 'NYB要员护卫'.
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
str: Commission genre, such as 'urgent_gem'.
|
|
|
|
|
"""
|
|
|
|
|
if self.is_event_commission():
|
|
|
|
|
return 'daily_event'
|
|
|
|
|
import jellyfish
|
|
|
|
|
min_key = ''
|
|
|
|
|
min_distance = 100
|
|
|
|
|
string = re.sub(r'[\x00-\x7F]', '', string)
|
|
|
|
|
for key, value in dictionary_jp.items():
|
|
|
|
|
for keyword in value:
|
|
|
|
|
distance = jellyfish.levenshtein_distance(keyword, string)
|
|
|
|
|
if distance < min_distance:
|
|
|
|
|
min_key = key
|
|
|
|
|
min_distance = distance
|
|
|
|
|
if min_distance < 3:
|
|
|
|
|
return min_key
|
|
|
|
|
|
|
|
|
|
logger.warning(f'Name with unknown genre: {string}')
|
|
|
|
|
self.valid = False
|
|
|
|
|
return ''
|
|
|
|
|
|
2021-11-14 13:00:50 +08:00
|
|
|
|
@Config.when(SERVER='tw')
|
|
|
|
|
def commission_name_parse(self, string):
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
string (str): Commission name, such as 'NYB要员护卫'.
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
str: Commission genre, such as 'urgent_gem'.
|
|
|
|
|
"""
|
|
|
|
|
if self.is_event_commission():
|
|
|
|
|
return 'daily_event'
|
|
|
|
|
for key, value in dictionary_tw.items():
|
|
|
|
|
for keyword in value:
|
|
|
|
|
if keyword in string:
|
|
|
|
|
return key
|
|
|
|
|
|
|
|
|
|
logger.warning(f'Name with unknown genre: {string}')
|
|
|
|
|
self.valid = False
|
|
|
|
|
return ''
|
|
|
|
|
|
2021-09-13 20:38:29 +08:00
|
|
|
|
@Config.when(SERVER=None)
|
|
|
|
|
def commission_name_parse(self, string):
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
string (str): Commission name, such as 'NYB要员护卫'.
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
str: Commission genre, such as 'urgent_gem'.
|
|
|
|
|
"""
|
|
|
|
|
if self.is_event_commission():
|
|
|
|
|
return 'daily_event'
|
|
|
|
|
for key, value in dictionary_cn.items():
|
|
|
|
|
for keyword in value:
|
|
|
|
|
if keyword in string:
|
|
|
|
|
return key
|
|
|
|
|
|
|
|
|
|
logger.warning(f'Name with unknown genre: {string}')
|
|
|
|
|
self.valid = False
|
|
|
|
|
return ''
|
|
|
|
|
|
|
|
|
|
def is_event_commission(self):
|
|
|
|
|
"""
|
|
|
|
|
Returns:
|
|
|
|
|
bool:
|
|
|
|
|
"""
|
|
|
|
|
# Event commission in Vacation Lane, with pink area on the left.
|
|
|
|
|
# area = area_offset((5, 5, 30, 30), self.area[0:2])
|
|
|
|
|
# return color_similar(color1=get_color(self.image, area), color2=(239, 166, 231))
|
|
|
|
|
|
|
|
|
|
# 2021.07.22 Event commissions in The Idol Master event, with
|
|
|
|
|
# area = area_offset((5, 5, 30, 30), self.area[0:2])
|
|
|
|
|
# return color_similar(color1=get_color(self.image, area), color2=(235, 173, 161))
|
|
|
|
|
|
2023-04-27 22:12:44 +08:00
|
|
|
|
# 2023.04.27 Vacation Lane Rerun, pink yellow gradient like Idol Master event
|
|
|
|
|
area = area_offset((5, 5, 30, 30), self.area[0:2])
|
|
|
|
|
if color_similar(color1=get_color(self.image, area), color2=(235, 173, 161), threshold=30):
|
|
|
|
|
return True
|
|
|
|
|
|
2021-09-13 20:38:29 +08:00
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
def convert_to_night(self):
|
|
|
|
|
if self.valid and self.category_str == 'extra':
|
|
|
|
|
self.category_str = 'night'
|
|
|
|
|
self.genre = f'{self.category_str}_{self.genre_str}'
|
|
|
|
|
|
|
|
|
|
def convert_to_running(self):
|
|
|
|
|
if self.valid:
|
|
|
|
|
self.status = 'running'
|
|
|
|
|
self.create_time = datetime.now()
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def finish_time(self):
|
|
|
|
|
if self.valid and self.status == 'running':
|
|
|
|
|
return (self.create_time + self.duration).replace(microsecond=0)
|
|
|
|
|
else:
|
|
|
|
|
return None
|
2021-09-20 09:33:35 +08:00
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def beautify_name(name):
|
|
|
|
|
name = name.strip()
|
|
|
|
|
name = re.sub(r'VI$', 'Ⅵ', name)
|
|
|
|
|
name = re.sub(r'IV$', 'Ⅳ', name)
|
|
|
|
|
name = re.sub(r'V$', 'Ⅴ', name)
|
|
|
|
|
name = re.sub(r'III$', 'Ⅲ', name)
|
|
|
|
|
name = re.sub(r'II$', 'Ⅱ', name)
|
|
|
|
|
name = re.sub(r'I$', 'Ⅰ', name)
|
|
|
|
|
return name
|