Add commands for toggling the altscreen (#62)

* Add commands EnterAltScreen and ExitAltScreen for toggling the altscreen
* Add altscreen toggling example
* ignore .idea

Co-authored-by: Christian Rocha <christian@rocha.is>
This commit is contained in:
kiyon 2021-03-09 01:48:34 +08:00 committed by GitHub
parent b3f62af8b5
commit 86b93a31c0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 113 additions and 1 deletions

1
.gitignore vendored
View File

@ -12,3 +12,4 @@ examples/textinputs/textinputs
examples/views/views examples/views/views
tutorials/basics/basics tutorials/basics/basics
tutorials/commands/commands tutorials/commands/commands
.idea

View File

@ -0,0 +1,76 @@
package main
import (
"fmt"
"os"
tea "github.com/charmbracelet/bubbletea"
"github.com/muesli/termenv"
)
var (
color = termenv.ColorProfile().Color
keyword = termenv.Style{}.Foreground(color("204")).Background(color("235")).Styled
)
type model struct {
altscreen bool
quitting bool
}
func (m model) Init() tea.Cmd {
return nil
}
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) {
case tea.KeyMsg:
switch msg.String() {
case "q", "ctrl+c", "esc":
m.quitting = true
return m, tea.Quit
case " ":
var cmd tea.Cmd
if m.altscreen {
cmd = tea.ExitAltScreen
} else {
cmd = tea.EnterAltScreen
}
m.altscreen = !m.altscreen
return m, cmd
}
}
return m, nil
}
func (m model) View() string {
if m.quitting {
return "Bye!\n"
}
const (
altscreenMode = " altscreen mode "
inlineMode = " inline mode "
)
var mode, otherMode string
if m.altscreen {
mode = altscreenMode
otherMode = inlineMode
} else {
mode = inlineMode
otherMode = altscreenMode
}
return fmt.Sprintf(
"\n You're in %s. Press %s to swich to %s.\n\n To exit press %s.\n",
keyword(mode), keyword(" space "), keyword(otherMode), keyword(" q "),
)
}
func main() {
if err := tea.NewProgram(model{}).Start(); err != nil {
fmt.Println("Error running program:", err)
os.Exit(1)
}
}

View File

@ -18,9 +18,9 @@ type tickMsg time.Time
func main() { func main() {
p := tea.NewProgram(model(5)) p := tea.NewProgram(model(5))
// Bubble Tea will automatically exit the alternate screen buffer.
p.EnterAltScreen() p.EnterAltScreen()
err := p.Start() err := p.Start()
p.ExitAltScreen()
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)

35
tea.go
View File

@ -168,6 +168,26 @@ func Quit() Msg {
// send a quitMsg with Quit. // send a quitMsg with Quit.
type quitMsg struct{} type quitMsg struct{}
// EnterAltScreen is a special command that tells the Bubble Tea program to enter
// alternate screen buffer.
func EnterAltScreen() Msg {
return enterAltScreenMsg{}
}
// enterAltScreenMsg in an internal message signals that the program should enter
// alternate screen buffer. You can send a enterAltScreenMsg with EnterAltScreen.
type enterAltScreenMsg struct{}
// ExitAltScreen is a special command that tells the Bubble Tea program to exit
// alternate screen buffer.
func ExitAltScreen() Msg {
return exitAltScreenMsg{}
}
// exitAltScreenMsg in an internal message signals that the program should exit
// alternate screen buffer. You can send a exitAltScreenMsg with ExitAltScreen.
type exitAltScreenMsg struct{}
// batchMsg is the internal message used to perform a bunch of commands. You // batchMsg is the internal message used to perform a bunch of commands. You
// can send a batchMsg with Batch. // can send a batchMsg with Batch.
type batchMsg []Cmd type batchMsg []Cmd
@ -366,9 +386,14 @@ func (p *Program) Start() error {
// Handle special messages // Handle special messages
switch msg.(type) { switch msg.(type) {
case quitMsg: case quitMsg:
p.ExitAltScreen()
p.renderer.stop() p.renderer.stop()
close(done) close(done)
return nil return nil
case enterAltScreenMsg:
p.EnterAltScreen()
case exitAltScreenMsg:
p.ExitAltScreen()
case hideCursorMsg: case hideCursorMsg:
hideCursor(p.output) hideCursor(p.output)
} }
@ -399,6 +424,11 @@ func (p *Program) Start() error {
func (p *Program) EnterAltScreen() { func (p *Program) EnterAltScreen() {
p.mtx.Lock() p.mtx.Lock()
defer p.mtx.Unlock() defer p.mtx.Unlock()
if p.altScreenActive {
return
}
fmt.Fprintf(p.output, te.CSI+te.AltScreenSeq) fmt.Fprintf(p.output, te.CSI+te.AltScreenSeq)
moveCursor(p.output, 0, 0) moveCursor(p.output, 0, 0)
@ -412,6 +442,11 @@ func (p *Program) EnterAltScreen() {
func (p *Program) ExitAltScreen() { func (p *Program) ExitAltScreen() {
p.mtx.Lock() p.mtx.Lock()
defer p.mtx.Unlock() defer p.mtx.Unlock()
if !p.altScreenActive {
return
}
fmt.Fprintf(p.output, te.CSI+te.ExitAltScreenSeq) fmt.Fprintf(p.output, te.CSI+te.ExitAltScreenSeq)
p.altScreenActive = false p.altScreenActive = false