bubbletea/tea.go

229 lines
4.7 KiB
Go
Raw Normal View History

package tea
2020-01-10 16:02:04 -05:00
import (
"fmt"
"os"
"sync"
2020-01-31 07:47:36 -05:00
te "github.com/muesli/termenv"
"golang.org/x/crypto/ssh/terminal"
2020-01-10 16:02:04 -05:00
)
// Msg represents an action and is usually the result of an IO operation. It's
2020-06-19 14:14:15 -04:00
// triggers the Update function, and henceforth, the UI.
2020-01-10 16:02:04 -05:00
type Msg interface{}
2020-04-17 19:02:25 -04:00
// Model contains the program's state.
2020-01-10 16:02:04 -05:00
type Model interface{}
2020-06-19 14:14:15 -04:00
// Cmd is an IO operation. If it's nil it's considered a no-op. Keep in mind
// that there's almost never a need to use a command to send a message to
// another part of your program.
type Cmd func() Msg
2020-03-31 16:08:03 -04:00
// Batch peforms a bunch of commands concurrently with no ordering guarantees
// about the results.
func Batch(cmds ...Cmd) Cmd {
if len(cmds) == 0 {
return nil
}
return func() Msg {
2020-03-31 16:08:03 -04:00
return batchMsg(cmds)
}
}
// Init is the first function that will be called. It returns your initial
// model and runs an optional command.
type Init func() (Model, Cmd)
2020-01-10 16:02:04 -05:00
// Update is called when a message is received. It may update the model and/or
// send a command.
type Update func(Msg, Model) (Model, Cmd)
// View produces a string which will be rendered to the terminal.
2020-01-10 16:02:04 -05:00
type View func(Model) string
// Program is a terminal user interface.
2020-01-10 16:02:04 -05:00
type Program struct {
init Init
update Update
view View
mtx sync.Mutex
output *os.File // where to send output. this will usually be os.Stdout.
2020-01-10 16:02:04 -05:00
}
// Quit is a special command that tells the program to exit.
func Quit() Msg {
2020-01-10 16:02:04 -05:00
return quitMsg{}
}
// quitMsg in an internal message signals that the program should quit. You can
// send a quitMsg with Quit.
2020-01-10 16:02:04 -05:00
type quitMsg struct{}
// batchMsg is the internal message used to perform a bunch of commands. You
// can send a batchMsg with Batch.
2020-03-31 16:08:03 -04:00
type batchMsg []Cmd
// WindowSizeMsg is used to report on the terminal size. It's sent once
2020-06-17 19:43:06 -04:00
// initially and then on every terminal resize.
type WindowSizeMsg struct {
Width int
Height int
}
// NewProgram creates a new Program.
func NewProgram(init Init, update Update, view View) *Program {
2020-01-10 16:02:04 -05:00
return &Program{
init: init,
update: update,
view: view,
output: os.Stdout,
2020-01-10 16:02:04 -05:00
}
}
// Start initializes the program.
2020-01-10 16:02:04 -05:00
func (p *Program) Start() error {
var (
model Model
cmd Cmd
cmds = make(chan Cmd)
msgs = make(chan Msg)
errs = make(chan error)
done = make(chan struct{})
mrRenderer = newRenderer(p.output, &p.mtx)
2020-01-10 16:02:04 -05:00
)
err := initTerminal()
2020-01-10 16:02:04 -05:00
if err != nil {
return err
}
defer restoreTerminal()
2020-01-10 16:02:04 -05:00
// Initialize program
model, cmd = p.init()
if cmd != nil {
go func() {
cmds <- cmd
}()
}
2020-06-05 12:40:44 -04:00
// Start renderer
mrRenderer.start()
2020-01-10 16:02:04 -05:00
// Render initial view
2020-06-05 12:40:44 -04:00
mrRenderer.write(p.view(model))
2020-01-10 16:02:04 -05:00
// Subscribe to user input
2020-01-10 16:02:04 -05:00
go func() {
for {
2020-06-22 20:30:16 -04:00
msg, err := ReadInput(os.Stdin)
if err != nil {
errs <- err
}
2020-06-22 20:30:16 -04:00
msgs <- msg
2020-01-10 16:02:04 -05:00
}
}()
// Get initial terminal size
go func() {
w, h, err := terminal.GetSize(int(p.output.Fd()))
if err != nil {
errs <- err
}
msgs <- WindowSizeMsg{w, h}
}()
// Listen for window resizes
go listenForResize(p.output, msgs, errs)
2020-01-10 16:02:04 -05:00
// Process commands
go func() {
for {
select {
case <-done:
return
case cmd := <-cmds:
if cmd != nil {
go func() {
msgs <- cmd()
2020-01-10 16:02:04 -05:00
}()
}
}
}
}()
// Handle updates and draw
for {
select {
case err := <-errs:
close(done)
return err
2020-01-10 16:02:04 -05:00
case msg := <-msgs:
2020-03-31 16:08:03 -04:00
// Handle quit message
2020-01-10 16:02:04 -05:00
if _, ok := msg.(quitMsg); ok {
mrRenderer.stop()
2020-01-10 16:02:04 -05:00
close(done)
return nil
}
// Process batch commands
if batchedCmds, ok := msg.(batchMsg); ok {
for _, cmd := range batchedCmds {
cmds <- cmd
}
continue
}
2020-06-17 19:43:06 -04:00
// Process internal messages for the renderer
mrRenderer.handleMessages(msg)
model, cmd = p.update(msg, model) // run update
cmds <- cmd // process command (if any)
mrRenderer.write(p.view(model)) // send view to renderer
2020-01-10 16:02:04 -05:00
}
}
}
// EnterAltScreen enters the alternate screen buffer.
func (p *Program) EnterAltScreen() {
p.mtx.Lock()
defer p.mtx.Unlock()
fmt.Fprintf(p.output, te.CSI+te.AltScreenSeq)
moveCursor(p.output, 0, 0)
}
// ExitAltScreen exits the alternate screen buffer.
func (p *Program) ExitAltScreen() {
p.mtx.Lock()
defer p.mtx.Unlock()
fmt.Fprintf(p.output, te.CSI+te.ExitAltScreenSeq)
}
2020-06-22 20:30:16 -04:00
func (p *Program) EnableMouseCellMotion() {
p.mtx.Lock()
defer p.mtx.Unlock()
fmt.Fprintf(p.output, te.CSI+"?1002h")
}
func (p *Program) DisableMouseCellMotion() {
p.mtx.Lock()
defer p.mtx.Unlock()
fmt.Fprintf(p.output, te.CSI+"?1002l")
}
func (p *Program) EnableMouseAllMotion() {
p.mtx.Lock()
defer p.mtx.Unlock()
fmt.Fprintf(p.output, te.CSI+"?1003h")
}
func (p *Program) DisableMouseAllMotion() {
p.mtx.Lock()
defer p.mtx.Unlock()
fmt.Fprintf(p.output, te.CSI+"?1003l")
}