-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathp3_pathfinder.py
165 lines (139 loc) · 6.31 KB
/
p3_pathfinder.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
# By Sam Reha and Patrick Russell
from heapq import heappush, heappop
from math import sqrt
def find_path(source_point, destination_point, mesh):
# A_star acting as Dijkstra's.
def A_star(src_box, dst_box, graph):
#dist = {}
#prev = {}
forward_visit = []
backward_visit = []
forward_dist = {}
backward_dist = {}
forward_prev = {}
backward_prev = {}
detail_points = {}
q = [] # Note, treat q as a priority queue
detail_points[src_box] = source_point
detail_points[dst_box] = destination_point
forward_dist[src_box] = 0
forward_prev[src_box] = None
backward_dist[dst_box] = 0
backward_prev[dst_box] = None
#Push the queue, the distance to the source, and the source into a heap
heappush(q, (forward_dist[src_box], src_box, "Destination"))
heappush(q, (backward_dist[dst_box], dst_box, "Source"))
#while the priority queue still has nodes
while len(q) > 0:
#Pop from the queue
priority, u, goal = heappop(q)
#Break if the destination is reached
"""if u == dst_box:
break"""
if goal == "Destination" and u in backward_visit:
break
elif goal == "Source" and u in forward_visit:
break
#Get the neighbours of the current node
neighborhood = graph['adj'].get(u, [])
for neighbor in neighborhood:
u_coord = detail_points[u]
detail_points[neighbor] = get_detail_point(u_coord, neighbor)
if goal == "Destination":
alt = forward_dist[u] + coordinate_distance(detail_points[u], detail_points[neighbor]) # Setting alt to 0 makes this behave like BFS
if neighbor not in forward_dist or alt < forward_dist[neighbor]:
visited_nodes.append(neighbor)
forward_visit.append(neighbor)
forward_dist[neighbor] = alt
forward_prev[neighbor] = u
#heappush(q, (alt + coordinate_distance(detail_points[neighbor], destination_point), neighbor, goal))
heappush(q, (alt, neighbor, goal))
else:
alt = backward_dist[u] + coordinate_distance(detail_points[u], detail_points[neighbor]) # Setting alt to 0 makes this behave like BFS
if neighbor not in backward_dist or alt < backward_dist[neighbor]:
visited_nodes.append(neighbor)
backward_visit.append(neighbor)
backward_dist[neighbor] = alt
backward_prev[neighbor] = u
#heappush(q, (alt + coordinate_distance(detail_points[neighbor], destination_point), neighbor, goal))
heappush(q, (alt, neighbor, goal))
#Draw the line between points
if u in forward_visit or u in backward_visit:
backward_u = u
forward_u = u
path = []
detail_points[src_box] = source_point
detail_points[dst_box] = destination_point
while forward_u:
prev_u = forward_prev[forward_u]
if prev_u is not None:
detail_points[prev_u] = get_detail_point(detail_points[forward_u], prev_u)
path.append((detail_points[forward_u], detail_points[prev_u]))
else:
path.append((detail_points[forward_u], source_point))
forward_u = prev_u
while backward_u:
prev_u = backward_prev[backward_u]
if prev_u is not None:
detail_points[prev_u] = get_detail_point(detail_points[backward_u], prev_u)
path.append((detail_points[backward_u], detail_points[prev_u]))
else:
path.append((detail_points[backward_u], destination_point))
backward_u = prev_u
# path.reverse() Uncomment this if we ever want to print the path out for the user
return path
else:
return []
# Gets the detail point of a box
def get_detail_point(starting_point, new_box):
u_x, u_y = starting_point
neighbor_x1 = new_box[0]
neighbor_x2 = new_box[1]
neighbor_y1 = new_box[2]
neighbor_y2 = new_box[3]
return (min(neighbor_x2-1,max(neighbor_x1,u_x)), min(neighbor_y2-1,max(neighbor_y1,u_y)))
#return (min(neighbor_x2,max(neighbor_x1,u_x)), min(neighbor_y2,max(neighbor_y1,u_y)))
#Gets the distance between two coordinates
def coordinate_distance(coord1, coord2):
x1 = coord1[0]
x2 = coord2[0]
y1 = coord1[1]
y2 = coord2[1]
return sqrt((x1-x2)**2+(y1-y2)**2)
#Gets the box the the point is in
def is_in_box(point, box):
x = point[0]
y = point[1]
box_x1 = box[0]
box_x2 = box[1]
box_y1 = box[2]
box_y2 = box[3]
if x > box_x1 and x <= box_x2 and y > box_y1 and y <= box_y2:
return True
else:
return False
path_list = []
visited_nodes = []
source_box = None
destination_box = None
found_source = False
found_dest = False
# Find source and destination points within mesh:
for box in mesh['boxes']:
if is_in_box(source_point, box) and not found_source:
source_box = box
path_list.append(source_point)
if is_in_box(destination_point, box) and not found_dest:
destination_box = box
path_list.append(destination_point)
if found_source and found_dest:
break
if source_box == destination_box:
path_list = [(source_point, destination_point)]
visited_nodes=[source_box]
else:
path_list = A_star(source_box, destination_box, mesh)
if len(path_list) is 0:
print "No path could be found!"
return (path_list, visited_nodes)