bubbletea/tty.go

92 lines
1.6 KiB
Go
Raw Normal View History

2020-10-13 12:52:30 -04:00
package tea
import (
"errors"
"io"
"time"
"github.com/muesli/cancelreader"
)
func (p *Program) initTerminal() error {
err := p.initInput()
if err != nil {
return err
2021-01-11 16:33:14 -05:00
}
if p.console != nil {
err = p.console.SetRaw()
2021-01-11 16:33:14 -05:00
if err != nil {
return err
}
}
p.renderer.hideCursor()
2020-10-13 12:52:30 -04:00
return nil
}
// restoreTerminalState restores the terminal to the state prior to running the
// Bubble Tea program.
func (p *Program) restoreTerminalState() error {
if p.renderer != nil {
p.renderer.showCursor()
}
if p.console != nil {
err := p.console.Reset()
if err != nil {
return err
}
2021-01-11 16:33:14 -05:00
}
return p.restoreInput()
2020-10-13 12:52:30 -04:00
}
// initCancelReader (re)commences reading inputs.
func (p *Program) initCancelReader() error {
var err error
p.cancelReader, err = cancelreader.NewReader(p.input)
if err != nil {
return err
}
p.readLoopDone = make(chan struct{})
go p.readLoop()
return nil
}
func (p *Program) readLoop() {
defer close(p.readLoopDone)
for {
if p.ctx.Err() != nil {
return
}
msgs, err := readInputs(p.cancelReader)
if err != nil {
if !errors.Is(err, io.EOF) && !errors.Is(err, cancelreader.ErrCanceled) {
p.errs <- err
}
return
}
for _, msg := range msgs {
p.msgs <- msg
}
}
}
// waitForReadLoop waits for the cancelReader to finish its read loop.
func (p *Program) waitForReadLoop() {
select {
case <-p.readLoopDone:
case <-time.After(500 * time.Millisecond):
// 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.
}
}