package apple2 import ( "image" "image/color" ) const ( loResPixelWidth = charWidth * 2 doubleLoResPixelWidth = charWidth loResPixelHeight = charHeight / 2 ) func getColorPatterns(light color.Color) [16][16]color.Color { /* For each lores pixel we have to fill 14 half mono pixels with the 4 bits of the color repeated. We will need to shift by 2 bits on the odd columns. Lets prepare 14+2 values for each color. */ var data [16][16]color.Color for ci := 0; ci < 16; ci++ { for cb := uint8(0); cb < 4; cb++ { bit := (ci >> cb) & 1 var colour color.Color if bit == 0 { colour = color.Black } else { colour = light } for i := uint8(0); i < 4; i++ { data[ci][cb+4*i] = colour } } } return data } func snapshotLoResModeMono(a *Apple2, isSecondPage bool, light color.Color) *image.RGBA { data := getTextFromMemory(a.mmu.physicalMainRAM, isSecondPage) return renderGrMode(data, false /*isMeres*/, light) } func snapshotMeResModeMono(a *Apple2, isSecondPage bool, light color.Color) *image.RGBA { data := getText80FromMemory(a, isSecondPage) return renderGrMode(data, true /*isMeres*/, light) } func renderGrMode(data []uint8, isDoubleResMode bool, light color.Color) *image.RGBA { grLines := textLines * 2 columns := len(data) / textLines pixelWidth := loResPixelWidth if isDoubleResMode { pixelWidth = doubleLoResPixelWidth } size := image.Rect(0, 0, columns*pixelWidth, grLines*loResPixelHeight) img := image.NewRGBA(size) patterns := getColorPatterns(light) for l := 0; l < grLines; l++ { for c := 0; c < columns; c++ { char := data[(l/2)*columns+c] grPixel := char >> 4 if l%2 == 0 { grPixel = char & 0xf } // We place pixelWidth mono pixels per graphic pixel. // The groups of 4 mono pixels need to be aligned with an offset to get plain surfaces offset := (c * pixelWidth) % 4 if isDoubleResMode && ((c % 2) == 0) { // See "Understanding the Apple II", page 8-44 // Even blocks color are rotated left one bit offset = offset + 3 // Equivalent to -1 } // Insert the pixelWidth pixels required for i := 0; i < pixelWidth; i++ { v := patterns[grPixel][i+offset] // Repeat the same color for 4 lines for r := 0; r < loResPixelHeight; r++ { img.Set(c*pixelWidth+i, l*4+r, v) } } } } return img }