forked from kirstenmg/disease-model
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathparticle-animation.py
226 lines (178 loc) · 7.64 KB
/
particle-animation.py
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
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
# import the modules we'll need:
import pygame #to draw window, particles
import random #to determine random movement
import math #to use math functions
import time #to keep track of infection time
# constants for this project
RADIUS = 15
(WIDTH, HEIGHT) = (500, 500) #height excludes text box
TEXT_BOX_HEIGHT = 100 # height of box with text at bottom of screen
# SPEED = 1
# COLLIDED_SPEED = 0.5
DEFAULT_COLOR = (0, 0, 255)
INFECTED_COLOR = (255, 0, 0)
RECOVERED_COLOR = (0, 255, 0)
RECOVERY = 10 #seconds
WINDOW_COLOR = (255, 255, 255)
WINDOW_TITLE = "Particles"
NUM_PARTICLES = 50
# activate pygame library
pygame.init()
# create pygame window object and display it
screen = pygame.display.set_mode((WIDTH, HEIGHT + TEXT_BOX_HEIGHT))
pygame.display.set_caption(WINDOW_TITLE)
screen.fill(WINDOW_COLOR)
#represents a particle
class Particle:
def __init__(self, x, y, angle):
self.angle = angle
self.speed = random.random() # a characteristic of the Particle
self.current_speed = self.speed # actual speed, adjusted after collisions
self.collision_time = 0 # time since latest collision
self.collided = False
self.x = x
self.y = y
self.radius = RADIUS
self.color = (0, 0, 100 + self.speed * 155)
self.status = "healthy"
self.onset = 0 # this doesn't become relevant until infection
# rates
self.recovery_probability = random.uniform(0.9, 1)
self.symptomatic_probability = random.uniform(0.5, 0.75)
self.infection_probability = random.uniform(0.9, 1)
# draw particle on screen
def display(self):
pygame.draw.circle(screen, self.color, ((int)(self.x),
(int)(self.y)), self.radius)
# update x and y position of particle
def move(self):
# check if speed should be reset - it is averaged, then restored to the
# particle's original speed
## if (self.collided and time.time() - self.collision_time >= 0.5):
## self.current_speed = self.speed
## self.collided = False
## elif (self.collided and time.time() - self.collision_time >= 0.25):
## self.current_speed = (self.current_speed + self.speed) / 2
# update position
self.x += math.cos(self.angle) * self.current_speed
self.y += -math.sin(self.angle) * self.current_speed
# check if particle needs to bounce off the wall
def bounce(self):
if self.x > WIDTH - self.radius:
self.x = 2 * (WIDTH - self.radius) - self.x
self.angle = math.pi - self.angle
elif self.x < self.radius:
self.x = 2 * self.radius - self.x
self.angle = math.pi - self.angle
if self.y > HEIGHT - self.radius:
self.y = 2 * (HEIGHT - self.radius) - self.y
self.angle = - self.angle
elif self.y < self.radius:
self.y = 2 * self.radius - self.y
self.angle = - self.angle
# infect this particle, if it is not already
def infect(self):
if self.status != "infected":
num = random.random()
if num <= self.infection_probability:
num = random.random()
if num <= self.symptomatic_probability:
self.color = (100 * self.speed + 155, 0, 0)
self.status = "infected"
self.onset = time.time()
# check if the particle should recover, if so, recover
def recover(self):
if self.status == "infected" and time.time() - self.onset > RECOVERY:
num = random.random()
if num < self.recovery_probability:
self.color = RECOVERED_COLOR
self.status = "recovered"
self.infection_probability = random.uniform(0, 0.01)
else:
self.status = "dead"
# adjust direction if centers of any two particles are closer than twice
# the radius
# parameter is list of all particles in the window
def collide(particles):
for i in range(0, len(particles) - 1):
target = particles[i]
for j in range(i + 1, len(particles)):
other = particles[j]
diff_x = target.x - other.x
diff_y = target.y - other.y
distance = math.sqrt(diff_x**2 + diff_y**2)
if distance < RADIUS * 2:
# adjust directions (switch velocities)
temp_angle = target.angle
temp_speed = target.current_speed
target.angle = other.angle
target.current_speed = other.current_speed
other.angle = temp_angle
other.current_speed = temp_speed
target.collision_time = time.time()
other.collision_time = time.time()
target.collided = True
other.collided = True
if (target.status == "infected" or other.status == "infected"):
target.infect()
other.infect()
# adjust target x and y to be next to other
target.x = other.x + diff_x / distance * 2 * RADIUS
target.y = other.y + diff_y / distance * 2 * RADIUS
def main():
particles = []
for i in range(NUM_PARTICLES):
x = random.randint(RADIUS, WIDTH - RADIUS)
y = random.randint(RADIUS, HEIGHT - RADIUS)
angle = random.uniform(0, 2 * math.pi)
particles.append(Particle(x, y, angle))
particles[0].infect()
# initialize status counts
healthy = 0
infected = 0
recovered = 0
dead = 0
# make screen persist until user closes it
running = True
while running:
# pause the program for 20 ms
pygame.time.delay(10)
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit() # we could also set running = False
# check if particles run into each other
collide(particles)
#reset particle status counts
healthy = 0
infected = 0
recovered = 0
# update each particle in our list and the display
for p in particles:
p.move()
p.bounce()
p.recover()
p.display()
if p.status == "healthy":
healthy += 1
elif p.status == "infected":
infected += 1
elif p.status == "recovered":
recovered += 1
elif p.status == "dead":
dead += 1
particles.remove(p)
# create font object
font = pygame.font.Font(pygame.font.get_default_font(), 15)
# create Surfaces with text and combine it with the screen surface
# to display status counts
text = font.render("healthy: " + str(healthy), True, (0, 0, 0))
screen.blit(text, (10, HEIGHT))
text = font.render("infected: " + str(infected), True, (0, 0, 0))
screen.blit(text, (10, HEIGHT + 20))
text = font.render("recovered: " + str(recovered), True, (0, 0, 0))
screen.blit(text, (10, HEIGHT + 40))
text = font.render("dead: " + str(dead), True, (0, 0, 0))
screen.blit(text, (10, HEIGHT + 60))
pygame.display.update()
screen.fill(WINDOW_COLOR)
main()