skyeweave/atlasimagecomposer/backend/compose.py

119 lines
4 KiB
Python
Raw Normal View History

2024-10-04 17:30:40 +00:00
import math
2024-08-09 16:57:33 +00:00
import pathlib
import sys
2024-10-04 18:18:56 +00:00
from typing import Tuple
2024-08-09 16:57:33 +00:00
import numpy as np
from PIL import Image
from tqdm.contrib import itertools as tqdm_itertools
from .atlas import fetch_data, fetch_image
2024-10-04 18:18:56 +00:00
from ..config import ExpressionDefaults, Paths
2024-08-09 16:57:33 +00:00
2024-10-04 18:18:56 +00:00
def compose():
2024-08-09 16:57:33 +00:00
Paths.IMAGES.mkdir(exist_ok=True)
Paths.OUTPUT.mkdir(exist_ok=True)
servantid = input("Enter servant ID: ")
try:
t = int(servantid)
2024-08-09 17:09:05 +00:00
if t <= 0:
2024-08-09 16:57:33 +00:00
raise ValueError
except ValueError:
print("Servant ID has to be a valid integer above 0")
sys.exit(1)
sprites = fetch_data(servantid)
for sprite in sprites:
fetch_image(servantid, sprite)
path = Paths.IMAGES / servantid
for f in path.iterdir():
2024-10-04 17:30:40 +00:00
process_sprite(f, sprites[f.stem]["position"], sprites[f.stem]["faceSize"], servantid)
2024-08-09 17:14:12 +00:00
print(f"Files have been saved at: {(Paths.OUTPUT / servantid).absolute()}")
2024-08-09 16:57:33 +00:00
2024-10-04 18:18:56 +00:00
def override_expressions(expressions: Image.Image, facesize: int) -> Tuple[Image.Image, int]:
exp_w, exp_h = expressions.size
2024-10-04 18:22:38 +00:00
composites, offset, floating_offset = [], facesize // 5, 0
2024-10-04 18:18:56 +00:00
per_composite = (ExpressionDefaults.faceSize * 4) // facesize
# Why ..........
composite_count = math.ceil((exp_h - 1280) / (offset + facesize * per_composite)) + 1
# Remove top offset
expressions = expressions.crop((0, 256, exp_w, exp_h))
for count in range(0, composite_count):
# width_topleft, height_topleft, width_botright, height_botright
composites.append((0, 3 * count * facesize + floating_offset, exp_w, (3 * count + 3) * facesize + floating_offset - 1 ))
floating_offset += offset
# Create new expression image, then add all compositions to it
overrides = Image.new("RGBA", (exp_w, facesize * composite_count * 3), (255, 0, 0, 0))
for idx, c in enumerate(composites):
crop = expressions.crop(c)
pos = (0, idx * 3 * facesize)
overrides.paste(crop, pos, crop)
# Override expressions sheet
return overrides.copy(), per_composite
# pylint: disable=too-many-locals
def process_sprite(filepath: pathlib.Path, position: tuple, facesize: int, servantid: str):
2024-08-09 16:57:33 +00:00
im = Image.open(filepath)
width, height = im.size
main_sprite = im.crop((0, 0, width, 768))
save_sprite(main_sprite, servantid, filepath.stem)
expressions = im.crop((0, 768, width, height))
2024-10-04 17:30:40 +00:00
expressionwidth, expressionheight = expressions.size
2024-10-04 18:18:56 +00:00
rows = expressionheight // facesize
expressions_p_row = expressionwidth // facesize
2024-10-04 17:30:40 +00:00
exp_per_row = 4
2024-10-04 18:18:56 +00:00
if facesize != ExpressionDefaults.faceSize:
expressions, exp_per_row = override_expressions(expressions, facesize)
2024-10-04 17:30:40 +00:00
for x, y in tqdm_itertools.product(range(0, rows), range(0, expressions_p_row, 1), ascii="-="):
2024-10-04 18:18:56 +00:00
img = generate_sprite(main_sprite, expressions, x, y, position, facesize)
2024-08-09 16:57:33 +00:00
if img is not None:
2024-10-04 17:30:40 +00:00
save_sprite(img, servantid, filepath.stem, (x, y, exp_per_row))
2024-08-09 16:57:33 +00:00
def is_empty(img: Image.Image):
data = np.asarray(img.crop((96, 96, 160, 160)).convert('LA'))
np.reshape(data, (-1, 1))
_, count_unique = np.unique(data, return_counts=True)
if count_unique.size < 10:
return True
return False
2024-10-04 18:18:56 +00:00
# pylint: disable=too-many-arguments
def generate_sprite(sprite: Image.Image, expressions: Image.Image, row: int, col: int, position: tuple, facesize: int) -> Image.Image | None:
2024-08-09 16:57:33 +00:00
area = (
2024-10-04 18:18:56 +00:00
col * facesize,
row * facesize,
(col + 1) * facesize - 1,
(row + 1) * facesize - 1
2024-08-09 16:57:33 +00:00
)
expression = expressions.crop(area)
if is_empty(expression):
return None
2024-10-04 18:18:56 +00:00
composition = sprite.copy()
composition.paste(expression, position, expression)
return composition
2024-08-09 16:57:33 +00:00
def save_sprite(image: Image.Image, folder: str, name: str, rowcol: tuple | None = None):
savefolder = Paths.OUTPUT / folder
if not savefolder.is_dir():
savefolder.mkdir()
2024-10-04 17:30:40 +00:00
postfix = f"_{rowcol[0] * rowcol[2] + rowcol[1] + 1}" if rowcol is not None else "_0"
2024-08-09 16:57:33 +00:00
outfile = savefolder / f"{name}{postfix}.png"
with open(outfile, 'wb') as file:
image.save(file)