Musicreater/msctPkgver/utils.py

223 lines
6.5 KiB
Python
Raw Normal View History

2023-01-20 00:31:45 +08:00
import math
import os
key = {
"x": [b"\x0f", b"\x0e", b"\x1c", b"\x14", b"\x15"],
"y": [b"\x11", b"\x10", b"\x1d", b"\x16", b"\x17"],
"z": [b"\x13", b"\x12", b"\x1e", b"\x18", b"\x19"],
}
"""key存储了方块移动指令的数据其中可以用key[x|y|z][0|1]来表示xyz的减或增
而key[][2+]是用来增加指定数目的"""
x = "x"
y = "y"
z = "z"
def move(axis: str, value: int):
if value == 0:
return b""
2023-01-20 00:31:45 +08:00
if abs(value) == 1:
return key[axis][0 if value == -1 else 1]
pointer = sum(
[
1 if i else 0
for i in (
value != -1,
value < -1 or value > 1,
value < -128 or value > 127,
value < -32768 or value > 32767,
)
]
)
return key[axis][pointer] + value.to_bytes(2 ** (pointer - 2), "big", signed=True)
2023-01-20 00:31:45 +08:00
2023-02-01 19:19:40 +08:00
def compress_zipfile(sourceDir, outFilename, compression=8, exceptFile=None):
2023-01-20 00:31:45 +08:00
"""使用compression指定的算法打包目录为zip文件\n
默认算法为DEFLATED(8),可用算法如下\n
STORED = 0\n
DEFLATED = 8\n
BZIP2 = 12\n
LZMA = 14\n
"""
import zipfile
zipf = zipfile.ZipFile(outFilename, "w", compression)
pre_len = len(os.path.dirname(sourceDir))
for parent, dirnames, filenames in os.walk(sourceDir):
for filename in filenames:
if filename == exceptFile:
continue
pathfile = os.path.join(parent, filename)
arc_name = pathfile[pre_len:].strip(os.path.sep) # 相对路径
zipf.write(pathfile, arc_name)
2023-01-20 00:31:45 +08:00
zipf.close()
2023-01-29 13:02:43 +08:00
def form_command_block_in_BDX_bytes(
command: str,
particularValue: int,
impluse: int = 0,
condition: bool = False,
needRedstone: bool = True,
tickDelay: int = 0,
customName: str = "",
executeOnFirstTick: bool = False,
trackOutput: bool = True,
2023-01-20 00:31:45 +08:00
):
"""
使用指定项目返回指定的指令方块放置指令项
:param command: `str`
指令
:param particularValue:
方块特殊值即朝向
:0 无条件
:1 无条件
:2 z轴负方向 无条件
:3 z轴正方向 无条件
:4 x轴负方向 无条件
:5 x轴正方向 无条件
:6 无条件
:7 无条件
:8 有条件
:9 有条件
:10 z轴负方向 有条件
:11 z轴正方向 有条件
:12 x轴负方向 有条件
:13 x轴正方向 有条件
:14 有条件
:14 有条件
注意此处特殊值中的条件会被下面condition参数覆写
:param impluse: `int 0|1|2`
方块类型
0脉冲 1循环 2连锁
:param condition: `bool`
是否有条件
:param needRedstone: `bool`
是否需要红石
:param tickDelay: `int`
执行延时
:param customName: `str`
悬浮字
lastOutput: `str`
上次输出字符串注意此处需要留空
:param executeOnFirstTick: `bool`
执行第一个已选项(循环指令方块是否激活后立即执行若为False则从激活时起延迟后第一次执行)
:param trackOutput: `bool`
是否输出
:return:str
"""
block = b"\x24" + particularValue.to_bytes(2, byteorder="big", signed=False)
2023-01-20 00:31:45 +08:00
for i in [
impluse.to_bytes(4, byteorder="big", signed=False),
bytes(command, encoding="utf-8") + b"\x00",
bytes(customName, encoding="utf-8") + b"\x00",
bytes("", encoding="utf-8") + b"\x00",
tickDelay.to_bytes(4, byteorder="big", signed=True),
executeOnFirstTick.to_bytes(1, byteorder="big"),
trackOutput.to_bytes(1, byteorder="big"),
condition.to_bytes(1, byteorder="big"),
needRedstone.to_bytes(1, byteorder="big"),
]:
block += i
return block
2023-02-01 19:19:40 +08:00
def bottem_side_length_of_smallest_square_bottom_box(total: int, maxHeight: int):
2023-01-20 00:31:45 +08:00
"""给定总方块数量和最大高度,返回所构成的图形外切正方形的边长
:param total: 总方块数量
:param maxHeight: 最大高度
:return: 外切正方形的边长 int"""
return math.ceil(math.sqrt(math.ceil(total / maxHeight)))
2023-01-29 13:02:43 +08:00
def to_BDX_bytes(
commands: list,
max_height: int = 64,
2023-01-20 00:31:45 +08:00
):
"""
:param commands: 指令列表(指令, 延迟)
:param max_height: 生成结构最大高度
2023-01-20 00:31:45 +08:00
:return 成功与否成功返回(True,未经过压缩的源,结构占用大小)失败返回(False,str失败原因)
"""
2023-02-01 19:19:40 +08:00
_sideLength = bottem_side_length_of_smallest_square_bottom_box(len(commands), max_height)
_bytes = b""
2023-01-20 00:31:45 +08:00
y_forward = True
z_forward = True
2023-01-20 00:31:45 +08:00
now_y = 0
now_z = 0
now_x = 0
2023-01-20 00:31:45 +08:00
for cmd, delay in commands:
impluse = 2
condition = False
needRedstone = False
tickDelay = delay
customName = ""
executeOnFirstTick = False
trackOutput = True
2023-01-29 13:02:43 +08:00
_bytes += form_command_block_in_BDX_bytes(
2023-01-20 00:31:45 +08:00
cmd,
(1 if y_forward else 0)
2023-01-20 00:31:45 +08:00
if (
((now_y != 0) and (not y_forward))
or (y_forward and (now_y != (max_height - 1)))
2023-01-20 00:31:45 +08:00
)
else (3 if z_forward else 2)
2023-01-20 00:31:45 +08:00
if (
((now_z != 0) and (not z_forward))
or (z_forward and (now_z != _sideLength))
2023-01-20 00:31:45 +08:00
)
else 5,
impluse=impluse,
condition=condition,
needRedstone=needRedstone,
tickDelay=tickDelay,
customName=customName,
executeOnFirstTick=executeOnFirstTick,
trackOutput=trackOutput,
2023-01-20 00:31:45 +08:00
)
now_y += 1 if y_forward else -1
2023-01-20 00:31:45 +08:00
if ((now_y >= max_height) and y_forward) or ((now_y < 0) and (not y_forward)):
now_y -= 1 if y_forward else -1
2023-01-20 00:31:45 +08:00
y_forward = not y_forward
2023-01-20 00:31:45 +08:00
now_z += 1 if z_forward else -1
2023-01-20 00:31:45 +08:00
if ((now_z > _sideLength) and z_forward) or (
(now_z < 0) and (not z_forward)
):
now_z -= 1 if z_forward else -1
z_forward = not z_forward
2023-01-20 00:31:45 +08:00
_bytes += key[x][1]
now_x += 1
2023-01-20 00:31:45 +08:00
else:
_bytes += key[z][int(z_forward)]
2023-01-20 00:31:45 +08:00
else:
_bytes += key[y][int(y_forward)]
2023-01-20 00:31:45 +08:00
return (
_bytes,
[
now_x + 1,
max_height if now_x or now_z else now_y,
_sideLength if now_x else now_z,
],
[now_x, now_y, now_z],
2023-01-20 00:31:45 +08:00
)