dep -> govendor
This commit is contained in:
248
vendor/github.com/faiface/pixel/text/atlas.go
generated
vendored
Normal file
248
vendor/github.com/faiface/pixel/text/atlas.go
generated
vendored
Normal file
@@ -0,0 +1,248 @@
|
||||
package text
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"image"
|
||||
"image/draw"
|
||||
"sort"
|
||||
"unicode"
|
||||
|
||||
"github.com/faiface/pixel"
|
||||
"golang.org/x/image/font"
|
||||
"golang.org/x/image/math/fixed"
|
||||
)
|
||||
|
||||
// Atlas7x13 is an Atlas using basicfont.Face7x13 with the ASCII rune set
|
||||
var Atlas7x13 *Atlas
|
||||
|
||||
// Glyph describes one glyph in an Atlas.
|
||||
type Glyph struct {
|
||||
Dot pixel.Vec
|
||||
Frame pixel.Rect
|
||||
Advance float64
|
||||
}
|
||||
|
||||
// Atlas is a set of pre-drawn glyphs of a fixed set of runes. This allows for efficient text drawing.
|
||||
type Atlas struct {
|
||||
face font.Face
|
||||
pic pixel.Picture
|
||||
mapping map[rune]Glyph
|
||||
ascent float64
|
||||
descent float64
|
||||
lineHeight float64
|
||||
}
|
||||
|
||||
// NewAtlas creates a new Atlas containing glyphs of the union of the given sets of runes (plus
|
||||
// unicode.ReplacementChar) from the given font face.
|
||||
//
|
||||
// Creating an Atlas is rather expensive, do not create a new Atlas each frame.
|
||||
//
|
||||
// Do not destroy or close the font.Face after creating the Atlas. Atlas still uses it.
|
||||
func NewAtlas(face font.Face, runeSets ...[]rune) *Atlas {
|
||||
seen := make(map[rune]bool)
|
||||
runes := []rune{unicode.ReplacementChar}
|
||||
for _, set := range runeSets {
|
||||
for _, r := range set {
|
||||
if !seen[r] {
|
||||
runes = append(runes, r)
|
||||
seen[r] = true
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fixedMapping, fixedBounds := makeSquareMapping(face, runes, fixed.I(2))
|
||||
|
||||
atlasImg := image.NewRGBA(image.Rect(
|
||||
fixedBounds.Min.X.Floor(),
|
||||
fixedBounds.Min.Y.Floor(),
|
||||
fixedBounds.Max.X.Ceil(),
|
||||
fixedBounds.Max.Y.Ceil(),
|
||||
))
|
||||
|
||||
for r, fg := range fixedMapping {
|
||||
dr, mask, maskp, _, _ := face.Glyph(fg.dot, r)
|
||||
draw.Draw(atlasImg, dr, mask, maskp, draw.Src)
|
||||
}
|
||||
|
||||
bounds := pixel.R(
|
||||
i2f(fixedBounds.Min.X),
|
||||
i2f(fixedBounds.Min.Y),
|
||||
i2f(fixedBounds.Max.X),
|
||||
i2f(fixedBounds.Max.Y),
|
||||
)
|
||||
|
||||
mapping := make(map[rune]Glyph)
|
||||
for r, fg := range fixedMapping {
|
||||
mapping[r] = Glyph{
|
||||
Dot: pixel.V(
|
||||
i2f(fg.dot.X),
|
||||
bounds.Max.Y-(i2f(fg.dot.Y)-bounds.Min.Y),
|
||||
),
|
||||
Frame: pixel.R(
|
||||
i2f(fg.frame.Min.X),
|
||||
bounds.Max.Y-(i2f(fg.frame.Min.Y)-bounds.Min.Y),
|
||||
i2f(fg.frame.Max.X),
|
||||
bounds.Max.Y-(i2f(fg.frame.Max.Y)-bounds.Min.Y),
|
||||
).Norm(),
|
||||
Advance: i2f(fg.advance),
|
||||
}
|
||||
}
|
||||
|
||||
return &Atlas{
|
||||
face: face,
|
||||
pic: pixel.PictureDataFromImage(atlasImg),
|
||||
mapping: mapping,
|
||||
ascent: i2f(face.Metrics().Ascent),
|
||||
descent: i2f(face.Metrics().Descent),
|
||||
lineHeight: i2f(face.Metrics().Height),
|
||||
}
|
||||
}
|
||||
|
||||
// Picture returns the underlying Picture containing an arrangement of all the glyphs contained
|
||||
// within the Atlas.
|
||||
func (a *Atlas) Picture() pixel.Picture {
|
||||
return a.pic
|
||||
}
|
||||
|
||||
// Contains reports wheter r in contained within the Atlas.
|
||||
func (a *Atlas) Contains(r rune) bool {
|
||||
_, ok := a.mapping[r]
|
||||
return ok
|
||||
}
|
||||
|
||||
// Glyph returns the description of r within the Atlas.
|
||||
func (a *Atlas) Glyph(r rune) Glyph {
|
||||
return a.mapping[r]
|
||||
}
|
||||
|
||||
// Kern returns the kerning distance between runes r0 and r1. Positive distance means that the
|
||||
// glyphs should be further apart.
|
||||
func (a *Atlas) Kern(r0, r1 rune) float64 {
|
||||
return i2f(a.face.Kern(r0, r1))
|
||||
}
|
||||
|
||||
// Ascent returns the distance from the top of the line to the baseline.
|
||||
func (a *Atlas) Ascent() float64 {
|
||||
return a.ascent
|
||||
}
|
||||
|
||||
// Descent returns the distance from the baseline to the bottom of the line.
|
||||
func (a *Atlas) Descent() float64 {
|
||||
return a.descent
|
||||
}
|
||||
|
||||
// LineHeight returns the recommended vertical distance between two lines of text.
|
||||
func (a *Atlas) LineHeight() float64 {
|
||||
return a.lineHeight
|
||||
}
|
||||
|
||||
// DrawRune returns parameters necessary for drawing a rune glyph.
|
||||
//
|
||||
// Rect is a rectangle where the glyph should be positioned. Frame is the glyph frame inside the
|
||||
// Atlas's Picture. NewDot is the new position of the dot.
|
||||
func (a *Atlas) DrawRune(prevR, r rune, dot pixel.Vec) (rect, frame, bounds pixel.Rect, newDot pixel.Vec) {
|
||||
if !a.Contains(r) {
|
||||
r = unicode.ReplacementChar
|
||||
}
|
||||
if !a.Contains(unicode.ReplacementChar) {
|
||||
return pixel.Rect{}, pixel.Rect{}, pixel.Rect{}, dot
|
||||
}
|
||||
if !a.Contains(prevR) {
|
||||
prevR = unicode.ReplacementChar
|
||||
}
|
||||
|
||||
if prevR >= 0 {
|
||||
dot.X += a.Kern(prevR, r)
|
||||
}
|
||||
|
||||
glyph := a.Glyph(r)
|
||||
|
||||
rect = glyph.Frame.Moved(dot.Sub(glyph.Dot))
|
||||
bounds = rect
|
||||
|
||||
if bounds.W()*bounds.H() != 0 {
|
||||
bounds = pixel.R(
|
||||
bounds.Min.X,
|
||||
dot.Y-a.Descent(),
|
||||
bounds.Max.X,
|
||||
dot.Y+a.Ascent(),
|
||||
)
|
||||
}
|
||||
|
||||
dot.X += glyph.Advance
|
||||
|
||||
return rect, glyph.Frame, bounds, dot
|
||||
}
|
||||
|
||||
type fixedGlyph struct {
|
||||
dot fixed.Point26_6
|
||||
frame fixed.Rectangle26_6
|
||||
advance fixed.Int26_6
|
||||
}
|
||||
|
||||
// makeSquareMapping finds an optimal glyph arrangement of the given runes, so that their common
|
||||
// bounding box is as square as possible.
|
||||
func makeSquareMapping(face font.Face, runes []rune, padding fixed.Int26_6) (map[rune]fixedGlyph, fixed.Rectangle26_6) {
|
||||
width := sort.Search(int(fixed.I(1024*1024)), func(i int) bool {
|
||||
width := fixed.Int26_6(i)
|
||||
_, bounds := makeMapping(face, runes, padding, width)
|
||||
return bounds.Max.X-bounds.Min.X >= bounds.Max.Y-bounds.Min.Y
|
||||
})
|
||||
return makeMapping(face, runes, padding, fixed.Int26_6(width))
|
||||
}
|
||||
|
||||
// makeMapping arranges glyphs of the given runes into rows in such a way, that no glyph is located
|
||||
// fully to the right of the specified width. Specifically, it places glyphs in a row one by one and
|
||||
// once it reaches the specified width, it starts a new row.
|
||||
func makeMapping(face font.Face, runes []rune, padding, width fixed.Int26_6) (map[rune]fixedGlyph, fixed.Rectangle26_6) {
|
||||
mapping := make(map[rune]fixedGlyph)
|
||||
bounds := fixed.Rectangle26_6{}
|
||||
|
||||
dot := fixed.P(0, 0)
|
||||
|
||||
for _, r := range runes {
|
||||
b, advance, ok := face.GlyphBounds(r)
|
||||
if !ok {
|
||||
fmt.Println(r)
|
||||
continue
|
||||
}
|
||||
|
||||
// this is important for drawing, artifacts arise otherwise
|
||||
frame := fixed.Rectangle26_6{
|
||||
Min: fixed.P(b.Min.X.Floor(), b.Min.Y.Floor()),
|
||||
Max: fixed.P(b.Max.X.Ceil(), b.Max.Y.Ceil()),
|
||||
}
|
||||
|
||||
dot.X -= frame.Min.X
|
||||
frame = frame.Add(dot)
|
||||
|
||||
mapping[r] = fixedGlyph{
|
||||
dot: dot,
|
||||
frame: frame,
|
||||
advance: advance,
|
||||
}
|
||||
bounds = bounds.Union(frame)
|
||||
|
||||
dot.X = frame.Max.X
|
||||
|
||||
// padding + align to integer
|
||||
dot.X += padding
|
||||
dot.X = fixed.I(dot.X.Ceil())
|
||||
|
||||
// width exceeded, new row
|
||||
if frame.Max.X >= width {
|
||||
dot.X = 0
|
||||
dot.Y += face.Metrics().Ascent + face.Metrics().Descent
|
||||
|
||||
// padding + align to integer
|
||||
dot.Y += padding
|
||||
dot.Y = fixed.I(dot.Y.Ceil())
|
||||
}
|
||||
}
|
||||
|
||||
return mapping, bounds
|
||||
}
|
||||
|
||||
func i2f(i fixed.Int26_6) float64 {
|
||||
return float64(i) / (1 << 6)
|
||||
}
|
Reference in New Issue
Block a user