# Copyright (C) 2016-2025 Jan Berges
# This program is free software under the terms of the BSD Zero Clause License.
"""Save and load RGB(A) graphics."""
from __future__ import division
import zlib, struct
[docs]
def save(filename, image):
"""Save image as 8-bit PNG without filtering.
Specified at https://www.w3.org/TR/PNG/.
Inspired by Blender thumbnailer code.
RGB images with less than 256 colors will be saved using indexed color.
Parameters
----------
filename : str
Name of PNG file to be written.
image : list of list of list
8-bit image data of shape (height, width, colors), where colors may be
1 (grayscale), 2 (grayscale and alpha), 3 (RGB), or 4 (RGBA).
"""
height = len(image)
width = len(image[0])
colors = len(image[0][0])
color = {1: 0, 2: 4, 3: 2, 4: 6}[colors]
image = [[tuple(min(max(int(round(x)), 0), 255) for x in col)
for col in row] for row in image]
if colors == 3:
pixels = [col for row in image for col in row]
plte = sorted(set(pixels))
# assign 0 to most common color so that 0 before row does not stand out:
count = dict()
for x in pixels:
count[x] = count.get(x, 0) + 1
imax = plte.index(max(count, key=count.get))
plte[0], plte[imax] = plte[imax], plte[0]
if len(plte) < 256:
color = 3
image = [[[plte.index(col)] for col in row] for row in image]
plte = [x for col in plte for x in col]
byte = [x for row in image for col in [[0]] + row for x in col]
# 0 before each row: https://www.w3.org/TR/PNG/#4Concepts.EncodingFiltering
with open(filename, 'wb') as png:
png.write(b'\x89PNG\r\n\x1a\n')
def chunk(name, data):
png.write(struct.pack('!I', len(data)))
png.write(name)
png.write(data)
png.write(struct.pack('!I', zlib.crc32(name + data) & 0xffffffff))
chunk(b'IHDR', struct.pack('!2I5B', width, height, 8, color, 0, 0, 0))
if color == 3:
chunk(b'PLTE', struct.pack('%dB' % len(plte), *plte))
chunk(b'IDAT',
zlib.compress(struct.pack('%dB' % len(byte), *byte), 9))
chunk(b'IEND', b'')
[docs]
def load(filename):
"""Load 8-bit PNG image.
Parameters
----------
filename : str
Name of PNG file to be written.
Returns
-------
list of list of list
8-bit image data of shape (height, width, colors), where colors may be
1 (grayscale), 2 (grayscale and alpha), 3 (RGB), or 4 (RGBA).
"""
with open(filename, 'rb') as png:
png.read(8)
idat = b''
while True:
size, = struct.unpack('!I', png.read(4))
name = png.read(4)
data = png.read(size)
csum = png.read(4)
if struct.pack('!I', zlib.crc32(name + data) & 0xffffffff) != csum:
print("Chunk '%s' corrupted!" % name)
elif name == b'IHDR':
width, height, _, color, _, _, _ = struct.unpack('!2I5B', data)
colors = {0: 1, 2: 3, 3: 1, 4: 2, 6: 4}[color]
elif name == b'PLTE':
byte = struct.unpack('%dB' % len(data), data)
plte = [[byte[3 * n + z]
for z in range(3)]
for n in range(len(data) // 3)]
elif name == b'IDAT':
idat += data
elif name == b'IEND':
break
data = zlib.decompress(idat)
byte = iter(struct.unpack('%dB' % len(data), data))
filters = []
image = [filters.append(next(byte)) or [[next(byte)
for z in range(colors)]
for x in range(width)]
for y in range(height)]
for y in range(height):
if filters[y] == 0:
continue
for x in range(width):
for z in range(colors):
A = image[y][x - 1][z] if x else 0
B = image[y - 1][x][z] if y else 0
C = image[y - 1][x - 1][z] if x and y else 0
if filters[y] == 1:
image[y][x][z] += A
elif filters[y] == 2:
image[y][x][z] += B
elif filters[y] == 3:
image[y][x][z] += (A + B) // 2
elif filters[y] == 4:
p = A + B - C
image[y][x][z] += min(A, B, C, key=lambda q: abs(p - q))
image[y][x][z] %= 256
if color == 3:
image = [[plte[image[y][x][0]]
for x in range(width)]
for y in range(height)]
return image