Rudimentary text selection with the mouse

This commit is contained in:
Sasha Koshka 2023-02-13 18:29:49 -05:00
parent 88502cf628
commit d18da8b07a
2 changed files with 96 additions and 20 deletions

View File

@ -10,6 +10,7 @@ import "git.tebibyte.media/sashakoshka/tomo/canvas"
type characterLayout struct { type characterLayout struct {
x int x int
width int
character rune character rune
} }
@ -45,6 +46,7 @@ type TextDrawer struct {
align Align align Align
wrap bool wrap bool
cut bool cut bool
metrics font.Metrics
layout []wordLayout layout []wordLayout
layoutClean bool layoutClean bool
@ -205,6 +207,38 @@ func (drawer *TextDrawer) PositionOf (index int) (position image.Point) {
return return
} }
// AtPosition returns the index at the specified position relative to the
// baseline.
func (drawer *TextDrawer) AtPosition (position image.Point) (index int) {
cursor := 0
if !drawer.layoutClean { drawer.recalculate() }
for _, word := range drawer.layout {
for _, character := range word.text {
bounds := drawer.boundsOfChar(character).Add(word.position)
if position.In(bounds) {
return cursor
}
cursor ++
}
for _, character := range word.whitespace {
bounds := drawer.boundsOfChar(character).Add(word.position)
if position.In(bounds) {
return cursor
}
cursor ++
}
}
return -1
}
func (drawer *TextDrawer) boundsOfChar (char characterLayout) (image.Rectangle) {
return image.Rect (
char.x, 0,
char.x + char.width,
drawer.metrics.Height.Ceil()).
Sub(image.Pt(0, drawer.metrics.Descent.Round()))
}
// Length returns the amount of runes in the drawer's text. // Length returns the amount of runes in the drawer's text.
func (drawer *TextDrawer) Length () (length int) { func (drawer *TextDrawer) Length () (length int) {
return len(drawer.runes) return len(drawer.runes)
@ -217,7 +251,7 @@ func (drawer *TextDrawer) recalculate () {
if drawer.runes == nil { return } if drawer.runes == nil { return }
if drawer.face == nil { return } if drawer.face == nil { return }
metrics := drawer.face.Metrics() drawer.metrics = drawer.face.Metrics()
dot := fixed.Point26_6 { 0, 0 } dot := fixed.Point26_6 { 0, 0 }
index := 0 index := 0
horizontalExtent := 0 horizontalExtent := 0
@ -241,6 +275,7 @@ func (drawer *TextDrawer) recalculate () {
word.text = append(word.text, characterLayout { word.text = append(word.text, characterLayout {
x: currentCharacterX.Round(), x: currentCharacterX.Round(),
character: character, character: character,
width: advance.Ceil(),
}) })
dot.X += advance dot.X += advance
@ -264,9 +299,9 @@ func (drawer *TextDrawer) recalculate () {
word.width + word.position.X > drawer.width && word.width + word.position.X > drawer.width &&
word.position.X > 0 { word.position.X > 0 {
word.position.Y += metrics.Height.Round() word.position.Y += drawer.metrics.Height.Round()
word.position.X = 0 word.position.X = 0
dot.Y += metrics.Height dot.Y += drawer.metrics.Height
dot.X = wordWidth dot.X = wordWidth
} }
@ -281,12 +316,13 @@ func (drawer *TextDrawer) recalculate () {
word.whitespace = append(word.whitespace, characterLayout { word.whitespace = append(word.whitespace, characterLayout {
x: currentCharacterX.Round(), x: currentCharacterX.Round(),
character: character, character: character,
width: advance.Ceil(),
}) })
spaceWidth += advance spaceWidth += advance
currentCharacterX += advance currentCharacterX += advance
if character == '\n' { if character == '\n' {
dot.Y += metrics.Height dot.Y += drawer.metrics.Height
dot.X = 0 dot.X = 0
word.breaksAfter ++ word.breaksAfter ++
break break
@ -309,8 +345,9 @@ func (drawer *TextDrawer) recalculate () {
// stop processing more words. and remove any words that have // stop processing more words. and remove any words that have
// also crossed the line. // also crossed the line.
if if
drawer.cut && drawer.cut && (
(dot.Y - metrics.Ascent - metrics.Descent).Round() > dot.Y - drawer.metrics.Ascent -
drawer.metrics.Descent).Round() >
drawer.height { drawer.height {
for for
@ -343,11 +380,15 @@ func (drawer *TextDrawer) recalculate () {
} }
if drawer.cut { if drawer.cut {
drawer.layoutBounds.Min.Y = 0 - metrics.Ascent.Round() drawer.layoutBounds.Min.Y = 0 - drawer.metrics.Ascent.Round()
drawer.layoutBounds.Max.Y = drawer.height - metrics.Ascent.Round() drawer.layoutBounds.Max.Y =
drawer.height -
drawer.metrics.Ascent.Round()
} else { } else {
drawer.layoutBounds.Min.Y = 0 - metrics.Ascent.Round() drawer.layoutBounds.Min.Y = 0 - drawer.metrics.Ascent.Round()
drawer.layoutBounds.Max.Y = dot.Y.Round() + metrics.Descent.Round() drawer.layoutBounds.Max.Y =
dot.Y.Round() +
drawer.metrics.Descent.Round()
} }
// TODO: // TODO:

View File

@ -15,6 +15,7 @@ type TextBox struct {
core core.CoreControl core core.CoreControl
focusableControl core.FocusableCoreControl focusableControl core.FocusableCoreControl
dragging bool
dot textmanip.Dot dot textmanip.Dot
scroll int scroll int
placeholder string placeholder string
@ -63,10 +64,44 @@ func (element *TextBox) handleResize () {
func (element *TextBox) HandleMouseDown (x, y int, button input.Button) { func (element *TextBox) HandleMouseDown (x, y int, button input.Button) {
if !element.Enabled() { return } if !element.Enabled() { return }
if !element.Focused() { element.Focus() } if !element.Focused() { element.Focus() }
if button == input.ButtonLeft {
point := image.Pt(x, y)
offset := element.Bounds().Min.Add (image.Pt (
element.config.Padding() - element.scroll,
element.config.Padding()))
runeIndex := element.valueDrawer.AtPosition(point.Sub(offset))
element.dragging = true
if runeIndex > -1 {
element.dot = textmanip.EmptyDot(runeIndex)
element.redo()
}
}
}
func (element *TextBox) HandleMouseMove (x, y int) {
if !element.Enabled() { return }
if !element.Focused() { element.Focus() }
if element.dragging {
point := image.Pt(x, y)
offset := element.Bounds().Min.Add (image.Pt (
element.config.Padding() - element.scroll,
element.config.Padding()))
runeIndex := element.valueDrawer.AtPosition(point.Sub(offset))
if runeIndex > -1 {
element.dot.End = runeIndex
element.redo()
}
}
}
func (element *TextBox) HandleMouseUp (x, y int, button input.Button) {
if button == input.ButtonLeft {
element.dragging = false
}
} }
func (element *TextBox) HandleMouseUp (x, y int, button input.Button) { }
func (element *TextBox) HandleMouseMove (x, y int) { }
func (element *TextBox) HandleMouseScroll (x, y int, deltaX, deltaY float64) { } func (element *TextBox) HandleMouseScroll (x, y int, deltaX, deltaY float64) { }
func (element *TextBox) HandleKeyDown(key input.Key, modifiers input.Modifiers) { func (element *TextBox) HandleKeyDown(key input.Key, modifiers input.Modifiers) {