bubbles/viewport/viewport.go
2020-06-15 21:21:08 -04:00

221 lines
4.3 KiB
Go

package viewport
import (
"os"
"strings"
tea "github.com/charmbracelet/bubbletea"
)
// MODEL
type Model struct {
Err error
Width 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
// UseInternalRenderer specifies whether or not to use the pager's internal,
// high performance renderer to paint the screen.
UseInternalRenderer bool
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.
func (m Model) ScrollPercent() float64 {
if m.Height >= len(m.lines) {
return 1.0
}
y := float64(m.YOffset)
h := float64(m.Height)
t := float64(len(m.lines))
return y / (t - h)
}
// SetContent set the pager's text content.
func (m *Model) SetContent(s string) {
s = strings.Replace(s, "\r\n", "\n", -1) // normalize line endings
m.lines = strings.Split(s, "\n")
lines := bounded(m.lines, m.YOffset, m.Height)
m.r.sync(strings.Join(lines, "\n"))
}
// ViewDown moves the view down by the number of lines in the viewport.
// Basically, "page down".
func (m *Model) ViewDown() {
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".
func (m *Model) ViewUp() {
m.YOffset = max(0, m.YOffset-m.Height)
}
// HalfViewUp moves the view up by half the height of the viewport.
func (m *Model) HalfViewUp() {
m.YOffset = max(0, m.YOffset-m.Height/2)
}
// HalfViewDown moves the view down by half the height of the viewport.
func (m *Model) HalfViewDown() {
m.YOffset = min(len(m.lines)-m.Height, m.YOffset+m.Height/2)
}
// LineDown moves the view up by the given number of lines.
func (m *Model) LineDown(n int) {
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])
}
}
// LineUp moves the view down by the given number of lines.
func (m *Model) LineUp(n int) {
if m.YOffset == 0 {
return
}
m.YOffset = max(0, m.YOffset-n)
if m.UseInternalRenderer {
m.r.insertTop(m.lines[m.YOffset])
}
}
// UPDATE
// Update runs the update loop with default keybindings. To define your own
// keybindings use the methods on Model and define your own update function.
func Update(msg tea.Msg, m Model) (Model, tea.Cmd) {
switch msg := msg.(type) {
case tea.KeyMsg:
switch msg.String() {
// Down one page
case "pgdown":
fallthrough
case " ": // spacebar
fallthrough
case "f":
m.ViewDown()
return m, nil
// Up one page
case "pgup":
fallthrough
case "b":
m.ViewUp()
return m, nil
// Down half page
case "d":
m.HalfViewDown()
return m, nil
// Up half page
case "u":
m.HalfViewUp()
return m, nil
// Down one line
case "down":
fallthrough
case "j":
m.LineDown(1)
return m, nil
// Up one line
case "up":
fallthrough
case "k":
m.LineUp(1)
return m, nil
}
}
return m, nil
}
// VIEW
// View renders the viewport into a 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 {
return m.Err.Error()
}
var lines []string
if len(m.lines) > 0 {
top := max(0, m.YOffset)
bottom := min(len(m.lines), m.YOffset+m.Height)
lines = m.lines[top:bottom]
}
// Fill empty space with newlines
extraLines := ""
if len(lines) < m.Height {
extraLines = strings.Repeat("\n", m.Height-len(lines))
}
return strings.Join(lines, "\n") + extraLines
}
// ETC
func min(a, b int) int {
if a < b {
return a
}
return b
}
func max(a, b int) int {
if a > b {
return a
}
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))]
}