2021-03-19 16:36:02 -04:00
|
|
|
package editor
|
2021-03-19 17:03:56 -04:00
|
|
|
|
|
|
|
import (
|
2021-03-24 16:23:17 -04:00
|
|
|
"timshome.page/gilo/terminal"
|
2021-03-19 17:03:56 -04:00
|
|
|
)
|
|
|
|
|
2021-03-24 15:09:28 -04:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// !Editor
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2021-03-25 12:27:48 -04:00
|
|
|
type cursor struct {
|
|
|
|
x int
|
|
|
|
y int
|
|
|
|
}
|
|
|
|
|
2021-03-24 14:17:29 -04:00
|
|
|
type editor struct {
|
2021-03-25 12:27:48 -04:00
|
|
|
screen *terminal.Screen
|
|
|
|
cursor *cursor
|
2021-03-30 16:05:33 -04:00
|
|
|
rows []*row
|
2021-03-23 15:51:59 -04:00
|
|
|
}
|
|
|
|
|
2021-03-24 14:17:29 -04:00
|
|
|
func New() *editor {
|
2021-03-25 12:27:48 -04:00
|
|
|
screen := terminal.Size()
|
|
|
|
cursor := &cursor { 0, 0 }
|
2021-03-30 16:05:33 -04:00
|
|
|
var rows []*row
|
2021-03-24 14:17:29 -04:00
|
|
|
|
2021-03-30 16:05:33 -04:00
|
|
|
return &editor{screen, cursor, rows }
|
2021-03-22 09:12:39 -04:00
|
|
|
}
|
|
|
|
|
2021-03-26 16:18:03 -04:00
|
|
|
func (e *editor) ProcessKeypress() bool {
|
2021-03-30 15:45:13 -04:00
|
|
|
var str string
|
2021-03-24 15:09:28 -04:00
|
|
|
|
2021-03-30 15:45:13 -04:00
|
|
|
ch, _ := terminal.ReadKey()
|
2021-03-30 16:22:35 -04:00
|
|
|
if ch == Ctrl('q') {
|
2021-03-30 15:45:13 -04:00
|
|
|
// Clean up on exit
|
|
|
|
terminal.Write(terminal.ClearScreen + terminal.ResetCursor)
|
2021-03-24 15:52:35 -04:00
|
|
|
|
2021-03-30 15:45:13 -04:00
|
|
|
return false
|
|
|
|
}
|
2021-03-30 14:42:59 -04:00
|
|
|
|
2021-03-30 15:45:13 -04:00
|
|
|
if ch == '\x1b' {
|
2021-03-30 15:47:42 -04:00
|
|
|
str = parseEscapeSequence()
|
2021-03-30 15:45:13 -04:00
|
|
|
} else {
|
|
|
|
str = string(ch)
|
2021-03-26 12:01:17 -04:00
|
|
|
}
|
|
|
|
|
2021-03-30 15:45:13 -04:00
|
|
|
switch str {
|
|
|
|
case keyUp, keyDown, keyLeft, keyRight, keyPageUp, keyPageDown, keyHome, keyEnd:
|
|
|
|
e.moveCursor(str)
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
// Do something later
|
|
|
|
terminal.Write("Code: %v", str)
|
2021-03-30 11:52:44 -04:00
|
|
|
}
|
2021-03-26 16:18:03 -04:00
|
|
|
|
2021-03-30 11:52:44 -04:00
|
|
|
return true
|
2021-03-26 12:01:17 -04:00
|
|
|
}
|
|
|
|
|
2021-03-30 14:42:59 -04:00
|
|
|
func (e *editor) moveCursor (key string) {
|
2021-03-25 12:46:53 -04:00
|
|
|
switch key {
|
2021-03-30 15:45:13 -04:00
|
|
|
case keyLeft:
|
2021-03-30 14:42:59 -04:00
|
|
|
if e.cursor.x != 0 {
|
|
|
|
e.cursor.x -= 1
|
|
|
|
}
|
2021-03-30 15:45:13 -04:00
|
|
|
case keyRight:
|
2021-03-30 14:42:59 -04:00
|
|
|
if e.cursor.x != e.screen.Cols-1 {
|
|
|
|
e.cursor.x += 1
|
|
|
|
}
|
2021-03-30 15:45:13 -04:00
|
|
|
case keyUp:
|
2021-03-30 14:42:59 -04:00
|
|
|
if e.cursor.y != 0 {
|
|
|
|
e.cursor.y -= 1
|
|
|
|
}
|
2021-03-30 15:45:13 -04:00
|
|
|
case keyDown:
|
2021-03-30 14:42:59 -04:00
|
|
|
if e.cursor.y != e.screen.Rows-1 {
|
|
|
|
e.cursor.y += 1
|
|
|
|
}
|
2021-03-30 15:45:13 -04:00
|
|
|
case keyPageUp:
|
2021-03-30 14:42:59 -04:00
|
|
|
e.cursor.y = 0
|
2021-03-30 15:45:13 -04:00
|
|
|
case keyPageDown:
|
2021-03-30 14:42:59 -04:00
|
|
|
e.cursor.y = e.screen.Rows
|
2021-03-30 15:45:13 -04:00
|
|
|
case keyHome:
|
|
|
|
e.cursor.x = 0
|
|
|
|
case keyEnd:
|
|
|
|
e.cursor.x = e.screen.Cols
|
2021-03-25 12:46:53 -04:00
|
|
|
}
|
2021-03-30 15:45:13 -04:00
|
|
|
}
|
|
|
|
|
2021-03-30 16:05:33 -04:00
|
|
|
// Convert the raw ANSI escape sequences to the type of key input
|
2021-03-30 15:45:13 -04:00
|
|
|
func parseEscapeSequence () string {
|
|
|
|
var runes []rune
|
|
|
|
|
|
|
|
for i := 0; i < 3; i++ {
|
|
|
|
ch, size := terminal.ReadKey()
|
|
|
|
if size == 0 {
|
|
|
|
return "\x1b"
|
|
|
|
}
|
|
|
|
runes = append(runes, ch)
|
|
|
|
|
|
|
|
if i == 1 && runes[1] >= 'A' {
|
2021-03-30 16:22:35 -04:00
|
|
|
// \eOH \eOF
|
2021-03-30 15:45:13 -04:00
|
|
|
if runes[0] == 'O' {
|
|
|
|
switch runes[1] {
|
|
|
|
case 'H':
|
|
|
|
return keyHome
|
|
|
|
case 'F':
|
|
|
|
return keyEnd
|
|
|
|
}
|
|
|
|
}
|
2021-03-30 16:22:35 -04:00
|
|
|
// \e[A
|
2021-03-30 15:45:13 -04:00
|
|
|
if runes[0] == '[' {
|
|
|
|
switch runes[1] {
|
|
|
|
case 'A':
|
|
|
|
return keyUp
|
|
|
|
case 'B':
|
|
|
|
return keyDown
|
|
|
|
case 'C':
|
|
|
|
return keyRight
|
|
|
|
case 'D':
|
|
|
|
return keyLeft
|
|
|
|
case 'H':
|
|
|
|
return keyHome
|
|
|
|
case 'F':
|
|
|
|
return keyEnd
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-30 16:22:35 -04:00
|
|
|
// \e[1~
|
2021-03-30 15:45:13 -04:00
|
|
|
if i == 2 && runes[0] == '[' && runes[2] == '~' {
|
|
|
|
switch runes[1] {
|
|
|
|
case '1':
|
|
|
|
return keyHome
|
|
|
|
case '3':
|
|
|
|
return keyDelete
|
|
|
|
case '4':
|
|
|
|
return keyEnd
|
|
|
|
case '5':
|
|
|
|
return keyPageUp
|
|
|
|
case '6':
|
|
|
|
return keyPageDown
|
|
|
|
case '7':
|
|
|
|
return keyHome
|
|
|
|
case '8':
|
|
|
|
return keyEnd
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return string('\x1b')
|
2021-03-24 15:52:35 -04:00
|
|
|
}
|