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 }