2020-07-30 11:29:20 -04:00
|
|
|
// Package tea provides a framework for building rich terminal user interfaces
|
|
|
|
// based on the paradigms of The Elm Architecture. It's well-suited for simple
|
|
|
|
// and complex terminal applications, either inline, full-window, or a mix of
|
|
|
|
// both. It's been battle-tested in several large projects and is
|
|
|
|
// production-ready.
|
2020-07-29 20:49:20 -04:00
|
|
|
//
|
|
|
|
// A tutorial is available at https://github.com/charmbracelet/bubbletea/tree/master/tutorials
|
|
|
|
//
|
|
|
|
// Example programs can be found at https://github.com/charmbracelet/bubbletea/tree/master/examples
|
2020-05-25 19:26:40 -04:00
|
|
|
package tea
|
2020-01-10 16:02:04 -05:00
|
|
|
|
|
|
|
import (
|
2020-06-22 15:55:52 -04:00
|
|
|
"fmt"
|
2020-01-25 01:15:56 -05:00
|
|
|
"os"
|
2020-09-21 13:09:32 -04:00
|
|
|
"runtime/debug"
|
2020-06-22 15:55:52 -04:00
|
|
|
"sync"
|
2020-01-31 07:47:36 -05:00
|
|
|
|
2020-06-22 15:55:52 -04:00
|
|
|
te "github.com/muesli/termenv"
|
2020-06-16 16:41:35 -04:00
|
|
|
"golang.org/x/crypto/ssh/terminal"
|
2020-01-10 16:02:04 -05:00
|
|
|
)
|
|
|
|
|
2020-04-10 15:43:18 -04: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-24 12:14:35 -04:00
|
|
|
// Cmd is an IO operation. If it's nil it's considered a no-op. Use it for
|
|
|
|
// things like HTTP requests, timers, saving and loading from disk, and so on.
|
|
|
|
//
|
|
|
|
// There's almost never a need to use a command to send a message to another
|
|
|
|
// part of your program. Instead, it can almost always be done in the update
|
|
|
|
// function.
|
2020-04-22 10:15:04 -04:00
|
|
|
type Cmd func() Msg
|
2020-03-31 16:27:36 -04:00
|
|
|
|
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 {
|
2020-04-01 12:05:05 -04:00
|
|
|
if len(cmds) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
2020-04-22 10:15:04 -04:00
|
|
|
return func() Msg {
|
2020-03-31 16:08:03 -04:00
|
|
|
return batchMsg(cmds)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-18 22:18:19 -05:00
|
|
|
// Init is the first function that will be called. It returns your initial
|
2020-05-25 08:02:46 -04:00
|
|
|
// model and runs an optional command.
|
2020-01-18 22:18:19 -05:00
|
|
|
type Init func() (Model, Cmd)
|
|
|
|
|
2020-06-24 12:14:35 -04:00
|
|
|
// Update is called when a message is received. Use it to inspect messages and,
|
2020-08-19 17:54:42 -04:00
|
|
|
// in response, update the model and/or send a command.
|
2020-01-10 16:02:04 -05:00
|
|
|
type Update func(Msg, Model) (Model, Cmd)
|
|
|
|
|
2020-07-29 20:54:15 -04:00
|
|
|
// View renders the program's UI, which is just a string. The view is rendered
|
|
|
|
// after every Update.
|
2020-01-10 16:02:04 -05:00
|
|
|
type View func(Model) string
|
|
|
|
|
2020-05-25 08:02:46 -04:00
|
|
|
// Program is a terminal user interface.
|
2020-01-10 16:02:04 -05:00
|
|
|
type Program struct {
|
2020-05-12 17:56:30 -04:00
|
|
|
init Init
|
|
|
|
update Update
|
|
|
|
view View
|
2020-06-22 15:55:52 -04:00
|
|
|
|
2020-07-13 11:39:04 -04:00
|
|
|
mtx sync.Mutex
|
|
|
|
output *os.File // where to send output. this will usually be os.Stdout.
|
|
|
|
renderer *renderer
|
|
|
|
altScreenActive bool
|
2020-09-21 13:09:32 -04:00
|
|
|
|
|
|
|
// CatchPanics is incredibly useful for restoring the terminal to a useable
|
|
|
|
// state after a panic occurs. When this is set, Bubble Tea will recover
|
|
|
|
// from panics, print the stack trace, and disable raw mode. This feature
|
|
|
|
// is on by default.
|
|
|
|
CatchPanics bool
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
|
2020-07-29 20:49:20 -04:00
|
|
|
// Quit is a special command that tells the Bubble Tea program to exit.
|
2020-04-22 10:15:04 -04:00
|
|
|
func Quit() Msg {
|
2020-01-10 16:02:04 -05:00
|
|
|
return quitMsg{}
|
|
|
|
}
|
|
|
|
|
2020-06-05 13:45:28 -04:00
|
|
|
// 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{}
|
|
|
|
|
2020-06-05 13:45:28 -04:00
|
|
|
// 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
|
|
|
|
|
2020-07-29 20:54:15 -04:00
|
|
|
// WindowSizeMsg is used to report on the terminal size. Its sent to Update
|
|
|
|
// once initially and then on every terminal resize.
|
2020-06-16 16:41:35 -04:00
|
|
|
type WindowSizeMsg struct {
|
2020-06-17 18:19:27 -04:00
|
|
|
Width int
|
|
|
|
Height int
|
2020-06-16 16:41:35 -04:00
|
|
|
}
|
|
|
|
|
2020-05-25 08:02:46 -04:00
|
|
|
// NewProgram creates a new Program.
|
2020-05-12 17:56:30 -04:00
|
|
|
func NewProgram(init Init, update Update, view View) *Program {
|
2020-01-10 16:02:04 -05:00
|
|
|
return &Program{
|
2020-05-12 17:56:30 -04:00
|
|
|
init: init,
|
|
|
|
update: update,
|
|
|
|
view: view,
|
2020-06-22 15:55:52 -04:00
|
|
|
|
2020-09-21 13:09:32 -04:00
|
|
|
output: os.Stdout,
|
|
|
|
CatchPanics: true,
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-25 08:02:46 -04:00
|
|
|
// Start initializes the program.
|
2020-01-10 16:02:04 -05:00
|
|
|
func (p *Program) Start() error {
|
|
|
|
var (
|
2020-07-13 11:39:04 -04:00
|
|
|
cmds = make(chan Cmd)
|
|
|
|
msgs = make(chan Msg)
|
|
|
|
errs = make(chan error)
|
|
|
|
done = make(chan struct{})
|
2020-01-10 16:02:04 -05:00
|
|
|
)
|
|
|
|
|
2020-09-21 13:09:32 -04:00
|
|
|
if p.CatchPanics {
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
p.ExitAltScreen()
|
|
|
|
fmt.Print("Caught panic. Restoring terminal...\n\n")
|
|
|
|
debug.PrintStack()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-07-13 11:39:04 -04:00
|
|
|
p.renderer = newRenderer(p.output, &p.mtx)
|
|
|
|
|
2020-01-25 01:15:29 -05:00
|
|
|
err := initTerminal()
|
2020-01-10 16:02:04 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-25 01:15:29 -05:00
|
|
|
defer restoreTerminal()
|
2020-01-10 16:02:04 -05:00
|
|
|
|
2020-01-18 22:18:19 -05:00
|
|
|
// Initialize program
|
2020-06-23 16:11:06 -04:00
|
|
|
model, initCmd := p.init()
|
|
|
|
if initCmd != nil {
|
2020-01-18 22:18:19 -05:00
|
|
|
go func() {
|
2020-06-23 16:11:06 -04:00
|
|
|
cmds <- initCmd
|
2020-01-18 22:18:19 -05:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-06-05 12:40:44 -04:00
|
|
|
// Start renderer
|
2020-07-13 11:39:04 -04:00
|
|
|
p.renderer.start()
|
|
|
|
p.renderer.altScreenActive = p.altScreenActive
|
2020-06-05 12:40:44 -04:00
|
|
|
|
2020-01-10 16:02:04 -05:00
|
|
|
// Render initial view
|
2020-07-13 11:39:04 -04:00
|
|
|
p.renderer.write(p.view(model))
|
2020-01-10 16:02:04 -05:00
|
|
|
|
2020-05-12 17:56:30 -04:00
|
|
|
// Subscribe to user input
|
2020-01-10 16:02:04 -05:00
|
|
|
go func() {
|
|
|
|
for {
|
2020-07-29 20:51:55 -04:00
|
|
|
msg, err := readInput(os.Stdin)
|
2020-04-22 10:32:18 -04:00
|
|
|
if err != nil {
|
|
|
|
errs <- err
|
|
|
|
}
|
2020-06-22 20:30:16 -04:00
|
|
|
msgs <- msg
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-06-16 16:41:35 -04:00
|
|
|
// Get initial terminal size
|
|
|
|
go func() {
|
2020-06-22 15:55:52 -04:00
|
|
|
w, h, err := terminal.GetSize(int(p.output.Fd()))
|
2020-06-16 16:41:35 -04:00
|
|
|
if err != nil {
|
|
|
|
errs <- err
|
|
|
|
}
|
|
|
|
msgs <- WindowSizeMsg{w, h}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Listen for window resizes
|
2020-06-22 15:55:52 -04:00
|
|
|
go listenForResize(p.output, msgs, errs)
|
2020-06-16 16:41:35 -04:00
|
|
|
|
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() {
|
2020-04-22 10:15:04 -04:00
|
|
|
msgs <- cmd()
|
2020-01-10 16:02:04 -05:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Handle updates and draw
|
|
|
|
for {
|
|
|
|
select {
|
2020-04-22 10:32:18 -04:00
|
|
|
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 {
|
2020-07-13 11:39:04 -04:00
|
|
|
p.renderer.stop()
|
2020-01-10 16:02:04 -05:00
|
|
|
close(done)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-06-17 12:10:03 -04:00
|
|
|
// 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
|
2020-07-13 11:39:04 -04:00
|
|
|
p.renderer.handleMessages(msg)
|
2020-06-23 16:11:06 -04:00
|
|
|
var cmd Cmd
|
2020-05-28 10:20:59 -04:00
|
|
|
model, cmd = p.update(msg, model) // run update
|
|
|
|
cmds <- cmd // process command (if any)
|
2020-07-13 11:39:04 -04:00
|
|
|
p.renderer.write(p.view(model)) // send view to renderer
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-29 20:49:20 -04:00
|
|
|
// EnterAltScreen enters the alternate screen buffer, which consumes the entire
|
|
|
|
// terminal window. ExitAltScreen will return the terminal to its former state.
|
2020-06-22 15:55:52 -04:00
|
|
|
func (p *Program) EnterAltScreen() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
|
|
|
fmt.Fprintf(p.output, te.CSI+te.AltScreenSeq)
|
2020-06-22 20:47:35 -04:00
|
|
|
moveCursor(p.output, 0, 0)
|
2020-07-13 11:39:04 -04:00
|
|
|
|
|
|
|
p.altScreenActive = true
|
|
|
|
if p.renderer != nil {
|
|
|
|
p.renderer.altScreenActive = p.altScreenActive
|
|
|
|
}
|
2020-02-01 21:02:12 -05:00
|
|
|
}
|
|
|
|
|
2020-06-17 14:28:08 -04:00
|
|
|
// ExitAltScreen exits the alternate screen buffer.
|
2020-06-22 15:55:52 -04:00
|
|
|
func (p *Program) ExitAltScreen() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
|
|
|
fmt.Fprintf(p.output, te.CSI+te.ExitAltScreenSeq)
|
2020-07-13 11:39:04 -04:00
|
|
|
|
|
|
|
p.altScreenActive = false
|
|
|
|
if p.renderer != nil {
|
|
|
|
p.renderer.altScreenActive = p.altScreenActive
|
|
|
|
}
|
2020-02-01 21:02:12 -05:00
|
|
|
}
|
2020-06-22 20:30:16 -04:00
|
|
|
|
2020-07-29 20:49:20 -04:00
|
|
|
// EnableMouseCellMotion enables mouse click, release, wheel and motion events
|
|
|
|
// if a button is pressed.
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) EnableMouseCellMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2020-06-25 12:42:31 -04:00
|
|
|
fmt.Fprintf(p.output, te.CSI+te.EnableMouseCellMotionSeq)
|
2020-06-22 20:30:16 -04:00
|
|
|
}
|
|
|
|
|
2020-07-29 20:49:20 -04:00
|
|
|
// DisableMouseCellMotion disables Mouse Cell Motion tracking. If you've
|
2020-06-23 12:01:23 -04:00
|
|
|
// enabled Cell Motion mouse trakcing be sure to call this as your program is
|
|
|
|
// exiting or your users will be very upset!
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) DisableMouseCellMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2020-06-25 12:42:31 -04:00
|
|
|
fmt.Fprintf(p.output, te.CSI+te.DisableMouseCellMotionSeq)
|
2020-06-22 20:30:16 -04:00
|
|
|
}
|
|
|
|
|
2020-06-23 12:01:23 -04:00
|
|
|
// EnableMouseAllMotion enables mouse click, release, wheel and motion events,
|
|
|
|
// regardless of whether a button is pressed. Many modern terminals support
|
|
|
|
// this, but not all.
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) EnableMouseAllMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2020-06-25 12:42:31 -04:00
|
|
|
fmt.Fprintf(p.output, te.CSI+te.EnableMouseAllMotionSeq)
|
2020-06-22 20:30:16 -04:00
|
|
|
}
|
|
|
|
|
2020-06-23 12:01:23 -04:00
|
|
|
// DisableMouseAllMotion disables All Motion mouse tracking. If you've enabled
|
|
|
|
// All Motion mouse tracking be sure you call this as your program is exiting
|
|
|
|
// or your users will be very upset!
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) DisableMouseAllMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2020-06-25 12:42:31 -04:00
|
|
|
fmt.Fprintf(p.output, te.CSI+te.DisableMouseAllMotionSeq)
|
2020-06-22 20:30:16 -04:00
|
|
|
}
|