[ebd3538] | 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 | }
|
---|