2020-05-12 16:39:08 -04:00
|
|
|
package boba
|
2020-01-10 16:02:04 -05:00
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
2020-01-25 01:15:56 -05:00
|
|
|
"os"
|
2020-01-10 16:02:04 -05:00
|
|
|
"strings"
|
2020-05-19 13:16:02 -04:00
|
|
|
"sync"
|
2020-01-31 07:47:36 -05:00
|
|
|
|
|
|
|
"github.com/muesli/termenv"
|
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-04-17 19:02:25 -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-05-17 19:28:12 -04:00
|
|
|
// Cmd is an IO operation. If it's nil it's considered a no-op.
|
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
|
|
|
|
// 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
|
|
|
|
type View func(Model) string
|
|
|
|
|
|
|
|
// Program is a terminal user interface
|
|
|
|
type Program struct {
|
2020-05-12 17:56:30 -04:00
|
|
|
init Init
|
|
|
|
update Update
|
|
|
|
view View
|
2020-05-19 13:16:02 -04:00
|
|
|
|
|
|
|
mutex sync.Mutex
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
|
2020-01-14 17:19:44 -05:00
|
|
|
// Quit is a command that tells the program to exit
|
2020-04-22 10:15:04 -04:00
|
|
|
func Quit() Msg {
|
2020-01-10 16:02:04 -05:00
|
|
|
return quitMsg{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Signals that the program should quit
|
|
|
|
type quitMsg struct{}
|
|
|
|
|
2020-03-31 16:08:03 -04:00
|
|
|
// batchMsg is used to perform a bunch of commands
|
|
|
|
type batchMsg []Cmd
|
|
|
|
|
2020-01-10 16:02:04 -05: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-05-19 13:16:02 -04:00
|
|
|
|
|
|
|
mutex: sync.Mutex{},
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start initializes the program
|
|
|
|
func (p *Program) Start() error {
|
|
|
|
var (
|
2020-04-22 10:25:49 -04:00
|
|
|
model Model
|
2020-01-22 19:15:30 -05:00
|
|
|
cmd Cmd
|
|
|
|
cmds = make(chan Cmd)
|
|
|
|
msgs = make(chan Msg)
|
2020-04-22 10:32:18 -04:00
|
|
|
errs = make(chan error)
|
2020-01-22 19:15:30 -05:00
|
|
|
done = make(chan struct{})
|
|
|
|
linesRendered int
|
2020-01-10 16:02:04 -05:00
|
|
|
)
|
|
|
|
|
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-04-22 10:25:49 -04:00
|
|
|
model, cmd = p.init()
|
2020-01-18 22:18:19 -05:00
|
|
|
if cmd != nil {
|
|
|
|
go func() {
|
|
|
|
cmds <- cmd
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-01-10 16:02:04 -05:00
|
|
|
// Render initial view
|
2020-04-22 10:25:49 -04:00
|
|
|
linesRendered = p.render(model, linesRendered)
|
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-04-22 10:32:18 -04:00
|
|
|
msg, err := ReadKey(os.Stdin)
|
|
|
|
if err != nil {
|
|
|
|
errs <- err
|
|
|
|
}
|
2020-01-17 20:46:34 -05:00
|
|
|
msgs <- KeyMsg(msg)
|
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 {
|
|
|
|
close(done)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-03-31 16:08:03 -04:00
|
|
|
// Process batch commands
|
2020-04-03 18:31:40 -04:00
|
|
|
if batchedCmds, ok := msg.(batchMsg); ok {
|
|
|
|
for _, cmd := range batchedCmds {
|
2020-03-31 16:08:03 -04:00
|
|
|
cmds <- cmd
|
|
|
|
}
|
2020-04-03 18:31:40 -04:00
|
|
|
continue
|
2020-03-31 16:08:03 -04:00
|
|
|
}
|
|
|
|
|
2020-04-22 10:25:49 -04:00
|
|
|
model, cmd = p.update(msg, model) // run update
|
|
|
|
cmds <- cmd // process command (if any)
|
|
|
|
linesRendered = p.render(model, linesRendered) // render to terminal
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-22 19:15:30 -05:00
|
|
|
// Render a view to the terminal. Returns the number of lines rendered.
|
|
|
|
func (p *Program) render(model Model, linesRendered int) int {
|
2020-05-11 23:05:24 -04:00
|
|
|
view := p.view(model)
|
2020-01-10 16:02:04 -05:00
|
|
|
|
2020-05-19 13:16:02 -04:00
|
|
|
p.mutex.Lock()
|
|
|
|
|
2020-01-10 16:02:04 -05:00
|
|
|
// We need to add carriage returns to ensure that the cursor travels to the
|
|
|
|
// start of a column after a newline
|
|
|
|
view = strings.Replace(view, "\n", "\r\n", -1)
|
|
|
|
|
2020-01-22 19:15:30 -05:00
|
|
|
if linesRendered > 0 {
|
2020-01-31 07:47:36 -05:00
|
|
|
termenv.ClearLines(linesRendered)
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
2020-04-17 14:35:54 -04:00
|
|
|
_, _ = io.WriteString(os.Stdout, view)
|
2020-05-19 13:16:02 -04:00
|
|
|
|
|
|
|
p.mutex.Unlock()
|
2020-01-22 19:15:30 -05:00
|
|
|
return strings.Count(view, "\r\n")
|
2020-01-10 16:02:04 -05:00
|
|
|
}
|
2020-01-25 21:16:41 -05:00
|
|
|
|
2020-02-10 12:44:04 -05:00
|
|
|
// AltScreen exits the altscreen. This is just a wrapper around the termenv
|
2020-02-01 21:02:12 -05:00
|
|
|
// function
|
|
|
|
func AltScreen() {
|
|
|
|
termenv.AltScreen()
|
|
|
|
}
|
|
|
|
|
2020-02-10 12:44:04 -05:00
|
|
|
// ExitAltScreen exits the altscreen. This is just a wrapper around the termenv
|
2020-02-01 21:02:12 -05:00
|
|
|
// function
|
|
|
|
func ExitAltScreen() {
|
|
|
|
termenv.ExitAltScreen()
|
|
|
|
}
|