113 lines
3.0 KiB
Python
113 lines
3.0 KiB
Python
# Copyright (c) 2023 Dominic Masters
|
|
#
|
|
# This software is released under the MIT License.
|
|
# https://opensource.org/licenses/MIT
|
|
|
|
|
|
from PIL import Image
|
|
import argparse
|
|
import os
|
|
|
|
# Args
|
|
parser = argparse.ArgumentParser(description='Converts image textures to internal game data format.')
|
|
parser.add_argument('-i', '--input');
|
|
parser.add_argument('-o', '--output');
|
|
parser.add_argument('-s', '--scale');
|
|
parser.add_argument('-sf', '--scale-filter');
|
|
parser.add_argument('-wx', '--wrap-x');
|
|
parser.add_argument('-wy', '--wrap-y');
|
|
parser.add_argument('-fi', '--filter-min')
|
|
parser.add_argument('-fg', '--filter-mag')
|
|
parser.add_argument('-csx', '--crop-start-x');
|
|
parser.add_argument('-csy', '--crop-start-y');
|
|
parser.add_argument('-cex', '--crop-end-x');
|
|
parser.add_argument('-cey', '--crop-end-y');
|
|
args = parser.parse_args()
|
|
|
|
# Ensure input exists
|
|
if not os.path.exists(args.input):
|
|
print(f"Input file '{args.input}' does not exist.")
|
|
exit(1)
|
|
|
|
# Open image
|
|
img = Image.open(args.input)
|
|
|
|
# Normalize the image
|
|
hasAlpha = 'transparency' in img.info
|
|
|
|
# Convert the image to RGB or RGBA mode based on alpha channel
|
|
if hasAlpha:
|
|
img = img.convert('RGBA')
|
|
else:
|
|
img = img.convert('RGB')
|
|
|
|
# Apply cropping
|
|
crop_box = [
|
|
int(args.crop_start_x) if args.crop_start_x not in (None, "") else 0,
|
|
int(args.crop_start_y) if args.crop_start_y not in (None, "") else 0,
|
|
int(args.crop_end_x) if args.crop_end_x not in (None, "") else img.width,
|
|
int(args.crop_end_y) if args.crop_end_y not in (None, "") else img.height
|
|
]
|
|
img = img.crop(crop_box)
|
|
|
|
# Apply scaling
|
|
if args.scale not in (None, ""):
|
|
scale = float(args.scale)
|
|
newSize = (int(img.width * scale), int(img.height * scale))
|
|
if args.scale_filter == 'NEAREST':
|
|
img = img.resize(newSize, Image.NEAREST)
|
|
elif args.scale_filter == 'BILINEAR':
|
|
img = img.resize(newSize, Image.BILINEAR)
|
|
elif args.scale_filter == 'BICUBIC':
|
|
img = img.resize(newSize, Image.BICUBIC)
|
|
elif args.scale_filter == 'LANCZOS':
|
|
img = img.resize(newSize, Image.LANCZOS)
|
|
else:
|
|
img = img.resize(newSize)
|
|
|
|
|
|
# Filter
|
|
if args.filter_min.lower() == 'NEAREST':
|
|
filterMin = 0
|
|
else:
|
|
filterMin = 1
|
|
|
|
if args.filter_mag.lower() == 'NEAREST':
|
|
filterMag = 0
|
|
else:
|
|
filterMag = 1
|
|
|
|
# Wrap
|
|
if args.wrap_x.lower() == 'repeat':
|
|
wrapX = 0
|
|
elif args.wrap_x.lower() == 'mirror':
|
|
wrapX = 1
|
|
elif args.wrap_x.lower() == 'clamp':
|
|
wrapX = 2
|
|
elif args.wrap_x.lower() == 'border':
|
|
wrapX = 3
|
|
else:
|
|
wrapX = 2
|
|
|
|
if args.wrap_y.lower() == 'repeat':
|
|
wrapY = 0
|
|
elif args.wrap_y.lower() == 'mirror':
|
|
wrapY = 1
|
|
elif args.wrap_y.lower() == 'clamp':
|
|
wrapY = 2
|
|
elif args.wrap_y.lower() == 'border':
|
|
wrapY = 3
|
|
else:
|
|
wrapY = 2
|
|
|
|
# Get raw pixel data
|
|
buffer = img.tobytes()
|
|
|
|
# Create the output directory if it doesn't exist
|
|
os.makedirs(os.path.dirname(args.output), exist_ok=True)
|
|
|
|
# Write the image metadata and pixel data to the output file
|
|
with open(args.output, 'wb') as f:
|
|
header = f"DT_2.00|{img.width}|{img.height}|{4 if hasAlpha else 3}|{wrapX}|{wrapY}|{filterMin}|{filterMag}|"
|
|
f.write(header.encode())
|
|
f.write(buffer) |