Source code for storylines.png

# 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