forked from microsoft/AirSim
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnavigate.py
96 lines (78 loc) · 3.09 KB
/
navigate.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
# use open cv to show new images from AirSim
from PythonClient import *
# requires Python 3.5.3 :: Anaconda 4.4.0
# pip install opencv-python
import cv2
import time
import math
import sys
client = MultirotorClient()
client.confirmConnection()
client.enableApiControl(True)
client.armDisarm(True)
client.takeoff()
# you must first press "1" in the AirSim view to turn on the depth capture
# get depth image
yaw = 0
pi = 3.14159265483
vx = 0
vy = 0
driving = 0
help = False
while True:
# this will return png width= 256, height= 144
result = client.simGetImage(0, AirSimImageType.DepthVis)
if (result == "\0"):
if (not help):
help = True
print("Please press '1' in the AirSim view to enable the Depth camera view")
else:
rawImage = np.fromstring(result, np.int8)
png = cv2.imdecode(rawImage, cv2.IMREAD_UNCHANGED)
gray = cv2.cvtColor(png, cv2.COLOR_BGR2GRAY)
# slice the image so we only check what we are headed into (and not what is down on the ground below us).
top = np.vsplit(gray, 2)[0]
# now look at 4 horizontal bands (far left, left, right, far right) and see which is most open.
# the depth map uses black for far away (0) and white for very close (255), so we invert that
# to get an estimate of distance.
bands = np.hsplit(top, [50,100,150,200]);
maxes = [np.max(x) for x in bands]
min = np.argmin(maxes)
distance = 255 - maxes[min]
# sanity check on what is directly in front of us (slot 2 in our hsplit)
current = 255 - maxes[2]
if (current < 20):
print("woops - we are about to crash, so stopping!")
client.hover().get()
sys.exit(0)
pitch, roll, yaw = client.getRollPitchYaw()
if (distance > current + 30):
# we have a 90 degree field of view (pi/2), we've sliced that into 5 chunks, each chunk then represents
# an angular delta of the following pi/10.
change = 0
driving = min
if (min == 0):
change = -2 * pi / 10
elif (min == 1):
change = -pi / 10
elif (min == 2):
change = 0 # center strip, go straight
elif (min == 3):
change = pi / 10
else:
change = 2*pi/10
yaw = (yaw + change)
vx = math.cos(yaw);
vy = math.sin(yaw);
print ("switching angle", math.degrees(yaw), vx, vy, min, distance, current)
if (vx == 0 and vy == 0):
vx = math.cos(yaw);
vy = math.sin(yaw);
print ("distance=", current)
client.moveByVelocityZ(vx, vy,-6, 1, DrivetrainType.ForwardOnly, YawMode(False, 0))
x = int(driving * 50)
cv2.rectangle(png, (x,0), (x+50,50), (0,255,0), 2)
cv2.imshow("Top", png)
key = cv2.waitKey(1) & 0xFF;
if (key == 27 or key == ord('q') or key == ord('x')):
break;