2020-10-03 23:38:26 +02:00
|
|
|
package izapple2
|
2019-04-21 18:18:43 +02:00
|
|
|
|
|
|
|
import (
|
2019-10-06 01:26:00 +02:00
|
|
|
"errors"
|
2020-10-17 20:10:48 +02:00
|
|
|
|
|
|
|
"github.com/ivanizag/izapple2/storage"
|
2019-04-21 18:18:43 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
/*
|
|
|
|
See:
|
2019-10-12 21:37:37 +02:00
|
|
|
https://mirrors.apple2.org.za/Apple%20II%20Documentation%20Project/Companies/Apple/Documentation/Apple%20Technical%20Information%20Library/a2til041.txt
|
2019-04-21 18:18:43 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
// CharacterGenerator represents the ROM wth the characters bitmaps
|
|
|
|
type CharacterGenerator struct {
|
2019-06-09 23:54:27 +02:00
|
|
|
data []uint8
|
|
|
|
columnMap charColumnMap
|
|
|
|
page int
|
2021-10-12 12:26:40 +02:00
|
|
|
pageSize int
|
2019-04-21 18:18:43 +02:00
|
|
|
}
|
|
|
|
|
2019-06-09 23:54:27 +02:00
|
|
|
type charColumnMap func(column int) int
|
|
|
|
|
2019-10-12 21:37:37 +02:00
|
|
|
func charGenColumnsMap2Plus(column int) int {
|
|
|
|
return 6 - column
|
|
|
|
}
|
|
|
|
|
|
|
|
func charGenColumnsMap2e(column int) int {
|
|
|
|
return column
|
|
|
|
}
|
|
|
|
|
2019-04-21 18:18:43 +02:00
|
|
|
const (
|
2021-10-12 12:26:40 +02:00
|
|
|
charGenPageSize2Plus = 2048
|
|
|
|
charGenPageSize2E = 2048 * 2
|
2019-04-21 18:18:43 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// NewCharacterGenerator instantiates a new Character Generator with the rom on the file given
|
2021-10-12 12:26:40 +02:00
|
|
|
func newCharacterGenerator(filename string, order charColumnMap, isApple2e bool) (*CharacterGenerator, error) {
|
2019-04-21 18:18:43 +02:00
|
|
|
var cg CharacterGenerator
|
2021-10-12 12:26:40 +02:00
|
|
|
cg.columnMap = order
|
|
|
|
cg.pageSize = charGenPageSize2Plus
|
|
|
|
if isApple2e {
|
|
|
|
cg.pageSize = charGenPageSize2E
|
|
|
|
}
|
|
|
|
|
2019-10-06 01:26:00 +02:00
|
|
|
err := cg.load(filename)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-10-12 12:26:40 +02:00
|
|
|
|
2019-10-06 01:26:00 +02:00
|
|
|
return &cg, nil
|
2019-04-21 18:18:43 +02:00
|
|
|
}
|
|
|
|
|
2019-10-06 01:26:00 +02:00
|
|
|
func (cg *CharacterGenerator) load(filename string) error {
|
2020-11-03 10:51:12 -06:00
|
|
|
bytes, _, err := storage.LoadResource(filename)
|
2019-10-06 01:26:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-05-15 16:01:04 +02:00
|
|
|
size := len(bytes)
|
2021-10-12 12:26:40 +02:00
|
|
|
if size < cg.pageSize {
|
2021-08-05 21:12:52 +02:00
|
|
|
return errors.New("character ROM size not supported")
|
2019-04-21 18:18:43 +02:00
|
|
|
}
|
2019-05-15 16:01:04 +02:00
|
|
|
cg.data = bytes
|
2019-10-06 01:26:00 +02:00
|
|
|
return nil
|
2019-04-21 18:18:43 +02:00
|
|
|
}
|
|
|
|
|
2019-06-09 23:54:27 +02:00
|
|
|
func (cg *CharacterGenerator) setPage(page int) {
|
|
|
|
// Some clones had a switch to change codepage with extra characters
|
2021-10-12 12:26:40 +02:00
|
|
|
pages := len(cg.data) / cg.pageSize
|
2019-06-09 23:54:27 +02:00
|
|
|
cg.page = page % pages
|
|
|
|
}
|
|
|
|
|
2021-10-12 12:26:40 +02:00
|
|
|
func (cg *CharacterGenerator) getPage() int {
|
|
|
|
return cg.page
|
|
|
|
}
|
|
|
|
|
2019-06-09 23:54:27 +02:00
|
|
|
func (cg *CharacterGenerator) nextPage() {
|
|
|
|
cg.setPage(cg.page + 1)
|
|
|
|
}
|
|
|
|
|
2019-04-21 21:04:02 +02:00
|
|
|
func (cg *CharacterGenerator) getPixel(char uint8, row int, column int) bool {
|
2021-10-12 12:26:40 +02:00
|
|
|
bits := cg.data[int(char)*8+row+cg.page*cg.pageSize]
|
2019-10-12 21:37:37 +02:00
|
|
|
bit := cg.columnMap(column)
|
2019-06-09 23:54:27 +02:00
|
|
|
value := bits >> uint(bit) & 1
|
|
|
|
return value == 1
|
2019-04-21 21:04:02 +02:00
|
|
|
}
|