1 | package astar;
|
---|
2 |
|
---|
3 | import java.awt.Rectangle;
|
---|
4 | import java.awt.Shape;
|
---|
5 | import java.awt.geom.AffineTransform;
|
---|
6 | import java.util.Collection;
|
---|
7 | import java.util.LinkedList;
|
---|
8 | import java.awt.geom.Point2D;
|
---|
9 | import java.awt.Point;
|
---|
10 |
|
---|
11 | import main.MapObject;
|
---|
12 |
|
---|
13 | import collision.Bound;
|
---|
14 |
|
---|
15 | public class AStarMap {
|
---|
16 | public QuadTree[][] map;
|
---|
17 | private int width;
|
---|
18 | private int height;
|
---|
19 | public static final int ROOT_SIZE = 40;
|
---|
20 | public static final int MAX_TREE_DEPTH = 4;
|
---|
21 |
|
---|
22 | public AStarMap(final int width, final int height) {
|
---|
23 | this.map = new QuadTree[width][height];
|
---|
24 | this.width = width;
|
---|
25 | this.height = height;
|
---|
26 | for (int x = 0; x < width; ++x) {
|
---|
27 | for (int y = 0; y < height; ++y) {
|
---|
28 | this.map[x][y] = new QuadTree(x * 40, y * 40, 40, 40);
|
---|
29 | }
|
---|
30 | }
|
---|
31 | }
|
---|
32 |
|
---|
33 | public AStarNode getNode(final int x, final int y) {
|
---|
34 | return this.getLeaf(this.map[x / 40][y / 40], x, y);
|
---|
35 | }
|
---|
36 |
|
---|
37 | public boolean validPath(final Point source, final Point dest) {
|
---|
38 | final double frac = 5.0 / source.distance(dest);
|
---|
39 | final int xTotal = dest.x - source.x;
|
---|
40 | final int yTotal = dest.y - source.y;
|
---|
41 | for (double xLen = frac * xTotal, yLen = frac * yTotal, xCur = 0.0, yCur = 0.0; Math.abs(xCur) < Math.abs(xTotal); xCur += xLen, yCur += yLen) {
|
---|
42 | final AStarNode curNode = this.getNode((int)xCur + source.x, (int)yCur + source.y);
|
---|
43 | if (!curNode.passable) {
|
---|
44 | return false;
|
---|
45 | }
|
---|
46 | }
|
---|
47 | return true;
|
---|
48 | }
|
---|
49 |
|
---|
50 | private AStarNode getLeaf(final QuadTree qt, final int x, final int y) {
|
---|
51 | if (qt.isLeaf()) {
|
---|
52 | return qt.getNode();
|
---|
53 | }
|
---|
54 | int child = -1;
|
---|
55 | if (x < qt.getX() + qt.getWidth() / 2) {
|
---|
56 | if (y < qt.getY() + qt.getHeight() / 2) {
|
---|
57 | child = 0;
|
---|
58 | }
|
---|
59 | else {
|
---|
60 | child = 2;
|
---|
61 | }
|
---|
62 | }
|
---|
63 | else if (y < qt.getY() + qt.getHeight() / 2) {
|
---|
64 | child = 1;
|
---|
65 | }
|
---|
66 | else {
|
---|
67 | child = 3;
|
---|
68 | }
|
---|
69 | return this.getLeaf(qt.getChildren()[child], x, y);
|
---|
70 | }
|
---|
71 |
|
---|
72 | public void generateAllNeighbors() {
|
---|
73 | for (int x = 0; x < this.width; ++x) {
|
---|
74 | for (int y = 0; y < this.height; ++y) {
|
---|
75 | this.generateNeighborsHelper(this.map[x][y]);
|
---|
76 | }
|
---|
77 | }
|
---|
78 | }
|
---|
79 |
|
---|
80 | public void generateNeighborsHelper(final QuadTree qt) {
|
---|
81 | if (qt.isLeaf()) {
|
---|
82 | qt.getNode().neighbors = this.getNeighbors(qt.getNode());
|
---|
83 | }
|
---|
84 | else {
|
---|
85 | this.generateNeighborsHelper(qt.getChildren()[0]);
|
---|
86 | this.generateNeighborsHelper(qt.getChildren()[1]);
|
---|
87 | this.generateNeighborsHelper(qt.getChildren()[2]);
|
---|
88 | this.generateNeighborsHelper(qt.getChildren()[3]);
|
---|
89 | }
|
---|
90 | }
|
---|
91 |
|
---|
92 | public LinkedList<AStarNode> getNeighbors(final AStarNode n) {
|
---|
93 | final LinkedList<AStarNode> neighbors = new LinkedList<AStarNode>();
|
---|
94 | final int xTop = n.getParent().getX() / 40;
|
---|
95 | final int yTop = (n.getParent().getY() - 1) / 40;
|
---|
96 | if (xTop >= 0 && xTop < this.width && yTop >= 0 && yTop < this.height) {
|
---|
97 | if (1 <= xTop) {
|
---|
98 | neighbors.addAll(this.getTopNeighbors(this.map[xTop - 1][yTop], n));
|
---|
99 | }
|
---|
100 | neighbors.addAll(this.getTopNeighbors(this.map[xTop][yTop], n));
|
---|
101 | if (xTop < this.width - 1) {
|
---|
102 | neighbors.addAll(this.getTopNeighbors(this.map[xTop + 1][yTop], n));
|
---|
103 | }
|
---|
104 | }
|
---|
105 | final int xBottom = n.getParent().getX() / 40;
|
---|
106 | final int yBottom = (n.getParent().getY() + n.getParent().getHeight()) / 40;
|
---|
107 | if (xBottom >= 0 && xBottom < this.width && yBottom >= 0 && yBottom < this.height) {
|
---|
108 | if (1 <= xBottom) {
|
---|
109 | neighbors.addAll(this.getBottomNeighbors(this.map[xBottom - 1][yBottom], n));
|
---|
110 | }
|
---|
111 | neighbors.addAll(this.getBottomNeighbors(this.map[xBottom][yBottom], n));
|
---|
112 | if (xBottom < this.width - 1) {
|
---|
113 | neighbors.addAll(this.getBottomNeighbors(this.map[xBottom + 1][yBottom], n));
|
---|
114 | }
|
---|
115 | }
|
---|
116 | final int xLeft = (n.getParent().getX() - 1) / 40;
|
---|
117 | final int yLeft = n.getParent().getY() / 40;
|
---|
118 | if (xLeft >= 0 && xLeft < this.width && yLeft >= 0 && yLeft < this.height) {
|
---|
119 | neighbors.addAll(this.getLeftNeighbors(this.map[xLeft][yLeft], n));
|
---|
120 | }
|
---|
121 | final int xRight = (n.getParent().getX() + n.getParent().getWidth()) / 40;
|
---|
122 | final int yRight = n.getParent().getY() / 40;
|
---|
123 | if (xRight >= 0 && xRight < this.width && yRight >= 0 && yRight < this.height) {
|
---|
124 | neighbors.addAll(this.getRightNeighbors(this.map[xRight][yRight], n));
|
---|
125 | }
|
---|
126 | return neighbors;
|
---|
127 | }
|
---|
128 |
|
---|
129 | private LinkedList<AStarNode> getTopNeighbors(final QuadTree p, final AStarNode n) {
|
---|
130 | final LinkedList<AStarNode> neighbors = new LinkedList<AStarNode>();
|
---|
131 | final int y = n.getParent().getY();
|
---|
132 | if (p.isLeaf()) {
|
---|
133 | final int xMin = n.getParent().getX();
|
---|
134 | final int xMax = xMin + n.getParent().getWidth();
|
---|
135 | if (p.getY() + p.getHeight() == y && xMax >= p.getX() && p.getX() + p.getWidth() >= xMin) {
|
---|
136 | neighbors.add(p.getNode());
|
---|
137 | }
|
---|
138 | }
|
---|
139 | else if (y <= p.getY() + p.getHeight() / 2) {
|
---|
140 | neighbors.addAll(this.getTopNeighbors(p.getChildren()[0], n));
|
---|
141 | neighbors.addAll(this.getTopNeighbors(p.getChildren()[1], n));
|
---|
142 | }
|
---|
143 | else {
|
---|
144 | neighbors.addAll(this.getTopNeighbors(p.getChildren()[2], n));
|
---|
145 | neighbors.addAll(this.getTopNeighbors(p.getChildren()[3], n));
|
---|
146 | }
|
---|
147 | return neighbors;
|
---|
148 | }
|
---|
149 |
|
---|
150 | private LinkedList<AStarNode> getBottomNeighbors(final QuadTree p, final AStarNode n) {
|
---|
151 | final LinkedList<AStarNode> neighbors = new LinkedList<AStarNode>();
|
---|
152 | final int y = n.getParent().getY() + n.getParent().getHeight();
|
---|
153 | if (p.isLeaf()) {
|
---|
154 | final int xMin = n.getParent().getX();
|
---|
155 | final int xMax = xMin + n.getParent().getWidth();
|
---|
156 | if (p.getY() == y && xMax >= p.getX() && p.getX() + p.getWidth() >= xMin) {
|
---|
157 | neighbors.add(p.getNode());
|
---|
158 | }
|
---|
159 | }
|
---|
160 | else if (y < p.getY() + p.getHeight() / 2) {
|
---|
161 | neighbors.addAll(this.getBottomNeighbors(p.getChildren()[0], n));
|
---|
162 | neighbors.addAll(this.getBottomNeighbors(p.getChildren()[1], n));
|
---|
163 | }
|
---|
164 | else {
|
---|
165 | neighbors.addAll(this.getBottomNeighbors(p.getChildren()[2], n));
|
---|
166 | neighbors.addAll(this.getBottomNeighbors(p.getChildren()[3], n));
|
---|
167 | }
|
---|
168 | return neighbors;
|
---|
169 | }
|
---|
170 |
|
---|
171 | private LinkedList<AStarNode> getLeftNeighbors(final QuadTree p, final AStarNode n) {
|
---|
172 | final LinkedList<AStarNode> neighbors = new LinkedList<AStarNode>();
|
---|
173 | final int x = n.getParent().getX();
|
---|
174 | if (p.isLeaf()) {
|
---|
175 | final int yMin = n.getParent().getY();
|
---|
176 | final int yMax = yMin + n.getParent().getHeight();
|
---|
177 | if (p.getX() + p.getWidth() == x && yMax >= p.getY() && p.getY() + p.getHeight() >= yMin) {
|
---|
178 | neighbors.add(p.getNode());
|
---|
179 | }
|
---|
180 | }
|
---|
181 | else if (x <= p.getX() + p.getWidth() / 2) {
|
---|
182 | neighbors.addAll(this.getLeftNeighbors(p.getChildren()[0], n));
|
---|
183 | neighbors.addAll(this.getLeftNeighbors(p.getChildren()[2], n));
|
---|
184 | }
|
---|
185 | else {
|
---|
186 | neighbors.addAll(this.getLeftNeighbors(p.getChildren()[1], n));
|
---|
187 | neighbors.addAll(this.getLeftNeighbors(p.getChildren()[3], n));
|
---|
188 | }
|
---|
189 | return neighbors;
|
---|
190 | }
|
---|
191 |
|
---|
192 | private LinkedList<AStarNode> getRightNeighbors(final QuadTree p, final AStarNode n) {
|
---|
193 | final LinkedList<AStarNode> neighbors = new LinkedList<AStarNode>();
|
---|
194 | final int x = n.getParent().getX() + n.getParent().getWidth();
|
---|
195 | if (p.isLeaf()) {
|
---|
196 | final int yMin = n.getParent().getY();
|
---|
197 | final int yMax = yMin + n.getParent().getHeight();
|
---|
198 | if (p.getX() == x && yMax >= p.getY() && p.getY() + p.getHeight() >= yMin) {
|
---|
199 | neighbors.add(p.getNode());
|
---|
200 | }
|
---|
201 | }
|
---|
202 | else if (x < p.getX() + p.getWidth() / 2) {
|
---|
203 | neighbors.addAll(this.getRightNeighbors(p.getChildren()[0], n));
|
---|
204 | neighbors.addAll(this.getRightNeighbors(p.getChildren()[2], n));
|
---|
205 | }
|
---|
206 | else {
|
---|
207 | neighbors.addAll(this.getRightNeighbors(p.getChildren()[1], n));
|
---|
208 | neighbors.addAll(this.getRightNeighbors(p.getChildren()[3], n));
|
---|
209 | }
|
---|
210 | return neighbors;
|
---|
211 | }
|
---|
212 |
|
---|
213 | public void addObstacle(final MapObject o) {
|
---|
214 | if (o.getBound() == null) {
|
---|
215 | return;
|
---|
216 | }
|
---|
217 | final Bound realBound = new Bound(o.getBound().createTransformedArea(new AffineTransform(1.0f, 0.0f, 0.0f, 1.0f, o.getBoundX(), o.getBoundY())));
|
---|
218 | final Rectangle rect = realBound.getBounds();
|
---|
219 | int lowXBound = rect.x / 40;
|
---|
220 | int lowYBound = rect.y / 40;
|
---|
221 | int highXBound = (rect.x + rect.width) / 40 + 1;
|
---|
222 | int highYBound = (rect.y + rect.height) / 40 + 1;
|
---|
223 | if (lowXBound < 0) {
|
---|
224 | lowXBound = 0;
|
---|
225 | }
|
---|
226 | if (highXBound > this.width - 1) {
|
---|
227 | highXBound = this.width - 1;
|
---|
228 | }
|
---|
229 | if (lowYBound < 0) {
|
---|
230 | lowYBound = 0;
|
---|
231 | }
|
---|
232 | if (highYBound > this.height - 1) {
|
---|
233 | highYBound = this.height - 1;
|
---|
234 | }
|
---|
235 | for (int x = lowXBound; x <= highXBound; ++x) {
|
---|
236 | for (int y = lowYBound; y <= highYBound; ++y) {
|
---|
237 | this.map[x][y].addObstacle(o);
|
---|
238 | }
|
---|
239 | }
|
---|
240 | }
|
---|
241 |
|
---|
242 | public void coalesceQuadTrees() {
|
---|
243 | for (int x = 0; x < this.width; ++x) {
|
---|
244 | for (int y = 0; y < this.height; ++y) {
|
---|
245 | this.map[x][y].coalesce();
|
---|
246 | }
|
---|
247 | }
|
---|
248 | }
|
---|
249 | }
|
---|