mirror of
https://github.com/Maks1mS/bubbles.git
synced 2024-12-23 22:32:59 +03:00
Add support for multiple stopwatches
In short, stopwatches will now ignore messages sent by other stopwatches.
This commit is contained in:
parent
8d3cfdf380
commit
86e0c53e88
@ -2,13 +2,35 @@
|
||||
package stopwatch
|
||||
|
||||
import (
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
tea "github.com/charmbracelet/bubbletea"
|
||||
)
|
||||
|
||||
var (
|
||||
lastID int
|
||||
idMtx sync.Mutex
|
||||
)
|
||||
|
||||
func nextID() int {
|
||||
idMtx.Lock()
|
||||
defer idMtx.Unlock()
|
||||
lastID++
|
||||
return lastID
|
||||
}
|
||||
|
||||
// TickMsg is a message that is sent on every timer tick.
|
||||
type TickMsg struct{}
|
||||
type TickMsg struct {
|
||||
// ID is the identifier of the stopwatch that send the message. This makes
|
||||
// it possible to determine which stopwatch a tick belongs to when there
|
||||
// are multiple stopwatches running.
|
||||
//
|
||||
// Note, however, that a stopwatch will reject ticks from other
|
||||
// stopwatches, so it's safe to flow all TickMsgs through all stopwatches
|
||||
// and hvae them still behave appropriately.
|
||||
ID int
|
||||
}
|
||||
|
||||
type startStopMsg struct {
|
||||
running bool
|
||||
@ -16,20 +38,22 @@ type startStopMsg struct {
|
||||
|
||||
type resetMsg struct{}
|
||||
|
||||
// Model of the timer component.
|
||||
// Model for the stopwatch component.
|
||||
type Model struct {
|
||||
d time.Duration
|
||||
|
||||
id int
|
||||
running bool
|
||||
|
||||
// How long to wait before every tick. Defaults to 1 second.
|
||||
Interval time.Duration
|
||||
}
|
||||
|
||||
// NewWithInterval creates a new stopwatch with the given timeout and tick interval.
|
||||
// NewWithInterval creates a new stopwatch with the given timeout and tick
|
||||
// interval.
|
||||
func NewWithInterval(interval time.Duration) Model {
|
||||
return Model{
|
||||
Interval: interval,
|
||||
id: nextID(),
|
||||
}
|
||||
}
|
||||
|
||||
@ -38,6 +62,11 @@ func New() Model {
|
||||
return NewWithInterval(time.Second)
|
||||
}
|
||||
|
||||
// ID returns the unique ID of the model.
|
||||
func (m Model) ID() int {
|
||||
return m.id
|
||||
}
|
||||
|
||||
// Init starts the stopwatch..
|
||||
func (m Model) Init() tea.Cmd {
|
||||
return m.Start()
|
||||
@ -47,7 +76,7 @@ func (m Model) Init() tea.Cmd {
|
||||
func (m Model) Start() tea.Cmd {
|
||||
return tea.Batch(func() tea.Msg {
|
||||
return startStopMsg{true}
|
||||
}, tick(m.Interval))
|
||||
}, tick(m.id, m.Interval))
|
||||
}
|
||||
|
||||
// Stop stops the stopwatch.
|
||||
@ -85,11 +114,11 @@ func (m Model) Update(msg tea.Msg) (Model, tea.Cmd) {
|
||||
case resetMsg:
|
||||
m.d = 0
|
||||
case TickMsg:
|
||||
if !m.running {
|
||||
if !m.running || (msg.ID != 0 && msg.ID != m.id) {
|
||||
break
|
||||
}
|
||||
m.d += m.Interval
|
||||
return m, tick(m.Interval)
|
||||
return m, tick(m.id, m.Interval)
|
||||
}
|
||||
|
||||
return m, nil
|
||||
@ -105,8 +134,8 @@ func (m Model) View() string {
|
||||
return m.d.String()
|
||||
}
|
||||
|
||||
func tick(d time.Duration) tea.Cmd {
|
||||
func tick(id int, d time.Duration) tea.Cmd {
|
||||
return tea.Tick(d, func(_ time.Time) tea.Msg {
|
||||
return TickMsg{}
|
||||
return TickMsg{ID: id}
|
||||
})
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user