457 lines
20 KiB
Python
457 lines
20 KiB
Python
# -------------------------------
|
||
# Difficult Rocket
|
||
# Copyright © 2020-2023 by shenjackyuanjie 3695888@qq.com
|
||
# All rights reserved
|
||
# -------------------------------
|
||
|
||
import math
|
||
import time
|
||
import random
|
||
import logging
|
||
import traceback
|
||
from xml.etree import ElementTree
|
||
from xml.etree.ElementTree import Element
|
||
from typing import List, TYPE_CHECKING, Union, Dict, Optional, Generator
|
||
|
||
# third party package
|
||
from defusedxml.ElementTree import parse
|
||
|
||
# pyglet
|
||
from pyglet.math import Vec4
|
||
from pyglet.text import Label
|
||
from pyglet.shapes import Line, Rectangle
|
||
from pyglet.sprite import Sprite
|
||
from pyglet.image import Texture
|
||
from pyglet.graphics import Batch, Group
|
||
|
||
from . import DR_mod_runtime
|
||
|
||
# Difficult Rocket
|
||
from Difficult_Rocket import DR_status
|
||
from Difficult_Rocket.utils.translate import tr
|
||
from Difficult_Rocket.api.types import Fonts, Options
|
||
from Difficult_Rocket.command.line import CommandText
|
||
from Difficult_Rocket.client.screen import BaseScreen
|
||
from .types import SR1Textures, SR1PartTexture, SR1PartData, SR1Rotation, xml_bool
|
||
|
||
if TYPE_CHECKING:
|
||
from Difficult_Rocket.client import ClientWindow
|
||
|
||
if DR_mod_runtime.use_DR_rust:
|
||
from .Difficult_Rocket_rs import CenterCamera_rs, SR1PartList_rs, SR1Ship_rs
|
||
|
||
logger = logging.getLogger('client')
|
||
|
||
|
||
def get_sr1_part(part_xml: Element) -> Optional[SR1PartData]:
|
||
if part_xml.tag != 'Part':
|
||
return None
|
||
# print(f"tag: {part.tag} attrib: {part.attrib}")
|
||
part_id = int(part_xml.attrib.get('id'))
|
||
part_type = part_xml.attrib.get('partType')
|
||
part_x = float(part_xml.attrib.get('x'))
|
||
part_y = float(part_xml.attrib.get('y'))
|
||
part_activate = xml_bool(part_xml.attrib.get('activated'))
|
||
part_angle = float(part_xml.attrib.get('angle'))
|
||
part_angle_v = float(part_xml.attrib.get('angleV'))
|
||
part_editor_angle = int(part_xml.attrib.get('editorAngle'))
|
||
part_flip_x = xml_bool(part_xml.attrib.get('flippedX'))
|
||
part_flip_y = xml_bool(part_xml.attrib.get('flippedY'))
|
||
part_explode = xml_bool(part_xml.attrib.get('exploded'))
|
||
if part_type not in SR1PartTexture.part_type_sprite:
|
||
part_textures = None
|
||
else:
|
||
part_textures = SR1PartTexture.get_textures_from_type(part_type)
|
||
# print(f'id: {part_id:<4} type: {part_type:<10} x: {part_x} y: {part_y} activated: {part_activate} '
|
||
# f'angle: {part_angle} angle_v: {part_angle_v} editor_angle: {part_editor_angle} '
|
||
# f'flip_x: {part_flip_x} flip_y: {part_flip_y} explode: {part_explode} '
|
||
# f'textures: {SR1PartTexture.get_textures_from_type(part_type)}')
|
||
return SR1PartData(x=part_x, y=part_y, id=part_id, p_type=part_type,
|
||
active=part_activate, angle=part_angle, angle_v=part_angle_v,
|
||
editor_angle=part_editor_angle, flip_x=part_flip_x,
|
||
flip_y=part_flip_y, explode=part_explode, textures=part_textures)
|
||
|
||
|
||
class _SR1ShipRender_Option(Options):
|
||
# debug option
|
||
debug_d_pos: bool = False
|
||
debug_mouse_pos: bool = False
|
||
debug_mouse_d_pos: bool = False
|
||
|
||
|
||
SR1ShipRender_Option = _SR1ShipRender_Option()
|
||
|
||
|
||
class SR1ShipRender(BaseScreen):
|
||
"""用于渲染 sr1 船的类"""
|
||
|
||
def __init__(self,
|
||
main_window: "ClientWindow"):
|
||
super().__init__(main_window)
|
||
logger.info(tr().client.sr1_render.setup.start())
|
||
load_start_time = time.time_ns()
|
||
self.rendered = False
|
||
self.focus = True
|
||
self.need_draw = False
|
||
self.drawing = False
|
||
self.gen_draw: Optional[Generator] = None
|
||
self.need_update_parts = False
|
||
self.dx = 0
|
||
self.dy = 0
|
||
self.debug_line = Line(main_window.width / 2, main_window.height / 2,
|
||
main_window.width / 2, main_window.height / 2,
|
||
width=3, color=(200, 10, 200, 255))
|
||
self.debug_line.visible = SR1ShipRender_Option.debug_d_pos
|
||
self.debug_mouse_line = Line(main_window.width / 2, main_window.height / 2,
|
||
main_window.width / 2, main_window.height / 2,
|
||
width=3, color=(10, 200, 200, 255))
|
||
self.debug_mouse_line.visible = SR1ShipRender_Option.debug_mouse_pos
|
||
self.debug_mouse_delta_line = Line(main_window.width / 2, main_window.height / 2,
|
||
main_window.width / 2, main_window.height / 2,
|
||
width=2, color=(200, 200, 10, 255))
|
||
self.debug_mouse_delta_line.visible = SR1ShipRender_Option.debug_mouse_d_pos
|
||
self.debug_d_pos_label = Label('debug label NODATA', font_name=Fonts.微软等宽无线,
|
||
x=main_window.width / 2, y=main_window.height / 2)
|
||
self.debug_d_pos_label.visible = SR1ShipRender_Option.debug_d_pos
|
||
self.debug_mouse_label = Label('debug mouse_label NODATA', font_name=Fonts.微软等宽无线,
|
||
x=main_window.width / 2, y=main_window.height / 2)
|
||
self.debug_mouse_label.visible = SR1ShipRender_Option.debug_mouse_pos
|
||
self.textures: Union[SR1Textures, None] = None
|
||
# self.xml_name = 'configs/dock1.xml'
|
||
# self.xml_doc: ElementTree = parse('configs/dock1.xml')
|
||
# self.xml_root: ElementTree.Element = self.xml_doc.getroot()
|
||
self.load_xml('configs/dock1.xml')
|
||
self.part_box_batch = Batch()
|
||
self.part_batch = Batch()
|
||
self.part_group = Group()
|
||
self.debug_label = Label(x=20, y=main_window.height - 20, font_size=DR_status.std_font_size,
|
||
text='SR1 render!', font_name=Fonts.微软等宽无线,
|
||
width=main_window.width - 20, height=20,
|
||
anchor_x='left', anchor_y='top')
|
||
self.part_data: Dict[int, SR1PartData] = {}
|
||
self.parts_sprite: Dict[int, Sprite] = {}
|
||
self.part_box_dict: Dict[int, Rectangle] = {}
|
||
load_end_time = time.time_ns()
|
||
logger.info(tr().client.sr1_render.setup.use_time().format(
|
||
(load_end_time - load_start_time) / 1000000000))
|
||
if DR_mod_runtime.use_DR_rust:
|
||
self.camera_rs = CenterCamera_rs(main_window,
|
||
min_zoom=(1 / 2) ** 10, max_zoom=10)
|
||
self.rust_parts = None
|
||
self.part_list_rs = SR1PartList_rs('configs/PartList.xml', 'default_part_list')
|
||
|
||
def load_xml(self, file_path: str) -> bool:
|
||
try:
|
||
start_time = time.time_ns()
|
||
logger.info(tr().client.sr1_render.xml.loading().format(file_path))
|
||
cache_doc = parse(file_path)
|
||
self.xml_doc = cache_doc
|
||
self.xml_root = self.xml_doc.getroot()
|
||
self.xml_name = file_path
|
||
if DR_mod_runtime.use_DR_rust:
|
||
try:
|
||
self.rust_ship = SR1Ship_rs(file_path, 'configs/PartList.xml', 'a_new_ship')
|
||
print(self.rust_ship.name)
|
||
print(self.rust_ship.img_pos)
|
||
except Exception:
|
||
traceback.print_exc()
|
||
logger.info(tr().client.sr1_render.xml.load_done())
|
||
logger.info(tr().client.sr1_render.xml.load_time().format(
|
||
(time.time_ns() - start_time) / 1000000000))
|
||
return True
|
||
except Exception as e:
|
||
print(e)
|
||
return False
|
||
|
||
def load_textures(self):
|
||
self.textures = SR1Textures()
|
||
|
||
def gen_sprite(self, part_datas: Dict[int, SR1PartData], each_count: int = 100) -> Generator:
|
||
count = 0
|
||
self.drawing = True
|
||
for part_id, part in part_datas.items():
|
||
# 下面就是调用 pyglet 去渲染的部分
|
||
# render_scale = DR_status.gui_scale # 这个是 DR 的缩放比例 可以调节的(
|
||
# 主要是 Windows 下有一个缩放系数嘛,我待会试试这玩意能不能获取(估计得 ctypes
|
||
# 在不缩放的情况下,XML的1个单位长度对应60个像素
|
||
render_x = part.x * 60
|
||
render_y = part.y * 60
|
||
# 你就这里改吧
|
||
cache_sprite = Sprite(img=self.textures.get_texture(part.textures),
|
||
x=render_x, y=render_y, z=random.random(),
|
||
batch=self.part_batch, group=self.part_group)
|
||
# 你得帮我换算一下 XML 里的 x y 和这里的屏幕像素的关系(OK
|
||
# 旋转啥的不是大问题, 我找你要那个渲染代码就是要 x y 的换算逻辑
|
||
cache_sprite.rotation = SR1Rotation.get_rotation(part.angle)
|
||
if part.flip_x:
|
||
cache_sprite.scale_x = -1 # 就是直接取反缩放,应该没问题····吧?(待会试试就知道了
|
||
if part.flip_y:
|
||
cache_sprite.scale_y = -1
|
||
cache_sprite.x = cache_sprite.x - cache_sprite.scale_x / 2
|
||
cache_sprite.y = cache_sprite.y - cache_sprite.scale_y / 2
|
||
self.parts_sprite[part.id] = cache_sprite
|
||
|
||
part_width = 100
|
||
part_height = 100
|
||
if DR_mod_runtime.use_DR_rust:
|
||
part_type = self.part_list_rs.get_part_type(part.p_type)
|
||
if part_type is not None:
|
||
part_width = part_type.width * 15
|
||
part_height = part_type.height * 15
|
||
part_box = Rectangle(x=render_x, y=render_y,
|
||
width=part_width, height=part_height,
|
||
batch=self.part_box_batch, group=self.part_group)
|
||
part_box.rotation = SR1Rotation.get_rotation(part.angle)
|
||
part_box.opacity = 50
|
||
self.part_box_dict[part.id] = part_box
|
||
# if not part_render: # 如果不渲染(渲染有毛病)
|
||
# self.parts_sprite[part.id].visible = False
|
||
count += 1
|
||
if count >= each_count:
|
||
count = 0
|
||
yield each_count
|
||
self.drawing = False
|
||
raise GeneratorExit
|
||
|
||
def render_ship(self):
|
||
if self.textures is None:
|
||
self.load_textures()
|
||
logger.info(tr().client.sr1_render.ship.load().format(self.xml_name))
|
||
start_time = time.perf_counter_ns()
|
||
self.part_data: Dict[int, SR1PartData] = {}
|
||
self.parts_sprite: Dict[int, Sprite] = {}
|
||
self.camera_rs.zoom = 1.0
|
||
if DR_mod_runtime.use_DR_rust:
|
||
self.camera_rs.dx = 0
|
||
self.camera_rs.dy = 0
|
||
parts = self.xml_root.find('Parts')
|
||
for part_xml in parts:
|
||
if part_xml.tag != 'Part':
|
||
continue # 如果不是部件,则跳过
|
||
part = get_sr1_part(part_xml)
|
||
if part.id in self.part_data:
|
||
print(f'hey! warning! id{part.id}')
|
||
self.part_data[part.id] = part
|
||
# 调用生成器 减少卡顿
|
||
try:
|
||
self.gen_draw = self.gen_sprite(self.part_data)
|
||
next(self.gen_draw)
|
||
except GeneratorExit:
|
||
self.drawing = False
|
||
self.need_draw = False
|
||
full_mass = 0
|
||
if DR_mod_runtime.use_DR_rust:
|
||
for part in self.part_data:
|
||
full_mass += self.part_list_rs.get_part_type(self.part_data[part].p_type).mass * 500
|
||
logger.info(tr().client.sr1_render.ship.load_time().format(
|
||
(time.perf_counter_ns() - start_time) / 1000000000))
|
||
logger.info(tr().client.sr1_render.ship.info().format(
|
||
len(self.part_data), f'{full_mass}kg' if DR_mod_runtime.use_DR_rust else tr().game.require_DR_rs()))
|
||
self.rendered = True
|
||
|
||
def update_parts(self) -> bool:
|
||
if not self.rendered:
|
||
return False
|
||
self.debug_line.x2, self.debug_line.y2 = self.camera_rs.dx + (
|
||
self.window_pointer.width / 2), self.camera_rs.dy + (
|
||
self.window_pointer.height / 2)
|
||
self.debug_d_pos_label.text = f'x: {self.camera_rs.dx} y: {self.camera_rs.dy}'
|
||
self.debug_d_pos_label.position = self.camera_rs.dx + (self.window_pointer.width / 2), self.camera_rs.dy + (
|
||
self.window_pointer.height / 2) + 10, 0
|
||
self.need_update_parts = False
|
||
|
||
def on_draw(self, window: "ClientWindow"):
|
||
if self.need_draw:
|
||
self.render_ship()
|
||
|
||
if self.drawing:
|
||
try:
|
||
next(self.gen_draw)
|
||
except GeneratorExit:
|
||
self.drawing = False
|
||
logger.info(tr().client.sr1_render.ship.render.done())
|
||
|
||
if self.need_update_parts:
|
||
self.update_parts()
|
||
self.need_update_parts = False
|
||
|
||
with self.camera_rs:
|
||
self.part_box_batch.draw()
|
||
self.part_batch.draw()
|
||
self.part_box_batch.draw()
|
||
|
||
self.debug_label.draw()
|
||
|
||
if SR1ShipRender_Option.debug_d_pos:
|
||
self.debug_line.draw()
|
||
self.debug_d_pos_label.draw()
|
||
if SR1ShipRender_Option.debug_mouse_pos:
|
||
self.debug_mouse_line.draw()
|
||
self.debug_mouse_label.draw()
|
||
if SR1ShipRender_Option.debug_mouse_d_pos:
|
||
self.debug_mouse_delta_line.draw()
|
||
|
||
def on_resize(self, width: int, height: int, window: "ClientWindow"):
|
||
if not self.rendered:
|
||
return
|
||
self.debug_line.x = width / 2
|
||
self.debug_line.y = height / 2
|
||
self.debug_mouse_line.x = width / 2
|
||
self.debug_mouse_line.y = height / 2
|
||
self.debug_mouse_delta_line.x = width / 2
|
||
self.debug_mouse_delta_line.y = height / 2
|
||
self.update_parts()
|
||
|
||
def on_mouse_scroll(self, x: int, y: int, scroll_x: int, scroll_y: int, window: "ClientWindow"):
|
||
if not self.rendered:
|
||
return
|
||
mouse_dx = x - (window.width / 2)
|
||
mouse_dy = y - (window.height / 2)
|
||
# 鼠标缩放位置相对于屏幕中心的位置
|
||
mouse_dx_d = mouse_dx - self.camera_rs.dx
|
||
mouse_dy_d = mouse_dy - self.camera_rs.dy
|
||
# 鼠标相对偏移量的偏移量
|
||
if scroll_y == 0:
|
||
zoom_d = 1
|
||
else:
|
||
zoom_d = ((2 ** scroll_y) - 1) * 0.5 + 1
|
||
# 缩放的变换量
|
||
if not (self.camera_rs.zoom == 10 and scroll_y > 0):
|
||
if self.camera_rs.zoom * zoom_d >= 10:
|
||
zoom_d = 10 / self.camera_rs.zoom
|
||
self.camera_rs.zoom = 10
|
||
else:
|
||
self.camera_rs.zoom *= zoom_d
|
||
mouse_dx_d *= (1 - zoom_d)
|
||
mouse_dy_d *= (1 - zoom_d)
|
||
self.camera_rs.dx += mouse_dx_d
|
||
self.camera_rs.dy += mouse_dy_d
|
||
|
||
self.debug_mouse_line.x2, self.debug_mouse_line.y2 = x, y
|
||
self.debug_mouse_delta_line.x2 = (mouse_dx - self.camera_rs.dx) * (1 - (0.5 ** scroll_y)) + (
|
||
window.width / 2)
|
||
self.debug_mouse_delta_line.y2 = (mouse_dy - self.camera_rs.dy) * (1 - (0.5 ** scroll_y)) + (
|
||
window.height / 2)
|
||
self.debug_mouse_label.text = f'x: {mouse_dx} y: {mouse_dy}'
|
||
self.debug_mouse_label.position = x, y + 10, 0
|
||
self.need_update_parts = True
|
||
# self.update_parts()
|
||
|
||
def on_command(self, command: CommandText, window: "ClientWindow"):
|
||
if command.find('render'):
|
||
if command.find('reset'):
|
||
self.camera_rs.zoom = 1
|
||
self.camera_rs.dx = 0
|
||
self.camera_rs.dy = 0
|
||
self.window_pointer.view = Vec4()
|
||
else:
|
||
self.need_draw = True
|
||
print('应该渲染飞船的')
|
||
elif command.find('debug'):
|
||
print('sr ?')
|
||
if command.find('delta'):
|
||
SR1ShipRender_Option.debug_d_pos = not SR1ShipRender_Option.debug_mouse_d_pos
|
||
self.debug_line.visible = SR1ShipRender_Option.debug_d_pos
|
||
self.debug_d_pos_label.visible = SR1ShipRender_Option.debug_d_pos
|
||
# print('sr1 delta')
|
||
elif command.find('mouse'):
|
||
if command.find('delta'):
|
||
SR1ShipRender_Option.debug_mouse_pos = not SR1ShipRender_Option.debug_mouse_pos
|
||
self.debug_mouse_line.visible = SR1ShipRender_Option.debug_mouse_pos
|
||
self.debug_mouse_label.visible = SR1ShipRender_Option.debug_mouse_pos
|
||
# print('sr1 mouse delta')
|
||
else:
|
||
SR1ShipRender_Option.debug_mouse_d_pos = not SR1ShipRender_Option.debug_mouse_d_pos
|
||
self.debug_mouse_delta_line.visible = SR1ShipRender_Option.debug_mouse_d_pos
|
||
# print('sr1 mouse')
|
||
|
||
elif command.find('get_buf'):
|
||
|
||
def screenshot(window):
|
||
from libs.pyglet.gl import GLubyte, GL_RGBA, GL_UNSIGNED_BYTE, \
|
||
glReadPixels
|
||
import pyglet
|
||
|
||
width = window.width
|
||
height = window.height
|
||
format_str = "RGBA"
|
||
|
||
buf = (GLubyte * (len(format_str) * width * height))()
|
||
glReadPixels(0, 0, width, height, GL_RGBA, GL_UNSIGNED_BYTE, buf)
|
||
return pyglet.image.ImageData(width, height, format_str, buf)
|
||
|
||
image_data = screenshot(self.window_pointer)
|
||
image_data.save('test.png')
|
||
elif command.find('gen_img'):
|
||
if not self.rendered:
|
||
return
|
||
if not DR_mod_runtime.use_DR_rust:
|
||
# 这个功能依赖于 DR rs (简称,我懒得在Python端实现)
|
||
return
|
||
img_box = self.rust_ship.img_pos
|
||
img_size = (img_box[2] - img_box[0] + 1000, img_box[3] - img_box[1] + 1000)
|
||
# img_center = (abs(img_box[0]), abs(img_box[1]))
|
||
# 中心点是左上角坐标
|
||
img_center = (abs(img_box[0]), abs(img_box[3]))
|
||
print(f"img_box: {img_box} img_size: {img_size} img_center: {img_center}")
|
||
try:
|
||
from PIL import Image
|
||
except ImportError:
|
||
traceback.print_exc()
|
||
print('PIL not found')
|
||
return
|
||
min_x = 0
|
||
min_y = 0
|
||
max_x = 0
|
||
max_y = 0
|
||
for part, sprite in self.parts_sprite.items():
|
||
sprite_img = sprite.image
|
||
print(f"sprite_img: {sprite_img} {sprite_img.width} {sprite_img.height}")
|
||
# 碰撞箱是居中的
|
||
# -x, -y, +x, +y
|
||
part_data = self.part_data[part]
|
||
bound_box = [-sprite_img.width / 2 + part_data.x, -sprite_img.height / 2 + part_data.y,
|
||
sprite_img.width / 2 + part_data.x, sprite_img.height / 2 + part_data.y]
|
||
min_x = min(min_x, bound_box[0])
|
||
min_y = min(min_y, bound_box[1])
|
||
max_x = max(max_x, bound_box[2])
|
||
max_y = max(max_y, bound_box[3])
|
||
print(f"min_x: {min_x} min_y: {min_y} max_x: {max_x} max_y: {max_y}")
|
||
img = Image.new('RGBA', img_size)
|
||
for part, sprite in self.parts_sprite.items():
|
||
sprite_img = sprite.image
|
||
print(f"sprite_img: {sprite_img} {sprite_img.width} {sprite_img.height}")
|
||
img_data = sprite_img.get_image_data()
|
||
fmt = img_data.format
|
||
if fmt != 'RGB':
|
||
fmt = 'RGBA'
|
||
pitch = -(img_data.width * len(fmt))
|
||
pil_image = Image.frombytes(fmt, (img_data.width, img_data.height), img_data.get_data(fmt, pitch))
|
||
pil_image = pil_image.rotate(SR1Rotation.get_rotation(self.part_data[part].angle), expand=True)
|
||
if self.part_data[part].flip_y:
|
||
pil_image.transpose(Image.FLIP_TOP_BOTTOM)
|
||
if self.part_data[part].flip_x:
|
||
pil_image.transpose(Image.FLIP_LEFT_RIGHT)
|
||
img.paste(pil_image, (int(self.part_data[part].x * 60 + img_center[0]), int(-self.part_data[part].y * 60 + img_center[1])), pil_image)
|
||
img.save(f'test{time.time()}.png', 'PNG')
|
||
|
||
def on_mouse_drag(self, x: int, y: int, dx: int, dy: int, buttons: int, modifiers: int, window: "ClientWindow"):
|
||
if not self.focus:
|
||
return
|
||
self.camera_rs.dx += dx
|
||
self.camera_rs.dy += dy
|
||
self.need_update_parts = True
|
||
# self.update_parts()
|
||
|
||
def on_file_drop(self, x: int, y: int, paths: List[str], window: "ClientWindow"):
|
||
for path in paths:
|
||
if self.load_xml(path): # 加载成功一个就停下
|
||
break
|
||
self.render_ship()
|
||
|
||
|
||
if __name__ == '__main__':
|
||
from objprint import op
|
||
|
||
op(SR1ShipRender_Option)
|