Files
lilygo-t5-template/lib/epdiy/scripts/imgconvert.py
Кобелев Андрей Андреевич a111352dc5 Прошло 2 года.
2026-03-10 22:54:23 +03:00

76 lines
2.9 KiB
Python
Executable File

#!python3
from PIL import Image
from argparse import ArgumentParser
import sys
import math
parser = ArgumentParser()
parser.add_argument('-i', action="store", dest="inputfile", required=True)
parser.add_argument('-n', action="store", dest="name", required=True)
parser.add_argument('-o', action="store", dest="outputfile", required=True)
parser.add_argument('-maxw', action="store", dest="max_width", default=1200, type=int)
parser.add_argument('-maxh', action="store", dest="max_height", default=825, type=int)
parser.add_argument('-nd', action="store_false", dest="dither", required=False, default=True)
parser.add_argument('-l', action="store", dest="levels", default=16, type=int)
args = parser.parse_args()
if args.levels > 16:
raise Exception("Maximum 16 levels of gray supported by this program")
if args.levels < 2:
# less than 2 doesn't make sense, there is at least on and off
arg.levels = 2
if args.max_height < 1:
raise Exception("Max height cannot be lower than 1")
if args.max_width < 1:
raise Exception("Max width cannot be lower than 1")
im = Image.open(args.inputfile)
if im.mode == 'I;16': # some pngs load like this
# so we fix it
im = im.point(lambda i:i*(1./256)).convert('L')
im.thumbnail((args.max_width, args.max_height), Image.LANCZOS)
if im.mode != 'L':
if im.mode != 'RGB':
im = im.convert(mode='RGB') # drop alpha
# now we can convert it to grayscale to get rid of colors
im = im.convert(mode='L')
# but the quant algorithm works much better on RGB, so go to RGB mode just for that (even though we're still grayscale)
im = im.convert(mode='RGB')
# prepare a palette for the levels of gray
paletteImage = Image.new(mode='P', size=[1,1])
# we make a palette that matches the indexed colors of the display, just repeating the same value for rgb
paletteColors = [round(i * 255.0 / (args.levels - 1)) for i in range(0,args.levels) for c in range(0,3)]
paletteImage.putpalette(paletteColors, rawmode='RGB')
im = im.quantize(colors=args.levels, palette=paletteImage, dither=Image.Dither.FLOYDSTEINBERG if args.dither else Image.Dither.NONE)
# now it's quantizied to the palette, 0 == dark, args.levels-1 == white
# Write out the output file.
with open(args.outputfile, 'w') as f:
f.write("const uint32_t {}_width = {};\n".format(args.name, im.size[0]))
f.write("const uint32_t {}_height = {};\n".format(args.name, im.size[1]))
f.write(
"const uint8_t {}_data[({}*{})/2] = {{\n".format(args.name, math.ceil(im.size[0] / 2) * 2, im.size[1])
)
for y in range(0, im.size[1]):
byte = 0
for x in range(0, im.size[0]):
l = im.getpixel((x, y))
if x % 2 == 0:
byte = l
else:
byte |= l << 4
f.write("0x{:02X}, ".format(byte))
if im.size[0] % 2 == 1:
f.write("0x{:02X}, ".format(byte))
f.write("\n\t")
f.write("};\n")