2020-06-16 00:50:13 +03:00
|
|
|
package viewport
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2020-06-16 01:32:18 +03:00
|
|
|
"strings"
|
2020-06-16 00:50:13 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type renderer struct {
|
|
|
|
Out io.Writer
|
|
|
|
Y int
|
|
|
|
Height int
|
|
|
|
TerminalHeight int
|
|
|
|
}
|
|
|
|
|
|
|
|
// clear clears the viewport region.
|
|
|
|
func (r *renderer) clear() {
|
2020-06-16 01:32:18 +03:00
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
moveTo(buf, r.Y, 0)
|
2020-06-16 00:50:13 +03:00
|
|
|
for i := 0; i < r.Height; i++ {
|
2020-06-16 01:32:18 +03:00
|
|
|
clearLine(buf)
|
|
|
|
cursorDown(buf, 1)
|
2020-06-16 00:50:13 +03:00
|
|
|
}
|
2020-06-16 01:32:18 +03:00
|
|
|
r.Out.Write(buf.Bytes())
|
|
|
|
}
|
|
|
|
|
|
|
|
// sync paints the whole area.
|
|
|
|
func (r *renderer) sync(lines []string) {
|
|
|
|
r.clear()
|
|
|
|
moveTo(r.Out, r.Y, 0)
|
2020-06-16 04:19:15 +03:00
|
|
|
r.writeLines(lines)
|
2020-06-16 00:50:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// write writes to the set writer.
|
2020-06-16 04:19:15 +03:00
|
|
|
func (r *renderer) writeLines(lines []string) {
|
2020-06-16 01:32:18 +03:00
|
|
|
if len(lines) == 0 {
|
2020-06-16 00:50:13 +03:00
|
|
|
return
|
|
|
|
}
|
2020-06-16 01:32:18 +03:00
|
|
|
io.WriteString(r.Out, strings.Join(lines, "\r\n"))
|
2020-06-16 00:50:13 +03:00
|
|
|
}
|
|
|
|
|
2020-06-16 04:19:15 +03:00
|
|
|
// Effectively scroll up. That is, insert a line at the top, pushing
|
2020-06-16 00:50:13 +03:00
|
|
|
// everything else down. This is roughly how ncurses does it.
|
2020-06-16 04:19:15 +03:00
|
|
|
func (r *renderer) insertTop(lines []string) {
|
2020-06-16 00:50:13 +03:00
|
|
|
changeScrollingRegion(r.Out, r.Y, r.Y+r.Height)
|
|
|
|
moveTo(r.Out, r.Y, 0)
|
2020-06-16 04:19:15 +03:00
|
|
|
insertLine(r.Out, len(lines))
|
|
|
|
r.writeLines(lines)
|
2020-06-16 21:28:01 +03:00
|
|
|
changeScrollingRegion(r.Out, 0, r.TerminalHeight)
|
2020-06-16 00:50:13 +03:00
|
|
|
}
|
|
|
|
|
2020-06-16 04:19:15 +03:00
|
|
|
// Effectively scroll down. That is, insert a line at the bottom, pushing
|
2020-06-16 00:50:13 +03:00
|
|
|
// everything else up. This is roughly how ncurses does it.
|
2020-06-16 04:19:15 +03:00
|
|
|
func (r *renderer) insertBottom(lines []string) {
|
2020-06-16 00:50:13 +03:00
|
|
|
changeScrollingRegion(r.Out, r.Y, r.Y+r.Height)
|
|
|
|
moveTo(r.Out, r.Y+r.Height, 0)
|
2020-06-16 04:19:15 +03:00
|
|
|
io.WriteString(r.Out, "\r\n"+strings.Join(lines, "\r\n"))
|
2020-06-16 21:28:01 +03:00
|
|
|
changeScrollingRegion(r.Out, 0, r.TerminalHeight)
|
2020-06-16 00:50:13 +03:00
|
|
|
}
|
|
|
|
|
2020-06-16 04:19:15 +03:00
|
|
|
// Terminal Control
|
2020-06-16 00:50:13 +03:00
|
|
|
|
|
|
|
const csi = "\x1b["
|
|
|
|
|
|
|
|
func changeScrollingRegion(w io.Writer, top, bottom int) {
|
|
|
|
fmt.Fprintf(w, csi+"%d;%dr", top, bottom)
|
|
|
|
}
|
|
|
|
|
|
|
|
func moveTo(w io.Writer, row, col int) {
|
|
|
|
fmt.Fprintf(w, csi+"%d;%dH", row, col)
|
|
|
|
}
|
|
|
|
|
|
|
|
func cursorDown(w io.Writer, numLines int) {
|
|
|
|
fmt.Fprintf(w, csi+"%dB", numLines)
|
|
|
|
}
|
|
|
|
|
|
|
|
func cursorDownString(numLines int) string {
|
|
|
|
return fmt.Sprintf(csi+"%dB", numLines)
|
|
|
|
}
|
|
|
|
|
|
|
|
func clearLine(w io.Writer) {
|
|
|
|
fmt.Fprint(w, csi+"2K")
|
|
|
|
}
|
|
|
|
|
|
|
|
func insertLine(w io.Writer, numLines int) {
|
|
|
|
fmt.Fprintf(w, csi+"%dL", numLines)
|
|
|
|
}
|