This repository has been archived on 2023-08-08. You can view files and clone it, but cannot push or open issues or pull requests.
tomo-old/elements/basic/textbox.go

306 lines
7.1 KiB
Go
Raw Normal View History

2023-01-18 05:38:58 +00:00
package basic
import "image"
import "git.tebibyte.media/sashakoshka/tomo"
import "git.tebibyte.media/sashakoshka/tomo/theme"
import "git.tebibyte.media/sashakoshka/tomo/artist"
import "git.tebibyte.media/sashakoshka/tomo/textmanip"
2023-01-18 05:38:58 +00:00
import "git.tebibyte.media/sashakoshka/tomo/elements/core"
2023-01-18 16:56:14 +00:00
// TextBox is a single-line text input.
2023-01-18 05:38:58 +00:00
type TextBox struct {
*core.Core
core core.CoreControl
enabled bool
selected bool
cursor int
scroll int
2023-01-18 05:38:58 +00:00
placeholder string
text []rune
2023-01-18 05:38:58 +00:00
placeholderDrawer artist.TextDrawer
valueDrawer artist.TextDrawer
onKeyDown func (tomo.Key, tomo.Modifiers, bool) (bool)
onChange func ()
2023-01-18 05:38:58 +00:00
}
2023-01-18 16:56:14 +00:00
// NewTextBox creates a new text box with the specified placeholder text, and
// a value. When the value is empty, the placeholder will be displayed in gray
// text.
func NewTextBox (placeholder, value string) (element *TextBox) {
2023-01-18 05:38:58 +00:00
element = &TextBox { enabled: true }
element.Core, element.core = core.NewCore(element)
element.placeholderDrawer.SetFace(theme.FontFaceRegular())
element.valueDrawer.SetFace(theme.FontFaceRegular())
element.placeholder = placeholder
element.placeholderDrawer.SetText([]rune(placeholder))
2023-01-18 05:38:58 +00:00
element.updateMinimumSize()
2023-01-18 16:56:14 +00:00
element.SetValue(value)
2023-01-18 05:38:58 +00:00
return
}
func (element *TextBox) Resize (width, height int) {
element.core.AllocateCanvas(width, height)
element.scrollToCursor()
2023-01-18 05:38:58 +00:00
element.draw()
}
func (element *TextBox) HandleMouseDown (x, y int, button tomo.Button) {
if !element.enabled { return }
2023-01-18 05:38:58 +00:00
element.Select()
}
func (element *TextBox) HandleMouseUp (x, y int, button tomo.Button) { }
func (element *TextBox) HandleMouseMove (x, y int) { }
func (element *TextBox) HandleScroll (x, y int, deltaX, deltaY float64) { }
func (element *TextBox) HandleKeyDown (
key tomo.Key,
modifiers tomo.Modifiers,
repeated bool,
) {
2023-01-18 16:56:14 +00:00
if element.onKeyDown != nil && element.onKeyDown(key, modifiers, repeated) {
return
}
altered := true
textChanged := false
2023-01-18 05:38:58 +00:00
switch {
case key == tomo.KeyBackspace:
if len(element.text) < 1 { break }
element.text, element.cursor = textmanip.Backspace (
element.text,
element.cursor,
modifiers.Control)
textChanged = true
case key == tomo.KeyDelete:
if len(element.text) < 1 { break }
element.text, element.cursor = textmanip.Delete (
element.text,
element.cursor,
modifiers.Control)
textChanged = true
case key == tomo.KeyLeft:
element.cursor = textmanip.MoveLeft (
element.text,
element.cursor,
modifiers.Control)
case key == tomo.KeyRight:
element.cursor = textmanip.MoveRight (
element.text,
element.cursor,
modifiers.Control)
2023-01-18 05:38:58 +00:00
case key.Printable():
element.text, element.cursor = textmanip.Type (
element.text,
element.cursor,
rune(key))
textChanged = true
default:
altered = false
}
if textChanged {
element.runOnChange()
element.valueDrawer.SetText(element.text)
}
if altered {
element.scrollToCursor()
}
if altered && element.core.HasImage () {
element.draw()
element.core.PushAll()
2023-01-18 05:38:58 +00:00
}
}
func (element *TextBox) HandleKeyUp(key tomo.Key, modifiers tomo.Modifiers) { }
func (element *TextBox) Selected () (selected bool) {
return element.selected
}
func (element *TextBox) Select () {
element.core.RequestSelection()
}
func (element *TextBox) HandleSelection (
direction tomo.SelectionDirection,
) (
accepted bool,
) {
direction = direction.Canon()
if !element.enabled { return false }
if element.selected && direction != tomo.SelectionDirectionNeutral {
return false
}
element.selected = true
if element.core.HasImage() {
element.draw()
element.core.PushAll()
}
return true
}
func (element *TextBox) HandleDeselection () {
element.selected = false
if element.core.HasImage() {
element.draw()
element.core.PushAll()
}
}
func (element *TextBox) SetEnabled (enabled bool) {
if element.enabled == enabled { return }
element.enabled = enabled
if element.core.HasImage () {
element.draw()
element.core.PushAll()
}
}
func (element *TextBox) SetPlaceholder (placeholder string) {
if element.placeholder == placeholder { return }
element.placeholder = placeholder
element.placeholderDrawer.SetText([]rune(placeholder))
2023-01-18 05:38:58 +00:00
element.updateMinimumSize()
if element.core.HasImage () {
element.draw()
element.core.PushAll()
}
}
func (element *TextBox) updateMinimumSize () {
textBounds := element.placeholderDrawer.LayoutBounds()
element.core.SetMinimumSize (
textBounds.Dx() +
theme.Padding() * 2,
element.placeholderDrawer.LineHeight().Round() +
theme.Padding() * 2)
}
2023-01-18 16:56:14 +00:00
func (element *TextBox) SetValue (text string) {
// if element.text == text { return }
2023-01-18 05:38:58 +00:00
element.text = []rune(text)
2023-01-18 16:56:14 +00:00
element.runOnChange()
element.valueDrawer.SetText(element.text)
2023-01-18 05:38:58 +00:00
if element.cursor > element.valueDrawer.Length() {
element.cursor = element.valueDrawer.Length()
}
element.scrollToCursor()
2023-01-18 05:38:58 +00:00
if element.core.HasImage () {
element.draw()
element.core.PushAll()
}
}
2023-01-18 06:29:59 +00:00
func (element *TextBox) Value () (value string) {
return string(element.text)
}
2023-01-18 16:56:14 +00:00
func (element *TextBox) Filled () (filled bool) {
return len(element.text) > 0
}
func (element *TextBox) OnKeyDown (
callback func (
key tomo.Key, modifiers tomo.Modifiers, repeated bool,
) (
handled bool,
),
) {
element.onKeyDown = callback
}
func (element *TextBox) OnChange (callback func ()) {
element.onChange = callback
}
func (element *TextBox) runOnChange () {
if element.onChange != nil {
element.onChange()
}
}
func (element *TextBox) scrollToCursor () {
if !element.core.HasImage() { return }
bounds := element.core.Bounds().Inset(theme.Padding())
bounds.Max.X -= element.valueDrawer.Em().Round()
cursorPosition := element.valueDrawer.PositionOf(element.cursor)
cursorPosition.X -= element.scroll
maxX := bounds.Max.X
minX := bounds.Min.X + bounds.Dx() / 2
if cursorPosition.X > maxX {
element.scroll += cursorPosition.X - maxX
} else if cursorPosition.X < minX {
element.scroll -= minX - cursorPosition.X
if element.scroll < 0 { element.scroll = 0 }
}
}
2023-01-18 05:38:58 +00:00
func (element *TextBox) draw () {
bounds := element.core.Bounds()
artist.FillRectangle (
element.core,
theme.InputPattern (
element.enabled,
element.Selected()),
bounds)
2023-01-18 06:20:04 +00:00
if len(element.text) == 0 && !element.selected {
2023-01-18 05:38:58 +00:00
// draw placeholder
textBounds := element.placeholderDrawer.LayoutBounds()
offset := image.Point {
X: theme.Padding(),
Y: theme.Padding(),
}
foreground := theme.ForegroundPattern(false)
element.placeholderDrawer.Draw (
element.core,
foreground,
offset.Sub(textBounds.Min))
} else {
// draw input value
textBounds := element.valueDrawer.LayoutBounds()
offset := image.Point {
X: theme.Padding() - element.scroll,
2023-01-18 05:38:58 +00:00
Y: theme.Padding(),
}
foreground := theme.ForegroundPattern(element.enabled)
element.valueDrawer.Draw (
element.core,
foreground,
offset.Sub(textBounds.Min))
if element.selected {
// cursor
cursorPosition := element.valueDrawer.PositionOf (
element.cursor)
artist.Line (
element.core,
theme.ForegroundPattern(true), 1,
cursorPosition.Add(offset),
image.Pt (
cursorPosition.X,
cursorPosition.Y + element.valueDrawer.
LineHeight().Round()).Add(offset))
}
}
}