123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441 |
- package main
- import (
- "bufio"
- "encoding/binary"
- "flag"
- "fmt"
- "os"
- "strconv"
- "strings"
- )
- func ListFonts(filename string) {
- f, err := os.Open(filename)
- if err != nil {
- fmt.Printf("Open(%s): %s\n", filename, err)
- panic(err)
- }
- defer f.Close()
- tdfonts := make([]byte, 20)
- f.Read(tdfonts)
- for true {
- fontdef := make([]byte, 4)
- read, _ := f.Read(fontdef)
- if read != 4 {
- fmt.Println("*END*")
- break
- }
- fontname := make([]byte, 13)
- f.Read(fontname)
- Name := strings.Trim(string(fontname[1:]), "\x00")
- // fmt.Printf("Font: %s\n", Name)
- f.Read(fontdef)
- single := make([]byte, 1)
- var FontType int8
- binary.Read(f, binary.LittleEndian, &FontType)
- // f.Read(single) // FontType
- // FontType := int(single[0])
- fmt.Printf("Font: %s (type %d)\n", Name, FontType)
- f.Read(single) // Spacing
- // blocksize := make([]byte, 2)
- // f.Read(blocksize)
- var BlockSize int16
- binary.Read(f, binary.LittleEndian, &BlockSize)
- // fmt.Printf("Size: %d / %x\n", BlockSize, BlockSize)
- letterOffsets := make([]int16, 94)
- binary.Read(f, binary.LittleEndian, &letterOffsets)
- if false {
- for idx, i := range letterOffsets {
- fmt.Printf(" %04X", i)
- if (idx+1)%10 == 0 {
- fmt.Println("")
- }
- }
- fmt.Println("")
- }
- data := make([]byte, BlockSize)
- binary.Read(f, binary.LittleEndian, &data)
- }
- }
- func byte_to_text(line []byte) string {
- var output string
- for _, ch := range line {
- output += fmt.Sprintf("0x%02x,", ch)
- }
- if len(output) > 0 {
- output = output[:len(output)-1]
- }
- return output
- }
- func text_to_hextext(line string) string {
- var output string
- // output = "\""
- for _, ch := range []byte(line) {
- // output += fmt.Sprintf("\\x%02x", ch)
- output += fmt.Sprintf("0x%02x,", ch)
- }
- if len(output) > 1 {
- output = output[:len(output)-1]
- }
- // output += "\""
- return output
- }
- func ExtractColor(name string, offsets []uint16, data []byte) {
- fmt.Printf("Extract Color Font: %s\n", name)
- var indexes []int
- var blocks [][][]byte
- var current [][]byte
- w := -1
- h := -1
- ch := -1
- var line []byte
- for idx, b := range data {
- if w == -1 {
- indexes = append(indexes, idx)
- current = make([][]byte, 0)
- line = make([]byte, 0)
- w = int(b)
- continue
- }
- if h == -1 {
- h = int(b)
- continue
- }
- if ch == -1 {
- ch = int(b)
- if ch == 0x0d {
- ch = -1
- current = append(current, line)
- line = make([]byte, 0)
- continue
- }
- if ch == 0 {
- current = append(current, line)
- blocks = append(blocks, current)
- current = make([][]byte, 0)
- line = make([]byte, 0)
- w = -1
- h = -1
- ch = -1
- continue
- }
- if ch == 0x26 {
- // & descender mark
- ch = -1
- continue
- }
- // line += string(rune(ch))
- line = append(line, byte(ch))
- ch = -1
- }
- }
- // offset optimization:
- var single []int
- for _, o := range offsets {
- if o == 65535 {
- single = append(single, -1)
- continue
- }
- for idx, i := range indexes {
- if o == uint16(i) {
- single = append(single, idx)
- break
- }
- }
- }
- // Handle Names with spaces
- filename := fmt.Sprintf("%s_font.go", strings.Replace(name, " ", "", -1))
- fp, err := os.Create(filename)
- if err != nil {
- panic(err)
- }
- fmt.Printf("Writing: %s\n", filename)
- defer fp.Close()
- writer := bufio.NewWriter(fp)
- // writer.WriteString("package main\n")
- writer.WriteString("// " + name + "\n\n")
- // Name := strings.ToUpper(name)
- Name := strings.Replace(name, " ", "", -1)
- writer.WriteString("func Font" + Name + "() ColorFont {\n")
- var output string
- output = " return ColorFont{characters: []int{"
- for _, s := range single {
- output += strconv.Itoa(s) + ", "
- }
- output = output[:len(output)-2] + "},\n"
- writer.WriteString(output)
- writer.Flush()
- output = " data: [][][]byte{"
- for _, blk := range blocks {
- output += "{"
- if len(blk) == 0 {
- output += "{},"
- } else {
- for _, inner := range blk {
- // output += text_to_hextext(b) + ","
- output += "{" + byte_to_text(inner) + "},"
- }
- output = output[:len(output)-1]
- }
- output += "},\n"
- writer.WriteString(output)
- output = " "
- }
- writer.WriteString(" }}\n")
- writer.WriteString("}\n")
- writer.Flush()
- }
- func ExtractBlock(name string, offsets []uint16, data []byte) {
- fmt.Printf("Extract Block Font: %s\n", name)
- var indexes []int
- var blocks [][][]byte
- var current [][]byte
- w := -1
- h := -1
- ch := -1
- var line []byte
- for idx, b := range data {
- if w == -1 {
- indexes = append(indexes, idx)
- current = make([][]byte, 0)
- line = make([]byte, 0)
- w = int(b)
- continue
- }
- if h == -1 {
- h = int(b)
- continue
- }
- if ch == -1 {
- ch = int(b)
- if ch == 0x0d {
- ch = -1
- current = append(current, line)
- line = make([]byte, 0)
- continue
- }
- if ch == 0 {
- current = append(current, line)
- blocks = append(blocks, current)
- current = make([][]byte, 0)
- line = make([]byte, 0)
- w = -1
- h = -1
- ch = -1
- continue
- }
- if ch == 0x26 {
- // & descender mark
- ch = -1
- continue
- }
- line = append(line, byte(ch))
- ch = -1
- }
- }
- // offset optimization:
- var single []int
- for _, o := range offsets {
- if o == 65535 {
- single = append(single, -1)
- continue
- }
- for idx, i := range indexes {
- if o == uint16(i) {
- single = append(single, idx)
- break
- }
- }
- }
- // Handle Names with spaces
- filename := fmt.Sprintf("%s_font.go", strings.Replace(name, " ", "", -1))
- fp, err := os.Create(filename)
- if err != nil {
- panic(err)
- }
- fmt.Printf("Writing: %s\n", filename)
- defer fp.Close()
- writer := bufio.NewWriter(fp)
- // writer.WriteString("package main\n")
- writer.WriteString("// " + name + "\n\n")
- // Name := strings.ToUpper(name)
- Name := strings.Replace(name, " ", "", -1)
- writer.WriteString("func Font" + Name + "() BlockFont {\n")
- var output string
- output = " return BlockFont{characters: []int{"
- for _, s := range single {
- output += strconv.Itoa(s) + ", "
- }
- output = output[:len(output)-2] + "},\n"
- writer.WriteString(output)
- writer.Flush()
- output = " data: [][][]byte{"
- for _, blk := range blocks {
- output += "{"
- if len(blk) == 0 {
- output += "{},"
- } else {
- for _, inner := range blk {
- output += "{" + byte_to_text(inner) + "},"
- }
- output = output[:len(output)-1]
- }
- output += "},\n"
- // output = output[:len(output)-1]
- // output += "},\n"
- writer.WriteString(output)
- output = " "
- }
- writer.WriteString(" }}\n")
- writer.WriteString("}\n")
- writer.Flush()
- }
- func ExtractFonts(filename string, fonts []string) {
- f, err := os.Open(filename)
- if err != nil {
- fmt.Printf("Open(%s): %s\n", filename, err)
- panic(err)
- }
- defer f.Close()
- tdfonts := make([]byte, 20)
- f.Read(tdfonts)
- for true {
- fontdef := make([]byte, 4)
- read, _ := f.Read(fontdef)
- if read != 4 {
- break
- }
- fontname := make([]byte, 13)
- f.Read(fontname)
- Name := strings.Trim(string(fontname[1:]), "\x00")
- // fmt.Printf("Font: %s\n", Name)
- f.Read(fontdef)
- single := make([]byte, 1)
- var FontType int8
- binary.Read(f, binary.LittleEndian, &FontType)
- // fmt.Printf("Font: %s (type %d)\n", Name, FontType)
- f.Read(single) // Spacing
- var BlockSize int16
- binary.Read(f, binary.LittleEndian, &BlockSize)
- letterOffsets := make([]uint16, 94)
- binary.Read(f, binary.LittleEndian, &letterOffsets)
- if false {
- for idx, i := range letterOffsets {
- fmt.Printf(" %04X", i)
- if (idx+1)%10 == 0 {
- fmt.Println("")
- }
- }
- fmt.Println("")
- }
- data := make([]byte, BlockSize)
- binary.Read(f, binary.LittleEndian, &data)
- // Special case where they are asking for all fonts
- if len(fonts) == 1 && fonts[0] == "*" {
- switch FontType {
- case 1:
- ExtractBlock(Name, letterOffsets, data)
- case 2:
- ExtractColor(Name, letterOffsets, data)
- default:
- fmt.Printf("Sorry, I can't handle Font: %s Type %d!\n", Name, FontType)
- }
- } else {
- for _, f := range fonts {
- if Name == f {
- switch FontType {
- case 1:
- ExtractBlock(Name, letterOffsets, data)
- case 2:
- ExtractColor(Name, letterOffsets, data)
- default:
- fmt.Printf("Sorry, I can't handle Font: %s Type %d!\n", Name, FontType)
- }
- break
- }
- }
- }
- }
- }
- func main() {
- fmt.Println("Font-Out - A TDF (TheDraw Font) file processor.")
- var fontfile string
- var fonts string
- var defaultPackage string = "main"
- var listFonts bool
- var allFonts bool
- flag.StringVar(&fontfile, "tdf", "", "TheDraw Font File")
- flag.StringVar(&fonts, "f", "", "Font(s) to extract")
- flag.BoolVar(&allFonts, "a", false, "Extract All Fonts")
- flag.StringVar(&defaultPackage, "p", "main", "Package name to use")
- flag.BoolVar(&listFonts, "l", false, "List Fonts")
- flag.Parse()
- if len(fontfile) == 0 {
- fmt.Println("I need a TDF filename.")
- flag.PrintDefaults()
- os.Exit(2)
- }
- if listFonts {
- ListFonts(fontfile)
- }
- var fontList []string
- if len(fonts) > 0 {
- fontList = strings.Split(fonts, ",")
- }
- if allFonts {
- fontList = make([]string, 0)
- fontList = append(fontList, "*")
- }
- if len(fontList) > 0 {
- ExtractFonts(fontfile, fontList)
- }
- }
|