forked from Mirrors/bubbletea
Open a TTY if input is not a TTY, unless the user has spec'd otherwise
This commit is contained in:
parent
4e2643f318
commit
0780601791
|
@ -0,0 +1,94 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
// An example of how to pipe in data to a Bubble Tea application. It's actually
|
||||||
|
// more of a proof that Bubble Tea will automatically listen for keystrokes
|
||||||
|
// when input is not a TTY, such as when data is piped or redirected in.
|
||||||
|
//
|
||||||
|
// In the case of this example we're listing for a single keystroke used to
|
||||||
|
// exit the program.
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/charmbracelet/bubbles/textinput"
|
||||||
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
stat, err := os.Stdin.Stat()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if stat.Mode()&os.ModeNamedPipe == 0 && stat.Size() == 0 {
|
||||||
|
fmt.Println("Try piping in some text.")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
reader := bufio.NewReader(os.Stdin)
|
||||||
|
var b strings.Builder
|
||||||
|
|
||||||
|
for {
|
||||||
|
r, _, err := reader.ReadRune()
|
||||||
|
if err != nil && err == io.EOF {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
_, err = b.WriteRune(r)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error getting input:", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
model := newModel(strings.TrimSpace(b.String()))
|
||||||
|
|
||||||
|
if err := tea.NewProgram(model).Start(); err != nil {
|
||||||
|
fmt.Println("Couldn't start program:", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type model struct {
|
||||||
|
userInput textinput.Model
|
||||||
|
}
|
||||||
|
|
||||||
|
func newModel(initialValue string) (m model) {
|
||||||
|
i := textinput.NewModel()
|
||||||
|
i.Prompt = ""
|
||||||
|
i.CursorColor = "63"
|
||||||
|
i.Width = 48
|
||||||
|
i.SetValue(initialValue)
|
||||||
|
i.CursorEnd()
|
||||||
|
i.Focus()
|
||||||
|
|
||||||
|
m.userInput = i
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m model) Init() tea.Cmd {
|
||||||
|
return textinput.Blink
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||||
|
if key, ok := msg.(tea.KeyMsg); ok {
|
||||||
|
switch key.Type {
|
||||||
|
case tea.KeyCtrlC, tea.KeyEscape, tea.KeyEnter:
|
||||||
|
return m, tea.Quit
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var cmd tea.Cmd
|
||||||
|
m.userInput, cmd = m.userInput.Update(msg)
|
||||||
|
return m, cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m model) View() string {
|
||||||
|
return fmt.Sprintf(
|
||||||
|
"\nYou piped in: %s\n\nPress ^C to exit",
|
||||||
|
m.userInput.View(),
|
||||||
|
)
|
||||||
|
}
|
41
tea.go
41
tea.go
|
@ -19,6 +19,7 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
|
"github.com/containerd/console"
|
||||||
isatty "github.com/mattn/go-isatty"
|
isatty "github.com/mattn/go-isatty"
|
||||||
te "github.com/muesli/termenv"
|
te "github.com/muesli/termenv"
|
||||||
"golang.org/x/crypto/ssh/terminal"
|
"golang.org/x/crypto/ssh/terminal"
|
||||||
|
@ -83,6 +84,7 @@ func WithOutput(output *os.File) ProgramOption {
|
||||||
func WithInput(input io.Reader) ProgramOption {
|
func WithInput(input io.Reader) ProgramOption {
|
||||||
return func(m *Program) {
|
return func(m *Program) {
|
||||||
m.input = input
|
m.input = input
|
||||||
|
m.inputStatus = customInput
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,6 +98,25 @@ func WithoutCatchPanics() ProgramOption {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// inputStatus indicates the current state of the input. By default, input is
|
||||||
|
// stdin, however we'll change this if input's not a TTY. The user can also set
|
||||||
|
// the input.
|
||||||
|
type inputStatus int
|
||||||
|
|
||||||
|
const (
|
||||||
|
defaultInput = iota // generally, this will be stdin
|
||||||
|
customInput // the user explicitly set the input
|
||||||
|
managedInput // we've opened a TTY for input
|
||||||
|
)
|
||||||
|
|
||||||
|
func (i inputStatus) String() string {
|
||||||
|
return [...]string{
|
||||||
|
"default input",
|
||||||
|
"custom input",
|
||||||
|
"managed input",
|
||||||
|
}[i]
|
||||||
|
}
|
||||||
|
|
||||||
// Program is a terminal user interface.
|
// Program is a terminal user interface.
|
||||||
type Program struct {
|
type Program struct {
|
||||||
initialModel Model
|
initialModel Model
|
||||||
|
@ -113,8 +134,15 @@ type Program struct {
|
||||||
// is on by default.
|
// is on by default.
|
||||||
CatchPanics bool
|
CatchPanics bool
|
||||||
|
|
||||||
|
inputStatus inputStatus
|
||||||
inputIsTTY bool
|
inputIsTTY bool
|
||||||
outputIsTTY bool
|
outputIsTTY bool
|
||||||
|
console console.Console
|
||||||
|
|
||||||
|
// 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.
|
||||||
|
windowsStdin *os.File
|
||||||
}
|
}
|
||||||
|
|
||||||
// Quit is a special command that tells the Bubble Tea program to exit.
|
// Quit is a special command that tells the Bubble Tea program to exit.
|
||||||
|
@ -195,6 +223,19 @@ func (p *Program) Start() error {
|
||||||
p.inputIsTTY = isatty.IsTerminal(f.Fd())
|
p.inputIsTTY = isatty.IsTerminal(f.Fd())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If input is not a terminal, and the user hasn't set a custom input, 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.
|
||||||
|
if !p.inputIsTTY && p.inputStatus != customInput {
|
||||||
|
f, err := openInputTTY()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
p.input = f
|
||||||
|
p.inputIsTTY = true
|
||||||
|
p.inputStatus = managedInput
|
||||||
|
}
|
||||||
|
|
||||||
// Listen for SIGINT. Note that in most cases ^C will not send an
|
// Listen for SIGINT. Note that in most cases ^C will not send an
|
||||||
// interrupt because the terminal will be in raw mode and thus capture
|
// interrupt because the terminal will be in raw mode and thus capture
|
||||||
// that keystroke and send it along to Program.Update. If input is not a
|
// that keystroke and send it along to Program.Update. If input is not a
|
||||||
|
|
36
tty.go
36
tty.go
|
@ -1,18 +1,22 @@
|
||||||
package tea
|
package tea
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/containerd/console"
|
"errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
var tty console.Console
|
var errInputIsNotAFile = errors.New("input is not a file")
|
||||||
|
|
||||||
func (p Program) initTerminal() error {
|
func (p *Program) initTerminal() error {
|
||||||
if p.outputIsTTY {
|
err := p.initInput()
|
||||||
tty = console.Current()
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if p.inputIsTTY {
|
if p.inputIsTTY {
|
||||||
err := tty.SetRaw()
|
if p.console == nil {
|
||||||
|
return errors.New("no console")
|
||||||
|
}
|
||||||
|
err = p.console.SetRaw()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -27,9 +31,21 @@ func (p Program) initTerminal() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p Program) restoreTerminal() error {
|
func (p Program) restoreTerminal() error {
|
||||||
if !p.outputIsTTY {
|
if p.outputIsTTY {
|
||||||
return nil
|
showCursor(p.output)
|
||||||
}
|
}
|
||||||
showCursor(p.output)
|
|
||||||
return tty.Reset()
|
if err := p.restoreInput(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Console will only be set if input is a TTY.
|
||||||
|
if p.console != nil {
|
||||||
|
err := p.console.Reset()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
54
tty_unix.go
54
tty_unix.go
|
@ -2,7 +2,59 @@
|
||||||
|
|
||||||
package tea
|
package tea
|
||||||
|
|
||||||
import "io"
|
import (
|
||||||
|
"errors"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/containerd/console"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (p *Program) initInput() error {
|
||||||
|
if !p.inputIsTTY {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// If input's a TTY this should always succeed.
|
||||||
|
f, ok := p.input.(*os.File)
|
||||||
|
if !ok {
|
||||||
|
return errInputIsNotAFile
|
||||||
|
}
|
||||||
|
|
||||||
|
c, err := console.ConsoleFromFile(f)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
p.console = c
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// On unix systems, RestoreInput closes any TTYs we opened for input. Note that
|
||||||
|
// we don't do this on Windows as it causes the prompt to not be drawn until the
|
||||||
|
// terminal receives a keypress rather than appearing promptly after the program
|
||||||
|
// exits.
|
||||||
|
func (p *Program) restoreInput() error {
|
||||||
|
if p.inputStatus == managedInput {
|
||||||
|
f, ok := p.input.(*os.File)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("could not close input")
|
||||||
|
}
|
||||||
|
err := f.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func openInputTTY() (*os.File, error) {
|
||||||
|
f, err := os.Open("/dev/tty")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return f, nil
|
||||||
|
}
|
||||||
|
|
||||||
// enableAnsiColors is only needed for Windows, so for other systems this is
|
// enableAnsiColors is only needed for Windows, so for other systems this is
|
||||||
// a no-op.
|
// a no-op.
|
||||||
|
|
|
@ -6,9 +6,55 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
|
"github.com/containerd/console"
|
||||||
"golang.org/x/sys/windows"
|
"golang.org/x/sys/windows"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func (p *Program) initInput() error {
|
||||||
|
if !p.inputIsTTY {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// If input's a TTY this should always succeed.
|
||||||
|
f, ok := p.input.(*os.File)
|
||||||
|
if !ok {
|
||||||
|
return errInputIsNotAFile
|
||||||
|
}
|
||||||
|
|
||||||
|
if p.inputStatus == managedInput {
|
||||||
|
// Save a reference to the current stdin then replace stdin with our
|
||||||
|
// input. We do this so we can hand input off to containerd/console to
|
||||||
|
// set raw mode, and do it in this fashion because the method
|
||||||
|
// console.ConsoleFromFile isn't supported on Windows.
|
||||||
|
p.windowsStdin = os.Stdin
|
||||||
|
os.Stdin = f
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note: this will panic if it fails.
|
||||||
|
c := console.Current()
|
||||||
|
p.console = c
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// restoreInput restores stdout in the event that we placed it aside to handle
|
||||||
|
// input with CONIN$, above.
|
||||||
|
func (p *Program) restoreInput() error {
|
||||||
|
if p.windowsStdin != nil {
|
||||||
|
os.Stdin = p.windowsStdin
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func openInputTTY() (*os.File, error) {
|
||||||
|
f, err := os.OpenFile("CONIN$", os.O_RDWR, 0644)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return f, nil
|
||||||
|
}
|
||||||
|
|
||||||
// enableAnsiColors enables support for ANSI color sequences in Windows
|
// enableAnsiColors enables support for ANSI color sequences in Windows
|
||||||
// default console. Note that this only works with Windows 10.
|
// default console. Note that this only works with Windows 10.
|
||||||
func enableAnsiColors(w io.Writer) {
|
func enableAnsiColors(w io.Writer) {
|
||||||
|
@ -20,6 +66,6 @@ func enableAnsiColors(w io.Writer) {
|
||||||
stdout := windows.Handle(f.Fd())
|
stdout := windows.Handle(f.Fd())
|
||||||
var originalMode uint32
|
var originalMode uint32
|
||||||
|
|
||||||
windows.GetConsoleMode(stdout, &originalMode)
|
_ = windows.GetConsoleMode(stdout, &originalMode)
|
||||||
windows.SetConsoleMode(stdout, originalMode|windows.ENABLE_VIRTUAL_TERMINAL_PROCESSING)
|
_ = windows.SetConsoleMode(stdout, originalMode|windows.ENABLE_VIRTUAL_TERMINAL_PROCESSING)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue