-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathtable_htmx.go
64 lines (56 loc) · 1.73 KB
/
table_htmx.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
package truetype
import (
"encoding/binary"
"errors"
)
var errInvalidMaxpTable = errors.New("invalid maxp table")
func parseTableMaxp(input []byte) (numGlyphs int, err error) {
if len(input) < 6 {
return 0, errInvalidMaxpTable
}
out := binary.BigEndian.Uint16(input[4:6])
return int(out), nil
}
type TableHVmtx []Metric // with length numGlyphs
// return the base side bearing, handling invalid glyph index
func (t TableHVmtx) getSideBearing(glyph GID) int16 {
if int(glyph) >= len(t) {
return 0
}
return t[glyph].SideBearing
}
type Metric struct {
Advance, SideBearing int16
}
// pad the width if numberOfHMetrics < numGlyphs
func parseHVmtxTable(input []byte, numberOfHMetrics, numGlyphs uint16) (TableHVmtx, error) {
if numberOfHMetrics == 0 {
return nil, errors.New("number of glyph metrics is 0")
}
if len(input) < 4*int(numberOfHMetrics) {
return nil, errors.New("invalid h/vmtx table (EOF)")
}
widths := make(TableHVmtx, numberOfHMetrics)
for i := range widths {
widths[i].Advance = int16(binary.BigEndian.Uint16(input[4*i:]))
widths[i].SideBearing = int16(binary.BigEndian.Uint16(input[4*i+2:]))
}
if left := int(numGlyphs) - int(numberOfHMetrics); left > 0 {
// avances are padded with the last value
// side bearings are given
widths = append(widths, make(TableHVmtx, numGlyphs-numberOfHMetrics)...)
lastWidth := widths[numberOfHMetrics-1]
for i := numberOfHMetrics; i < numGlyphs; i++ {
widths[i] = lastWidth
}
input = input[4*int(numberOfHMetrics):]
if len(input) < 2*left {
return nil, errors.New("invalid h/vmtx table (EOF)")
}
subslice := widths[numberOfHMetrics:]
for i := range subslice {
subslice[i].SideBearing = int16(binary.BigEndian.Uint16(input[2*i:]))
}
}
return widths, nil
}