text,widget/material: make font collections explicit

Before this change, package font implemented a global font registry,
with the usual problems of package global state.

This change deletes the global registry and introduces the text.Collection
type for representing a list of fonts and their faces. Collection exports
Lookup that finds the closest match and its face.

The existing FontRegistry is renamed to Cache to reflect its new limited
functionality: a cache of shapes and measurements on top of a Collection.

Then, material.NewTheme is changed to take a Collection and initialize
a Cache.

Updates gio#19 because multiple windows require a separate (writable) Cache per
window, while (read-only) Collections may be shared.

Signed-off-by: Elias Naur <mail@eliasnaur.com>
This commit is contained in:
Elias Naur
2020-06-07 16:20:44 +02:00
parent c2c31a4d00
commit b07d34354e
4 changed files with 110 additions and 113 deletions
-36
View File
@@ -1,36 +0,0 @@
// SPDX-License-Identifier: Unlicense OR MIT
// Package font implements a central font registry.
package font
import (
"sync"
"gioui.org/text"
)
var (
mu sync.Mutex
initialized bool
shaper = new(text.FontRegistry)
)
// Default returns a singleton *text.Shaper that contains
// the registered fonts.
func Default() *text.FontRegistry {
mu.Lock()
defer mu.Unlock()
initialized = true
return shaper
}
// Register a face. Register panics if Default has been
// called.
func Register(font text.Font, face text.Face) {
mu.Lock()
defer mu.Unlock()
if initialized {
panic("Register must be called before Default")
}
shaper.Register(font, face)
}
+28 -18
View File
@@ -1,6 +1,6 @@
// SPDX-License-Identifier: Unlicense OR MIT
// Package gofont registers the Go fonts in the font registry.
// Package gofont exports the Go fonts as a text.Collection.
//
// See https://blog.golang.org/go-fonts for a description of the
// fonts, and the golang.org/x/image/font/gofont packages for the
@@ -9,8 +9,8 @@ package gofont
import (
"fmt"
"sync"
"gioui.org/font"
"gioui.org/font/opentype"
"gioui.org/text"
"golang.org/x/image/font/gofont/gobold"
@@ -27,27 +27,37 @@ import (
"golang.org/x/image/font/gofont/gosmallcapsitalic"
)
func Register() {
register(text.Font{}, goregular.TTF)
register(text.Font{Style: text.Italic}, goitalic.TTF)
register(text.Font{Weight: text.Bold}, gobold.TTF)
register(text.Font{Style: text.Italic, Weight: text.Bold}, gobolditalic.TTF)
register(text.Font{Weight: text.Medium}, gomedium.TTF)
register(text.Font{Weight: text.Medium, Style: text.Italic}, gomediumitalic.TTF)
register(text.Font{Variant: "Mono"}, gomono.TTF)
register(text.Font{Variant: "Mono", Weight: text.Bold}, gomonobold.TTF)
register(text.Font{Variant: "Mono", Weight: text.Bold, Style: text.Italic}, gomonobolditalic.TTF)
register(text.Font{Variant: "Mono", Style: text.Italic}, gomonoitalic.TTF)
register(text.Font{Variant: "Mono", Style: text.Italic}, gomonoitalic.TTF)
register(text.Font{Variant: "Smallcaps"}, gosmallcaps.TTF)
register(text.Font{Variant: "Smallcaps", Style: text.Italic}, gosmallcapsitalic.TTF)
var (
once sync.Once
collection *text.Collection
)
func Collection() *text.Collection {
once.Do(func() {
c := new(text.Collection)
register(c, text.Font{}, goregular.TTF)
register(c, text.Font{Style: text.Italic}, goitalic.TTF)
register(c, text.Font{Weight: text.Bold}, gobold.TTF)
register(c, text.Font{Style: text.Italic, Weight: text.Bold}, gobolditalic.TTF)
register(c, text.Font{Weight: text.Medium}, gomedium.TTF)
register(c, text.Font{Weight: text.Medium, Style: text.Italic}, gomediumitalic.TTF)
register(c, text.Font{Variant: "Mono"}, gomono.TTF)
register(c, text.Font{Variant: "Mono", Weight: text.Bold}, gomonobold.TTF)
register(c, text.Font{Variant: "Mono", Weight: text.Bold, Style: text.Italic}, gomonobolditalic.TTF)
register(c, text.Font{Variant: "Mono", Style: text.Italic}, gomonoitalic.TTF)
register(c, text.Font{Variant: "Mono", Style: text.Italic}, gomonoitalic.TTF)
register(c, text.Font{Variant: "Smallcaps"}, gosmallcaps.TTF)
register(c, text.Font{Variant: "Smallcaps", Style: text.Italic}, gosmallcapsitalic.TTF)
collection = c
})
return collection
}
func register(fnt text.Font, ttf []byte) {
func register(c *text.Collection, fnt text.Font, ttf []byte) {
face, err := opentype.Parse(ttf)
if err != nil {
panic(fmt.Sprintf("failed to parse font: %v", err))
}
fnt.Typeface = "Go"
font.Register(fnt, face)
c.Register(fnt, face)
}