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 (
|
2021-01-11 16:33:14 -05:00
|
|
|
"context"
|
2020-06-22 15:55:52 -04:00
|
|
|
"fmt"
|
2020-12-23 12:56:13 -05:00
|
|
|
"io"
|
2020-01-25 01:15:56 -05:00
|
|
|
"os"
|
2021-01-11 16:33:14 -05:00
|
|
|
"os/signal"
|
2020-09-21 13:09:32 -04:00
|
|
|
"runtime/debug"
|
2020-06-22 15:55:52 -04:00
|
|
|
"sync"
|
2021-01-11 16:33:14 -05:00
|
|
|
"syscall"
|
2021-09-28 13:30:11 -04:00
|
|
|
"time"
|
2020-01-31 07:47:36 -05:00
|
|
|
|
2021-02-26 18:38:52 -05:00
|
|
|
"github.com/containerd/console"
|
2020-12-23 12:56:13 -05:00
|
|
|
isatty "github.com/mattn/go-isatty"
|
2022-02-06 00:40:50 -05:00
|
|
|
"github.com/muesli/cancelreader"
|
2022-06-04 09:14:03 -04:00
|
|
|
"github.com/muesli/termenv"
|
2021-04-29 08:24:25 -04:00
|
|
|
"golang.org/x/term"
|
2020-01-10 16:02:04 -05:00
|
|
|
)
|
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Msg contain data from the result of a IO operation. Msgs trigger the update
|
|
|
|
// function and, henceforth, the UI.
|
2020-01-10 16:02:04 -05:00
|
|
|
type Msg interface{}
|
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Model contains the program's state as well as its core functions.
|
2020-10-15 16:08:19 -04:00
|
|
|
type Model interface {
|
2020-10-18 12:43:41 -04:00
|
|
|
// Init is the first function that will be called. It returns an optional
|
|
|
|
// initial command. To not perform an initial command return nil.
|
2020-10-15 16:08:19 -04:00
|
|
|
Init() Cmd
|
|
|
|
|
|
|
|
// Update is called when a message is received. Use it to inspect messages
|
|
|
|
// and, in response, update the model and/or send a command.
|
|
|
|
Update(Msg) (Model, Cmd)
|
|
|
|
|
|
|
|
// View renders the program's UI, which is just a string. The view is
|
|
|
|
// rendered after every Update.
|
|
|
|
View() string
|
|
|
|
}
|
2020-01-10 16:02:04 -05:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Cmd is an IO operation that returns a message when it's complete. 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.
|
2020-06-24 12:14:35 -04:00
|
|
|
//
|
2021-09-28 13:54:25 -04:00
|
|
|
// Note that there's almost never a reason to use a command to send a message
|
|
|
|
// to another part of your program. That 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
|
|
|
|
2021-07-29 16:47:13 -04:00
|
|
|
// Options to customize the program during its initialization. These are
|
|
|
|
// generally set with ProgramOptions.
|
2021-05-19 20:52:01 -04:00
|
|
|
//
|
|
|
|
// The options here are treated as bits.
|
|
|
|
type startupOptions byte
|
|
|
|
|
2021-07-29 15:43:03 -04:00
|
|
|
func (s startupOptions) has(option startupOptions) bool {
|
|
|
|
return s&option != 0
|
|
|
|
}
|
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
const (
|
|
|
|
withAltScreen startupOptions = 1 << iota
|
|
|
|
withMouseCellMotion
|
|
|
|
withMouseAllMotion
|
2021-07-29 15:43:03 -04:00
|
|
|
withInputTTY
|
2021-07-29 16:47:13 -04:00
|
|
|
withCustomInput
|
2021-10-30 12:29:30 -04:00
|
|
|
withANSICompressor
|
2021-05-19 20:52:01 -04:00
|
|
|
)
|
|
|
|
|
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-10-15 16:08:19 -04:00
|
|
|
initialModel Model
|
2020-06-22 15:55:52 -04:00
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// Configuration options that will set as the program is initializing,
|
2021-07-29 15:29:09 -04:00
|
|
|
// treated as bits. These options can be set via various ProgramOptions.
|
2021-05-19 20:52:01 -04:00
|
|
|
startupOptions startupOptions
|
|
|
|
|
2022-04-12 10:23:10 -04:00
|
|
|
ctx context.Context
|
2021-09-28 13:30:11 -04:00
|
|
|
mtx *sync.Mutex
|
2020-12-23 12:56:13 -05:00
|
|
|
|
2022-04-12 10:23:10 -04:00
|
|
|
msgs chan Msg
|
|
|
|
errs chan error
|
|
|
|
readLoopDone chan struct{}
|
2021-06-16 21:20:56 -04:00
|
|
|
|
2022-06-04 09:14:03 -04:00
|
|
|
output *termenv.Output // where to send output. this will usually be os.Stdout.
|
|
|
|
restoreOutput func() error
|
|
|
|
input io.Reader // this will usually be os.Stdin.
|
|
|
|
cancelReader cancelreader.CancelReader
|
2022-04-12 10:23:10 -04:00
|
|
|
|
|
|
|
renderer renderer
|
|
|
|
altScreenActive bool
|
|
|
|
altScreenWasActive bool // was the altscreen active before releasing the terminal?
|
2020-09-21 13:09:32 -04:00
|
|
|
|
2021-01-17 09:53:47 -05:00
|
|
|
// CatchPanics is incredibly useful for restoring the terminal to a usable
|
2020-09-21 13:09:32 -04:00
|
|
|
// 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
|
2021-01-11 16:33:14 -05:00
|
|
|
|
2022-04-12 10:23:10 -04:00
|
|
|
ignoreSignals bool
|
|
|
|
|
2022-02-04 10:09:17 -05:00
|
|
|
killc chan bool
|
|
|
|
|
2021-07-29 17:42:03 -04:00
|
|
|
console console.Console
|
2021-02-26 18:38:52 -05:00
|
|
|
|
|
|
|
// Stores the original reference to stdin for cases where input is not a
|
|
|
|
// TTY on windows and we've automatically opened CONIN$ to receive input.
|
|
|
|
// When the program exits this will be restored.
|
2021-05-31 10:27:59 -04:00
|
|
|
//
|
|
|
|
// Lint ignore note: the linter will find false positive on unix systems
|
|
|
|
// as this value only comes into play on Windows, hence the ignore comment
|
|
|
|
// below.
|
|
|
|
windowsStdin *os.File //nolint:golint,structcheck,unused
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// Batch performs a bunch of commands concurrently with no ordering guarantees
|
|
|
|
// about the results. Use a Batch to return several commands.
|
|
|
|
//
|
|
|
|
// Example:
|
|
|
|
//
|
2022-08-15 05:58:40 -04:00
|
|
|
// func (m model) Init() Cmd {
|
|
|
|
// return tea.Batch(someCommand, someOtherCommand)
|
|
|
|
// }
|
2021-05-19 20:52:01 -04:00
|
|
|
func Batch(cmds ...Cmd) Cmd {
|
2022-02-03 10:08:34 -05:00
|
|
|
var validCmds []Cmd
|
|
|
|
for _, c := range cmds {
|
|
|
|
if c == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
validCmds = append(validCmds, c)
|
|
|
|
}
|
|
|
|
if len(validCmds) == 0 {
|
2021-05-19 20:52:01 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return func() Msg {
|
2022-02-03 10:08:34 -05:00
|
|
|
return batchMsg(validCmds)
|
2021-05-19 20:52:01 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// batchMsg is the internal message used to perform a bunch of commands. You
|
|
|
|
// can send a batchMsg with Batch.
|
|
|
|
type batchMsg []Cmd
|
|
|
|
|
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{}
|
|
|
|
|
2021-05-03 14:28:45 -04:00
|
|
|
// EnterAltScreen is a special command that tells the Bubble Tea program to
|
2021-09-28 13:54:25 -04:00
|
|
|
// enter the alternate screen buffer.
|
2021-05-19 20:52:01 -04:00
|
|
|
//
|
|
|
|
// Because commands run asynchronously, this command should not be used in your
|
|
|
|
// model's Init function. To initialize your program with the altscreen enabled
|
|
|
|
// use the WithAltScreen ProgramOption instead.
|
2021-03-08 12:48:34 -05:00
|
|
|
func EnterAltScreen() Msg {
|
|
|
|
return enterAltScreenMsg{}
|
|
|
|
}
|
|
|
|
|
2021-05-03 14:28:45 -04:00
|
|
|
// enterAltScreenMsg in an internal message signals that the program should
|
|
|
|
// enter alternate screen buffer. You can send a enterAltScreenMsg with
|
|
|
|
// EnterAltScreen.
|
2021-03-08 12:48:34 -05:00
|
|
|
type enterAltScreenMsg struct{}
|
|
|
|
|
|
|
|
// ExitAltScreen is a special command that tells the Bubble Tea program to exit
|
2021-05-19 20:52:01 -04:00
|
|
|
// the alternate screen buffer. This command should be used to exit the
|
|
|
|
// alternate screen buffer while the program is running.
|
|
|
|
//
|
|
|
|
// Note that the alternate screen buffer will be automatically exited when the
|
|
|
|
// program quits.
|
2021-03-08 12:48:34 -05:00
|
|
|
func ExitAltScreen() Msg {
|
|
|
|
return exitAltScreenMsg{}
|
|
|
|
}
|
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// exitAltScreenMsg in an internal message signals that the program should exit
|
|
|
|
// alternate screen buffer. You can send a exitAltScreenMsg with ExitAltScreen.
|
|
|
|
type exitAltScreenMsg struct{}
|
2021-05-03 14:27:20 -04:00
|
|
|
|
|
|
|
// EnableMouseCellMotion is a special command that enables mouse click,
|
2021-05-19 20:52:01 -04:00
|
|
|
// release, and wheel events. Mouse movement events are also captured if
|
|
|
|
// a mouse button is pressed (i.e., drag events).
|
|
|
|
//
|
|
|
|
// Because commands run asynchronously, this command should not be used in your
|
|
|
|
// model's Init function. Use the WithMouseCellMotion ProgramOption instead.
|
2021-05-03 14:27:20 -04:00
|
|
|
func EnableMouseCellMotion() Msg {
|
|
|
|
return enableMouseCellMotionMsg{}
|
|
|
|
}
|
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// enableMouseCellMotionMsg is a special command that signals to start
|
|
|
|
// listening for "cell motion" type mouse events (ESC[?1002l). To send an
|
|
|
|
// enableMouseCellMotionMsg, use the EnableMouseCellMotion command.
|
|
|
|
type enableMouseCellMotionMsg struct{}
|
2021-05-03 14:27:20 -04:00
|
|
|
|
|
|
|
// EnableMouseAllMotion is a special command that enables mouse click, release,
|
2021-05-19 20:52:01 -04:00
|
|
|
// wheel, and motion events, which are delivered regardless of whether a mouse
|
|
|
|
// button is pressed, effectively enabling support for hover interactions.
|
|
|
|
//
|
|
|
|
// Many modern terminals support this, but not all. If in doubt, use
|
|
|
|
// EnableMouseCellMotion instead.
|
|
|
|
//
|
|
|
|
// Because commands run asynchronously, this command should not be used in your
|
|
|
|
// model's Init function. Use the WithMouseAllMotion ProgramOption instead.
|
2021-05-03 14:27:20 -04:00
|
|
|
func EnableMouseAllMotion() Msg {
|
|
|
|
return enableMouseAllMotionMsg{}
|
|
|
|
}
|
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// enableMouseAllMotionMsg is a special command that signals to start listening
|
|
|
|
// for "all motion" type mouse events (ESC[?1003l). To send an
|
|
|
|
// enableMouseAllMotionMsg, use the EnableMouseAllMotion command.
|
|
|
|
type enableMouseAllMotionMsg struct{}
|
2021-05-03 14:27:20 -04:00
|
|
|
|
|
|
|
// DisableMouse is a special command that stops listening for mouse events.
|
|
|
|
func DisableMouse() Msg {
|
|
|
|
return disableMouseMsg{}
|
|
|
|
}
|
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// disableMouseMsg is an internal message that that signals to stop listening
|
|
|
|
// for mouse events. To send a disableMouseMsg, use the DisableMouse command.
|
|
|
|
type disableMouseMsg struct{}
|
2020-03-31 16:08:03 -04:00
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// WindowSizeMsg is used to report the terminal size. It's sent to Update once
|
|
|
|
// initially and then on every terminal resize. Note that Windows does not
|
|
|
|
// have support for reporting when resizes occur as it does not support the
|
|
|
|
// SIGWINCH signal.
|
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-12-03 10:44:10 -05:00
|
|
|
// HideCursor is a special command for manually instructing Bubble Tea to hide
|
|
|
|
// the cursor. In some rare cases, certain operations will cause the terminal
|
|
|
|
// to show the cursor, which is normally hidden for the duration of a Bubble
|
2021-05-03 14:27:20 -04:00
|
|
|
// Tea program's lifetime. You will most likely not need to use this command.
|
2020-12-03 10:44:10 -05:00
|
|
|
func HideCursor() Msg {
|
|
|
|
return hideCursorMsg{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// hideCursorMsg is an internal command used to hide the cursor. You can send
|
|
|
|
// this message with HideCursor.
|
|
|
|
type hideCursorMsg struct{}
|
|
|
|
|
2020-05-25 08:02:46 -04:00
|
|
|
// NewProgram creates a new Program.
|
2020-12-23 12:56:13 -05:00
|
|
|
func NewProgram(model Model, opts ...ProgramOption) *Program {
|
|
|
|
p := &Program{
|
2021-02-26 15:11:23 -05:00
|
|
|
mtx: &sync.Mutex{},
|
2020-10-15 16:08:19 -04:00
|
|
|
initialModel: model,
|
2020-12-23 12:56:13 -05:00
|
|
|
input: os.Stdin,
|
2022-01-04 02:33:28 -05:00
|
|
|
msgs: make(chan Msg),
|
2020-10-15 16:08:19 -04:00
|
|
|
CatchPanics: true,
|
2022-02-04 10:09:17 -05:00
|
|
|
killc: make(chan bool, 1),
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
2020-12-23 12:56:13 -05:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Apply all options to the program.
|
2020-12-23 12:56:13 -05:00
|
|
|
for _, opt := range opts {
|
|
|
|
opt(p)
|
|
|
|
}
|
|
|
|
|
2022-06-04 09:14:03 -04:00
|
|
|
// if no output was set, set it to stdout
|
|
|
|
if p.output == nil {
|
|
|
|
p.output = termenv.DefaultOutput()
|
|
|
|
|
|
|
|
// cache detected color values
|
|
|
|
termenv.WithColorCache(true)(p.output)
|
|
|
|
}
|
|
|
|
|
|
|
|
p.restoreOutput, _ = termenv.EnableVirtualTerminalProcessing(p.output)
|
|
|
|
|
2020-12-23 12:56:13 -05:00
|
|
|
return p
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
|
2021-09-22 07:26:58 -04:00
|
|
|
// StartReturningModel initializes the program. Returns the final model.
|
|
|
|
func (p *Program) StartReturningModel() (Model, error) {
|
2022-04-12 10:23:10 -04:00
|
|
|
cmds := make(chan Cmd)
|
|
|
|
p.errs = make(chan error)
|
2020-01-10 16:02:04 -05:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Channels for managing goroutine lifecycles.
|
2021-09-28 13:30:11 -04:00
|
|
|
var (
|
|
|
|
sigintLoopDone = make(chan struct{})
|
|
|
|
cmdLoopDone = make(chan struct{})
|
|
|
|
resizeLoopDone = make(chan struct{})
|
|
|
|
initSignalDone = make(chan struct{})
|
|
|
|
|
|
|
|
waitForGoroutines = func(withReadLoop bool) {
|
|
|
|
if withReadLoop {
|
|
|
|
select {
|
2022-04-12 10:23:10 -04:00
|
|
|
case <-p.readLoopDone:
|
2021-09-28 13:30:11 -04:00
|
|
|
case <-time.After(500 * time.Millisecond):
|
2021-09-28 13:54:25 -04:00
|
|
|
// The read loop hangs, which means the input
|
|
|
|
// cancelReader's cancel function has returned true even
|
|
|
|
// though it was not able to cancel the read.
|
2021-09-28 13:30:11 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
<-cmdLoopDone
|
|
|
|
<-resizeLoopDone
|
|
|
|
<-sigintLoopDone
|
|
|
|
<-initSignalDone
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2022-04-12 10:23:10 -04:00
|
|
|
var cancelContext context.CancelFunc
|
|
|
|
p.ctx, cancelContext = context.WithCancel(context.Background())
|
2021-09-28 13:30:11 -04:00
|
|
|
defer cancelContext()
|
2021-01-11 16:33:14 -05:00
|
|
|
|
2021-07-29 17:01:38 -04:00
|
|
|
switch {
|
|
|
|
case p.startupOptions.has(withInputTTY):
|
|
|
|
// Open a new TTY, by request
|
2021-07-29 15:43:03 -04:00
|
|
|
f, err := openInputTTY()
|
|
|
|
if err != nil {
|
2021-09-22 07:26:58 -04:00
|
|
|
return p.initialModel, err
|
2021-07-29 15:43:03 -04:00
|
|
|
}
|
2021-09-28 13:30:11 -04:00
|
|
|
|
|
|
|
defer f.Close() // nolint:errcheck
|
|
|
|
|
2021-07-29 15:43:03 -04:00
|
|
|
p.input = f
|
|
|
|
|
2021-07-29 17:01:38 -04:00
|
|
|
case !p.startupOptions.has(withCustomInput):
|
|
|
|
// If the user hasn't set a custom input, and input's not a terminal,
|
|
|
|
// open a TTY so we can capture input as normal. This will allow things
|
|
|
|
// to "just work" in cases where data was piped or redirected into this
|
|
|
|
// application.
|
|
|
|
f, isFile := p.input.(*os.File)
|
|
|
|
if !isFile {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
if isatty.IsTerminal(f.Fd()) {
|
|
|
|
break
|
2021-02-26 18:38:52 -05:00
|
|
|
}
|
2021-07-29 17:01:38 -04:00
|
|
|
|
|
|
|
f, err := openInputTTY()
|
|
|
|
if err != nil {
|
2021-09-22 07:26:58 -04:00
|
|
|
return p.initialModel, err
|
2021-07-29 17:01:38 -04:00
|
|
|
}
|
2021-09-28 13:30:11 -04:00
|
|
|
|
|
|
|
defer f.Close() // nolint:errcheck
|
|
|
|
|
2021-07-29 17:01:38 -04:00
|
|
|
p.input = f
|
2021-02-26 18:38:52 -05:00
|
|
|
}
|
|
|
|
|
2022-08-24 23:32:12 -04:00
|
|
|
// Listen for SIGINT and SIGTERM.
|
|
|
|
//
|
|
|
|
// In most cases ^C will not send an interrupt because the terminal will be
|
|
|
|
// in raw mode and ^C will be captured as a keystroke and sent along to
|
|
|
|
// Program.Update as a KeyMsg. When input is not a TTY, however, ^C will be
|
|
|
|
// caught here.
|
|
|
|
//
|
|
|
|
// SIGTERM is sent by unix utilities (like kill) to terminate a process.
|
2021-01-11 16:33:14 -05:00
|
|
|
go func() {
|
|
|
|
sig := make(chan os.Signal, 1)
|
2022-08-24 23:32:12 -04:00
|
|
|
signal.Notify(sig, syscall.SIGINT, syscall.SIGTERM)
|
2021-09-28 13:30:11 -04:00
|
|
|
defer func() {
|
|
|
|
signal.Stop(sig)
|
|
|
|
close(sigintLoopDone)
|
|
|
|
}()
|
2021-01-11 16:33:14 -05:00
|
|
|
|
2022-04-12 10:23:10 -04:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-p.ctx.Done():
|
|
|
|
return
|
|
|
|
case <-sig:
|
|
|
|
if !p.ignoreSignals {
|
|
|
|
p.msgs <- quitMsg{}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2021-01-11 16:33:14 -05:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-09-21 13:09:32 -04:00
|
|
|
if p.CatchPanics {
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
2021-05-19 21:35:36 -04:00
|
|
|
p.shutdown(true)
|
2020-10-08 07:54:53 -04:00
|
|
|
fmt.Printf("Caught panic:\n\n%s\n\nRestoring terminal...\n\n", r)
|
2020-09-21 13:09:32 -04:00
|
|
|
debug.PrintStack()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-12-23 12:56:13 -05:00
|
|
|
// Check if output is a TTY before entering raw mode, hiding the cursor and
|
|
|
|
// so on.
|
2021-07-29 12:39:13 -04:00
|
|
|
if err := p.initTerminal(); err != nil {
|
2021-09-22 07:26:58 -04:00
|
|
|
return p.initialModel, err
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
|
2021-05-19 20:52:01 -04:00
|
|
|
// If no renderer is set use the standard one.
|
|
|
|
if p.renderer == nil {
|
2021-10-30 12:29:30 -04:00
|
|
|
p.renderer = newRenderer(p.output, p.mtx, p.startupOptions.has(withANSICompressor))
|
2021-05-19 20:52:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Honor program startup options.
|
|
|
|
if p.startupOptions&withAltScreen != 0 {
|
|
|
|
p.EnterAltScreen()
|
|
|
|
}
|
|
|
|
if p.startupOptions&withMouseCellMotion != 0 {
|
|
|
|
p.EnableMouseCellMotion()
|
|
|
|
} else if p.startupOptions&withMouseAllMotion != 0 {
|
|
|
|
p.EnableMouseAllMotion()
|
|
|
|
}
|
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Initialize the program.
|
2020-10-15 16:08:19 -04:00
|
|
|
model := p.initialModel
|
2021-09-07 15:29:13 -04:00
|
|
|
if initCmd := model.Init(); initCmd != nil {
|
2020-01-18 22:18:19 -05:00
|
|
|
go func() {
|
2021-09-28 13:30:11 -04:00
|
|
|
defer close(initSignalDone)
|
|
|
|
select {
|
|
|
|
case cmds <- initCmd:
|
2022-04-12 10:23:10 -04:00
|
|
|
case <-p.ctx.Done():
|
2021-09-28 13:30:11 -04:00
|
|
|
}
|
2020-01-18 22:18:19 -05:00
|
|
|
}()
|
2021-09-28 13:30:11 -04:00
|
|
|
} else {
|
|
|
|
close(initSignalDone)
|
2020-01-18 22:18:19 -05:00
|
|
|
}
|
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Start the renderer.
|
2020-07-13 11:39:04 -04:00
|
|
|
p.renderer.start()
|
2021-02-09 13:33:25 -05:00
|
|
|
p.renderer.setAltScreen(p.altScreenActive)
|
2020-06-05 12:40:44 -04:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Render the initial view.
|
2020-10-15 16:08:19 -04:00
|
|
|
p.renderer.write(model.View())
|
2020-01-10 16:02:04 -05:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Subscribe to user input.
|
2021-07-29 16:47:13 -04:00
|
|
|
if p.input != nil {
|
2022-04-12 10:23:10 -04:00
|
|
|
if err := p.initCancelReader(); err != nil {
|
|
|
|
return model, err
|
|
|
|
}
|
2021-09-28 13:30:11 -04:00
|
|
|
} else {
|
2022-04-12 10:23:10 -04:00
|
|
|
defer close(p.readLoopDone)
|
2020-12-23 12:56:13 -05:00
|
|
|
}
|
2022-04-12 10:23:10 -04:00
|
|
|
defer p.cancelReader.Close() // nolint:errcheck
|
2020-01-10 16:02:04 -05:00
|
|
|
|
2022-06-04 09:14:03 -04:00
|
|
|
if f, ok := p.output.TTY().(*os.File); ok && isatty.IsTerminal(f.Fd()) {
|
2021-09-28 13:54:25 -04:00
|
|
|
// Get the initial terminal size and send it to the program.
|
2020-12-29 22:21:17 -05:00
|
|
|
go func() {
|
2021-07-29 17:42:03 -04:00
|
|
|
w, h, err := term.GetSize(int(f.Fd()))
|
2020-12-29 22:21:17 -05:00
|
|
|
if err != nil {
|
2022-04-12 10:23:10 -04:00
|
|
|
p.errs <- err
|
2020-12-29 22:21:17 -05:00
|
|
|
}
|
2021-09-28 13:30:11 -04:00
|
|
|
|
|
|
|
select {
|
2022-04-12 10:23:10 -04:00
|
|
|
case <-p.ctx.Done():
|
2021-09-28 13:30:11 -04:00
|
|
|
case p.msgs <- WindowSizeMsg{w, h}:
|
|
|
|
}
|
2020-12-29 22:21:17 -05:00
|
|
|
}()
|
2020-06-16 16:41:35 -04:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Listen for window resizes.
|
2022-04-12 10:23:10 -04:00
|
|
|
go listenForResize(p.ctx, f, p.msgs, p.errs, resizeLoopDone)
|
2021-09-28 13:30:11 -04:00
|
|
|
} else {
|
|
|
|
close(resizeLoopDone)
|
2020-12-29 22:21:17 -05:00
|
|
|
}
|
2020-06-16 16:41:35 -04:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Process commands.
|
2020-01-10 16:02:04 -05:00
|
|
|
go func() {
|
2021-09-28 13:30:11 -04:00
|
|
|
defer close(cmdLoopDone)
|
|
|
|
|
2020-01-10 16:02:04 -05:00
|
|
|
for {
|
|
|
|
select {
|
2022-04-12 10:23:10 -04:00
|
|
|
case <-p.ctx.Done():
|
2021-09-28 13:30:11 -04:00
|
|
|
|
2020-01-10 16:02:04 -05:00
|
|
|
return
|
|
|
|
case cmd := <-cmds:
|
2021-09-28 13:30:11 -04:00
|
|
|
if cmd == nil {
|
|
|
|
continue
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
2021-09-28 13:30:11 -04:00
|
|
|
|
|
|
|
// Don't wait on these goroutines, otherwise the shutdown
|
|
|
|
// latency would get too large as a Cmd can run for some time
|
|
|
|
// (e.g. tick commands that sleep for half a second). It's not
|
|
|
|
// possible to cancel them so we'll have to leak the goroutine
|
|
|
|
// until Cmd returns.
|
|
|
|
go func() {
|
|
|
|
select {
|
|
|
|
case p.msgs <- cmd():
|
2022-04-12 10:23:10 -04:00
|
|
|
case <-p.ctx.Done():
|
2021-09-28 13:30:11 -04:00
|
|
|
}
|
|
|
|
}()
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Handle updates and draw.
|
2020-01-10 16:02:04 -05:00
|
|
|
for {
|
|
|
|
select {
|
2022-02-04 10:09:17 -05:00
|
|
|
case <-p.killc:
|
|
|
|
return nil, nil
|
2022-04-12 10:23:10 -04:00
|
|
|
case err := <-p.errs:
|
2021-09-28 13:30:11 -04:00
|
|
|
cancelContext()
|
2022-04-12 10:23:10 -04:00
|
|
|
waitForGoroutines(p.cancelReader.Cancel())
|
2021-05-19 21:35:36 -04:00
|
|
|
p.shutdown(false)
|
2021-09-22 07:26:58 -04:00
|
|
|
return model, err
|
2021-09-28 13:30:11 -04:00
|
|
|
|
2021-06-16 21:20:56 -04:00
|
|
|
case msg := <-p.msgs:
|
2020-03-31 16:08:03 -04:00
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Handle special internal messages.
|
2021-05-06 13:48:46 -04:00
|
|
|
switch msg := msg.(type) {
|
2020-12-03 10:44:10 -05:00
|
|
|
case quitMsg:
|
2021-09-28 13:30:11 -04:00
|
|
|
cancelContext()
|
2022-04-12 10:23:10 -04:00
|
|
|
waitForGoroutines(p.cancelReader.Cancel())
|
2021-05-19 21:35:36 -04:00
|
|
|
p.shutdown(false)
|
2021-09-22 07:26:58 -04:00
|
|
|
return model, nil
|
2021-05-03 14:27:20 -04:00
|
|
|
|
2021-05-06 13:48:46 -04:00
|
|
|
case batchMsg:
|
|
|
|
for _, cmd := range msg {
|
|
|
|
cmds <- cmd
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
|
2021-05-28 21:41:42 -04:00
|
|
|
case WindowSizeMsg:
|
2022-06-05 06:58:33 -04:00
|
|
|
p.mtx.Lock()
|
2021-05-28 21:41:42 -04:00
|
|
|
p.renderer.repaint()
|
2022-06-05 06:58:33 -04:00
|
|
|
p.mtx.Unlock()
|
2021-05-28 21:41:42 -04:00
|
|
|
|
2021-03-08 12:48:34 -05:00
|
|
|
case enterAltScreenMsg:
|
|
|
|
p.EnterAltScreen()
|
2021-05-03 14:27:20 -04:00
|
|
|
|
2021-03-08 12:48:34 -05:00
|
|
|
case exitAltScreenMsg:
|
|
|
|
p.ExitAltScreen()
|
2021-05-03 14:27:20 -04:00
|
|
|
|
|
|
|
case enableMouseCellMotionMsg:
|
|
|
|
p.EnableMouseCellMotion()
|
|
|
|
|
|
|
|
case enableMouseAllMotionMsg:
|
|
|
|
p.EnableMouseAllMotion()
|
|
|
|
|
|
|
|
case disableMouseMsg:
|
|
|
|
p.DisableMouseCellMotion()
|
|
|
|
p.DisableMouseAllMotion()
|
|
|
|
|
2020-12-03 10:44:10 -05:00
|
|
|
case hideCursorMsg:
|
2022-06-04 09:14:03 -04:00
|
|
|
p.output.HideCursor()
|
2022-04-12 10:23:10 -04:00
|
|
|
|
|
|
|
case execMsg:
|
2022-06-01 20:12:21 -04:00
|
|
|
// NB: this blocks.
|
2022-04-12 10:23:10 -04:00
|
|
|
p.exec(msg.cmd, msg.fn)
|
2022-08-24 14:29:14 -04:00
|
|
|
|
|
|
|
case sequenceMsg:
|
|
|
|
go func() {
|
|
|
|
// Execute commands one at a time, in order.
|
|
|
|
for _, cmd := range msg {
|
|
|
|
select {
|
|
|
|
case p.msgs <- cmd():
|
|
|
|
case <-p.ctx.Done():
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
|
2021-09-28 13:54:25 -04:00
|
|
|
// Process internal messages for the renderer.
|
2021-02-09 13:33:25 -05:00
|
|
|
if r, ok := p.renderer.(*standardRenderer); ok {
|
|
|
|
r.handleMessages(msg)
|
|
|
|
}
|
|
|
|
|
2020-06-23 16:11:06 -04:00
|
|
|
var cmd Cmd
|
2020-10-15 16:08:19 -04:00
|
|
|
model, cmd = model.Update(msg) // run update
|
|
|
|
cmds <- cmd // process command (if any)
|
|
|
|
p.renderer.write(model.View()) // send view to renderer
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-22 07:26:58 -04:00
|
|
|
// Start initializes the program. Ignores the final model.
|
|
|
|
func (p *Program) Start() error {
|
|
|
|
_, err := p.StartReturningModel()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-06-16 21:20:56 -04:00
|
|
|
// Send sends a message to the main update function, effectively allowing
|
|
|
|
// messages to be injected from outside the program for interoperability
|
|
|
|
// purposes.
|
|
|
|
//
|
|
|
|
// If the program is not running this this will be a no-op, so it's safe to
|
|
|
|
// send messages if the program is unstarted, or has exited.
|
|
|
|
func (p *Program) Send(msg Msg) {
|
2022-01-04 02:33:28 -05:00
|
|
|
p.msgs <- msg
|
2021-06-16 21:20:56 -04:00
|
|
|
}
|
|
|
|
|
2021-10-04 18:26:07 -04:00
|
|
|
// Quit is a convenience function for quitting Bubble Tea programs. Use it
|
|
|
|
// when you need to shut down a Bubble Tea program from the outside.
|
|
|
|
//
|
|
|
|
// If you wish to quit from within a Bubble Tea program use the Quit command.
|
|
|
|
//
|
|
|
|
// If the program is not running this will be a no-op, so it's safe to call
|
|
|
|
// if the program is unstarted or has already exited.
|
|
|
|
func (p *Program) Quit() {
|
|
|
|
p.Send(Quit())
|
|
|
|
}
|
|
|
|
|
2022-02-04 10:09:17 -05:00
|
|
|
// Kill stops the program immediately and restores the former terminal state.
|
|
|
|
// The final render that you would normally see when quitting will be skipped.
|
|
|
|
func (p *Program) Kill() {
|
|
|
|
p.killc <- true
|
|
|
|
p.shutdown(true)
|
|
|
|
}
|
|
|
|
|
2021-05-19 21:35:36 -04:00
|
|
|
// shutdown performs operations to free up resources and restore the terminal
|
|
|
|
// to its original state.
|
|
|
|
func (p *Program) shutdown(kill bool) {
|
2022-02-04 10:09:17 -05:00
|
|
|
if p.renderer != nil {
|
|
|
|
if kill {
|
|
|
|
p.renderer.kill()
|
|
|
|
} else {
|
|
|
|
p.renderer.stop()
|
|
|
|
}
|
2021-05-19 21:35:36 -04:00
|
|
|
}
|
|
|
|
p.ExitAltScreen()
|
|
|
|
p.DisableMouseCellMotion()
|
|
|
|
p.DisableMouseAllMotion()
|
2022-04-12 10:23:10 -04:00
|
|
|
_ = p.restoreTerminalState()
|
2022-06-04 09:14:03 -04:00
|
|
|
|
|
|
|
if p.restoreOutput != nil {
|
|
|
|
_ = p.restoreOutput()
|
|
|
|
}
|
2021-05-19 21:35:36 -04: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.
|
2021-05-03 14:28:45 -04:00
|
|
|
//
|
2022-06-08 08:55:54 -04:00
|
|
|
// Deprecated: Use the WithAltScreen ProgramOption instead.
|
2020-06-22 15:55:52 -04:00
|
|
|
func (p *Program) EnterAltScreen() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2021-03-08 12:48:34 -05:00
|
|
|
|
|
|
|
if p.altScreenActive {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-06-04 09:14:03 -04:00
|
|
|
p.output.AltScreen()
|
2022-08-25 13:14:52 -04:00
|
|
|
p.output.MoveCursor(1, 1)
|
2020-07-13 11:39:04 -04:00
|
|
|
|
|
|
|
p.altScreenActive = true
|
|
|
|
if p.renderer != nil {
|
2021-02-09 13:33:25 -05:00
|
|
|
p.renderer.setAltScreen(p.altScreenActive)
|
2020-07-13 11:39:04 -04:00
|
|
|
}
|
2020-02-01 21:02:12 -05:00
|
|
|
}
|
|
|
|
|
2020-06-17 14:28:08 -04:00
|
|
|
// ExitAltScreen exits the alternate screen buffer.
|
2021-05-03 14:28:45 -04:00
|
|
|
//
|
2022-06-08 08:55:54 -04:00
|
|
|
// Deprecated: The altscreen will exited automatically when the program exits.
|
2020-06-22 15:55:52 -04:00
|
|
|
func (p *Program) ExitAltScreen() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2021-03-08 12:48:34 -05:00
|
|
|
|
|
|
|
if !p.altScreenActive {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-06-04 09:14:03 -04:00
|
|
|
p.output.ExitAltScreen()
|
2020-07-13 11:39:04 -04:00
|
|
|
|
|
|
|
p.altScreenActive = false
|
|
|
|
if p.renderer != nil {
|
2021-02-09 13:33:25 -05:00
|
|
|
p.renderer.setAltScreen(p.altScreenActive)
|
2020-07-13 11:39:04 -04:00
|
|
|
}
|
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
|
2021-05-03 14:27:20 -04:00
|
|
|
// if a mouse button is pressed (i.e., drag events).
|
|
|
|
//
|
2022-06-08 08:55:54 -04:00
|
|
|
// Deprecated: Use the WithMouseCellMotion ProgramOption instead.
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) EnableMouseCellMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2022-06-04 09:14:03 -04:00
|
|
|
|
|
|
|
p.output.EnableMouseCellMotion()
|
2020-06-22 20:30:16 -04:00
|
|
|
}
|
|
|
|
|
2021-05-03 14:27:20 -04:00
|
|
|
// DisableMouseCellMotion disables Mouse Cell Motion tracking. This will be
|
|
|
|
// called automatically when exiting a Bubble Tea program.
|
|
|
|
//
|
2022-06-08 08:55:54 -04:00
|
|
|
// Deprecated: The mouse will automatically be disabled when the program exits.
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) DisableMouseCellMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2022-06-04 09:14:03 -04:00
|
|
|
|
|
|
|
p.output.DisableMouseCellMotion()
|
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,
|
2021-05-03 14:27:20 -04:00
|
|
|
// regardless of whether a mouse button is pressed. Many modern terminals
|
|
|
|
// support this, but not all.
|
|
|
|
//
|
2022-06-08 08:55:54 -04:00
|
|
|
// Deprecated: Use the WithMouseAllMotion ProgramOption instead.
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) EnableMouseAllMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2022-06-04 09:14:03 -04:00
|
|
|
|
|
|
|
p.output.EnableMouseAllMotion()
|
2020-06-22 20:30:16 -04:00
|
|
|
}
|
|
|
|
|
2021-05-03 14:27:20 -04:00
|
|
|
// DisableMouseAllMotion disables All Motion mouse tracking. This will be
|
|
|
|
// called automatically when exiting a Bubble Tea program.
|
|
|
|
//
|
2022-06-08 08:55:54 -04:00
|
|
|
// Deprecated: The mouse will automatically be disabled when the program exits.
|
2020-06-22 20:30:16 -04:00
|
|
|
func (p *Program) DisableMouseAllMotion() {
|
|
|
|
p.mtx.Lock()
|
|
|
|
defer p.mtx.Unlock()
|
2022-06-04 09:14:03 -04:00
|
|
|
|
|
|
|
p.output.DisableMouseAllMotion()
|
2020-06-22 20:30:16 -04:00
|
|
|
}
|
2022-04-12 10:23:10 -04:00
|
|
|
|
|
|
|
// ReleaseTerminal restores the original terminal state and cancels the input
|
|
|
|
// reader. You can return control to the Program with RestoreTerminal.
|
|
|
|
func (p *Program) ReleaseTerminal() error {
|
|
|
|
p.ignoreSignals = true
|
|
|
|
p.cancelInput()
|
|
|
|
p.altScreenWasActive = p.altScreenActive
|
|
|
|
if p.altScreenActive {
|
|
|
|
p.ExitAltScreen()
|
|
|
|
time.Sleep(time.Millisecond * 10) // give the terminal a moment to catch up
|
|
|
|
}
|
|
|
|
return p.restoreTerminalState()
|
|
|
|
}
|
|
|
|
|
|
|
|
// RestoreTerminal reinitializes the Program's input reader, restores the
|
|
|
|
// terminal to the former state when the program was running, and repaints.
|
|
|
|
// Use it to reinitialize a Program after running ReleaseTerminal.
|
|
|
|
func (p *Program) RestoreTerminal() error {
|
|
|
|
p.ignoreSignals = false
|
|
|
|
|
|
|
|
if err := p.initTerminal(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := p.initCancelReader(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.altScreenWasActive {
|
|
|
|
p.EnterAltScreen()
|
|
|
|
}
|
|
|
|
|
|
|
|
go p.Send(repaintMsg{})
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2022-06-22 12:53:02 -04:00
|
|
|
|
2022-06-28 15:34:22 -04:00
|
|
|
// Println prints above the Program. This output is unmanaged by the program
|
|
|
|
// and will persist across renders by the Program.
|
2022-06-22 12:53:02 -04:00
|
|
|
//
|
|
|
|
// If the altscreen is active no output will be printed.
|
|
|
|
func (p *Program) Println(args ...interface{}) {
|
|
|
|
p.msgs <- printLineMessage{
|
|
|
|
messageBody: fmt.Sprint(args...),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Printf prints above the Program. It takes a format template followed by
|
|
|
|
// values similar to fmt.Printf. This output is unmanaged by the program and
|
|
|
|
// will persist across renders by the Program.
|
|
|
|
//
|
|
|
|
// Unlike fmt.Printf (but similar to log.Printf) the message will be print on
|
|
|
|
// its own line.
|
|
|
|
//
|
|
|
|
// If the altscreen is active no output will be printed.
|
|
|
|
func (p *Program) Printf(template string, args ...interface{}) {
|
|
|
|
p.msgs <- printLineMessage{
|
|
|
|
messageBody: fmt.Sprintf(template, args...),
|
|
|
|
}
|
|
|
|
}
|
2022-08-24 14:29:14 -04:00
|
|
|
|
|
|
|
// sequenceMsg is used interally to run the the given commands in order.
|
|
|
|
type sequenceMsg []Cmd
|
|
|
|
|
|
|
|
// Sequence runs the given commands one at a time, in order. Contrast this with
|
|
|
|
// Batch, which runs commands concurrently.
|
|
|
|
func Sequence(cmds ...Cmd) Cmd {
|
|
|
|
return func() Msg {
|
|
|
|
return sequenceMsg(cmds)
|
|
|
|
}
|
|
|
|
}
|