-
Notifications
You must be signed in to change notification settings - Fork 34
/
Copy pathDogFight.py
534 lines (505 loc) · 20.3 KB
/
DogFight.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
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
#!/usr/bin/python
from __future__ import absolute_import, division, print_function, unicode_literals
"""quite a complicated demo showing many features of pi3d as well as
communication between players using httpRequest (see rpi_json.sql and
rpi_json.php) json serialisation and threading.
"""
import sys
import time, math, glob, random, threading, json
import demo
import pi3d
if sys.version_info[0] == 3:
from urllib import request as urllib_request
from urllib import parse as urllib_parse
else:
import urllib
urllib_request = urllib
urllib_parse = urllib
#display, camera, shader
DISPLAY = pi3d.Display.create(x=100, y=100, frames_per_second=20)
#a default camera is created automatically but we might need a 2nd 2D camera
#for displaying the instruments etc. Also, because the landscape is large
#we need to set the far plane to 10,000
CAMERA = pi3d.Camera(lens=(1.0, 10000.0, 55.0, 1.6))
CAMERA2D = pi3d.Camera(is_3d=False)
print("""===================================
== W increase power, S reduce power
== V view mode, C control mode
== B brakes
== mouse movement joystick
== Left button fire!
== X jumps to location of 1st enemy in list
================================""")
SHADER = pi3d.Shader("uv_bump") #for objects to look 3D
ELEVSH = pi3d.Shader("uv_elev_map") # for multi textured terrain
FLATSH = pi3d.Shader("uv_flat") #for 'unlit' objects like the background
GRAVITY = 9.8 #m/s**2
LD = 10 #lift/drag ratio
DAMPING = 0.95 #reduce roll and pitch rate each update_variables
BOOSTER = 1.5 #extra manoevreability boost to defy 1st Low of Thermodynamics.
#load bullet images
BULLET_TEX = [] #list to hold Texture refs
iFiles = glob.glob(sys.path[0] + "/textures/biplane/bullet??.png")
iFiles.sort() # order is vital to animation!
for f in iFiles:
BULLET_TEX.append(pi3d.Texture(f))
DAMAGE_FACTOR = 50 #dived by distance of shoot()
NR_TM = 1.0 #check much less frequently until something comes back
FA_TM = 5.0
NR_DIST = 250
FA_DIST = 1500
P_FACTOR = 0.001
I_FACTOR = 0.00001
#define Aeroplane class
class Aeroplane(object):
def __init__(self, model, recalc_time, refid):
self.refid = refid
self.recalc_time = recalc_time #in theory use different values for enemy
self.x, self.y, self. z = 0.0, 0.0, 0.0
self.x_perr, self.y_perr, self.z_perr = 0.0, 0.0, 0.0
self.x_ierr, self.y_ierr, self.z_ierr = 0.0, 0.0, 0.0
self.d_err = 0.0
self.v_speed, self.h_speed = 0.0, 0.0
self.rollrate, self.pitchrate, self.yaw = 0.0, 0.0, 0.0
self.direction, self.roll, self.pitch = 0.0, 0.0, 0.0
self.max_roll, self.max_pitch = 65, 30 #limit rotations
self.ailerons, self.elevator = 0.0, 0.0
self.max_ailerons, self.max_elevator = 10.0, 10.0 #limit conrol surface movement
self.VNE = 120 #max speed (Velocity Not to be Exceeded)
self.mass = 300
self.a_factor, self.e_factor = 10, 10
self.roll_inrta, self.pitch_inrta = 100, 100
self.max_power = 2000 #force units of thrust really
self.lift_factor = 20.0 #randomly adjusted to give required performance!
self.power_setting = 0.0
self.throttle_step = 20
self.last_time = time.time()
self.last_pos_time = self.last_time
self.del_time = None #difference in pi time for other aero c.f. main one
self.rtime = 60
self.nearest = None
self.other_damage = 0.0 #done to nearest others since last json_load
self.damage = 0.0 #done to this aeroplane by others
#create the actual model
self.model = pi3d.Model(file_string=model, camera=CAMERA)
self.model.set_shader(SHADER)
#create the bullets
plane = pi3d.Plane(h=25, w=1)
self.bullets = pi3d.MergeShape(camera=CAMERA)
#the merge method does rotations 1st Z, 2nd X, 3rd Y for some reason
#for multi axis rotations you need to figure it out by rotating a
#sheet of paper in the air in front of you (angles counter clockwise)!
self.bullets.merge([[plane, -2.0, 0.5, 8.0, 90,0,0, 1,1,1],
[plane, -2.0, 0.5, 8.0, 0,90,90, 1,1,1],
[plane, 2.0, 0.5, 8.0, 90,0,0, 1,1,1],
[plane, 2.0, 0.5, 8.0, 0,90,90, 1,1,1]])
self.num_b = len(BULLET_TEX)
self.seq_b = self.num_b
self.bullets.set_draw_details(FLATSH, [BULLET_TEX[0]])
def set_ailerons(self, dx):
self.ailerons = dx
if abs(self.ailerons) > self.max_ailerons:
self.ailerons = math.copysign(self.max_ailerons, self.ailerons)
def set_elevator(self, dy):
self.elevator = dy
if abs(self.elevator) > self.max_elevator:
self.elevator = math.copysign(self.max_elevator, self.elevator)
def set_power(self, incr):
self.power_setting += incr * self.throttle_step
if self.power_setting < 0:
self.power_setting = 0
elif self.power_setting > self.max_power:
self.power_setting = self.max_power
def shoot(self, target):
#only shoot if animation seq. ended
if self.seq_b < self.num_b:
return 0.0
#animate bullets
self.seq_b = 0
#check for hit
#components of direction vector
diag_xz = math.cos(math.radians(self.pitch))
drn_x = diag_xz * math.sin(math.radians(self.direction))
drn_y = math.sin(math.radians(self.pitch))
drn_z = diag_xz * math.cos(math.radians(self.direction))
#this will already be a unit vector
#vector from target to aeroplane
a_x = target[0] - self.x
a_y = target[1] - self.y
a_z = target[2] - self.z
#dot product
dot_p = drn_x * a_x + drn_y * a_y + drn_z * a_z
dx = a_x - dot_p * drn_x
dy = a_y - dot_p * drn_y
dz = a_z - dot_p * drn_z
distance = math.sqrt(dx**2 + dy**2 + dz**2)
print("distance={0:.2f}".format(distance))
return DAMAGE_FACTOR / distance if distance > 0.0 else 2.0 * DAMAGE_FACTOR
def home(self, target):
#turn towards target location, mainly for AI control of enemy aircraft
dir_t = math.degrees(math.atan2((target[0] - self.x), (target[2] - self.z)))
#make sure the direction is alway a value between +/- 180 degrees
#roll so bank is half direction,
self.roll = -((dir_t - self.direction + 180) % 360 - 180) / 2
#find angle between self and target
pch_t = math.degrees(math.atan2((target[1] - self.y),
math.sqrt((target[2] - self.z)**2 + (target[0] - self.x)**2)))
self.pitch = pch_t
return True
def update_variables(self):
#time
tm = time.time()
dt = tm - self.last_time
if dt < self.recalc_time: # don't need to do all this physics every loop
return
self.last_time = tm
#force from ailerons and elevators to get rotational accelerations
spsq = self.v_speed**2 + self.h_speed**2 #speed squared
a_force = self.a_factor * self.ailerons * spsq #ailerons force (moment really)
roll_acc = a_force / self.roll_inrta #good old Newton
e_force = self.e_factor * self.elevator * spsq #elevator
pitch_acc = e_force / self.pitch_inrta
#velocities and positions
if abs(self.roll) > self.max_roll: #make it easier to do flight control
self.roll = math.copysign(self.max_roll, self.roll)
self.rollrate = 0.0
if abs(self.pitch) > self.max_pitch:
self.pitch = math.copysign(self.max_pitch, self.pitch)
self.pitchrate = 0.0
self.roll += self.rollrate * dt #update roll position
self.pitch += self.pitchrate * dt #update roll rate
self.rollrate += roll_acc * dt
self.rollrate *= DAMPING # to stop going out of contol while looking around!
self.pitchrate += pitch_acc * dt
self.pitchrate *= DAMPING
#angle of attack
aofa = math.atan2(self.v_speed, self.h_speed)
aofa = math.radians(self.pitch) - aofa # approximation to sin difference
lift = self.lift_factor * spsq * aofa
drag = lift / LD
if spsq < 100: #stall!
lift *= 0.9
drag *= 1.3
cos_pitch = math.cos(math.radians(self.pitch))
sin_pitch = math.sin(math.radians(self.pitch))
cos_roll = math.cos(math.radians(self.roll))
sin_roll = math.sin(math.radians(self.roll))
h_force = (self.power_setting - drag) * cos_pitch - lift * sin_pitch
v_force = lift * cos_pitch * cos_roll - self.mass * GRAVITY
h_acc = h_force / self.mass
v_acc = v_force / self.mass
self.h_speed += h_acc * dt
if self.h_speed > self.VNE:
self.h_speed = self.VNE
elif self.h_speed < 0:
self.h_speed = 0
self.v_speed += v_acc * dt
if abs(self.v_speed) > self.VNE:
self.v_speed = math.copysign(self.VNE, self.v_speed)
turn_force = -lift * sin_roll * 1.5
radius = self.mass * spsq / turn_force if turn_force != 0.0 else 0.0
self.yaw = math.sqrt(spsq) / radius if radius != 0.0 else 0.0
def update_position(self, height):
#time
tm = time.time()
dt = tm - self.last_pos_time
self.last_pos_time = tm
self.x += (self.h_speed * math.sin(math.radians(self.direction)) * dt -
self.x_perr * P_FACTOR - self.x_ierr * I_FACTOR)
self.y += self.v_speed * dt - self.y_perr * P_FACTOR - self.y_ierr * I_FACTOR
if self.y < (height + 3):
self.y = height + 3
self.v_speed = 0
self.pitch = 2.5
#self.roll = 0
self.z += (self.h_speed * math.cos(math.radians(self.direction)) * dt -
self.z_perr * P_FACTOR - self.z_ierr * I_FACTOR)
self.direction += math.degrees(self.yaw) * dt - self.d_err * P_FACTOR
#set values of model
sin_d = math.sin(math.radians(self.direction))
cos_d = math.cos(math.radians(self.direction))
sin_r = math.sin(math.radians(self.roll))
cos_r = math.cos(math.radians(self.roll))
sin_p = math.sin(math.radians(self.pitch))
cos_p = math.cos(math.radians(self.pitch))
absroll = math.degrees(math.asin(sin_r * cos_d + cos_r * sin_p * sin_d))
abspitch = math.degrees(math.asin(sin_r * sin_d - cos_r * sin_p * cos_d))
self.model.position(self.x, self.y, self.z)
self.model.rotateToX(abspitch)
self.model.rotateToY(self.direction)
self.model.rotateToZ(absroll)
#set values for bullets
if self.seq_b < self.num_b:
self.bullets.position(self.x, self.y, self.z)
self.bullets.rotateToX(abspitch)
self.bullets.rotateToY(self.direction)
self.bullets.rotateToZ(absroll)
#set values for camera
return (self.x - 10.0 * sin_d, self.y + 4, self.z - 10.0 * cos_d, self.direction)
def draw(self):
self.model.draw()
#draw the bullet sequence if not finished
if self.seq_b < self.num_b:
self.bullets.buf[0].textures[0] = BULLET_TEX[self.seq_b]
self.bullets.draw()
self.seq_b += 1
#define Instruments class
class Instruments(object):
def __init__(self):
wd = DISPLAY.width
ht = DISPLAY.height
asi_tex = pi3d.Texture("textures/airspeed_indicator.png")
alt_tex = pi3d.Texture("textures/altimeter.png")
rad_tex = pi3d.Texture("textures/radar.png")
dot_tex = pi3d.Texture("textures/radar_dot.png")
ndl_tex = pi3d.Texture("textures/instrument_needle.png")
self.asi = pi3d.ImageSprite(asi_tex, FLATSH, camera=CAMERA2D,
w=128, h=128, x=-128, y=-ht/2+64, z=2)
self.alt = pi3d.ImageSprite(alt_tex, FLATSH, camera=CAMERA2D,
w=128, h=128, x=0, y=-ht/2+64, z=2)
self.rad = pi3d.ImageSprite(rad_tex, FLATSH, camera=CAMERA2D,
w=128, h=128, x=128, y=-ht/2+64, z=2)
self.dot = pi3d.ImageSprite(dot_tex, FLATSH, camera=CAMERA2D,
w=16, h=16, z=1)
self.ndl1 = pi3d.ImageSprite(ndl_tex, FLATSH, camera=CAMERA2D,
w=128, h=128, x=-128, y=-ht/2+64, z=1)
self.ndl2 = pi3d.ImageSprite(ndl_tex, FLATSH, camera=CAMERA2D,
w=128, h=128, x=0, y=-ht/2+64, z=1)
self.ndl3 = pi3d.ImageSprite(ndl_tex, FLATSH, camera=CAMERA2D,
w=128, h=128, x=128, y=-ht/2+64, z=1)
self.dot_list = []
self.update_time = 0.0
def draw(self):
self.asi.draw()
self.alt.draw()
self.rad.draw()
for i in self.dot_list:
self.dot.position(i[1] + 128, i[2] + self.rad.y(), 1)
self.dot.draw()
self.ndl1.draw()
self.ndl2.draw()
self.ndl3.draw()
def update(self, ae, others):
self.ndl1.rotateToZ(-360*ae.h_speed/140)
self.ndl2.rotateToZ(-360*ae.y/3000)
self.ndl3.rotateToZ(-ae.direction)
self.dot_list = []
for i in others:
if i == "start":
continue
o = others[i]
dx = (o.x - ae.x) / 50
dy = (o.z - ae.z) / 50
d = math.hypot(dx, dy)
if d > 40:
dx *= 40 / d
dy *= 40 / d
self.dot_list.append([o.refid, dx, dy])
self.update_time = ae.last_pos_time
def json_load(ae, others):
"""httprequest other players. Sends own data and gets back array of all
other players within sight. This function runs in a background thread
"""
#TODO pass nearest, nearest.hp and own hp merge in some way
tm_now = time.time()
jstring = json.dumps([ae.refid, ae.last_time, ae.x, ae.y, ae.z,
ae.h_speed, ae.v_speed, ae.pitch, ae.direction, ae.roll,
ae.pitchrate, ae.yaw, ae.rollrate, ae.power_setting, ae.damage], separators=(',',':'))
if ae.nearest:
n_id = ae.nearest.refid
n_damage = ae.nearest.other_damage
ae.nearest.other_damage = 0.0
else:
n_id = ""
n_damage = 0.0
params = urllib_parse.urlencode({"id":ae.refid, "tm":tm_now, "x":ae.x, "z":ae.z,
"json":jstring, "nearest":n_id, "damage":n_damage})
others["start"] = tm_now #used for polling freqency
urlstring = "http://www.eldwick.org.uk/sharecalc/rpi_json.php?{0}".format(params)
try:
r = urllib_request.urlopen(urlstring)
if r.getcode() == 200: #good response
jstring = r.read().decode("utf-8")
if len(jstring) > 50: #error messages are shorter than this
olist = json.loads(jstring)
#smooth time offset value
ae.del_time = ae.del_time * 0.9 + olist[0] * 0.1 if ae.del_time else olist[0]
#own damage is cumulative and not reset on server until dead!
ae.damage = olist[1]
#if ae.damage > 2.0 * DAMAGE_FACTOR: #explode return to GO etc
#print(ae.damage)
olist = olist[2:]
"""
synchronisation system: sends time.time() which is used to calculate
an offset on the server and which is inserted as the second term
in the json string. When the list of other players comes back from
the server it is preceded by the same offset time inserted in this json.
This is used to adjust the last_time for all
the other avatars.
"""
nearest = None
ae.rtime = 60
for o in olist:
if not(o[0] in others):
others[o[0]] = Aeroplane("models/biplane.obj", 0.1, o[0])
oa = others[o[0]] #oa is other aeroplane, ae is this one!
oa.refif = o[0]
#exponential smooth time offset values
oa.del_time = oa.del_time * 0.9 + o[1] * 0.1 if oa.del_time else o[1]
oa.last_time = o[2] + oa.del_time - ae.del_time # o[1] inserted by server code
dt = tm_now - oa.last_time
if oa.x == 0.0:
oa.x, oa.y, oa.z = o[3], o[4], o[5]
nx = o[3] + o[6] * math.sin(math.radians(o[9])) * dt
ny = o[4] + o[7] * dt
nz = o[5] + o[6] * math.cos(math.radians(o[9])) * dt
distance = math.hypot(nx - ae.x, nz - ae.z)
if not nearest or distance < nearest:
nearest = distance
ae.nearest = oa
oa.x_perr, oa.y_perr, oa.z_perr = oa.x - nx, oa.y - ny, oa.z - nz
oa.x_ierr += oa.x_perr
oa.y_ierr += oa.y_perr
oa.z_ierr += oa.z_perr
oa.d_err = ((oa.direction - (o[9] + o[12] * dt) + 180) % 360 - 180) / 2
oa.h_speed = o[6]
oa.v_speed = o[7]
oa.pitch = o[8]
oa.roll = o[10]
oa.pitchrate = o[11]
oa.yaw = o[12]
oa.rollrate = o[13]
oa.power_setting = o[14]
oa.damage = o[15]
if nearest:
ae.rtime = NR_TM + (max(min(nearest, FA_DIST), NR_DIST) - NR_DIST) / \
(FA_DIST - NR_DIST) * (FA_TM - NR_TM)
#TODO tidy up inactive others; flag not to draw, delete if inactive for long enough
return True
else:
print(jstring)
return False
else:
print(r.getcode())
return False
except Exception as e:
print("exception:", e)
#MAC address
try:
refid = (open("/sys/class/net/eth0/address").read()).strip()
except:
try:
refid = (open("/sys/class/net/wlan0/address").read()).strip()
except:
refid = "00:00:00:00:00:00"
#create the instances of Aeroplane
a = Aeroplane("models/biplane.obj", 0.02, refid)
a.z, a.direction = 900, 180
#create instance of instruments
inst = Instruments()
others = {"start": 0.0} #contains a dictionary of other players keyed by refid
thr = threading.Thread(target=json_load, args=(a, others))
thr.daemon = True #allows the program to exit even if a Thread is still running
thr.start()
# Load textures for the environment cube
ectex = pi3d.loadECfiles("textures/ecubes", "sbox")
myecube = pi3d.EnvironmentCube(size=7000.0, maptype="FACES", camera=CAMERA)
myecube.set_draw_details(FLATSH, ectex)
myecube.set_fog((0.5,0.5,0.5,1.0), 4000)
# Create elevation map
mapwidth = 10000.0
mapdepth = 10000.0
mapheight = 1000.0
mountimg1 = pi3d.Texture('textures/mountains3_512.jpg') # diffuse textures
roadimg = pi3d.Texture('textures/Roof.png')
grassimg = pi3d.Texture('textures/grass.jpg')
rockimg = pi3d.Texture('textures/rock1.jpg')
mudbmp = pi3d.Texture('textures/mudnormal.jpg') #normal textures
grassbmp = pi3d.Texture('textures/grasstile_n.jpg')
rockbmp = pi3d.Texture('textures/rocktile2.jpg')
mymap = pi3d.ElevationMap("textures/mountainsHgt.jpg", name="map",
width=mapwidth, depth=mapdepth, height=mapheight,
divx=64, divy=64, camera=CAMERA, texmap='textures/roads.jpg')
mymap.set_draw_details(ELEVSH, [grassimg, grassbmp,
rockimg, rockbmp,
mountimg1, rockbmp,
roadimg, mudbmp], 64.0, 0.0, umult=96.0, vmult=96.0)
mymap.set_fog((0.5, 0.5, 0.5, 1.0), 4000)
# init events
inputs = pi3d.InputEvents()
inputs.get_mouse_movement()
CAMERA.position((0.0, 0.0, -10.0))
cam_rot, cam_pitch = 0, 0
cam_toggle = True #control mode
while DISPLAY.loop_running() and not inputs.key_state("KEY_ESC"):
inputs.do_input_events()
#""" mouse input
mx, my, mv, mh, md = inputs.get_mouse_movement()
if cam_toggle:
a.set_ailerons(-mx * 0.001)
a.set_elevator(my * 0.001)
else:
cam_rot -= mx * 0.1
cam_pitch -= my * 0.1
#"""
""" joystick input
mx, my = inputs.get_joystickR()
if cam_toggle:
a.set_ailerons(-mx * 0.06)
a.set_elevator(my * 0.02)
else:
cam_rot -= mx * 2.0
cam_pitch -= my * 2.0
"""
if inputs.key_state("KEY_W") or inputs.get_hat()[1] == -1: #increase throttle
a.set_power(1)
if inputs.key_state("KEY_S") or inputs.get_hat()[1] == 1: #throttle back
a.set_power(-1)
if inputs.key_state("KEY_X"): #jump to first enemy!
for i in others:
if i != "start":
b = others[i]
a.x, a.y, a.z = b.x, b.y + 5, b.z
break
if inputs.key_state("KEY_B") or inputs.key_state("BTN_BASE2"): #brakes
a.h_speed *= 0.99
if inputs.key_state("KEY_V") or inputs.key_state("BTN_TOP2"): #view mode
cam_toggle = False
a.set_ailerons(0)
a.set_elevator(0)
if inputs.key_state("KEY_C") or inputs.key_state("BTN_BASE"): #control mode
cam_toggle = True
cam_rot, cam_pitch = 0, 0
if inputs.key_state("BTN_LEFT") or inputs.key_state("BTN_PINKIE"): #shoot
#target is always nearest others set during last json_load()
#tx, ty, tz = 0., 0.0, 0.0
if a.nearest:
tx, ty, tz = a.nearest.x, a.nearest.y, a.nearest.z
a.nearest.other_damage += a.shoot([tx, ty, tz])
a.update_variables()
loc = a.update_position(mymap.calcHeight(a.x, a.z))
CAMERA.reset()
#CAMERA.rotate(-20 + cam_pitch, -loc[3] + cam_rot, 0) #unreal view
CAMERA.rotate(-20 + cam_pitch, -loc[3] + cam_rot, -a.roll) #air-sick view
CAMERA.position((loc[0], loc[1], loc[2]))
inst.draw()
a.draw()
for i in others:
if i == "start":
continue
b = others[i]
b.update_variables()
b.update_position(mymap.calcHeight(b.x, b.z))
b.draw()
#do httprequest if thread not already started and enough time has elapsed
if not (thr.is_alive()) and (a.last_pos_time > (others["start"] + a.rtime)):
thr = threading.Thread(target=json_load, args=(a, others))
thr.daemon = True #allows the program to exit even if a Thread is still running
thr.start()
if a.last_pos_time > (inst.update_time + NR_TM):
inst.update(a, others)
mymap.draw()
myecube.position(loc[0], loc[1], loc[2])
myecube.draw()
inputs.release()
DISPLAY.destroy()