210 lines
5.2 KiB
Go
Raw Normal View History

2016-09-11 17:55:19 +02:00
package components
2016-09-27 22:05:44 +02:00
import (
2017-09-23 13:56:45 +02:00
"github.com/erroneousboat/termui"
runewidth "github.com/mattn/go-runewidth"
2016-09-27 22:05:44 +02:00
)
2016-09-11 17:55:19 +02:00
2016-10-02 14:53:00 +02:00
// Input is the definition of an Input component
2016-09-11 17:55:19 +02:00
type Input struct {
Par *termui.Par
Text []rune
CursorPositionScreen int
CursorPositionText int
Offset int
2016-09-11 17:55:19 +02:00
}
2016-09-13 21:13:51 +02:00
// CreateInput is the constructor of the Input struct
2017-12-01 23:52:25 +01:00
func CreateInputComponent() *Input {
2016-09-11 17:55:19 +02:00
input := &Input{
Par: termui.NewPar(""),
Text: make([]rune, 0),
CursorPositionScreen: 0,
CursorPositionText: 0,
Offset: 0,
2016-09-11 17:55:19 +02:00
}
2016-09-13 21:02:02 +02:00
input.Par.Height = 3
2016-09-11 17:55:19 +02:00
return input
}
2016-09-13 21:13:51 +02:00
// Buffer implements interface termui.Bufferer
2016-09-11 17:55:19 +02:00
func (i *Input) Buffer() termui.Buffer {
2016-09-13 21:02:02 +02:00
buf := i.Par.Buffer()
// Set visible cursor, get char at screen cursor position
char := buf.At(i.Par.InnerX()+i.CursorPositionScreen, i.Par.Block.InnerY())
2016-09-13 21:02:02 +02:00
buf.Set(
i.Par.InnerX()+i.CursorPositionScreen,
2016-09-13 21:02:02 +02:00
i.Par.Block.InnerY(),
2016-10-11 19:28:37 +02:00
termui.Cell{
Ch: char.Ch,
Fg: i.Par.TextBgColor,
Bg: i.Par.TextFgColor,
},
2016-09-13 21:02:02 +02:00
)
return buf
2016-09-11 17:55:19 +02:00
}
2016-09-13 21:13:51 +02:00
// GetHeight implements interface termui.GridBufferer
2016-09-11 17:55:19 +02:00
func (i *Input) GetHeight() int {
2016-09-13 21:02:02 +02:00
return i.Par.Block.GetHeight()
2016-09-11 17:55:19 +02:00
}
2016-09-13 21:13:51 +02:00
// SetWidth implements interface termui.GridBufferer
2016-09-11 17:55:19 +02:00
func (i *Input) SetWidth(w int) {
2016-09-13 21:02:02 +02:00
i.Par.SetWidth(w)
2016-09-11 17:55:19 +02:00
}
2016-09-13 21:13:51 +02:00
// SetX implements interface termui.GridBufferer
2016-09-11 17:55:19 +02:00
func (i *Input) SetX(x int) {
2016-09-13 21:02:02 +02:00
i.Par.SetX(x)
2016-09-11 17:55:19 +02:00
}
2016-09-13 21:13:51 +02:00
// SetY implements interface termui.GridBufferer
2016-09-11 17:55:19 +02:00
func (i *Input) SetY(y int) {
2016-09-13 21:02:02 +02:00
i.Par.SetY(y)
2016-09-11 17:55:19 +02:00
}
// Insert will insert a given key at the place of the current CursorPositionText
2016-10-21 21:23:25 +02:00
func (i *Input) Insert(key rune) {
// Append key to the left side
left := make([]rune, len(i.Text[0:i.CursorPositionText]))
copy(left, i.Text[0:i.CursorPositionText])
left = append(left, key)
2016-10-21 21:23:25 +02:00
// Combine left and right side
i.Text = append(left, i.Text[i.CursorPositionText:]...)
i.MoveCursorRight()
2016-09-11 17:55:19 +02:00
}
// Backspace will remove a character in front of the CursorPositionText
2016-10-09 14:59:48 +02:00
func (i *Input) Backspace() {
if i.CursorPositionText > 0 {
2016-09-12 22:08:44 +02:00
i.MoveCursorLeft()
i.Text = append(i.Text[0:i.CursorPositionText], i.Text[i.CursorPositionText+1:]...)
i.Par.Text = string(i.Text[i.Offset:])
2016-09-11 17:55:19 +02:00
}
}
// Delete will remove a character at the CursorPositionText
2016-10-09 14:59:48 +02:00
func (i *Input) Delete() {
if i.CursorPositionText < len(i.Text) {
i.Text = append(i.Text[0:i.CursorPositionText], i.Text[i.CursorPositionText+1:]...)
i.Par.Text = string(i.Text[i.Offset:])
2016-10-09 14:59:48 +02:00
}
}
// MoveCursorRight will increase the current CursorPositionText with 1
2016-09-11 17:55:19 +02:00
func (i *Input) MoveCursorRight() {
if i.CursorPositionText < len(i.Text) {
i.CursorPositionText++
i.ScrollRight()
2016-09-11 17:55:19 +02:00
}
i.Par.Text = string(i.Text[i.Offset:])
2016-09-11 17:55:19 +02:00
}
// MoveCursorLeft will decrease the current CursorPositionText with 1
2016-09-11 17:55:19 +02:00
func (i *Input) MoveCursorLeft() {
if i.CursorPositionText > 0 {
i.CursorPositionText--
i.ScrollLeft()
}
i.Par.Text = string(i.Text[i.Offset:])
}
func (i *Input) ScrollLeft() {
// Is the cursor at the far left of the Input component?
if i.CursorPositionScreen == 0 {
// Decrease offset to show what is on the left side
if i.Offset > 0 {
i.Offset--
}
} else {
i.CursorPositionScreen -= i.GetRuneWidthRight()
}
}
func (i *Input) ScrollRight() {
// Is the cursor at the far right of the Input component, cursor
// isn't at the end of the text
if (i.CursorPositionScreen + i.GetRuneWidthLeft()) > i.Par.InnerBounds().Dx()-1 {
// Increase offset to show what is on the right side
if i.Offset < len(i.Text) {
i.Offset = i.CalculateOffset()
i.CursorPositionScreen = i.GetRuneWidthOffsetToCursor()
}
} else {
i.CursorPositionScreen += i.GetRuneWidthLeft()
2016-09-11 17:55:19 +02:00
}
}
// CalculateOffset will, based on the width of the runes on the
// left of the text cursor, calculate the offset that needs to
// be used by the Inpute Component
func (i *Input) CalculateOffset() int {
var offset int
var currentRuneWidth int
for j := (i.CursorPositionText - 1); currentRuneWidth < i.GetMaxWidth()-1; j-- {
currentRuneWidth += runewidth.RuneWidth(i.Text[j])
offset = j
}
return offset
}
// GetRunWidthOffsetToCursor will get the rune width of all
// the runes from the offset until the text cursor
func (i *Input) GetRuneWidthOffsetToCursor() int {
return runewidth.StringWidth(string(i.Text[i.Offset:i.CursorPositionText]))
}
// GetRuneWidthLeft will get the width of a rune on the left side
// of the CursorPositionText
func (i *Input) GetRuneWidthLeft() int {
return runewidth.RuneWidth(i.Text[i.CursorPositionText-1])
}
// GetRuneWidthLeft will get the width of a rune on the right side
// of the CursorPositionText
func (i *Input) GetRuneWidthRight() int {
return runewidth.RuneWidth(i.Text[i.CursorPositionText])
}
2016-09-13 21:13:51 +02:00
// IsEmpty will return true when the input is empty
2016-09-11 17:55:19 +02:00
func (i *Input) IsEmpty() bool {
2016-09-13 21:02:02 +02:00
if i.Par.Text == "" {
2016-09-11 17:55:19 +02:00
return true
}
return false
}
2016-09-13 21:13:51 +02:00
// Clear will empty the input and move the cursor to the start position
2016-09-11 17:55:19 +02:00
func (i *Input) Clear() {
2016-10-22 10:51:21 +02:00
i.Text = make([]rune, 0)
2016-09-13 21:02:02 +02:00
i.Par.Text = ""
i.CursorPositionScreen = 0
i.CursorPositionText = 0
i.Offset = 0
2016-09-11 17:55:19 +02:00
}
2016-10-21 21:23:25 +02:00
// GetText returns the text currently in the input
func (i *Input) GetText() string {
2016-09-13 21:02:02 +02:00
return i.Par.Text
2016-09-11 17:55:19 +02:00
}
// GetMaxWidth returns the maximum number of positions
// the Input component can display
func (i *Input) GetMaxWidth() int {
return i.Par.InnerBounds().Dx() - 1
}