-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathshaping.go
205 lines (184 loc) · 5.82 KB
/
shaping.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
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
// SPDX-License-Identifier: Unlicense OR BSD-3-Clause
package shaping
import (
"github.com/go-text/typesetting/di"
"github.com/go-text/typesetting/harfbuzz"
"golang.org/x/image/math/fixed"
)
// HarfbuzzShaper implements the Shaper interface using harfbuzz.
// Reusing this shaper type across multiple shaping operations is
// faster and more memory-efficient than creating a new shaper
// for each operation.
type HarfbuzzShaper struct {
buf *harfbuzz.Buffer
fonts fontLRU
features []harfbuzz.Feature
}
// SetFontCacheSize adjusts the size of the font cache within the shaper.
// It is safe to adjust the size after using the shaper, though shrinking
// it may result in many evictions on the next shaping.
func (h *HarfbuzzShaper) SetFontCacheSize(size int) {
h.fonts.maxSize = size
}
var _ Shaper = (*HarfbuzzShaper)(nil)
// Shaper describes the signature of a font shaping operation.
type Shaper interface {
// Shape takes an Input and shapes it into the Output.
Shape(Input) Output
}
const (
// scaleShift is the power of 2 with which to automatically scale
// up the input coordinate space of the shaper. This factor will
// be removed prior to returning dimensions. This ensures that the
// returned glyph dimensions take advantage of all of the precision
// that a fixed.Int26_6 can provide.
scaleShift = 6
)
// clamp ensures val is in the inclusive range [low,high].
func clamp(val, low, high int) int {
if val < low {
return low
}
if val > high {
return high
}
return val
}
// Shape turns an input into an output.
func (t *HarfbuzzShaper) Shape(input Input) Output {
// Prepare to shape the text.
if t.buf == nil {
t.buf = harfbuzz.NewBuffer()
} else {
t.buf.Clear()
}
runes, start, end := input.Text, input.RunStart, input.RunEnd
if end < start {
// Try to guess what the caller actually wanted.
end, start = start, end
}
start = clamp(start, 0, len(runes))
end = clamp(end, 0, len(runes))
t.buf.AddRunes(runes, start, end-start)
// handle vertical sideways text
isSideways := false
if input.Direction.IsSideways() {
// temporarily switch to horizontal
input.Direction = input.Direction.SwitchAxis()
isSideways = true
}
t.buf.Props.Direction = input.Direction.Harfbuzz()
t.buf.Props.Language = input.Language
t.buf.Props.Script = input.Script
// reuse font when possible
font, ok := t.fonts.Get(input.Face.Font)
if !ok { // create a new font and cache it
font = harfbuzz.NewFont(input.Face)
t.fonts.Put(input.Face.Font, font)
}
// adjust the user provided fields
font.XScale = int32(input.Size.Ceil()) << scaleShift
font.YScale = font.XScale
if L := len(input.FontFeatures); cap(t.features) < L {
t.features = make([]harfbuzz.Feature, L)
} else {
t.features = t.features[0:L]
}
for i, f := range input.FontFeatures {
t.features[i] = harfbuzz.Feature{
Tag: f.Tag,
Value: f.Value,
Start: harfbuzz.FeatureGlobalStart,
End: harfbuzz.FeatureGlobalEnd,
}
}
// Actually use harfbuzz to shape the text.
t.buf.Shape(font, t.features)
// Convert the shaped text into an Output.
glyphs := make([]Glyph, len(t.buf.Info))
for i := range glyphs {
g := t.buf.Info[i].Glyph
glyphs[i] = Glyph{
ClusterIndex: t.buf.Info[i].Cluster,
GlyphID: g,
Mask: t.buf.Info[i].Mask,
}
extents, ok := font.GlyphExtents(g)
if !ok {
// Leave the glyph having zero size if it isn't in the font. There
// isn't really anything we can do to recover from such an error.
continue
}
glyphs[i].Width = fixed.I(int(extents.Width)) >> scaleShift
glyphs[i].Height = fixed.I(int(extents.Height)) >> scaleShift
glyphs[i].XBearing = fixed.I(int(extents.XBearing)) >> scaleShift
glyphs[i].YBearing = fixed.I(int(extents.YBearing)) >> scaleShift
glyphs[i].XAdvance = fixed.I(int(t.buf.Pos[i].XAdvance)) >> scaleShift
glyphs[i].YAdvance = fixed.I(int(t.buf.Pos[i].YAdvance)) >> scaleShift
glyphs[i].XOffset = fixed.I(int(t.buf.Pos[i].XOffset)) >> scaleShift
glyphs[i].YOffset = fixed.I(int(t.buf.Pos[i].YOffset)) >> scaleShift
}
countClusters(glyphs, input.RunEnd, input.Direction.Progression())
out := Output{
Glyphs: glyphs,
Direction: input.Direction,
Face: input.Face,
Size: input.Size,
}
out.Runes.Offset = input.RunStart
out.Runes.Count = input.RunEnd - input.RunStart
if isSideways {
// set the Direction to the correct value.
// this is required here so that the following call to ExtentsForDirection
// returns the vertical data.
out.sideways()
}
fontExtents := font.ExtentsForDirection(out.Direction.Harfbuzz())
out.LineBounds = Bounds{
Ascent: fixed.I(int(fontExtents.Ascender)) >> scaleShift,
Descent: fixed.I(int(fontExtents.Descender)) >> scaleShift,
Gap: fixed.I(int(fontExtents.LineGap)) >> scaleShift,
}
out.RecalculateAll()
return out
}
// countClusters tallies the number of runes and glyphs in each cluster
// and updates the relevant fields on the provided glyph slice.
func countClusters(glyphs []Glyph, textLen int, dir di.Progression) {
currentCluster := -1
runesInCluster := 0
glyphsInCluster := 0
previousCluster := textLen
for i := range glyphs {
g := glyphs[i].ClusterIndex
if g != currentCluster {
// If we're processing a new cluster, count the runes and glyphs
// that compose it.
runesInCluster = 0
glyphsInCluster = 1
currentCluster = g
nextCluster := -1
glyphCountLoop:
for k := i + 1; k < len(glyphs); k++ {
if glyphs[k].ClusterIndex == g {
glyphsInCluster++
} else {
nextCluster = glyphs[k].ClusterIndex
break glyphCountLoop
}
}
if nextCluster == -1 {
nextCluster = textLen
}
switch dir {
case di.FromTopLeft:
runesInCluster = nextCluster - currentCluster
case di.TowardTopLeft:
runesInCluster = previousCluster - currentCluster
}
previousCluster = g
}
glyphs[i].GlyphCount = glyphsInCluster
glyphs[i].RuneCount = runesInCluster
}
}