mirror of
https://github.com/Maks1mS/bubbles.git
synced 2025-01-26 13:51:04 +03:00
Broad (working) pass at viewport renderer
This commit is contained in:
parent
786ec557d4
commit
0b19d41e0a
98
viewport/renderer.go
Normal file
98
viewport/renderer.go
Normal 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)
|
||||||
|
}
|
@ -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))]
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user