mirror of
https://git.sr.ht/~eliasnaur/gio
synced 2026-07-05 17:35:36 +00:00
text,widget,opentype: change text.Face.Shape to return a clip.PathSpec
With this change, the Shape function returns a clip.PathSpec instead of a clip.Outline op. It is then possible to create a clip.Outline or clip.Stroke op to fill the text path or draw its stroke. Signed-off-by: Christophe Meessen <meessen@cppm.in2p3.fr>
This commit is contained in:
committed by
Elias Naur
parent
3db11cbaad
commit
a34e239c04
@@ -119,7 +119,7 @@ func (f *Font) Layout(ppem fixed.Int26_6, maxWidth int, txt io.Reader) ([]text.L
|
|||||||
return layoutText(&buf, ppem, maxWidth, fonts, glyphs)
|
return layoutText(&buf, ppem, maxWidth, fonts, glyphs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *Font) Shape(ppem fixed.Int26_6, str text.Layout) clip.Op {
|
func (f *Font) Shape(ppem fixed.Int26_6, str text.Layout) clip.PathSpec {
|
||||||
var buf sfnt.Buffer
|
var buf sfnt.Buffer
|
||||||
return textPath(&buf, ppem, []*opentype{{Font: f.font, Hinting: font.HintingFull}}, str)
|
return textPath(&buf, ppem, []*opentype{{Font: f.font, Hinting: font.HintingFull}}, str)
|
||||||
}
|
}
|
||||||
@@ -139,7 +139,7 @@ func (c *Collection) Layout(ppem fixed.Int26_6, maxWidth int, txt io.Reader) ([]
|
|||||||
return layoutText(&buf, ppem, maxWidth, c.fonts, glyphs)
|
return layoutText(&buf, ppem, maxWidth, c.fonts, glyphs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Collection) Shape(ppem fixed.Int26_6, str text.Layout) clip.Op {
|
func (c *Collection) Shape(ppem fixed.Int26_6, str text.Layout) clip.PathSpec {
|
||||||
var buf sfnt.Buffer
|
var buf sfnt.Buffer
|
||||||
return textPath(&buf, ppem, c.fonts, str)
|
return textPath(&buf, ppem, c.fonts, str)
|
||||||
}
|
}
|
||||||
@@ -261,12 +261,11 @@ func toLayout(glyphs []glyph) text.Layout {
|
|||||||
return text.Layout{Text: buf.String(), Advances: advs}
|
return text.Layout{Text: buf.String(), Advances: advs}
|
||||||
}
|
}
|
||||||
|
|
||||||
func textPath(buf *sfnt.Buffer, ppem fixed.Int26_6, fonts []*opentype, str text.Layout) clip.Op {
|
func textPath(buf *sfnt.Buffer, ppem fixed.Int26_6, fonts []*opentype, str text.Layout) clip.PathSpec {
|
||||||
var lastPos f32.Point
|
var lastPos f32.Point
|
||||||
var builder clip.Path
|
var builder clip.Path
|
||||||
ops := new(op.Ops)
|
|
||||||
var x fixed.Int26_6
|
var x fixed.Int26_6
|
||||||
builder.Begin(ops)
|
builder.Begin(new(op.Ops))
|
||||||
rune := 0
|
rune := 0
|
||||||
for _, r := range str.Text {
|
for _, r := range str.Text {
|
||||||
if !unicode.IsSpace(r) {
|
if !unicode.IsSpace(r) {
|
||||||
@@ -323,9 +322,7 @@ func textPath(buf *sfnt.Buffer, ppem fixed.Int26_6, fonts []*opentype, str text.
|
|||||||
x += str.Advances[rune]
|
x += str.Advances[rune]
|
||||||
rune++
|
rune++
|
||||||
}
|
}
|
||||||
return clip.Outline{
|
return builder.End()
|
||||||
Path: builder.End(),
|
|
||||||
}.Op()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func readGlyphs(r io.Reader) ([]glyph, error) {
|
func readGlyphs(r io.Reader) ([]glyph, error) {
|
||||||
|
|||||||
@@ -70,7 +70,7 @@ func TestCollectionAsFace(t *testing.T) {
|
|||||||
|
|
||||||
// All shapes from the original fonts should be distinct because the glyphs are distinct, including the replacement
|
// All shapes from the original fonts should be distinct because the glyphs are distinct, including the replacement
|
||||||
// glyphs.
|
// glyphs.
|
||||||
distinctShapes := []clip.Op{shapeValid1, shapeInvalid1, shapeValid2, shapeInvalid2}
|
distinctShapes := []clip.PathSpec{shapeValid1, shapeInvalid1, shapeValid2, shapeInvalid2}
|
||||||
for i := 0; i < len(distinctShapes); i++ {
|
for i := 0; i < len(distinctShapes); i++ {
|
||||||
for j := i + 1; j < len(distinctShapes); j++ {
|
for j := i + 1; j < len(distinctShapes); j++ {
|
||||||
if areShapesEqual(distinctShapes[i], distinctShapes[j]) {
|
if areShapesEqual(distinctShapes[i], distinctShapes[j]) {
|
||||||
@@ -174,23 +174,23 @@ func mergeFonts(ttf1, ttf2 []byte) []byte {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// shapeRune uses a given Face to shape exactly one rune at a fixed size, then returns the resulting shape data.
|
// shapeRune uses a given Face to shape exactly one rune at a fixed size, then returns the resulting shape data.
|
||||||
func shapeRune(f text.Face, r rune) (clip.Op, error) {
|
func shapeRune(f text.Face, r rune) (clip.PathSpec, error) {
|
||||||
ppem := fixed.I(200)
|
ppem := fixed.I(200)
|
||||||
lines, err := f.Layout(ppem, 2000, strings.NewReader(string(r)))
|
lines, err := f.Layout(ppem, 2000, strings.NewReader(string(r)))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return clip.Op{}, err
|
return clip.PathSpec{}, err
|
||||||
}
|
}
|
||||||
if len(lines) != 1 {
|
if len(lines) != 1 {
|
||||||
return clip.Op{}, fmt.Errorf("unexpected rendering for \"U+%08X\": got %d lines (expected: 1)", r, len(lines))
|
return clip.PathSpec{}, fmt.Errorf("unexpected rendering for \"U+%08X\": got %d lines (expected: 1)", r, len(lines))
|
||||||
}
|
}
|
||||||
return f.Shape(ppem, lines[0].Layout), nil
|
return f.Shape(ppem, lines[0].Layout), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// areShapesEqual returns true iff both given text shapes are produced with identical operations.
|
// areShapesEqual returns true iff both given text shapes are produced with identical operations.
|
||||||
func areShapesEqual(shape1, shape2 clip.Op) bool {
|
func areShapesEqual(shape1, shape2 clip.PathSpec) bool {
|
||||||
var ops1, ops2 op.Ops
|
var ops1, ops2 op.Ops
|
||||||
shape1.Push(&ops1).Pop()
|
clip.Outline{Path: shape1}.Op().Push(&ops1).Pop()
|
||||||
shape2.Push(&ops2).Pop()
|
clip.Outline{Path: shape2}.Op().Push(&ops2).Pop()
|
||||||
var r1, r2 ops.Reader
|
var r1, r2 ops.Reader
|
||||||
r1.Reset(&ops1.Internal)
|
r1.Reset(&ops1.Internal)
|
||||||
r2.Reset(&ops2.Internal)
|
r2.Reset(&ops2.Internal)
|
||||||
|
|||||||
+4
-4
@@ -26,7 +26,7 @@ type layoutElem struct {
|
|||||||
type path struct {
|
type path struct {
|
||||||
next, prev *path
|
next, prev *path
|
||||||
key pathKey
|
key pathKey
|
||||||
val clip.Op
|
val clip.PathSpec
|
||||||
}
|
}
|
||||||
|
|
||||||
type layoutKey struct {
|
type layoutKey struct {
|
||||||
@@ -81,16 +81,16 @@ func (l *layoutCache) insert(lt *layoutElem) {
|
|||||||
lt.next.prev = lt
|
lt.next.prev = lt
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *pathCache) Get(k pathKey) (clip.Op, bool) {
|
func (c *pathCache) Get(k pathKey) (clip.PathSpec, bool) {
|
||||||
if v, ok := c.m[k]; ok {
|
if v, ok := c.m[k]; ok {
|
||||||
c.remove(v)
|
c.remove(v)
|
||||||
c.insert(v)
|
c.insert(v)
|
||||||
return v.val, true
|
return v.val, true
|
||||||
}
|
}
|
||||||
return clip.Op{}, false
|
return clip.PathSpec{}, false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *pathCache) Put(k pathKey, v clip.Op) {
|
func (c *pathCache) Put(k pathKey, v clip.PathSpec) {
|
||||||
if c.m == nil {
|
if c.m == nil {
|
||||||
c.m = make(map[pathKey]*path)
|
c.m = make(map[pathKey]*path)
|
||||||
c.head = new(path)
|
c.head = new(path)
|
||||||
|
|||||||
+1
-1
@@ -24,7 +24,7 @@ func TestLayoutLRU(t *testing.T) {
|
|||||||
func TestPathLRU(t *testing.T) {
|
func TestPathLRU(t *testing.T) {
|
||||||
c := new(pathCache)
|
c := new(pathCache)
|
||||||
put := func(i int) {
|
put := func(i int) {
|
||||||
c.Put(pathKey{str: strconv.Itoa(i)}, clip.Op{})
|
c.Put(pathKey{str: strconv.Itoa(i)}, clip.PathSpec{})
|
||||||
}
|
}
|
||||||
get := func(i int) bool {
|
get := func(i int) bool {
|
||||||
_, ok := c.Get(pathKey{str: strconv.Itoa(i)})
|
_, ok := c.Get(pathKey{str: strconv.Itoa(i)})
|
||||||
|
|||||||
+4
-4
@@ -18,7 +18,7 @@ type Shaper interface {
|
|||||||
// LayoutString is Layout for strings.
|
// LayoutString is Layout for strings.
|
||||||
LayoutString(font Font, size fixed.Int26_6, maxWidth int, str string) []Line
|
LayoutString(font Font, size fixed.Int26_6, maxWidth int, str string) []Line
|
||||||
// Shape a line of text and return a clipping operation for its outline.
|
// Shape a line of text and return a clipping operation for its outline.
|
||||||
Shape(font Font, size fixed.Int26_6, layout Layout) clip.Op
|
Shape(font Font, size fixed.Int26_6, layout Layout) clip.PathSpec
|
||||||
}
|
}
|
||||||
|
|
||||||
// A FontFace is a Font and a matching Face.
|
// A FontFace is a Font and a matching Face.
|
||||||
@@ -121,7 +121,7 @@ func (c *Cache) LayoutString(font Font, size fixed.Int26_6, maxWidth int, str st
|
|||||||
|
|
||||||
// Shape is a caching implementation of the Shaper interface. Shape assumes that the layout
|
// Shape is a caching implementation of the Shaper interface. Shape assumes that the layout
|
||||||
// argument is unchanged from a call to Layout or LayoutString.
|
// argument is unchanged from a call to Layout or LayoutString.
|
||||||
func (c *Cache) Shape(font Font, size fixed.Int26_6, layout Layout) clip.Op {
|
func (c *Cache) Shape(font Font, size fixed.Int26_6, layout Layout) clip.PathSpec {
|
||||||
cache := c.lookup(font)
|
cache := c.lookup(font)
|
||||||
return cache.shape(size, layout)
|
return cache.shape(size, layout)
|
||||||
}
|
}
|
||||||
@@ -143,9 +143,9 @@ func (f *faceCache) layout(ppem fixed.Int26_6, maxWidth int, str string) []Line
|
|||||||
return l
|
return l
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *faceCache) shape(ppem fixed.Int26_6, layout Layout) clip.Op {
|
func (f *faceCache) shape(ppem fixed.Int26_6, layout Layout) clip.PathSpec {
|
||||||
if f == nil {
|
if f == nil {
|
||||||
return clip.Op{}
|
return clip.PathSpec{}
|
||||||
}
|
}
|
||||||
pk := pathKey{
|
pk := pathKey{
|
||||||
ppem: ppem,
|
ppem: ppem,
|
||||||
|
|||||||
+1
-1
@@ -48,7 +48,7 @@ type Font struct {
|
|||||||
// methods must be safe for concurrent use.
|
// methods must be safe for concurrent use.
|
||||||
type Face interface {
|
type Face interface {
|
||||||
Layout(ppem fixed.Int26_6, maxWidth int, txt io.Reader) ([]Line, error)
|
Layout(ppem fixed.Int26_6, maxWidth int, txt io.Reader) ([]Line, error)
|
||||||
Shape(ppem fixed.Int26_6, str Layout) clip.Op
|
Shape(ppem fixed.Int26_6, str Layout) clip.PathSpec
|
||||||
}
|
}
|
||||||
|
|
||||||
// Typeface identifies a particular typeface design. The empty
|
// Typeface identifies a particular typeface design. The empty
|
||||||
|
|||||||
+2
-2
@@ -531,8 +531,8 @@ func (e *Editor) layout(gtx layout.Context, content layout.Widget) layout.Dimens
|
|||||||
if !ok {
|
if !ok {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
path := e.shaper.Shape(e.font, e.textSize, layout)
|
op := clip.Outline{Path: e.shaper.Shape(e.font, e.textSize, layout)}.Op()
|
||||||
e.shapes = append(e.shapes, line{off, path, selected, yOffs, size})
|
e.shapes = append(e.shapes, line{off, op, selected, yOffs, size})
|
||||||
}
|
}
|
||||||
|
|
||||||
key.InputOp{Tag: &e.eventKey, Hint: e.InputHint}.Add(gtx.Ops)
|
key.InputOp{Tag: &e.eventKey, Hint: e.InputHint}.Add(gtx.Ops)
|
||||||
|
|||||||
+1
-1
@@ -179,7 +179,7 @@ func (l Label) Layout(gtx layout.Context, s text.Shaper, font text.Font, size un
|
|||||||
}
|
}
|
||||||
t := op.Offset(layout.FPt(off)).Push(gtx.Ops)
|
t := op.Offset(layout.FPt(off)).Push(gtx.Ops)
|
||||||
rcl := clip.Rect(cl.Sub(off)).Push(gtx.Ops)
|
rcl := clip.Rect(cl.Sub(off)).Push(gtx.Ops)
|
||||||
cl := s.Shape(font, textSize, l).Push(gtx.Ops)
|
cl := clip.Outline{Path: s.Shape(font, textSize, l)}.Op().Push(gtx.Ops)
|
||||||
paint.PaintOp{}.Add(gtx.Ops)
|
paint.PaintOp{}.Add(gtx.Ops)
|
||||||
cl.Pop()
|
cl.Pop()
|
||||||
rcl.Pop()
|
rcl.Pop()
|
||||||
|
|||||||
Reference in New Issue
Block a user