mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2024-12-26 11:48:55 +01:00
text: Optimization: reduce ColorM calc
This commit is contained in:
parent
5257e6f9a7
commit
059bab0b13
104
text/text.go
104
text/text.go
@ -53,59 +53,16 @@ type colorMCacheEntry struct {
|
|||||||
atime int64
|
atime int64
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
func drawGlyph(dst *ebiten.Image, face font.Face, r rune, img *glyphImage, x, y fixed.Int26_6, clr ebiten.ColorM) {
|
||||||
colorMCache = map[colorMCacheKey]*colorMCacheEntry{}
|
|
||||||
)
|
|
||||||
|
|
||||||
func drawGlyph(dst *ebiten.Image, face font.Face, r rune, img *glyphImage, x, y fixed.Int26_6, clr color.Color) {
|
|
||||||
if img == nil {
|
if img == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// RGBA() is in [0 - 0xffff]. Adjust them in [0 - 0xff].
|
|
||||||
cr, cg, cb, ca := clr.RGBA()
|
|
||||||
cr >>= 8
|
|
||||||
cg >>= 8
|
|
||||||
cb >>= 8
|
|
||||||
ca >>= 8
|
|
||||||
if ca == 0 {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
b := getGlyphBounds(face, r)
|
b := getGlyphBounds(face, r)
|
||||||
op := &ebiten.DrawImageOptions{}
|
op := &ebiten.DrawImageOptions{}
|
||||||
op.GeoM.Translate(fixed26_6ToFloat64(x+b.Min.X), fixed26_6ToFloat64(y+b.Min.Y))
|
op.GeoM.Translate(fixed26_6ToFloat64(x+b.Min.X), fixed26_6ToFloat64(y+b.Min.Y))
|
||||||
|
|
||||||
key := colorMCacheKey(uint32(cr) | (uint32(cg) << 8) | (uint32(cb) << 16) | (uint32(ca) << 24))
|
op.ColorM = clr
|
||||||
e, ok := colorMCache[key]
|
|
||||||
if ok {
|
|
||||||
e.atime = now()
|
|
||||||
} else {
|
|
||||||
if len(colorMCache) > cacheLimit {
|
|
||||||
oldest := int64(math.MaxInt64)
|
|
||||||
oldestKey := colorMCacheKey(0)
|
|
||||||
for key, c := range colorMCache {
|
|
||||||
if c.atime < oldest {
|
|
||||||
oldestKey = key
|
|
||||||
oldest = c.atime
|
|
||||||
}
|
|
||||||
}
|
|
||||||
delete(colorMCache, oldestKey)
|
|
||||||
}
|
|
||||||
|
|
||||||
cm := ebiten.ColorM{}
|
|
||||||
rf := float64(cr) / float64(ca)
|
|
||||||
gf := float64(cg) / float64(ca)
|
|
||||||
bf := float64(cb) / float64(ca)
|
|
||||||
af := float64(ca) / 0xff
|
|
||||||
cm.Scale(rf, gf, bf, af)
|
|
||||||
e = &colorMCacheEntry{
|
|
||||||
m: cm,
|
|
||||||
atime: now(),
|
|
||||||
}
|
|
||||||
colorMCache[key] = e
|
|
||||||
}
|
|
||||||
op.ColorM = e.m
|
|
||||||
re := image.Rect(img.x, img.y, img.x+img.width, img.y+img.height)
|
re := image.Rect(img.x, img.y, img.x+img.width, img.y+img.height)
|
||||||
op.SourceRect = &re
|
op.SourceRect = &re
|
||||||
|
|
||||||
@ -191,6 +148,7 @@ func getGlyphImages(face font.Face, runes []rune) []*glyphImage {
|
|||||||
neededGlyphs[i] = b
|
neededGlyphs[i] = b
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: What if w2 is too big (e.g. > 4096)?
|
||||||
w2 := 0
|
w2 := 0
|
||||||
h2 := 0
|
h2 := 0
|
||||||
for _, b := range neededGlyphs {
|
for _, b := range neededGlyphs {
|
||||||
@ -236,6 +194,58 @@ func getGlyphImages(face font.Face, runes []rune) []*glyphImage {
|
|||||||
|
|
||||||
var textM sync.Mutex
|
var textM sync.Mutex
|
||||||
|
|
||||||
|
var (
|
||||||
|
colorMCache = map[colorMCacheKey]*colorMCacheEntry{}
|
||||||
|
emptyColorM ebiten.ColorM
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
emptyColorM.Scale(0, 0, 0, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func colorToColorM(clr color.Color) ebiten.ColorM {
|
||||||
|
// RGBA() is in [0 - 0xffff]. Adjust them in [0 - 0xff].
|
||||||
|
cr, cg, cb, ca := clr.RGBA()
|
||||||
|
cr >>= 8
|
||||||
|
cg >>= 8
|
||||||
|
cb >>= 8
|
||||||
|
ca >>= 8
|
||||||
|
if ca == 0 {
|
||||||
|
return emptyColorM
|
||||||
|
}
|
||||||
|
key := colorMCacheKey(uint32(cr) | (uint32(cg) << 8) | (uint32(cb) << 16) | (uint32(ca) << 24))
|
||||||
|
e, ok := colorMCache[key]
|
||||||
|
if ok {
|
||||||
|
e.atime = now()
|
||||||
|
return e.m
|
||||||
|
}
|
||||||
|
if len(colorMCache) > cacheLimit {
|
||||||
|
oldest := int64(math.MaxInt64)
|
||||||
|
oldestKey := colorMCacheKey(0)
|
||||||
|
for key, c := range colorMCache {
|
||||||
|
if c.atime < oldest {
|
||||||
|
oldestKey = key
|
||||||
|
oldest = c.atime
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete(colorMCache, oldestKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
cm := ebiten.ColorM{}
|
||||||
|
rf := float64(cr) / float64(ca)
|
||||||
|
gf := float64(cg) / float64(ca)
|
||||||
|
bf := float64(cb) / float64(ca)
|
||||||
|
af := float64(ca) / 0xff
|
||||||
|
cm.Scale(rf, gf, bf, af)
|
||||||
|
e = &colorMCacheEntry{
|
||||||
|
m: cm,
|
||||||
|
atime: now(),
|
||||||
|
}
|
||||||
|
colorMCache[key] = e
|
||||||
|
|
||||||
|
return e.m
|
||||||
|
}
|
||||||
|
|
||||||
// Draw draws a given text on a given destination image dst.
|
// Draw draws a given text on a given destination image dst.
|
||||||
//
|
//
|
||||||
// face is the font for text rendering.
|
// face is the font for text rendering.
|
||||||
@ -258,11 +268,13 @@ func Draw(dst *ebiten.Image, text string, face font.Face, x, y int, clr color.Co
|
|||||||
|
|
||||||
runes := []rune(text)
|
runes := []rune(text)
|
||||||
glyphImgs := getGlyphImages(face, runes)
|
glyphImgs := getGlyphImages(face, runes)
|
||||||
|
colorm := colorToColorM(clr)
|
||||||
|
|
||||||
for i, r := range runes {
|
for i, r := range runes {
|
||||||
if prevR >= 0 {
|
if prevR >= 0 {
|
||||||
fx += face.Kern(prevR, r)
|
fx += face.Kern(prevR, r)
|
||||||
}
|
}
|
||||||
drawGlyph(dst, face, r, glyphImgs[i], fx, fixed.I(y), clr)
|
drawGlyph(dst, face, r, glyphImgs[i], fx, fixed.I(y), colorm)
|
||||||
fx += glyphAdvance(face, r)
|
fx += glyphAdvance(face, r)
|
||||||
|
|
||||||
prevR = r
|
prevR = r
|
||||||
|
Loading…
Reference in New Issue
Block a user