Broad (working) pass at viewport renderer

This commit is contained in:
Christian Rocha 2020-06-15 17:50:13 -04:00
parent 786ec557d4
commit 0b19d41e0a
No known key found for this signature in database
GPG Key ID: D6CC7A16E5878018
2 changed files with 168 additions and 19 deletions

98
viewport/renderer.go Normal file
View File

@ -0,0 +1,98 @@
package viewport
import (
"bytes"
"fmt"
"io"
)
type renderer struct {
Out io.Writer
Y int
Height int
TerminalWidth int
TerminalHeight int
}
// sync paints the whole area.
func (r *renderer) sync(content string) {
r.clear()
moveTo(r.Out, r.Y, 0)
r.write(content)
}
// clear clears the viewport region.
func (r *renderer) clear() {
b := new(bytes.Buffer)
moveTo(b, r.Y, 0)
for i := 0; i < r.Height; i++ {
clearLine(b)
cursorDown(b, 1)
}
r.Out.Write(b.Bytes())
}
// write writes to the set writer.
func (r *renderer) write(s string) {
if len(s) == 0 {
return
}
buf := new(bytes.Buffer)
for _, r := range []rune(s) {
if r == '\n' {
buf.WriteString("\r\n")
continue
}
buf.WriteRune(r)
}
r.Out.Write(buf.Bytes())
}
// Effectively scroll up. That is, insert a line at the top, scrolling
// everything else down. This is roughly how ncurses does it.
func (r *renderer) insertTop(line string) {
changeScrollingRegion(r.Out, r.Y, r.Y+r.Height)
moveTo(r.Out, r.Y, 0)
insertLine(r.Out, 1)
io.WriteString(r.Out, line)
changeScrollingRegion(r.Out, r.TerminalWidth, r.TerminalHeight)
}
// Effectively scroll down. That is, insert a line at the bottom, scrolling
// everything else up. This is roughly how ncurses does it.
func (r *renderer) insertBottom(line string) {
changeScrollingRegion(r.Out, r.Y, r.Y+r.Height)
moveTo(r.Out, r.Y+r.Height, 0)
io.WriteString(r.Out, "\n"+line)
changeScrollingRegion(r.Out, r.TerminalWidth, r.TerminalHeight)
}
// Screen command
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)
}

View File

@ -1,6 +1,7 @@
package viewport package viewport
import ( import (
"os"
"strings" "strings"
tea "github.com/charmbracelet/bubbletea" tea "github.com/charmbracelet/bubbletea"
@ -12,9 +13,36 @@ type Model struct {
Err error Err error
Width int Width int
Height int Height int
// YOffset is the vertical scroll position.
YOffset int
// Y is the position of the viewport in relation to the terminal window.
// It's used in high performance rendering.
Y int Y int
// UseInternalRenderer specifies whether or not to use the pager's internal,
// high performance renderer to paint the screen.
UseInternalRenderer bool
lines []string lines []string
r renderer
}
func NewModel(yPos, width, height, terminalWidth, terminalHeight int) Model {
return Model{
Width: width,
Height: height,
UseInternalRenderer: true,
r: renderer{
Y: yPos,
Height: height,
TerminalWidth: terminalWidth,
TerminalHeight: terminalHeight,
Out: os.Stdout,
},
}
} }
// Scrollpercent returns the amount scrolled as a float between 0 and 1. // Scrollpercent returns the amount scrolled as a float between 0 and 1.
@ -22,7 +50,7 @@ func (m Model) ScrollPercent() float64 {
if m.Height >= len(m.lines) { if m.Height >= len(m.lines) {
return 1.0 return 1.0
} }
y := float64(m.Y) y := float64(m.YOffset)
h := float64(m.Height) h := float64(m.Height)
t := float64(len(m.lines)) t := float64(len(m.lines))
return y / (t - h) return y / (t - h)
@ -32,51 +60,60 @@ func (m Model) ScrollPercent() float64 {
func (m *Model) SetContent(s string) { func (m *Model) SetContent(s string) {
s = strings.Replace(s, "\r\n", "\n", -1) // normalize line endings s = strings.Replace(s, "\r\n", "\n", -1) // normalize line endings
m.lines = strings.Split(s, "\n") m.lines = strings.Split(s, "\n")
}
// NewModel creates a new pager model. Pass the dimensions of the pager. lines := bounded(m.lines, m.YOffset, m.Height)
func NewModel(width, height int) Model { m.r.sync(strings.Join(lines, "\n"))
return Model{
Width: width,
Height: height,
}
} }
// ViewDown moves the view down by the number of lines in the viewport. // ViewDown moves the view down by the number of lines in the viewport.
// Basically, "page down". // Basically, "page down".
func (m *Model) ViewDown() { func (m *Model) ViewDown() {
m.Y = min(len(m.lines)-m.Height, m.Y+m.Height) m.YOffset = min(len(m.lines)-m.Height, m.YOffset+m.Height)
} }
// ViewUp moves the view up by one height of the viewport. Basically, "page up". // ViewUp moves the view up by one height of the viewport. Basically, "page up".
func (m *Model) ViewUp() { func (m *Model) ViewUp() {
m.Y = max(0, m.Y-m.Height) m.YOffset = max(0, m.YOffset-m.Height)
} }
// HalfViewUp moves the view up by half the height of the viewport. // HalfViewUp moves the view up by half the height of the viewport.
func (m *Model) HalfViewUp() { func (m *Model) HalfViewUp() {
m.Y = max(0, m.Y-m.Height/2) m.YOffset = max(0, m.YOffset-m.Height/2)
} }
// HalfViewDown moves the view down by half the height of the viewport. // HalfViewDown moves the view down by half the height of the viewport.
func (m *Model) HalfViewDown() { func (m *Model) HalfViewDown() {
m.Y = min(len(m.lines)-m.Height, m.Y+m.Height/2) m.YOffset = min(len(m.lines)-m.Height, m.YOffset+m.Height/2)
} }
// LineDown moves the view up by the given number of lines. // LineDown moves the view up by the given number of lines.
func (m *Model) LineDown(n int) { func (m *Model) LineDown(n int) {
m.Y = min(len(m.lines)-m.Height, m.Y+n) if m.YOffset >= len(m.lines)-m.Height-1 {
return
}
m.YOffset = min(len(m.lines)-m.Height, m.YOffset+n)
if m.UseInternalRenderer {
m.r.insertBottom(m.lines[m.YOffset+m.Height-1])
}
} }
// LineDown moves the view down by the given number of lines. // LineUp moves the view down by the given number of lines.
func (m *Model) LineUp(n int) { func (m *Model) LineUp(n int) {
m.Y = max(0, m.Y-n) if m.YOffset == 0 {
return
}
m.YOffset = max(0, m.YOffset-n)
if m.UseInternalRenderer {
m.r.insertTop(m.lines[m.YOffset])
}
} }
// UPDATE // UPDATE
// Update runs the update loop with default keybindings. To define your own // Update runs the update loop with default keybindings. To define your own
// keybindings use the methods on Model. // keybindings use the methods on Model and define your own update function.
func Update(msg tea.Msg, m Model) (Model, tea.Cmd) { func Update(msg tea.Msg, m Model) (Model, tea.Cmd) {
switch msg := msg.(type) { switch msg := msg.(type) {
@ -131,6 +168,12 @@ func Update(msg tea.Msg, m Model) (Model, tea.Cmd) {
// View renders the viewport into a string. // View renders the viewport into a string.
func View(m Model) string { func View(m Model) string {
if m.UseInternalRenderer {
// Skip over the area that would normally be rendered
return cursorDownString(m.Height)
}
if m.Err != nil { if m.Err != nil {
return m.Err.Error() return m.Err.Error()
} }
@ -138,8 +181,8 @@ func View(m Model) string {
var lines []string var lines []string
if len(m.lines) > 0 { if len(m.lines) > 0 {
top := max(0, m.Y) top := max(0, m.YOffset)
bottom := min(len(m.lines), m.Y+m.Height) bottom := min(len(m.lines), m.YOffset+m.Height)
lines = m.lines[top:bottom] lines = m.lines[top:bottom]
} }
@ -167,3 +210,11 @@ func max(a, b int) int {
} }
return b return b
} }
func clamp(val, low, high int) int {
return max(low, min(high, val))
}
func bounded(s []string, start, end int) []string {
return s[clamp(start, 0, len(s)):clamp(end, 0, len(s))]
}