forked from tanjeffreyz/auto-maple
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlayout.py
290 lines (243 loc) · 9.61 KB
/
layout.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
"""A module for saving map layouts and determining shortest paths."""
import config
import settings
import utils
import cv2
import math
import pickle
from os.path import join, isfile, splitext, basename
from heapq import heappush, heappop
class Node:
"""Represents a vertex on a quadtree."""
def __init__(self, x, y):
"""
Creates a new Node at (X, Y). Also initializes the Node's children.
:param x: The x position of the node.
:param y: The y position of the node.
"""
self.x = x
self.y = y
self.up_left = None
self.up_right = None
self.down_left = None
self.down_right = None
def children(self):
"""
Returns an array of this Node's children if they exist.
:return: This Node's children.
"""
result = []
if self.up_left:
result.append(self.up_left)
if self.up_right:
result.append(self.up_right)
if self.down_left:
result.append(self.down_left)
if self.down_right:
result.append(self.down_right)
return result
def __str__(self):
"""
Returns a string representation of this Node as a coordinate point.
:return: A string of the form '(x, y)'.
"""
return str(tuple(self))
def __iter__(self):
"""
Support converting a Node into a tuple.
:return: This Node's x and y positions.
"""
yield self.x
yield self.y
class Layout:
"""Uses a quadtree to represent possible player positions in a map layout."""
LAYOUTS_DIR = 'layouts'
TOLERANCE = settings.move_tolerance / 2
def __init__(self, name):
"""
Creates a new Layout object with the given NAME.
:param name: The name of this layout.
"""
self.name = name
self.root = None
@utils.run_if_enabled
def add(self, x, y):
"""
Adds a Node to the quadtree at position (X, Y) if it does not already exist.
:param x: The x-position of the new point.
:param y: The y-position of the new point.
:return: None
"""
def add_helper(node):
if not node:
return Node(x, y)
if y >= node.y and x < node.x:
node.up_left = add_helper(node.up_left)
elif y >= node.y and x >= node.x:
node.up_right = add_helper(node.up_right)
elif y < node.y and x < node.x:
node.down_left = add_helper(node.down_left)
else:
node.down_right = add_helper(node.down_right)
return node
def check_collision(point):
return utils.distance(tuple(point), (x, y)) >= Layout.TOLERANCE
checks = map(check_collision, self.search(x - Layout.TOLERANCE,
x + Layout.TOLERANCE,
y - Layout.TOLERANCE,
y + Layout.TOLERANCE))
if all(checks):
self.root = add_helper(self.root)
def search(self, x_min, x_max, y_min, y_max):
"""
Returns a list of all Nodes bounded horizontally by X_MIN and X_MAX, and bounded
vertically by Y_MIN and Y_MAX.
:param x_min: The left boundary of the range.
:param x_max: The right boundary of the range.
:param y_min: The bottom boundary of the range.
:param y_max: The top boundary of the range.
:return: A list of all Nodes in the range.
"""
nodes = []
def search_helper(node):
if node:
if x_min <= node.x <= x_max and y_min <= node.y <= y_max:
nodes.append(node)
if x_min < node.x:
if y_min < node.y:
search_helper(node.down_left)
if y_max >= node.y:
search_helper(node.up_left)
if x_max >= node.x:
if y_min < node.y:
search_helper(node.down_right)
if y_max >= node.y:
search_helper(node.up_right)
search_helper(self.root)
return nodes
def shortest_path(self, source, target):
"""
Returns the shortest path from A to B using horizontal and vertical teleports.
This method uses a variant of the A* search algorithm.
:param source: The position to start at.
:param target: The destination.
:return: A list of all Nodes on the shortest path in order.
"""
fringe = []
vertices = [source]
distances = [0]
edge_to = [0]
def push_neighbors(index):
"""
Adds possible Nodes that can be reached from POINT (using only one or
two teleports) to the fringe. The Nodes that are returned are all closer
to TARGET than POINT is.
:param index: The index of the current position.
:return: None
"""
point = vertices[index]
def push_best(nodes):
"""
Pushes the Node closest to TARGET to the fringe.
:param nodes: A list of points to compare.
:return: None
"""
if nodes:
points = [tuple(n) for n in nodes]
closest = utils.closest_point(points, target)
# Push to the fringe
distance = distances[index] + utils.distance(point, closest)
heuristic = distance + utils.distance(closest, target)
heappush(fringe, (heuristic, len(vertices)))
# Update vertex and edge lists to include the new node
vertices.append(closest)
distances.append(distance)
edge_to.append(index)
x_error = (target[0] - point[0])
y_error = (target[1] - point[1])
delta = settings.move_tolerance / math.sqrt(2)
# Push best possible node using horizontal teleport
if abs(x_error) > settings.move_tolerance:
if x_error > 0:
x_min = point[0] + settings.move_tolerance / 4
x_max = point[0] + settings.move_tolerance * 2
else:
x_min = point[0] - settings.move_tolerance * 2
x_max = point[0] - settings.move_tolerance / 4
candidates = self.search(x_min,
x_max,
point[1] - delta,
point[1] + delta)
push_best(candidates)
# Push best possible node using vertical teleport
if abs(y_error) > settings.move_tolerance:
if y_error > 0:
y_min = point[1] + settings.move_tolerance / 4
y_max = 1
else:
y_min = 0
y_max = point[1] - settings.move_tolerance / 4
candidates = self.search(point[0] - delta,
point[0] + delta,
y_min,
y_max)
push_best(candidates)
# Perform the A* search algorithm
i = 0
while utils.distance(vertices[i], target) > settings.move_tolerance:
push_neighbors(i)
if len(fringe) == 0:
break
i = heappop(fringe)[1]
# Extract and return shortest path
path = [target]
while i != 0:
path.append(vertices[i])
i = edge_to[i]
path.append(source)
path = list(reversed(path))
config.path = path.copy()
return path
def draw(self, image):
"""
Draws the points in this QuadTree onto IMAGE using in-order traversal.
:param image: The image to draw on.
:return: None
"""
def draw_helper(node):
if node:
draw_helper(node.up_left)
draw_helper(node.down_left)
center = utils.convert_to_absolute(tuple(node), image)
cv2.circle(image, center, 1, (255, 165, 0), -1)
draw_helper(node.up_right)
draw_helper(node.down_right)
draw_helper(self.root)
@staticmethod
def load(routine):
"""
Loads the Layout object associated with ROUTINE. Creates and returns a
new Layout if the specified Layout does not exist.
:param routine: The routine associated with the desired Layout.
:return: A Layout instance.
"""
layout_name = splitext(basename(routine))[0]
target = f'{Layout.LAYOUTS_DIR}/{layout_name}'
if isfile(target):
print(f" ~ Found existing Layout file at '{target}'.")
with open(target, 'rb') as file:
return pickle.load(file)
else:
print(f" ~ Created new Layout file at '{target}'.")
new_layout = Layout(layout_name)
new_layout.save()
return new_layout
@utils.run_if_enabled
def save(self):
"""
Pickles this Layout instance to a file that is named after the routine in which
this Layout was generated.
:return: None
"""
with open(join(Layout.LAYOUTS_DIR, self.name), 'wb') as file:
pickle.dump(self, file)