bubbletea/tea.go

203 lines
4.3 KiB
Go
Raw Normal View History

2020-01-10 16:02:04 -05:00
package tea
import (
"errors"
2020-01-10 16:02:04 -05:00
"io"
"log"
"log/syslog"
2020-01-10 16:02:04 -05:00
"strings"
"github.com/pkg/term"
)
// Msg represents an action. It's used by Update to update the UI.
type Msg interface{}
// Model contains the updatable data for an application
2020-01-10 16:02:04 -05:00
type Model interface{}
// Cmd is an IO operation. If it's nil it's considered a no-op.
type Cmd func(Model) Msg
2020-01-10 16:02:04 -05:00
// Sub is an event subscription. If it returns nil it's considered a no-op.
type Sub func(Model) Msg
// 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 {
init Init
update Update
view View
subscriptions []Sub
rw io.ReadWriter
2020-01-10 16:02:04 -05:00
}
// ErrMsg is just a regular message containing an error. We handle it in Update
// just like a regular message by case switching. Of course, the developer
// could also define her own errors as well.
type ErrMsg struct {
error
}
func (e ErrMsg) String() string {
return e.Error()
}
// NewErrMsg is a convenience function for creating a generic ErrMsg
func NewErrMsg(s string) ErrMsg {
return ErrMsg{errors.New(s)}
}
// NewErrMsgFromErr is a convenience function for creating an ErrMsg from an
// existing error
func NewErrMsgFromErr(e error) ErrMsg {
return ErrMsg{e}
}
// Quit is a command that tells the program to exit
func Quit(_ Model) Msg {
2020-01-10 16:02:04 -05:00
return quitMsg{}
}
// Signals that the program should quit
type quitMsg struct{}
// NewProgram creates a new Program
func NewProgram(init Init, update Update, view View, subs []Sub) *Program {
2020-01-10 16:02:04 -05:00
return &Program{
init: init,
update: update,
view: view,
subscriptions: subs,
2020-01-10 16:02:04 -05:00
}
}
// Start initializes the program
func (p *Program) Start() error {
var (
2020-01-22 19:15:30 -05:00
model Model
cmd Cmd
cmds = make(chan Cmd)
msgs = make(chan Msg)
done = make(chan struct{})
linesRendered int
2020-01-10 16:02:04 -05:00
)
tty, err := term.Open("/dev/tty")
if err != nil {
return err
}
p.rw = tty
tty.SetRaw()
hideCursor()
2020-01-10 16:02:04 -05:00
defer func() {
showCursor()
tty.Restore()
}()
// Initialize program
model, cmd = p.init()
if cmd != nil {
go func() {
cmds <- cmd
}()
}
2020-01-10 16:02:04 -05:00
// Render initial view
2020-01-22 19:15:30 -05:00
linesRendered = p.render(model, linesRendered)
2020-01-10 16:02:04 -05:00
// Subscribe to user input. We could move this out of here and offer it
// as a subscription, but it blocks nicely and seems to be a common enough
// need that we're enabling it by default.
2020-01-10 16:02:04 -05:00
go func() {
for {
msg, _ := ReadKey(p.rw)
msgs <- KeyMsg(msg)
2020-01-10 16:02:04 -05:00
}
}()
// Process subscriptions
go func() {
if len(p.subscriptions) > 0 {
for _, sub := range p.subscriptions {
go func(s Sub) {
for {
msgs <- s(model)
}
}(sub)
}
}
}()
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(model)
2020-01-10 16:02:04 -05:00
}()
}
}
}
}()
// Handle updates and draw
for {
select {
case msg := <-msgs:
if _, ok := msg.(quitMsg); ok {
close(done)
return nil
}
model, cmd = p.update(msg, model)
cmds <- cmd // process command (if any)
2020-01-22 19:15:30 -05:00
linesRendered = p.render(model, linesRendered)
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 {
view := p.view(model) + "\n"
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 {
clearLines(linesRendered)
2020-01-10 16:02:04 -05:00
}
io.WriteString(p.rw, view)
2020-01-22 19:15:30 -05:00
return strings.Count(view, "\r\n")
2020-01-10 16:02:04 -05:00
}
2020-01-18 10:34:28 -05:00
// UseSysLog sets up logging to log the system log. This becomes helpful when
// debugging since we can't easily print to the terminal since our TUI is
// occupying it!
//
// On macOS this is a just a matter of: tail -f /var/log/system.log
func UseSysLog(programName string) error {
l, err := syslog.New(syslog.LOG_NOTICE, programName)
if err != nil {
return err
}
log.SetOutput(l)
return nil
}