mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 14:22:55 +08:00
159 lines
5.7 KiB
Python
159 lines
5.7 KiB
Python
|
|
from pathlib import Path
|
|||
|
|
from configs.path_config import IMAGE_PATH, TXT_PATH
|
|||
|
|
from utils.image_utils import CreateImg
|
|||
|
|
from typing import Tuple
|
|||
|
|
from math import sqrt, pow
|
|||
|
|
import random
|
|||
|
|
|
|||
|
|
try:
|
|||
|
|
import ujson as json
|
|||
|
|
except ModuleNotFoundError:
|
|||
|
|
import json
|
|||
|
|
|
|||
|
|
icon_path = Path(IMAGE_PATH) / "genshin" / "genshin_icon"
|
|||
|
|
map_path = Path(IMAGE_PATH) / "genshin" / "map" / "map.png"
|
|||
|
|
resource_label_file = Path(TXT_PATH) / "genshin" / "resource_label_file.json"
|
|||
|
|
resource_point_file = Path(TXT_PATH) / "genshin" / "resource_point_file.json"
|
|||
|
|
|
|||
|
|
|
|||
|
|
class Map:
|
|||
|
|
"""
|
|||
|
|
原神资源生成类
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
def __init__(
|
|||
|
|
self,
|
|||
|
|
resource_name: str,
|
|||
|
|
center_point: Tuple[int, int],
|
|||
|
|
deviation: Tuple[int, int] = (25, 51),
|
|||
|
|
padding: int = 100,
|
|||
|
|
planning_route: bool = False,
|
|||
|
|
):
|
|||
|
|
"""
|
|||
|
|
参数:
|
|||
|
|
:param resource_name: 资源名称
|
|||
|
|
:param center_point: 中心点
|
|||
|
|
:param deviation: 坐标误差
|
|||
|
|
:param padding: 截图外边距
|
|||
|
|
:param planning_route: 是否规划最佳线路
|
|||
|
|
"""
|
|||
|
|
self.map = CreateImg(0, 0, background=map_path)
|
|||
|
|
self.resource_name = resource_name
|
|||
|
|
self.center_x = center_point[0]
|
|||
|
|
self.center_y = center_point[1]
|
|||
|
|
self.deviation = deviation
|
|||
|
|
self.padding = padding
|
|||
|
|
self.planning_route = planning_route
|
|||
|
|
|
|||
|
|
data = json.load(open(resource_label_file, "r", encoding="utf8"))
|
|||
|
|
# 资源 id
|
|||
|
|
self.resource_id = [
|
|||
|
|
data[x]["id"]
|
|||
|
|
for x in data
|
|||
|
|
if x != "CENTER_POINT" and data[x]["name"] == resource_name
|
|||
|
|
][0]
|
|||
|
|
# 传送锚点 id
|
|||
|
|
self.teleport_anchor_id = [
|
|||
|
|
data[x]["id"]
|
|||
|
|
for x in data
|
|||
|
|
if x != "CENTER_POINT" and data[x]["name"] == "传送锚点"
|
|||
|
|
][0]
|
|||
|
|
# 神像 id
|
|||
|
|
self.teleport_god_id = [
|
|||
|
|
data[x]["id"]
|
|||
|
|
for x in data
|
|||
|
|
if x != "CENTER_POINT" and data[x]["name"] == "七天神像"
|
|||
|
|
][0]
|
|||
|
|
# 资源坐标
|
|||
|
|
data = json.load(open(resource_point_file, "r", encoding="utf8"))
|
|||
|
|
self.resource_point = [
|
|||
|
|
(
|
|||
|
|
self.center_x + int(data[x]["x_pos"]),
|
|||
|
|
self.center_y + int(data[x]["y_pos"]),
|
|||
|
|
)
|
|||
|
|
for x in data
|
|||
|
|
if x != "CENTER_POINT" and data[x]["label_id"] == self.resource_id
|
|||
|
|
]
|
|||
|
|
# 传送锚点坐标
|
|||
|
|
self.teleport_anchor_point = [
|
|||
|
|
(
|
|||
|
|
self.center_x + int(data[x]["x_pos"]),
|
|||
|
|
self.center_y + int(data[x]["y_pos"]),
|
|||
|
|
)
|
|||
|
|
for x in data
|
|||
|
|
if x != "CENTER_POINT" and data[x]["label_id"] == self.teleport_anchor_id
|
|||
|
|
]
|
|||
|
|
# 神像坐标
|
|||
|
|
self.teleport_god_point = [
|
|||
|
|
(
|
|||
|
|
self.center_x + int(data[x]["x_pos"]),
|
|||
|
|
self.center_y + int(data[x]["y_pos"]),
|
|||
|
|
)
|
|||
|
|
for x in data
|
|||
|
|
if x != "CENTER_POINT" and data[x]["label_id"] == self.teleport_god_id
|
|||
|
|
]
|
|||
|
|
|
|||
|
|
# 将地图上生成资源图标
|
|||
|
|
def generate_resource_icon_in_map(self) -> int:
|
|||
|
|
x_list = [x[0] for x in self.resource_point]
|
|||
|
|
y_list = [x[1] for x in self.resource_point]
|
|||
|
|
min_width = min(x_list) - self.padding
|
|||
|
|
max_width = max(x_list) + self.padding
|
|||
|
|
min_height = min(y_list) - self.padding
|
|||
|
|
max_height = max(y_list) + self.padding
|
|||
|
|
if self.planning_route:
|
|||
|
|
self._generate_best_route()
|
|||
|
|
self._generate_transfer_icon((min_width, min_height, max_width, max_height))
|
|||
|
|
for x, y in self.resource_point:
|
|||
|
|
icon = self._get_icon_image(self.resource_id)
|
|||
|
|
self.map.paste(icon, (x - self.deviation[0], y - self.deviation[1]), True)
|
|||
|
|
self.map.crop((min_width, min_height, max_width, max_height))
|
|||
|
|
rand = random.randint(1, 10000)
|
|||
|
|
self.map.save(f'{IMAGE_PATH}/temp/genshin_map_{rand}.png')
|
|||
|
|
return rand
|
|||
|
|
|
|||
|
|
# 资源数量
|
|||
|
|
def get_resource_count(self) -> int:
|
|||
|
|
return len(self.resource_point)
|
|||
|
|
|
|||
|
|
# 生成传送锚点和神像
|
|||
|
|
def _generate_transfer_icon(self, box: Tuple[int, int, int, int]):
|
|||
|
|
min_width, min_height, max_width, max_height = box
|
|||
|
|
for points in [self.teleport_anchor_point, self.teleport_god_point]:
|
|||
|
|
id_ = (
|
|||
|
|
self.teleport_anchor_id
|
|||
|
|
if points == self.teleport_anchor_point
|
|||
|
|
else self.teleport_god_id
|
|||
|
|
)
|
|||
|
|
for x, y in points:
|
|||
|
|
if min_width < x < max_width and min_height < y < max_height:
|
|||
|
|
icon = self._get_icon_image(id_)
|
|||
|
|
self.map.paste(
|
|||
|
|
icon, (x - self.deviation[0], y - self.deviation[1]), True
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
# 生成最优路线(说是最优其实就是直线最短)
|
|||
|
|
def _generate_best_route(self):
|
|||
|
|
for x, y in self.resource_point:
|
|||
|
|
min_deviation = 999999
|
|||
|
|
xy = None
|
|||
|
|
for points in [
|
|||
|
|
self.resource_point,
|
|||
|
|
self.teleport_anchor_point,
|
|||
|
|
self.teleport_god_point,
|
|||
|
|
]:
|
|||
|
|
for r_x, r_y in points:
|
|||
|
|
distance = int(sqrt(pow(abs(r_x - x), 2) + pow(abs(r_y - y), 2)))
|
|||
|
|
if distance < min_deviation and x != r_x and y != r_y:
|
|||
|
|
min_deviation = distance
|
|||
|
|
xy = (x, y, r_x, r_y)
|
|||
|
|
|
|||
|
|
self.map.line(xy, (255, 0, 0), width=3)
|
|||
|
|
|
|||
|
|
# 获取资源图标
|
|||
|
|
def _get_icon_image(self, id_: int) -> "CreateImg":
|
|||
|
|
icon = icon_path / f"{id_}.png"
|
|||
|
|
if icon.exists():
|
|||
|
|
return CreateImg(50, 50, background=icon)
|
|||
|
|
return CreateImg(50, 50, background=f"{icon_path}/box.png")
|