izapple2/screenHiRes.go
2019-11-09 18:13:19 +01:00

112 lines
2.6 KiB
Go

package apple2
import (
"image"
"image/color"
)
const (
hiResWidth = 280
hiResLineBytes = hiResWidth / 7
doubleHiResWidth = 2 * hiResWidth
hiResHeight = 192
hiResHeightMixed = 160
hiResPage1Address = uint16(0x2000)
hiResPage2Address = uint16(0x4000)
)
func getHiResLineOffset(line int) uint16 {
// See "Understanding the Apple II", page 5-14
// http://www.applelogic.org/files/UNDERSTANDINGTHEAII.pdf
section := line >> 6 // Top, middle and bottom
outerEigth := (line >> 3) & 0x07
innerEigth := line & 0x07
return uint16(section*40 + outerEigth*0x80 + innerEigth*0x400)
}
func getHiResLine(a *Apple2, line int, isSecondPage bool, auxMem bool) []uint8 {
address := hiResPage1Address
if isSecondPage {
address = hiResPage2Address
}
address += getHiResLineOffset(line)
return a.mmu.getPhysicalMainRAM(auxMem).subRange(address, address+hiResLineBytes)
}
func snapshotHiResModeMono(a *Apple2, isSecondPage bool, mixedMode bool, light color.Color) *image.RGBA {
// As described in "Undertanding the Apple II", with half pixel shifts
height := hiResHeight
if mixedMode {
height = hiResHeightMixed
}
size := image.Rect(0, 0, 2*hiResWidth, height)
img := image.NewRGBA(size)
for y := 0; y < height; y++ {
bytes := getHiResLine(a, y, isSecondPage, false /*auxMem*/)
x := 0
var previousColour color.Color = color.Black
for _, b := range bytes {
shifted := b>>7 == 1
for j := uint(0); j < 7; j++ {
bit := (b >> j) & 1
colour := light
if bit == 0 {
colour = color.Black
}
if shifted {
img.Set(x, y, previousColour)
} else {
img.Set(x, y, colour)
}
img.Set(x+1, y, colour)
previousColour = colour
x += 2
}
}
}
return img
}
func snapshotDoubleHiResModeMono(a *Apple2, isSecondPage bool, mixedMode bool, light color.Color) *image.RGBA {
// As described in "Inside the Apple IIe"
height := hiResHeight
if mixedMode {
height = hiResHeightMixed
}
size := image.Rect(0, 0, doubleHiResWidth, height)
img := image.NewRGBA(size)
for y := 0; y < height; y++ {
lineParts := [][]uint8{
getHiResLine(a, y, isSecondPage, true),
getHiResLine(a, y, isSecondPage, false),
}
x := 0
// For the NTSC filter to work we have to insert an initial black pixel and skip the last one
img.Set(x, y, color.Black)
x++
for iByte := 0; iByte < hiResLineBytes-1; iByte++ {
for iPart := 0; iPart < 2; iPart++ {
b := lineParts[iPart][iByte]
for j := uint(0); j < 7; j++ {
bit := (b >> j) & 1
colour := light
if bit == 0 {
colour = color.Black
}
img.Set(x, y, colour)
x++
}
}
}
}
return img
}