2021-03-26 16:18:03 -04:00
|
|
|
// Editor methods involved in drawing to the console
|
|
|
|
package editor
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
"timshome.page/gilo/terminal"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// !Editor Methods
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
func (e *editor) RefreshScreen() {
|
2021-03-31 09:43:47 -04:00
|
|
|
e.scroll()
|
|
|
|
|
2021-03-26 16:18:03 -04:00
|
|
|
ab := newBuffer()
|
|
|
|
|
|
|
|
ab.append(terminal.HideCursor)
|
|
|
|
ab.append(terminal.ResetCursor)
|
|
|
|
|
|
|
|
e.drawRows(ab)
|
2021-03-31 14:56:46 -04:00
|
|
|
ab.append(terminal.MoveCursor(e.renderX - e.offset.x, e.cursor.y - e.offset.y))
|
2021-03-26 16:18:03 -04:00
|
|
|
|
|
|
|
ab.append(terminal.ShowCursor)
|
|
|
|
|
|
|
|
terminal.Write(ab.toString())
|
|
|
|
}
|
|
|
|
|
2021-03-31 09:43:47 -04:00
|
|
|
func (e *editor) scroll() {
|
2021-03-31 14:56:46 -04:00
|
|
|
e.renderX = 0
|
|
|
|
|
|
|
|
if e.cursor.y < e.rowCount() {
|
|
|
|
e.renderX = e.rows[e.cursor.y].cursorXToRenderX(e.cursor.x)
|
|
|
|
}
|
|
|
|
|
2021-03-31 09:43:47 -04:00
|
|
|
if e.cursor.y < e.offset.y {
|
|
|
|
e.offset.y = e.cursor.y
|
|
|
|
}
|
|
|
|
|
|
|
|
if e.cursor.y >= e.offset.y + e.screen.Rows {
|
|
|
|
e.offset.y = e.cursor.y - e.screen.Rows + 1
|
|
|
|
}
|
2021-03-31 11:12:09 -04:00
|
|
|
|
2021-03-31 14:56:46 -04:00
|
|
|
if e.renderX < e.offset.x {
|
|
|
|
e.offset.x = e.renderX
|
2021-03-31 11:12:09 -04:00
|
|
|
}
|
|
|
|
|
2021-03-31 14:56:46 -04:00
|
|
|
if e.renderX >= e.offset.x + e.screen.Cols {
|
|
|
|
e.offset.x = e.renderX - e.screen.Cols
|
2021-03-31 11:12:09 -04:00
|
|
|
}
|
2021-03-31 09:43:47 -04:00
|
|
|
}
|
|
|
|
|
2021-03-26 16:18:03 -04:00
|
|
|
func (e *editor) drawRows(ab *buffer) {
|
2021-03-30 18:43:46 -04:00
|
|
|
for y :=0; y < e.screen.Rows; y++ {
|
2021-03-31 09:43:47 -04:00
|
|
|
fileRow := y + e.offset.y
|
|
|
|
|
2021-03-31 14:56:46 -04:00
|
|
|
if fileRow >= e.rowCount() {
|
2021-03-31 09:28:39 -04:00
|
|
|
e.drawPlaceholderRow(y, ab)
|
2021-03-26 16:18:03 -04:00
|
|
|
} else {
|
2021-03-31 11:12:09 -04:00
|
|
|
rawRow := e.rows[fileRow]
|
|
|
|
|
|
|
|
// If the column offset is greater than the length of the row,
|
|
|
|
// just display an empty row
|
2021-03-31 14:32:43 -04:00
|
|
|
if e.offset.x > rawRow.size() {
|
2021-03-31 11:12:09 -04:00
|
|
|
ab.append("")
|
|
|
|
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:32:43 -04:00
|
|
|
rowLen := e.rows[fileRow].rSize() - e.offset.x
|
|
|
|
outputRow := truncateString(string(e.rows[fileRow].render[e.offset.x:]), rowLen)
|
|
|
|
ab.append(outputRow)
|
2021-03-26 16:18:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ab.append(terminal.ClearLine)
|
|
|
|
|
|
|
|
if y < (e.screen.Rows - 1) {
|
|
|
|
ab.append("\r\n")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-31 09:28:39 -04:00
|
|
|
func (e *editor) drawPlaceholderRow(y int, ab *buffer) {
|
2021-03-31 14:56:46 -04:00
|
|
|
if e.rowCount() == 0 && y == e.screen.Rows / 3 {
|
2021-03-31 09:28:39 -04:00
|
|
|
welcome := fmt.Sprintf("Gilo editor -- version %s", KiloVersion)
|
|
|
|
if len(welcome) > e.screen.Cols {
|
|
|
|
welcome = truncateString(welcome, e.screen.Cols)
|
|
|
|
}
|
|
|
|
|
|
|
|
padding := (e.screen.Cols - len(welcome)) / 2
|
|
|
|
if padding > 0 {
|
|
|
|
ab.appendRune('~')
|
|
|
|
padding--
|
|
|
|
}
|
|
|
|
|
|
|
|
for padding > 0 {
|
|
|
|
padding--
|
|
|
|
ab.appendRune(' ')
|
|
|
|
}
|
|
|
|
|
|
|
|
ab.append(welcome)
|
|
|
|
} else {
|
|
|
|
ab.appendRune('~')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-26 16:18:03 -04:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// !Output Buffer
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
type buffer struct {
|
|
|
|
buf *strings.Builder
|
|
|
|
}
|
|
|
|
|
|
|
|
func newBuffer() *buffer {
|
|
|
|
var buf strings.Builder
|
|
|
|
|
|
|
|
b := new(buffer)
|
|
|
|
b.buf = &buf
|
|
|
|
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *buffer) appendRune(r rune) int {
|
|
|
|
size, _ := b.buf.WriteRune(r)
|
|
|
|
|
|
|
|
return size
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *buffer) append(s string) int {
|
|
|
|
size, _ := b.buf.WriteString(s)
|
|
|
|
|
|
|
|
return size
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *buffer) appendLn(s string) int {
|
|
|
|
str := fmt.Sprintf("%s\r\n", s)
|
|
|
|
size, _ := b.buf.WriteString(str)
|
|
|
|
|
|
|
|
return size
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *buffer) toString() string {
|
|
|
|
return b.buf.String()
|
|
|
|
}
|