mirror of
https://github.com/charmbracelet/gum.git
synced 2024-10-26 07:10:39 +03:00
fb6849ca16
* Add `--select-if-one` flag to `choose`/`filter`. * Remove accidental commit of other changes. * fix: use o.Options --------- Co-authored-by: Maas Lalani <maas@lalani.dev>
173 lines
4.4 KiB
Go
173 lines
4.4 KiB
Go
package choose
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/alecthomas/kong"
|
|
"github.com/charmbracelet/bubbles/paginator"
|
|
tea "github.com/charmbracelet/bubbletea"
|
|
"github.com/charmbracelet/lipgloss"
|
|
"github.com/mattn/go-isatty"
|
|
|
|
"github.com/charmbracelet/gum/ansi"
|
|
"github.com/charmbracelet/gum/internal/exit"
|
|
"github.com/charmbracelet/gum/internal/stdin"
|
|
"github.com/charmbracelet/gum/style"
|
|
)
|
|
|
|
var (
|
|
subduedStyle = lipgloss.NewStyle().Foreground(lipgloss.AdaptiveColor{Light: "#847A85", Dark: "#979797"})
|
|
verySubduedStyle = lipgloss.NewStyle().Foreground(lipgloss.AdaptiveColor{Light: "#DDDADA", Dark: "#3C3C3C"})
|
|
)
|
|
|
|
// Run provides a shell script interface for choosing between different through
|
|
// options.
|
|
func (o Options) Run() error {
|
|
if len(o.Options) == 0 {
|
|
input, _ := stdin.Read()
|
|
if input == "" {
|
|
return errors.New("no options provided, see `gum choose --help`")
|
|
}
|
|
o.Options = strings.Split(strings.TrimSuffix(input, "\n"), "\n")
|
|
}
|
|
|
|
if o.SelectIfOne && len(o.Options) == 1 {
|
|
print(o.Options[0])
|
|
return nil
|
|
}
|
|
|
|
// We don't need to display prefixes if we are only picking one option.
|
|
// Simply displaying the cursor is enough.
|
|
if o.Limit == 1 && !o.NoLimit {
|
|
o.SelectedPrefix = ""
|
|
o.UnselectedPrefix = ""
|
|
o.CursorPrefix = ""
|
|
}
|
|
|
|
// If we've set no limit then we can simply select as many options as there
|
|
// are so let's set the limit to the number of options.
|
|
if o.NoLimit {
|
|
o.Limit = len(o.Options) + 1
|
|
}
|
|
|
|
if len(o.Selected) > o.Limit {
|
|
return errors.New("number of selected options cannot be greater than the limit")
|
|
}
|
|
|
|
// Keep track of the selected items.
|
|
currentSelected := 0
|
|
// Check if selected items should be used.
|
|
hasSelectedItems := len(o.Selected) > 0
|
|
|
|
startingIndex := 0
|
|
currentOrder := 0
|
|
|
|
items := make([]item, len(o.Options))
|
|
|
|
for i, option := range o.Options {
|
|
var order int
|
|
// Check if the option should be selected.
|
|
isSelected := hasSelectedItems && currentSelected < o.Limit && arrayContains(o.Selected, option)
|
|
// If the option is selected then increment the current selected count.
|
|
if isSelected {
|
|
if o.Limit == 1 {
|
|
// When the user can choose only one option don't select the option but
|
|
// start with the cursor hovering over it.
|
|
startingIndex = i
|
|
isSelected = false
|
|
} else {
|
|
currentSelected++
|
|
order = currentOrder
|
|
currentOrder++
|
|
}
|
|
}
|
|
|
|
items[i] = item{text: option, selected: isSelected, order: order}
|
|
}
|
|
|
|
// Use the pagination model to display the current and total number of
|
|
// pages.
|
|
pager := paginator.New()
|
|
pager.SetTotalPages((len(items) + o.Height - 1) / o.Height)
|
|
pager.PerPage = o.Height
|
|
pager.Type = paginator.Dots
|
|
pager.ActiveDot = subduedStyle.Render("•")
|
|
pager.InactiveDot = verySubduedStyle.Render("•")
|
|
pager.KeyMap = paginator.KeyMap{}
|
|
pager.Page = startingIndex / o.Height
|
|
|
|
// Disable Keybindings since we will control it ourselves.
|
|
tm, err := tea.NewProgram(model{
|
|
index: startingIndex,
|
|
currentOrder: currentOrder,
|
|
height: o.Height,
|
|
cursor: o.Cursor,
|
|
header: o.Header,
|
|
selectedPrefix: o.SelectedPrefix,
|
|
unselectedPrefix: o.UnselectedPrefix,
|
|
cursorPrefix: o.CursorPrefix,
|
|
items: items,
|
|
limit: o.Limit,
|
|
paginator: pager,
|
|
cursorStyle: o.CursorStyle.ToLipgloss(),
|
|
headerStyle: o.HeaderStyle.ToLipgloss(),
|
|
itemStyle: o.ItemStyle.ToLipgloss(),
|
|
selectedItemStyle: o.SelectedItemStyle.ToLipgloss(),
|
|
numSelected: currentSelected,
|
|
hasTimeout: o.Timeout > 0,
|
|
timeout: o.Timeout,
|
|
}, tea.WithOutput(os.Stderr)).Run()
|
|
|
|
if err != nil {
|
|
return fmt.Errorf("failed to start tea program: %w", err)
|
|
}
|
|
|
|
m := tm.(model)
|
|
if m.aborted {
|
|
return exit.ErrAborted
|
|
}
|
|
|
|
if o.Ordered && o.Limit > 1 {
|
|
sort.Slice(m.items, func(i, j int) bool {
|
|
return m.items[i].order < m.items[j].order
|
|
})
|
|
}
|
|
|
|
var s strings.Builder
|
|
|
|
for _, item := range m.items {
|
|
if item.selected {
|
|
s.WriteString(item.text)
|
|
s.WriteRune('\n')
|
|
}
|
|
}
|
|
|
|
if isatty.IsTerminal(os.Stdout.Fd()) {
|
|
fmt.Print(s.String())
|
|
} else {
|
|
fmt.Print(ansi.Strip(s.String()))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// BeforeReset hook. Used to unclutter style flags.
|
|
func (o Options) BeforeReset(ctx *kong.Context) error {
|
|
style.HideFlags(ctx)
|
|
return nil
|
|
}
|
|
|
|
// Check if an array contains a value.
|
|
func arrayContains(strArray []string, value string) bool {
|
|
for _, str := range strArray {
|
|
if str == value {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|