mirror of
https://github.com/maaslalani/confetty.git
synced 2024-11-24 16:46:43 +00:00
123 lines
3.1 KiB
Go
123 lines
3.1 KiB
Go
package fireworks
|
|
|
|
import (
|
|
"math"
|
|
"math/rand"
|
|
"time"
|
|
|
|
"github.com/maaslalani/confetty/array"
|
|
"github.com/maaslalani/confetty/simulation"
|
|
|
|
tea "github.com/charmbracelet/bubbletea"
|
|
"github.com/charmbracelet/harmonica"
|
|
"github.com/charmbracelet/lipgloss"
|
|
)
|
|
|
|
const (
|
|
framesPerSecond = 30.0
|
|
numParticles = 50
|
|
)
|
|
|
|
var (
|
|
colors = []string{"#a864fd", "#29cdff", "#78ff44", "#ff718d", "#fdff6a"}
|
|
characters = []string{"+", "*", "•"}
|
|
head = "▄"
|
|
tail = "│"
|
|
)
|
|
|
|
type frameMsg time.Time
|
|
|
|
func animate() tea.Cmd {
|
|
return tea.Tick(time.Second/framesPerSecond, func(t time.Time) tea.Msg {
|
|
return frameMsg(t)
|
|
})
|
|
}
|
|
|
|
type model struct {
|
|
system *simulation.System
|
|
}
|
|
|
|
func SpawnShoot(width, height int) *simulation.Particle {
|
|
color := lipgloss.Color(array.Sample(colors))
|
|
v := float64(rand.Intn(15) + 15.0)
|
|
x := rand.Float64() * float64(width)
|
|
p := simulation.Particle{
|
|
Physics: harmonica.NewProjectile(
|
|
harmonica.FPS(framesPerSecond),
|
|
harmonica.Point{X: x, Y: float64(height)},
|
|
harmonica.Vector{X: 0, Y: -v},
|
|
harmonica.TerminalGravity,
|
|
),
|
|
Char: lipgloss.NewStyle().Foreground(color).Render(head),
|
|
TailChar: lipgloss.NewStyle().Foreground(color).Render(tail),
|
|
Color: color,
|
|
Shooting: true,
|
|
ExplosionCall: SpawnExplosion,
|
|
}
|
|
return &p
|
|
}
|
|
|
|
func SpawnExplosion(color lipgloss.Color, x, y float64, width, height int) []*simulation.Particle {
|
|
v := float64(rand.Intn(10) + 20.0)
|
|
particles := []*simulation.Particle{}
|
|
for i := 0; i < numParticles; i++ {
|
|
p := simulation.Particle{
|
|
Physics: harmonica.NewProjectile(
|
|
harmonica.FPS(framesPerSecond),
|
|
harmonica.Point{X: x, Y: y},
|
|
harmonica.Vector{X: math.Cos(float64(i)) * v, Y: math.Sin(float64(i)) * v / 2},
|
|
harmonica.TerminalGravity,
|
|
),
|
|
Char: lipgloss.NewStyle().Foreground(color).Render(array.Sample(characters)),
|
|
Shooting: false,
|
|
}
|
|
particles = append(particles, &p)
|
|
}
|
|
return particles
|
|
}
|
|
|
|
func InitialModel() model {
|
|
return model{system: &simulation.System{
|
|
Particles: []*simulation.Particle{},
|
|
Frame: simulation.Frame{},
|
|
}}
|
|
}
|
|
|
|
// Init initializes the confetti after a small delay
|
|
func (m model) Init() tea.Cmd {
|
|
return animate()
|
|
}
|
|
|
|
// Update updates the model every frame, it handles the animation loop and
|
|
// updates the particle physics every frame
|
|
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":
|
|
return m, tea.Quit
|
|
}
|
|
m.system.Particles = append(m.system.Particles, SpawnShoot(m.system.Frame.Width, m.system.Frame.Height))
|
|
|
|
return m, nil
|
|
case frameMsg:
|
|
m.system.Update()
|
|
return m, animate()
|
|
case tea.WindowSizeMsg:
|
|
if m.system.Frame.Width == 0 && m.system.Frame.Height == 0 {
|
|
// For the first frameMsg spawn a system of particles
|
|
m.system.Particles = append(m.system.Particles, SpawnShoot(msg.Width, msg.Height))
|
|
}
|
|
m.system.Frame.Width = msg.Width
|
|
m.system.Frame.Height = msg.Height
|
|
return m, nil
|
|
default:
|
|
return m, nil
|
|
}
|
|
}
|
|
|
|
// View displays all the particles on the screen
|
|
func (m model) View() string {
|
|
return m.system.Render()
|
|
}
|