-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathparticle_system.rs
134 lines (118 loc) · 4.66 KB
/
particle_system.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
extern crate sdl2;
use crate::two_d;
// extern crate gl;
use sdl2::rect::Rect;
use sdl2::render::Canvas;
use rand::Rng;
#[derive(Clone, Copy)]
pub enum ParticleShape {
Rect, // Rectangle shape
Circle, // Circle shape
}
// particle system +
// Shader +
pub struct Particle {
pub x: f32,
pub y: f32,
pub x_vel: f32,
pub y_vel: f32,
pub life: f32,
pub size: u32,
pub color: two_d::Color,
pub alpha: u8,
pub shape: ParticleShape,
}
#[allow(dead_code)]
impl Particle {
pub fn new(x: f32, y: f32, x_vel: f32, y_vel: f32, life: f32, color: two_d::Color, shape: ParticleShape) -> Self {
Particle {
x,
y,
x_vel,
y_vel,
life,
size: 2,
color,
alpha: 255,
shape, // Set the shape
}
}
pub fn update(&mut self, delta_time: f32, screen_height: u32) {
self.x += self.x_vel * delta_time;
self.y += self.y_vel * delta_time;
self.life -= delta_time;
self.alpha = (self.life * 255.0).max(0.0).min(255.0) as u8;
// If particle reaches bottom of the screen, respawn it at the top
if self.y as u32 > screen_height {
self.y = 0.0;
self.life = 5.0;
}
}
pub fn render<T: sdl2::render::RenderTarget>(&self, canvas: &mut Canvas<T>) {
canvas.set_draw_color(sdl2::pixels::Color::RGBA(self.color.r(), self.color.g(), self.color.b(), self.alpha));
match self.shape {
ParticleShape::Rect => {
canvas.fill_rect(Rect::new(self.x as i32, self.y as i32, self.size, self.size)).unwrap();
}
ParticleShape::Circle => {
let center_x = self.x as f32;
let center_y = self.y as f32;
let radius = self.size as f32 / 2.0;
// Draw filled circle
for angle in 0..360 {
let radian = angle as f32 * std::f32::consts::PI / 180.0;
let x = radian.cos() * radius + center_x;
let y = radian.sin() * radius + center_y;
// Calculate the topmost point for the vertical line to fill the circle
let top_y = center_y - (y - center_y).abs();
for fill_y in top_y as i32..=y as i32 {
canvas.draw_point(sdl2::rect::Point::new(x as i32, fill_y)).unwrap();
}
}
}
}
}
}
#[allow(dead_code)]
pub fn spawn_particles_sparks(particles: &mut Vec<Particle>, x: i32, y: i32, count: usize, shape: ParticleShape) {
let mut rng = rand::thread_rng();
for _ in 0..count {
let angle = rng.gen_range(0.0..std::f32::consts::TAU);
let speed = rng.gen_range(50.0..200.0);
let x_vel = angle.cos() * speed;
let y_vel = angle.sin() * speed;
let life = rng.gen_range(0.5..2.5);
let color = two_d::Color::new_rgba(123, 56, 89, 255);
particles.push(Particle::new(x as f32, y as f32, x_vel, y_vel, life, color, shape));
}
}
#[allow(dead_code)]
pub fn spawn_particles_fires(particles: &mut Vec<Particle>, x: i32, y: i32, count: usize, shape: ParticleShape) {
let mut rng = rand::thread_rng();
for _ in 0..count {
let angle = rng.gen_range(std::f32::consts::PI..std::f32::consts::TAU);
let speed = rng.gen_range(50.0..200.0);
let x_vel = angle.cos() * speed;
let y_vel = angle.sin() * speed;
let life = rng.gen_range(0.5..2.5);
let color = match rng.gen_range(0..3) {
0 => two_d::Color::new(254, 95, 85),
1 => two_d::Color::new(254, 207, 92),
_ => two_d::Color::new(254, 253, 153),
};
particles.push(Particle::new(x as f32, y as f32, x_vel, y_vel, life, color, shape));
}
}
#[allow(dead_code)]
pub fn spawn_particles_rain(particles: &mut Vec<Particle>, screen_width: u32, count: usize, shape: ParticleShape) {
let mut rng = rand::thread_rng();
for _ in 0..count {
let x = rng.gen_range(0..screen_width) as f32; // random horizontal position
let y = 0.0; // start at the top of the screen
let x_vel = rng.gen_range(-5.0..5.0); // slight horizontal movement
let y_vel = rng.gen_range(50.0..100.0); // vertical falling movement
let life = rng.gen_range(2.0..5.0); // life of raindrop
let color = two_d::Color::new_rgba(0, 0, 255, 255); // blue color for rain
particles.push(Particle::new(x, y, x_vel, y_vel, life, color, shape));
}
}