izapple2/screen/text.go

200 lines
5.1 KiB
Go
Raw Normal View History

2020-10-16 20:41:34 +02:00
package screen
2019-05-03 20:09:53 +02:00
import (
"fmt"
2019-05-03 20:09:53 +02:00
"image"
"image/color"
"strings"
2019-05-03 20:09:53 +02:00
)
const (
2020-10-04 19:21:49 +02:00
charWidth = 7
charHeight = 8
text40Columns = 40
textLines = 24
2019-05-03 20:09:53 +02:00
)
2020-10-16 20:41:34 +02:00
func snapshotText40(vs VideoSource, isSecondPage bool, light color.Color) *image.RGBA {
text := getTextFromMemory(vs, isSecondPage, false)
return renderText(vs, text, nil /*colorMap*/, light)
}
2019-05-03 20:09:53 +02:00
2020-10-16 20:41:34 +02:00
func snapshotText80(vs VideoSource, isSecondPage bool, light color.Color) *image.RGBA {
text := getText80FromMemory(vs, isSecondPage)
return renderText(vs, text, nil /*colorMap*/, light)
}
2020-10-16 20:41:34 +02:00
func snapshotText40RGB(vs VideoSource, isSecondPage bool) *image.RGBA {
text := getTextFromMemory(vs, isSecondPage, false)
colorMap := getTextFromMemory(vs, isSecondPage, true)
return renderText(vs, text, colorMap, nil)
}
2020-10-16 20:41:34 +02:00
func snapshotText40RGBColors(vs VideoSource, isSecondPage bool) *image.RGBA {
colorMap := getTextFromMemory(vs, isSecondPage, true)
return renderText(vs, nil /*text*/, colorMap, nil)
}
2020-10-16 20:41:34 +02:00
func getText80FromMemory(vs VideoSource, isSecondPage bool) []uint8 {
text40Columns := getTextFromMemory(vs, isSecondPage, false)
text40ColumnsAlt := getTextFromMemory(vs, isSecondPage, true)
2019-11-08 23:56:54 +01:00
// Merge the two 40 cols to return 80 cols
text80Columns := make([]uint8, 2*len(text40Columns))
for i := 0; i < len(text40Columns); i++ {
text80Columns[2*i] = text40ColumnsAlt[i]
text80Columns[2*i+1] = text40Columns[i]
}
return text80Columns
2019-11-08 23:56:54 +01:00
}
2019-05-03 20:09:53 +02:00
2020-10-16 20:41:34 +02:00
func getTextFromMemory(vs VideoSource, isSecondPage bool, isExt bool) []uint8 {
data := vs.GetTextMemory(isSecondPage, isExt)
2019-11-08 23:56:54 +01:00
2020-10-04 19:21:49 +02:00
text := make([]uint8, textLines*text40Columns)
for l := 0; l < textLines; l++ {
2020-10-04 19:21:49 +02:00
for c := 0; c < text40Columns; c++ {
char := data[getTextCharOffset(c, l)]
2020-10-04 19:21:49 +02:00
text[text40Columns*l+c] = char
2019-11-08 23:56:54 +01:00
}
}
return text
}
2020-10-16 20:41:34 +02:00
func getTextCharOffset(col int, line int) uint16 {
// See "Understanding the Apple II", page 5-10
// http://www.applelogic.org/files/UNDERSTANDINGTHEAII.pdf
section := line / 8 // Top, middle and bottom
eighth := line % 8
return uint16(section*40 + eighth*0x80 + col)
}
func getRGBTextColor(pixel bool, colorKey uint8) color.Color {
if pixel {
colorKey >>= 4
}
colorKey &= 0x0f
2020-08-09 17:42:47 +02:00
return ntscColorMap[colorKey]
}
2020-10-16 20:41:34 +02:00
func renderText(vs VideoSource, text []uint8, colorMap []uint8, light color.Color) *image.RGBA {
columns := len(text) / textLines
if text == nil {
2020-10-04 19:21:49 +02:00
columns = text40Columns
}
2019-11-08 23:56:54 +01:00
width := columns * charWidth
height := textLines * charHeight
2020-10-04 19:21:49 +02:00
size := image.Rect(0, 0, 2*hiResWidth, hiResHeight)
2019-05-03 20:09:53 +02:00
img := image.NewRGBA(size)
for x := 0; x < width; x++ {
for y := 0; y < height; y++ {
2019-11-08 23:56:54 +01:00
line := y / charHeight
2019-05-03 20:09:53 +02:00
col := x / charWidth
rowInChar := y % charHeight
colInChar := x % charWidth
charIndex := line*columns + col
var char uint8
if text != nil {
char = text[charIndex]
} else {
char = 79 + 128 // Debug screen filed with O
}
2020-10-16 20:41:34 +02:00
pixel := vs.GetCharacterPixel(char, rowInChar, colInChar)
2019-05-03 20:09:53 +02:00
var colour color.Color
if colorMap != nil {
colour = getRGBTextColor(pixel, colorMap[charIndex])
} else if pixel {
2019-05-03 21:45:29 +02:00
colour = light
2019-05-03 20:09:53 +02:00
} else {
colour = color.Black
}
2020-10-04 19:21:49 +02:00
if columns == text40Columns {
img.Set(x*2, y, colour)
img.Set(x*2+1, y, colour)
} else {
img.Set(x, y, colour)
}
2019-05-03 20:09:53 +02:00
}
}
return img
}
2020-10-16 20:41:34 +02:00
// RenderTextModeAnsi returns the text mode contents using ANSI escape codes for reverse and flash
func RenderTextModeAnsi(vs VideoSource, is80Columns bool, isSecondPage bool, isAltText bool) string {
//func DumpTextModeAnsi(a *Apple2) string {
// is80Columns := a.io.isSoftSwitchActive(ioFlag80Col)
// isSecondPage := a.io.isSoftSwitchActive(ioFlagSecondPage) && !a.mmu.store80Active
// isAltText := a.isApple2e && a.io.isSoftSwitchActive(ioFlagAltChar)
var text []uint8
if is80Columns {
2020-10-16 20:41:34 +02:00
text = getText80FromMemory(vs, isSecondPage)
} else {
2020-10-16 20:41:34 +02:00
text = getTextFromMemory(vs, isSecondPage, false)
}
columns := len(text) / textLines
2019-11-08 23:56:54 +01:00
content := "\n"
2019-11-08 23:56:54 +01:00
content += fmt.Sprintln(strings.Repeat("#", columns+4))
for l := 0; l < textLines; l++ {
line := ""
2019-11-08 23:56:54 +01:00
for c := 0; c < columns; c++ {
char := text[l*columns+c]
line += textMemoryByteToString(char, isAltText)
}
content += fmt.Sprintf("# %v #\n", line)
}
2019-11-08 23:56:54 +01:00
content += fmt.Sprintln(strings.Repeat("#", columns+4))
return content
}
func textMemoryByteToString(value uint8, isAltCharSet bool) string {
// See https://en.wikipedia.org/wiki/Apple_II_character_set
// Supports the new lowercase characters in the Apple2e
// Only ascii from 0x20 to 0x5F is visible
topBits := value >> 6
isInverse := topBits == 0
isFlash := topBits == 1
if isFlash && isAltCharSet {
// On the Apple2e with lowercase chars there is not flash mode.
isFlash = false
isInverse = true
}
if isAltCharSet {
value = value & 0x7F
} else {
value = value & 0x3F
}
if value < 0x20 {
value += 0x40
}
if value == 0x7f {
// DEL is full box
value = '_'
}
if isFlash {
if value == ' ' {
// Flashing space in Apple is the full box. It can't be done with ANSI codes
value = '_'
}
return fmt.Sprintf("\033[5m%v\033[0m", string(value))
} else if isInverse {
return fmt.Sprintf("\033[7m%v\033[0m", string(value))
} else {
return string(value)
}
}