forked from Mirrors/bubbletea
140 lines
2.9 KiB
Go
140 lines
2.9 KiB
Go
package main
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"log"
|
|
"math/rand"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/charmbracelet/bubbles/spinner"
|
|
tea "github.com/charmbracelet/bubbletea"
|
|
"github.com/mattn/go-isatty"
|
|
"github.com/muesli/reflow/indent"
|
|
"github.com/muesli/termenv"
|
|
)
|
|
|
|
var (
|
|
color = termenv.ColorProfile().Color
|
|
)
|
|
|
|
func main() {
|
|
rand.Seed(time.Now().UTC().UnixNano())
|
|
|
|
var (
|
|
daemonMode bool
|
|
showHelp bool
|
|
opts []tea.ProgramOption
|
|
)
|
|
|
|
flag.BoolVar(&daemonMode, "d", false, "run as a daemon")
|
|
flag.BoolVar(&showHelp, "h", false, "show help")
|
|
flag.Parse()
|
|
|
|
if showHelp {
|
|
flag.Usage()
|
|
os.Exit(0)
|
|
}
|
|
|
|
if daemonMode || !isatty.IsTerminal(os.Stdout.Fd()) {
|
|
// If we're in daemon mode don't render the TUI
|
|
opts = []tea.ProgramOption{tea.WithoutRenderer()}
|
|
} else {
|
|
// If we're in TUI mode, discard log output
|
|
log.SetOutput(ioutil.Discard)
|
|
}
|
|
|
|
p := tea.NewProgram(newModel(), opts...)
|
|
if err := p.Start(); err != nil {
|
|
fmt.Println("Error starting Bubble Tea program:", err)
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
type result struct {
|
|
duration time.Duration
|
|
emoji string
|
|
}
|
|
|
|
type model struct {
|
|
spinner spinner.Model
|
|
results []result
|
|
quitting bool
|
|
}
|
|
|
|
func newModel() model {
|
|
const showLastResults = 5
|
|
|
|
return model{
|
|
spinner: spinner.NewModel(),
|
|
results: make([]result, showLastResults),
|
|
}
|
|
}
|
|
|
|
func (m model) Init() tea.Cmd {
|
|
log.Println("Starting work...")
|
|
return tea.Batch(
|
|
spinner.Tick,
|
|
runPretendProcess,
|
|
)
|
|
}
|
|
|
|
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|
switch msg := msg.(type) {
|
|
case tea.KeyMsg:
|
|
m.quitting = true
|
|
return m, tea.Quit
|
|
case spinner.TickMsg:
|
|
var cmd tea.Cmd
|
|
m.spinner, cmd = m.spinner.Update(msg)
|
|
return m, cmd
|
|
case processFinishedMsg:
|
|
d := time.Duration(msg)
|
|
res := result{emoji: randomEmoji(), duration: d}
|
|
log.Printf("%s Job finished in %s", res.emoji, res.duration)
|
|
m.results = append(m.results[1:], res)
|
|
return m, runPretendProcess
|
|
default:
|
|
return m, nil
|
|
}
|
|
}
|
|
|
|
func (m model) View() string {
|
|
s := "\n" +
|
|
termenv.String(m.spinner.View()).Foreground(color("206")).String() +
|
|
" Doing some work...\n\n"
|
|
|
|
for _, res := range m.results {
|
|
if res.duration == 0 {
|
|
s += "........................\n"
|
|
} else {
|
|
s += fmt.Sprintf("%s Job finished in %s\n", res.emoji, res.duration)
|
|
}
|
|
}
|
|
|
|
s += termenv.String("\nPress any key to exit\n").Foreground(color("240")).String()
|
|
|
|
if m.quitting {
|
|
s += "\n"
|
|
}
|
|
|
|
return indent.String(s, 1)
|
|
}
|
|
|
|
// processFinishedMsg is send when a pretend process completes.
|
|
type processFinishedMsg time.Duration
|
|
|
|
// pretendProcess simulates a long-running process.
|
|
func runPretendProcess() tea.Msg {
|
|
pause := time.Duration(rand.Int63n(899)+100) * time.Millisecond
|
|
time.Sleep(pause)
|
|
return processFinishedMsg(pause)
|
|
}
|
|
|
|
func randomEmoji() string {
|
|
emojis := []rune("🍦🧋🍡🤠👾😭🦊🐯🦆🥨🎏🍔🍒🍥🎮📦🦁🐶🐸🍕🥐🧲🚒🥇🏆🌽")
|
|
return string(emojis[rand.Intn(len(emojis))])
|
|
}
|