2021-01-25 23:16:46 +00:00
|
|
|
"""RGB colour palettes to target for Apple II image conversions."""
|
|
|
|
|
2021-07-15 13:25:32 +00:00
|
|
|
import colour
|
2021-01-15 22:18:25 +00:00
|
|
|
import numpy as np
|
|
|
|
import image
|
2021-07-19 11:55:50 +00:00
|
|
|
import palette_ntsc
|
2021-01-15 22:18:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Palette:
|
2021-11-02 22:15:13 +00:00
|
|
|
# How many successive screen pixels are used to compute output pixel
|
|
|
|
# palette index.
|
|
|
|
PALETTE_DEPTH = None
|
|
|
|
|
|
|
|
# These next three dictionaries are all indexed by a tuple of (n-bit pixel
|
|
|
|
# value, NTSC phase), where:
|
|
|
|
# n == PALETTE_DEPTH
|
|
|
|
# MSB of the pixel value represents the current pixel on/off state
|
|
|
|
# LSB of the pixel value is the on/off state of the pixel n-1 positions
|
|
|
|
# to the left of current
|
|
|
|
# NTSC phase = 0 .. 3 (= x position % 4)
|
|
|
|
#
|
|
|
|
# The choice of LSB --> MSB increasing from left to right across the
|
|
|
|
# screen matches the ordering used by the mapping of double hi-res memory
|
|
|
|
# to screen pixels.
|
|
|
|
#
|
|
|
|
# Dictionary values are the colour of the corresponding pixel in various
|
|
|
|
# colour spaces.
|
|
|
|
|
|
|
|
# Values are pixel colour in sRGB colour space. Palettes are defined in
|
|
|
|
# this colour space.
|
2021-01-15 22:18:25 +00:00
|
|
|
SRGB = None
|
2021-11-02 22:15:13 +00:00
|
|
|
|
|
|
|
# Values are pixel colour in (linear) RGB colour space. Dithering is
|
|
|
|
# performed in this colour space.
|
2021-11-02 15:14:22 +00:00
|
|
|
RGB = {}
|
2021-01-15 22:18:25 +00:00
|
|
|
|
2021-11-02 22:15:13 +00:00
|
|
|
# Values are pixel colour in CAM16-UCS colour space. This is used for
|
|
|
|
# computing perceptual differences between colour values when optimizing
|
|
|
|
# the image dithering.
|
|
|
|
CAM16UCS = {}
|
2021-01-15 22:18:25 +00:00
|
|
|
|
2021-07-19 17:35:44 +00:00
|
|
|
def __init__(self):
|
2021-01-25 22:28:00 +00:00
|
|
|
self.RGB = {}
|
|
|
|
for k, v in self.SRGB.items():
|
|
|
|
self.RGB[k] = (np.clip(image.srgb_to_linear_array(v / 255), 0.0,
|
|
|
|
1.0) * 255).astype(np.uint8)
|
2021-07-19 17:35:44 +00:00
|
|
|
with colour.utilities.suppress_warnings(colour_usage_warnings=True):
|
|
|
|
self.CAM16UCS[k] = colour.convert(
|
|
|
|
v / 255, "sRGB", "CAM16UCS").astype(np.float32)
|
2021-07-15 12:58:22 +00:00
|
|
|
|
2021-11-02 22:15:13 +00:00
|
|
|
@staticmethod
|
|
|
|
def _pixel_phase_shifts(phase_3_srgb):
|
|
|
|
"""Constructs dictionary of 4-bit pixel sequences for each NTSC phase.
|
2021-11-02 22:24:47 +00:00
|
|
|
Assumes PALETTE_DEPTH == 4
|
2021-11-02 22:15:13 +00:00
|
|
|
|
|
|
|
Args:
|
|
|
|
phase_3_rgb: dict mapping 4-bit pixel sequence to sRGB values,
|
|
|
|
for NTSC phase 3.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
dict mapping (shifted 4-bit pixel sequence, phase 0..3) to sRGB
|
|
|
|
values
|
|
|
|
"""
|
|
|
|
srgb_phases = {}
|
|
|
|
for pixels, srgb in phase_3_srgb.items():
|
|
|
|
srgb_phases[pixels, 3] = srgb
|
|
|
|
# Rotate to compute 4-bit pixel sequences that produce the same
|
|
|
|
# colour for NTSC phases 0..2
|
|
|
|
for phase in range(0, 3):
|
|
|
|
lsb = pixels & 1
|
|
|
|
pixels >>= 1
|
|
|
|
pixels |= lsb << 3
|
|
|
|
srgb_phases[pixels, phase] = srgb
|
|
|
|
return srgb_phases
|
2021-11-02 15:45:20 +00:00
|
|
|
|
|
|
|
def bitmap_to_idx(self, pixels: np.array) -> int:
|
|
|
|
"""Converts a bitmap of pixels into integer representation.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
pixels: 1-D array of booleans, representing a window of pixels from
|
|
|
|
L to R. Must be of size <= 8
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
8-bit integer representation of pixels, suitable for use as an
|
|
|
|
index into palette arrays
|
|
|
|
"""
|
2021-11-02 15:14:22 +00:00
|
|
|
return np.packbits(
|
|
|
|
# numpy uses big-endian representation which is the opposite
|
|
|
|
# order to screen representation (i.e. LSB is the left-most
|
2021-11-02 15:45:20 +00:00
|
|
|
# screen pixel), so we need to flip the order
|
2021-11-02 15:14:22 +00:00
|
|
|
np.flip(pixels, axis=0)
|
2021-11-02 22:15:13 +00:00
|
|
|
)[0] >> (8 - pixels.shape[0])
|
2021-02-14 23:34:25 +00:00
|
|
|
|
2021-01-25 22:28:00 +00:00
|
|
|
|
2021-07-19 17:35:44 +00:00
|
|
|
class ToHgrPalette(Palette):
|
|
|
|
"""4-bit palette used as default by other DHGR image converters."""
|
|
|
|
PALETTE_DEPTH = 4
|
|
|
|
|
|
|
|
# Default tohgr/bmp2dhr palette
|
2021-11-02 22:15:13 +00:00
|
|
|
SRGB = Palette._pixel_phase_shifts({
|
2021-07-19 17:35:44 +00:00
|
|
|
0: np.array((0, 0, 0)), # Black
|
|
|
|
8: np.array((148, 12, 125)), # Magenta
|
|
|
|
4: np.array((99, 77, 0)), # Brown
|
|
|
|
12: np.array((249, 86, 29)), # Orange
|
|
|
|
2: np.array((51, 111, 0)), # Dark green
|
|
|
|
10: np.array((126, 126, 126)), # Grey2
|
|
|
|
6: np.array((67, 200, 0)), # Green
|
|
|
|
14: np.array((221, 206, 23)), # Yellow
|
|
|
|
1: np.array((32, 54, 212)), # Dark blue
|
|
|
|
9: np.array((188, 55, 255)), # Violet
|
|
|
|
5: np.array((126, 126, 126)), # Grey1
|
|
|
|
13: np.array((255, 129, 236)), # Pink
|
|
|
|
3: np.array((7, 168, 225)), # Med blue
|
|
|
|
11: np.array((158, 172, 255)), # Light blue
|
|
|
|
7: np.array((93, 248, 133)), # Aqua
|
|
|
|
15: np.array((255, 255, 255)), # White
|
2021-11-02 22:15:13 +00:00
|
|
|
})
|
2021-07-19 17:35:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
class OpenEmulatorPalette(Palette):
|
|
|
|
"""4-bit palette chosen to approximately match OpenEmulator output."""
|
|
|
|
PALETTE_DEPTH = 4
|
|
|
|
|
|
|
|
# OpenEmulator
|
2021-11-02 22:15:13 +00:00
|
|
|
SRGB = Palette._pixel_phase_shifts({
|
2021-07-19 17:35:44 +00:00
|
|
|
0: np.array((0, 0, 0)), # Black
|
|
|
|
8: np.array((203, 0, 121)), # Magenta
|
|
|
|
4: np.array((99, 103, 0)), # Brown
|
|
|
|
12: np.array((244, 78, 0)), # Orange
|
|
|
|
2: np.array((0, 150, 0)), # Dark green
|
|
|
|
10: np.array((130, 130, 130)), # Grey2
|
|
|
|
6: np.array((0, 235, 0)), # Green
|
|
|
|
14: np.array((214, 218, 0)), # Yellow
|
|
|
|
1: np.array((20, 0, 246)), # Dark blue
|
|
|
|
9: np.array((230, 0, 244)), # Violet
|
|
|
|
5: np.array((130, 130, 130)), # Grey1
|
|
|
|
13: np.array((244, 105, 235)), # Pink
|
|
|
|
3: np.array((0, 174, 243)), # Med blue
|
|
|
|
11: np.array((160, 156, 244)), # Light blue
|
|
|
|
7: np.array((25, 243, 136)), # Aqua
|
|
|
|
15: np.array((244, 247, 244)), # White
|
2021-11-02 22:15:13 +00:00
|
|
|
})
|
2021-07-19 17:35:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
class VirtualIIPalette(Palette):
|
|
|
|
"""4-bit palette exactly matching Virtual II emulator output."""
|
|
|
|
PALETTE_DEPTH = 4
|
|
|
|
|
2021-11-02 22:15:13 +00:00
|
|
|
SRGB = Palette._pixel_phase_shifts({
|
2021-07-19 17:35:44 +00:00
|
|
|
0: np.array((0, 0, 0)), # Black
|
|
|
|
8: np.array((231, 36, 66)), # Magenta
|
|
|
|
4: np.array((154, 104, 0)), # Brown
|
|
|
|
12: np.array((255, 124, 0)), # Orange
|
|
|
|
2: np.array((0, 135, 45)), # Dark green
|
|
|
|
10: np.array((104, 104, 104)), # Grey2
|
|
|
|
6: np.array((0, 222, 0)), # Green
|
|
|
|
14: np.array((255, 252, 0)), # Yellow
|
|
|
|
1: np.array((1, 30, 169)), # Dark blue
|
|
|
|
9: np.array((230, 73, 228)), # Violet
|
|
|
|
5: np.array((185, 185, 185)), # Grey1
|
|
|
|
13: np.array((255, 171, 153)), # Pink
|
|
|
|
3: np.array((47, 69, 255)), # Med blue
|
|
|
|
11: np.array((120, 187, 255)), # Light blue
|
|
|
|
7: np.array((83, 250, 208)), # Aqua
|
|
|
|
15: np.array((255, 255, 255)), # White
|
2021-11-02 22:15:13 +00:00
|
|
|
})
|
2021-01-25 22:28:00 +00:00
|
|
|
|
|
|
|
|
2021-02-14 23:34:25 +00:00
|
|
|
class NTSCPalette(Palette):
|
2021-03-15 10:45:33 +00:00
|
|
|
"""8-bit NTSC palette computed by averaging chroma signal over 8 pixels."""
|
2021-02-14 23:34:25 +00:00
|
|
|
PALETTE_DEPTH = 8
|
|
|
|
|
|
|
|
# Computed using ntsc_colours.py
|
2021-07-19 11:55:50 +00:00
|
|
|
SRGB = palette_ntsc.SRGB
|
2021-02-14 23:34:25 +00:00
|
|
|
|
2021-07-19 16:54:46 +00:00
|
|
|
|
2021-01-25 22:28:00 +00:00
|
|
|
PALETTES = {
|
2021-07-19 17:35:44 +00:00
|
|
|
'openemulator': OpenEmulatorPalette,
|
|
|
|
'virtualii': VirtualIIPalette,
|
|
|
|
'tohgr': ToHgrPalette,
|
2021-02-14 23:34:25 +00:00
|
|
|
'ntsc': NTSCPalette
|
2021-01-25 22:28:00 +00:00
|
|
|
}
|
|
|
|
|
2021-07-19 11:55:50 +00:00
|
|
|
DEFAULT_PALETTE = 'ntsc'
|