gum/confirm/confirm.go

122 lines
2.8 KiB
Go
Raw Normal View History

2022-07-26 21:43:26 +02:00
// Package confirm provides an interface to ask a user to confirm an action.
// The user is provided with an interface to choose an affirmative or negative
// answer, which is then reflected in the exit code for use in scripting.
//
// If the user selects the affirmative answer, the program exits with 0. If the
// user selects the negative answer, the program exits with 1.
//
// I.e. confirm if the user wants to delete a file
//
2022-08-05 00:45:19 +02:00
// $ gum confirm "Are you sure?" && rm file.txt
2022-07-26 21:43:26 +02:00
package confirm
import (
"time"
"github.com/charmbracelet/gum/timeout"
2022-07-26 21:43:26 +02:00
tea "github.com/charmbracelet/bubbletea"
"github.com/charmbracelet/lipgloss"
)
type model struct {
prompt string
affirmative string
negative string
quitting bool
2022-12-13 21:46:43 +01:00
aborted bool
hasTimeout bool
timeout time.Duration
confirmation bool
2022-07-26 21:43:26 +02:00
defaultSelection bool
2022-07-26 21:43:26 +02:00
// styles
promptStyle lipgloss.Style
selectedStyle lipgloss.Style
unselectedStyle lipgloss.Style
}
func (m model) Init() tea.Cmd {
return timeout.Init(m.timeout, m.defaultSelection)
}
2022-07-26 21:43:26 +02:00
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) {
case tea.WindowSizeMsg:
return m, nil
case tea.KeyMsg:
switch msg.String() {
case "ctrl+c":
m.confirmation = false
2022-12-13 21:46:43 +01:00
m.aborted = true
fallthrough
case "esc":
m.confirmation = false
2022-12-13 21:46:43 +01:00
m.quitting = true
return m, tea.Quit
case "q", "n", "N":
m.confirmation = false
m.quitting = true
return m, tea.Quit
case "left", "h", "ctrl+p", "tab",
"right", "l", "ctrl+n", "shift+tab":
if m.negative == "" {
break
}
m.confirmation = !m.confirmation
2022-07-26 21:43:26 +02:00
case "enter":
m.quitting = true
return m, tea.Quit
case "y", "Y":
m.quitting = true
m.confirmation = true
return m, tea.Quit
}
case timeout.TickTimeoutMsg:
if msg.TimeoutValue <= 0 {
2022-07-26 21:43:26 +02:00
m.quitting = true
m.confirmation = m.defaultSelection
2022-07-26 21:43:26 +02:00
return m, tea.Quit
}
m.timeout = msg.TimeoutValue
return m, timeout.Tick(msg.TimeoutValue, msg.Data)
2022-07-26 21:43:26 +02:00
}
return m, nil
}
func (m model) View() string {
if m.quitting {
return ""
}
var aff, neg, timeoutStrYes, timeoutStrNo string
timeoutStrNo = ""
timeoutStrYes = ""
if m.hasTimeout {
if m.defaultSelection {
timeoutStrYes = timeout.Str(m.timeout)
} else {
timeoutStrNo = timeout.Str(m.timeout)
}
}
if m.confirmation {
aff = m.selectedStyle.Render(m.affirmative + timeoutStrYes)
neg = m.unselectedStyle.Render(m.negative + timeoutStrNo)
2022-07-26 21:43:26 +02:00
} else {
aff = m.unselectedStyle.Render(m.affirmative + timeoutStrYes)
neg = m.selectedStyle.Render(m.negative + timeoutStrNo)
2022-07-26 21:43:26 +02:00
}
// If the option is intentionally empty, do not show it.
if m.negative == "" {
neg = ""
}
return lipgloss.JoinVertical(lipgloss.Center, m.promptStyle.Render(m.prompt), lipgloss.JoinHorizontal(lipgloss.Left, aff, neg))
2022-07-26 21:43:26 +02:00
}