60 lines
1.7 KiB
Python
60 lines
1.7 KiB
Python
import os
|
|
import sys
|
|
|
|
from PIL import Image
|
|
|
|
def intensity_to_index(px):
|
|
indices = {0x00: 0, 0x55: 1, 0xaa: 2, 0xff: 3}
|
|
assert px in indices, px
|
|
return indices[px]
|
|
|
|
def convert(image, bpp):
|
|
assert bpp in {8, 4, 2}, bpp
|
|
px_per_byte = 8 // bpp
|
|
px_per_row = 8
|
|
bits_per_byte = 8
|
|
bytes_per_row = (px_per_row // (bits_per_byte // bpp))
|
|
|
|
assert image.mode == 'L', image.mode
|
|
width, height = image.size
|
|
|
|
buf = bytearray(width * height // px_per_byte)
|
|
|
|
for cell_y in range(height//8):
|
|
for cell_x in range(width//8):
|
|
for y in range(8):
|
|
for x in range(8):
|
|
px = im.getpixel((cell_x * 8 + x, cell_y * 8 + y))
|
|
index = intensity_to_index(px)
|
|
buf_ix = x//px_per_byte + (bytes_per_row * (cell_x * 8 + (cell_y * width) + y))
|
|
buf[buf_ix] |= (index << bpp * ((px_per_byte - 1) - (x % px_per_byte)))
|
|
return buf
|
|
|
|
def debug(buf, bpp):
|
|
assert bpp in {8, 4, 2}, bpp
|
|
px_per_byte = 8 // bpp
|
|
px_per_row = 8
|
|
bits_per_byte = 8
|
|
bytes_per_row = (px_per_row // (bits_per_byte // bpp))
|
|
bit_mask = (2 ** bpp) - 1
|
|
|
|
for row in range(len(buf) // bytes_per_row):
|
|
for x in range(bytes_per_row):
|
|
px = buf[row * bytes_per_row + x]
|
|
for shift in reversed(range(0, px_per_row, bpp)):
|
|
print((px >> shift) & bit_mask, end='')
|
|
print()
|
|
if (row % 8 == 7):
|
|
print()
|
|
|
|
in_path = sys.argv[1]
|
|
bpp = int(sys.argv[2])
|
|
out_path = sys.argv[3]
|
|
|
|
im = Image.open(in_path)
|
|
buf = convert(im, bpp)
|
|
if 'NBPP_DEBUG' in os.environ:
|
|
debug(buf, bpp)
|
|
with open(out_path, 'wb') as f:
|
|
f.write(buf)
|