bubbletea/examples/textinputs/main.go

176 lines
3.7 KiB
Go
Raw Normal View History

package main
// A simple example demonstrating the use of multiple text input components
// from the Bubbles component library.
import (
"fmt"
"os"
"github.com/charmbracelet/bubbles/textinput"
tea "github.com/charmbracelet/bubbletea"
"github.com/charmbracelet/lipgloss"
)
var (
focusedStyle = lipgloss.NewStyle().Foreground(lipgloss.Color("205"))
blurredButtonStyle = lipgloss.NewStyle().Foreground(lipgloss.Color("240"))
noStyle = lipgloss.NewStyle()
focusedSubmitButton = "[ " + focusedStyle.Render("Submit") + " ]"
blurredSubmitButton = "[ " + blurredButtonStyle.Render("Submit") + " ]"
)
func main() {
if err := tea.NewProgram(initialModel()).Start(); err != nil {
fmt.Printf("could not start program: %s\n", err)
os.Exit(1)
}
}
2020-07-30 12:40:37 -04:00
type model struct {
index int
2020-10-30 03:36:52 -04:00
nameInput textinput.Model
emailInput textinput.Model
passwordInput textinput.Model
submitButton string
}
func initialModel() model {
2020-10-30 03:36:52 -04:00
name := textinput.NewModel()
name.Placeholder = "Nickname"
name.Focus()
name.PromptStyle = focusedStyle
name.TextStyle = focusedStyle
name.CharLimit = 32
2020-10-30 03:36:52 -04:00
email := textinput.NewModel()
email.Placeholder = "Email"
email.CharLimit = 64
2020-10-30 03:36:52 -04:00
password := textinput.NewModel()
password.Placeholder = "Password"
password.EchoMode = textinput.EchoPassword
password.EchoCharacter = '•'
password.CharLimit = 32
return model{0, name, email, password, blurredSubmitButton}
}
func (m model) Init() tea.Cmd {
return textinput.Blink
}
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
var cmd tea.Cmd
switch msg := msg.(type) {
case tea.KeyMsg:
switch msg.String() {
2021-05-01 09:28:58 -04:00
case "ctrl+c", "esc":
return m, tea.Quit
// Cycle between inputs
2020-07-30 12:40:37 -04:00
case "tab", "shift+tab", "enter", "up", "down":
2020-10-30 03:36:52 -04:00
inputs := []textinput.Model{
m.nameInput,
m.emailInput,
m.passwordInput,
}
s := msg.String()
// Did the user press enter while the submit button was focused?
// If so, exit.
if s == "enter" && m.index == len(inputs) {
return m, tea.Quit
}
// Cycle indexes
if s == "up" || s == "shift+tab" {
m.index--
} else {
m.index++
}
if m.index > len(inputs) {
m.index = 0
} else if m.index < 0 {
m.index = len(inputs)
}
for i := 0; i <= len(inputs)-1; i++ {
if i == m.index {
2020-07-30 12:40:37 -04:00
// Set focused state
inputs[i].Focus()
inputs[i].PromptStyle = focusedStyle
inputs[i].TextStyle = focusedStyle
continue
}
2020-07-30 12:40:37 -04:00
// Remove focused state
inputs[i].Blur()
inputs[i].PromptStyle = noStyle
inputs[i].TextStyle = noStyle
}
m.nameInput = inputs[0]
m.emailInput = inputs[1]
m.passwordInput = inputs[2]
if m.index == len(inputs) {
m.submitButton = focusedSubmitButton
} else {
m.submitButton = blurredSubmitButton
}
return m, nil
}
}
2020-07-30 12:40:37 -04:00
// Handle character input and blinks
m, cmd = updateInputs(msg, m)
return m, cmd
}
2020-07-30 12:40:37 -04:00
// Pass messages and models through to text input components. Only text inputs
// with Focus() set will respond, so it's safe to simply update all of them
// here without any further logic.
func updateInputs(msg tea.Msg, m model) (model, tea.Cmd) {
var (
cmd tea.Cmd
cmds []tea.Cmd
)
2020-07-30 12:40:37 -04:00
m.nameInput, cmd = m.nameInput.Update(msg)
cmds = append(cmds, cmd)
2020-07-30 12:40:37 -04:00
m.emailInput, cmd = m.emailInput.Update(msg)
cmds = append(cmds, cmd)
2020-07-30 12:40:37 -04:00
m.passwordInput, cmd = m.passwordInput.Update(msg)
cmds = append(cmds, cmd)
2020-07-30 12:40:37 -04:00
return m, tea.Batch(cmds...)
}
func (m model) View() string {
s := "\n"
inputs := []string{
m.nameInput.View(),
m.emailInput.View(),
m.passwordInput.View(),
}
for i := 0; i < len(inputs); i++ {
s += inputs[i]
if i < len(inputs)-1 {
s += "\n"
}
}
s += "\n\n" + m.submitButton + "\n"
return s
}