forked from Mirrors/bubbletea
Rename 'countdown' example to 'timer' and use timer Bubble
This commit is contained in:
parent
99ba63fe8a
commit
03d1aa6c08
|
@ -1,69 +0,0 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
"time"
|
||||
|
||||
tea "github.com/charmbracelet/bubbletea"
|
||||
)
|
||||
|
||||
var (
|
||||
duration = time.Second * 10
|
||||
interval = time.Millisecond
|
||||
)
|
||||
|
||||
func main() {
|
||||
m := model{
|
||||
timeout: time.Now().Add(duration),
|
||||
}
|
||||
|
||||
if err := tea.NewProgram(m).Start(); err != nil {
|
||||
fmt.Println("Oh no, it didn't work:", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
}
|
||||
|
||||
type tickMsg time.Time
|
||||
|
||||
type model struct {
|
||||
timeout time.Time
|
||||
lastTick time.Time
|
||||
}
|
||||
|
||||
func (m model) Init() tea.Cmd {
|
||||
return tick()
|
||||
}
|
||||
|
||||
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||
switch msg := msg.(type) {
|
||||
case tea.KeyMsg:
|
||||
switch msg.String() {
|
||||
case "ctrl+c", "q", "esc":
|
||||
return m, tea.Quit
|
||||
}
|
||||
|
||||
case tickMsg:
|
||||
t := time.Time(msg)
|
||||
if t.After(m.timeout) {
|
||||
return m, tea.Quit
|
||||
}
|
||||
m.lastTick = t
|
||||
return m, tick()
|
||||
}
|
||||
|
||||
return m, nil
|
||||
}
|
||||
|
||||
func (m model) View() string {
|
||||
t := m.timeout.Sub(m.lastTick).Milliseconds()
|
||||
secs := t / 1000
|
||||
millis := t % 1000 / 10
|
||||
return fmt.Sprintf("This program will quit in %02d:%02d\n", secs, millis)
|
||||
}
|
||||
|
||||
func tick() tea.Cmd {
|
||||
return tea.Tick(time.Duration(interval), func(t time.Time) tea.Msg {
|
||||
return tickMsg(t)
|
||||
})
|
||||
}
|
|
@ -0,0 +1,122 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
"time"
|
||||
|
||||
"github.com/charmbracelet/bubbles/help"
|
||||
"github.com/charmbracelet/bubbles/key"
|
||||
"github.com/charmbracelet/bubbles/timer"
|
||||
tea "github.com/charmbracelet/bubbletea"
|
||||
)
|
||||
|
||||
const timeout = time.Second * 5
|
||||
|
||||
type model struct {
|
||||
timer timer.Model
|
||||
keymap keymap
|
||||
help help.Model
|
||||
quitting bool
|
||||
}
|
||||
|
||||
type keymap struct {
|
||||
start key.Binding
|
||||
stop key.Binding
|
||||
reset key.Binding
|
||||
quit key.Binding
|
||||
}
|
||||
|
||||
func (m model) Init() tea.Cmd {
|
||||
return m.timer.Init()
|
||||
}
|
||||
|
||||
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||
switch msg := msg.(type) {
|
||||
case timer.TickMsg:
|
||||
var cmd tea.Cmd
|
||||
m.timer, cmd = m.timer.Update(msg)
|
||||
return m, cmd
|
||||
|
||||
case timer.StartStopMsg:
|
||||
var cmd tea.Cmd
|
||||
m.timer, cmd = m.timer.Update(msg)
|
||||
m.keymap.stop.SetEnabled(m.timer.Running())
|
||||
m.keymap.start.SetEnabled(!m.timer.Running())
|
||||
return m, cmd
|
||||
|
||||
case timer.TimeoutMsg:
|
||||
m.quitting = true
|
||||
return m, tea.Quit
|
||||
|
||||
case tea.KeyMsg:
|
||||
switch {
|
||||
case key.Matches(msg, m.keymap.quit):
|
||||
m.quitting = true
|
||||
return m, tea.Quit
|
||||
case key.Matches(msg, m.keymap.reset):
|
||||
m.timer.Timeout = timeout
|
||||
case key.Matches(msg, m.keymap.start, m.keymap.stop):
|
||||
return m, m.timer.Toggle()
|
||||
}
|
||||
}
|
||||
|
||||
return m, nil
|
||||
}
|
||||
|
||||
func (m model) helpView() string {
|
||||
return "\n" + m.help.ShortHelpView([]key.Binding{
|
||||
m.keymap.start,
|
||||
m.keymap.stop,
|
||||
m.keymap.reset,
|
||||
m.keymap.quit,
|
||||
})
|
||||
}
|
||||
|
||||
func (m model) View() string {
|
||||
// For a more detailed timer view you could read m.timer.Timeout to get
|
||||
// the remaining time as a time.Duration and skip calling m.timer.View()
|
||||
// entirely.
|
||||
s := m.timer.View()
|
||||
|
||||
if m.timer.Timedout() {
|
||||
s = "All done!"
|
||||
}
|
||||
s += "\n"
|
||||
if !m.quitting {
|
||||
s = "Exiting in " + s
|
||||
s += m.helpView()
|
||||
}
|
||||
return s
|
||||
}
|
||||
|
||||
func main() {
|
||||
m := model{
|
||||
timer: timer.NewWithInterval(timeout, time.Millisecond),
|
||||
keymap: keymap{
|
||||
start: key.NewBinding(
|
||||
key.WithKeys("s"),
|
||||
key.WithHelp("s", "start"),
|
||||
),
|
||||
stop: key.NewBinding(
|
||||
key.WithKeys("s"),
|
||||
key.WithHelp("s", "stop"),
|
||||
),
|
||||
reset: key.NewBinding(
|
||||
key.WithKeys("r"),
|
||||
key.WithHelp("r", "reset"),
|
||||
),
|
||||
quit: key.NewBinding(
|
||||
key.WithKeys("q", "ctrl+c"),
|
||||
key.WithHelp("q", "quit"),
|
||||
),
|
||||
},
|
||||
help: help.NewModel(),
|
||||
}
|
||||
m.keymap.start.SetEnabled(false)
|
||||
|
||||
if err := tea.NewProgram(m).Start(); err != nil {
|
||||
fmt.Println("Uh oh, we encountered an error:", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue