2020-10-16 18:41:34 +00:00
|
|
|
package screen
|
2019-05-03 18:09:53 +00:00
|
|
|
|
|
|
|
import (
|
2020-10-04 17:21:49 +00:00
|
|
|
"fmt"
|
2019-05-03 18:09:53 +00:00
|
|
|
"image"
|
|
|
|
"image/color"
|
|
|
|
)
|
|
|
|
|
2020-08-09 13:39:41 +00:00
|
|
|
/*
|
2024-01-06 20:48:23 +00:00
|
|
|
See:
|
|
|
|
|
|
|
|
https://mrob.com/pub/xapple2/colors.html
|
|
|
|
https://archive.org/details/IIgs_2523063_Master_Color_Values
|
2020-08-09 13:39:41 +00:00
|
|
|
*/
|
|
|
|
var ntscColorMap = [16]color.Color{
|
|
|
|
color.RGBA{0, 0, 0, 255}, // Black
|
|
|
|
color.RGBA{227, 30, 96, 255}, // Magenta
|
|
|
|
color.RGBA{96, 78, 189, 255}, // Dark Blue
|
|
|
|
color.RGBA{255, 68, 253, 255}, // Purple
|
|
|
|
color.RGBA{0, 163, 96, 255}, // Dark Green
|
|
|
|
color.RGBA{156, 156, 156, 255}, // Grey 1
|
|
|
|
color.RGBA{20, 207, 253, 255}, // Medium Blue
|
|
|
|
color.RGBA{208, 195, 255, 255}, // Light Blue
|
|
|
|
color.RGBA{96, 114, 3, 255}, // Brown
|
|
|
|
color.RGBA{255, 106, 60, 255}, // Orange
|
|
|
|
color.RGBA{156, 156, 156, 255}, // Grey 2
|
|
|
|
color.RGBA{255, 160, 208, 255}, // Pink
|
|
|
|
color.RGBA{20, 245, 60, 255}, // Green
|
|
|
|
color.RGBA{208, 221, 141, 255}, // Yellow
|
|
|
|
color.RGBA{114, 255, 208, 255}, // Aquamarine
|
|
|
|
color.RGBA{255, 255, 255, 255}, // White
|
|
|
|
}
|
2019-05-12 17:22:32 +00:00
|
|
|
|
2022-05-09 12:34:47 +00:00
|
|
|
var attenuatedColorMap = buildAttenuatedColorMap(ntscColorMap)
|
|
|
|
|
|
|
|
func buildAttenuatedColorMap(colorMap [16]color.Color) [16]color.Color {
|
|
|
|
colors := [16]color.Color{}
|
|
|
|
for i := 0; i < len(colorMap); i++ {
|
|
|
|
r, g, b, _ := colorMap[i].RGBA()
|
|
|
|
colors[i] = color.RGBA64{
|
|
|
|
uint16(r / 2), uint16(g / 2), uint16(b / 2),
|
|
|
|
65535,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return colors
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2020-08-09 13:39:41 +00:00
|
|
|
var rgbColorMap = [16]color.Color{
|
|
|
|
color.RGBA{0, 0, 0, 255}, // Black
|
|
|
|
color.RGBA{221, 0, 51, 255}, // Magenta
|
|
|
|
color.RGBA{0, 0, 153, 255}, // Dark Blue
|
|
|
|
color.RGBA{221, 34, 221, 255}, // Purple
|
|
|
|
color.RGBA{0, 119, 34, 255}, // Dark Green
|
|
|
|
color.RGBA{85, 85, 85, 255}, // Grey 1
|
|
|
|
color.RGBA{34, 34, 255, 255}, // Medium Blue
|
|
|
|
color.RGBA{102, 170, 255, 255}, // Light Blue
|
|
|
|
color.RGBA{136, 85, 0, 255}, // Brown
|
|
|
|
color.RGBA{255, 102, 0, 255}, // Orange
|
|
|
|
color.RGBA{170, 170, 170, 255}, // Grey 2
|
|
|
|
color.RGBA{255, 153, 136, 255}, // Pink
|
|
|
|
color.RGBA{17, 221, 0, 255}, // Green
|
|
|
|
color.RGBA{255, 255, 0, 255}, // Yellow
|
|
|
|
color.RGBA{68, 255, 153, 255}, // Aquamarine
|
|
|
|
color.RGBA{255, 255, 255, 255}, // White
|
2019-05-03 18:09:53 +00:00
|
|
|
}
|
2022-05-09 12:34:47 +00:00
|
|
|
*/
|
2019-05-03 18:09:53 +00:00
|
|
|
|
2020-10-16 13:05:38 +00:00
|
|
|
func filterNTSCColor(in *image.RGBA, mask *image.Alpha, screenMode int) *image.RGBA {
|
2020-08-09 13:39:41 +00:00
|
|
|
colorMap := ntscColorMap // or rgbColorMap
|
2022-05-09 12:34:47 +00:00
|
|
|
colorMapLow := ntscColorMap
|
2020-10-16 13:05:38 +00:00
|
|
|
if screenMode == ScreenModeNTSC {
|
2022-05-09 12:34:47 +00:00
|
|
|
colorMapLow = attenuatedColorMap
|
2020-10-04 14:10:34 +00:00
|
|
|
}
|
2019-05-03 18:09:53 +00:00
|
|
|
|
|
|
|
b := in.Bounds()
|
2020-08-09 14:42:16 +00:00
|
|
|
width := b.Dx()
|
|
|
|
height := b.Dy()
|
|
|
|
size := image.Rect(0, 0, width+4, height)
|
2019-05-03 18:09:53 +00:00
|
|
|
out := image.NewRGBA(size)
|
|
|
|
|
2020-10-04 17:21:49 +00:00
|
|
|
if width < 2*hiResWidth {
|
|
|
|
panic(fmt.Sprintf("The image has width %v. We can't apply the NTSC filter.", width))
|
|
|
|
}
|
|
|
|
|
2020-08-09 14:42:16 +00:00
|
|
|
for y := 0; y < height; y++ {
|
2019-11-09 16:44:04 +00:00
|
|
|
// We store the last four bits. We start with 0000
|
2019-05-05 10:51:30 +00:00
|
|
|
v := 0
|
2020-08-09 14:42:16 +00:00
|
|
|
for x := 0; x < width; x++ {
|
2019-05-03 18:09:53 +00:00
|
|
|
cIn := in.At(x, y)
|
|
|
|
r, _, _, _ := cIn.RGBA()
|
|
|
|
|
2020-08-09 13:39:41 +00:00
|
|
|
pos := 1 << uint(x%4)
|
2019-05-03 18:09:53 +00:00
|
|
|
if r != 0 {
|
|
|
|
v |= pos
|
|
|
|
} else {
|
|
|
|
v &^= pos
|
2020-08-06 16:35:34 +00:00
|
|
|
}
|
|
|
|
|
2020-10-04 14:10:34 +00:00
|
|
|
var cOut color.Color
|
|
|
|
if r != 0 {
|
|
|
|
cOut = colorMap[v]
|
|
|
|
} else {
|
2022-05-09 12:34:47 +00:00
|
|
|
cOut = colorMapLow[v]
|
2020-10-04 14:10:34 +00:00
|
|
|
}
|
2020-08-06 16:35:34 +00:00
|
|
|
if mask != nil {
|
2020-10-04 14:10:34 +00:00
|
|
|
// RGB mode7
|
2020-08-06 16:35:34 +00:00
|
|
|
_, _, _, a := mask.At(x, y).RGBA()
|
|
|
|
if a > 0 {
|
|
|
|
cOut = cIn
|
2019-05-03 18:09:53 +00:00
|
|
|
}
|
|
|
|
}
|
2020-08-06 16:35:34 +00:00
|
|
|
|
2019-05-03 18:09:53 +00:00
|
|
|
out.Set(x, y, cOut)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We fade for the last three positions
|
2020-08-09 14:42:16 +00:00
|
|
|
for x := width; x < width+4; x++ {
|
2020-08-09 13:39:41 +00:00
|
|
|
v >>= 1
|
2019-05-03 18:09:53 +00:00
|
|
|
cOut := colorMap[v]
|
|
|
|
out.Set(x, y, cOut)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|