forked from najibelmokhtari/basic-epidemic-simulation
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathepidemic.py
204 lines (152 loc) · 5.36 KB
/
epidemic.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
# 0: healthy
# 1: immune
# 10 - 20 : infected time
# -1: dead
from random import randrange
import pygame
import sys
PROBA_DEATH = 100 # the black plague was 100%. smallpox was ~30%-35%
CONTAGION_RATE = 4.5 # This is the R0 factor. Number of people an individual will infect on average
PROBA_INFECT = CONTAGION_RATE * 10
VACCINATION_RATE = 0
SIMULATION_SPEED = 25 # time between days in milliseconds. 0: fastest.
# 500 means every day the simulation pauses for 500 ms
# 25 is good for watching
nb_rows = 50
nb_cols = 50
global states, states_temp
states = [[0] * nb_cols for i1 in range(nb_rows)]
states_temp = [[0] * nb_cols for i1 in range(nb_rows)]
def get_neighbour(x, y):
incx = randrange(3)
incy = randrange(3)
incx = (incx * 1) - 1
incy = (incy * 1) - 1
x2 = x + incx
y2 = y + incy
if x2 < 0:
x2 = 0
if x2 >= nb_cols:
x2 = nb_cols - 1
if y2 < 0:
y2 = 0
if y2 >= nb_rows:
y2 = nb_rows - 1
return [x2, y2]
global display
global myfont
global initial_pause
WHITE = (255, 255, 255)
BLUE = (0, 0, 255)
GREEN = (0, 100, 0)
BLACK = (0, 0, 0)
def init_screen():
global display
global myfont
myfont = pygame.font.SysFont('Calibri', 40)
display.fill(WHITE)
image = pygame.image.load("death_toll.png").convert_alpha()
image = pygame.transform.rotozoom(image, 0, .35)
display.blit(image, (540, 23))
def vaccinate():
for x in range(nb_cols):
for y in range(nb_rows):
if randrange(99) < VACCINATION_RATE:
states[x][y] = 1
def count_dead():
global states
count = 0
for x in range(nb_cols):
for y in range(nb_rows):
if states[x][y] == -1:
count = count + 1
return count
def main():
pygame.init()
pygame.font.init()
global display
display=pygame.display.set_mode((800,750),0,32)
pygame.display.set_caption("Virus Epidemic")
init_screen()
global states, states_temp
states[5][5] = 10
vaccinate()
image = pygame.image.load("death_toll.png").convert_alpha()
image = pygame.transform.rotozoom(image, 0, .35)
display.blit(image, (540, 23))
global initial_pause
initial_pause = True
it = 0
death_toll = 0
while True:
pygame.time.delay(SIMULATION_SPEED)
it = it + 1
if it <= 10000 and it >= 2:
states_temp = states.copy()
for x in range(nb_cols):
for y in range(nb_rows):
state = states[x][y]
if state == -1:
pass
if state >= 10:
states_temp[x][y] = state + 1
if state >= 20:
if randrange(99) < PROBA_DEATH:
states_temp[x][y] = -1
else:
states_temp[x][y] = 1
if state >= 10 and state <= 20:
if randrange(99) < PROBA_INFECT:
neighbour = get_neighbour(x, y)
x2 = neighbour[0]
y2 = neighbour[1]
neigh_state = states[x2][y2]
if neigh_state == 0:
states_temp[x2][y2] = 10
states = states_temp.copy()
death_toll = count_dead()
pygame.draw.rect(display, WHITE, (450, 30, 80, 50))
textsurface = myfont.render(str(death_toll), False, (0, 0, 0))
display.blit(textsurface, (450, 30))
for x in range(nb_cols):
for y in range(nb_rows):
if states[x][y] == 0:
color = BLUE
if states[x][y] == 1:
color = GREEN
if states[x][y] >= 10:
color = (states[x][y] * 12, 50, 50)
if states[x][y] == -1:
color = WHITE
pygame.draw.circle(display, color, (100 + x * 12 + 5, 100 + y * 12 + 5), 5)
pygame.draw.rect(display, WHITE, (100 + x * 12 + 3, 100 + y * 12 + 4, 1, 1))
pygame.draw.rect(display, WHITE, (100 + x * 12 + 5, 100 + y * 12 + 4, 1, 1))
for event in pygame.event.get():
if event.type==pygame.QUIT:
pygame.quit()
sys.exit()
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_SPACE:
states = [[0] * nb_cols for i1 in range(nb_rows)]
states_temp = [[0] * nb_cols for i1 in range(nb_rows)]
vaccinate()
states_temp = states.copy()
states[5][5] = 10
init_screen()
it = 0
death_toll = 0
initial_pause = True
pygame.display.update()
if it == 1:
initial_pause = True
if initial_pause:
while initial_pause:
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_SPACE:
initial_pause = False
break
main()