diff options
Diffstat (limited to 'core/src')
| -rw-r--r-- | core/src/ch/asynk/tankontank/engine/PossiblePaths.java | 297 | 
1 files changed, 297 insertions, 0 deletions
| diff --git a/core/src/ch/asynk/tankontank/engine/PossiblePaths.java b/core/src/ch/asynk/tankontank/engine/PossiblePaths.java new file mode 100644 index 0000000..7f2d430 --- /dev/null +++ b/core/src/ch/asynk/tankontank/engine/PossiblePaths.java @@ -0,0 +1,297 @@ +package ch.asynk.tankontank.engine; + +import java.util.List; +import java.util.Iterator; +import java.util.ArrayList; +import java.util.LinkedList; + +import com.badlogic.gdx.math.Vector2; +import com.badlogic.gdx.math.Vector3; + +public class PossiblePaths implements Iterable<Vector3> +{ +    private final Board board; + +    private Pawn pawn; +    private Tile to; +    private Orientation lastO; +    private List<Tile> stack; +    private List<Tile> ctrlTiles; +    private List<ArrayList<Tile>> paths; +    private List<ArrayList<Tile>> filteredPaths; +    private Board.TileCollection tiles; + +    public PossiblePaths(Board board, int tSize, int stSize, int ftSize, int vectSize) +    { +        this.board = board; +        this.tiles = new TileSet(board, tSize); +        this.stack = new ArrayList<Tile>(stSize); +        this.ctrlTiles = new ArrayList<Tile>(ftSize); +        this.paths = new LinkedList<ArrayList<Tile>>(); +        this.filteredPaths = new LinkedList<ArrayList<Tile>>(); +        this.lastO = Orientation.KEEP; +    } + +    public void setLastOrientation(Orientation lastO) +    { +        this.lastO = lastO; +    } + +    public int init(Pawn pawn, Tile to) +    { +        this.pawn = pawn; +        this.to = to; +        clear(); + +        return build(); +    } + +    public void clear() +    { +        for (List<Tile> tiles : this.paths) tiles.clear(); +        for (List<Tile> tiles : this.filteredPaths) tiles.clear(); +        this.stack.clear(); +        this.paths.clear(); +        this.ctrlTiles.clear(); +        this.filteredPaths.clear(); +        this.tiles.clear(); +    } + +    public int size() +    { +        if (ctrlTiles.size() == 0) +            return paths.size(); +        return filteredPaths.size(); +    } + +    public boolean contains(Tile tile) +    { +        return tiles.contains(tile); +    } + +    public void enable(int i, boolean enable) +    { +        for (Tile tile : tiles) +            board.enableOverlayOn(tile, i, enable); +    } + +    private int build() +    { +        // from and to are not part of the path +        Tile from = pawn.getTile(); +        if (board.distance(from, to) == 1) { +            ArrayList<Tile> temp = new ArrayList<Tile>(0); +            // temp.add(from); +            // temp.add(to); +            paths.add(temp); +            for (Tile tile : temp) tiles.add(tile); +        } else { +            // stack.add(from); +            findAllPaths(from, pawn.getMovementPoints(), true); +        } + +        // printToErr("paths", paths); +        stack.clear(); +        return paths.size(); +    } + +    private void findAllPaths(Tile from, int mvtLeft, boolean roadMarch) +    { +        Tile moves[] = new Tile[6]; +        board.setAdjacentTiles(from, moves); + +        for(int i = 0; i < 6; i++) { +            Tile next = moves[i]; +            if (next == null) continue; + +            boolean road = next.road(board.getSide(i)); +            int cost = next.costFrom(pawn, board.getSide(i), road); +            int r = (mvtLeft - cost); +            if (roadMarch & road) r += pawn.getRoadMarchBonus(); + +            if ((board.distance(next, to) <= r)) { +                if (next == to) { +                    ArrayList<Tile> temp = new ArrayList<Tile>(stack.size() + 1); +                    for (Tile t: stack) +                        temp.add(t); +                    // temp.add(next); +                    paths.add(temp); +                    for (Tile tile : temp) tiles.add(tile); +                } else { +                    stack.add(next); +                    findAllPaths(next, (mvtLeft - cost), (roadMarch & road)); +                    stack.remove(stack.size() - 1); +                } +            } +        } +    } + +    public int toggleCtrlTile(Tile tile) +    { +        if (ctrlTiles.contains(tile)) +            ctrlTiles.remove(tile); +        else +            ctrlTiles.add(tile); +        return filterPaths(); +    } + +    private int filterPaths() +    { +        int s = ctrlTiles.size(); + +        tiles.clear(); +        filteredPaths.clear(); +        for (ArrayList<Tile> path : paths) { +            int ok = 0; +            for (Tile filter : ctrlTiles) { +                if (path.contains(filter)) +                    ok += 1; +            } +            if (ok == s) { +                if (path.size() == (s + 0)) { // from and to are not part of the path +                    filteredPaths.clear(); +                    filteredPaths.add(path); +                    tiles.clear(); +                    for (Tile tile : path) tiles.add(tile); +                    break; +                } else { +                    filteredPaths.add(path); +                    for (Tile tile : path) tiles.add(tile); +                } +            } +        } + +        // printToErr("filteredPaths", filteredPaths); +        return filteredPaths.size(); +    } + +    public int pathCost(int i) +    { +        int cost = 0; +        boolean roadMarch = true; +        Tile prev = null; + +        for (Tile next : paths.get(i)) { +            if (prev != null) { +                Orientation o = Orientation.fromMove(next.col, next.row, prev.col, prev.row); +                boolean road = next.road(o); +                cost += next.costFrom(pawn, o, road); +                roadMarch &= road; +            } +            prev = next; +        } + +        if (roadMarch) +            cost -= pawn.getRoadMarchBonus(); +        if (cost < 1) +            cost = 1; + +        return cost; +    } + +    public int pathSteps(int idx) +    { +        int steps = 0; + +        Tile tile = pawn.getTile(); +        Orientation o = pawn.getOrientation(); +        for (Tile next : filteredPaths.get(idx)) { +            Orientation nextO = Orientation.fromMove(tile.col, tile.row, next.col, next.row); +            if (nextO != o) { +                steps += 2; +                o = nextO; +            } else +                steps += 1; +            tile = next; +        } +        if (lastO != Orientation.fromMove(tile.col, tile.row, to.col, to.row)) +            steps += 2; +        else +            steps +=1; + +        return steps; +    } + +    @Override +    public Iterator<Vector3> iterator() +    { +        return new Vector3Iterator(pawn, to, lastO, filteredPaths.get(0)); +    } + +    private void printToErr(String what, List<ArrayList<Tile>> paths) +    { +        System.err.println(what + " ("+paths.size()+")"); +        for (ArrayList<Tile> path : paths) { +            System.err.println(" - path"); +            for(Tile tile : path) +                System.err.println("   " + tile); +        } +        System.err.println(); +    } +} + +class Vector3Iterator implements Iterator<Vector3> +{ +    private Pawn pawn; +    private Tile to; +    private Orientation o; +    private Orientation lastO; +    private Tile tile; +    private Vector2 pos = new Vector2(); +    private Vector3 v = new Vector3(); +    private int i; +    private List<Tile> path; + +    public Vector3Iterator(Pawn pawn, Tile to, Orientation lastO, List<Tile> path) +    { +        this.pawn = pawn; +        this.to = to; +        this.lastO = lastO; +        this.path = path; +        this.tile = pawn.getTile(); +        this.o = pawn.getOrientation(); +        this.v.set(pawn.getPosition().x, pawn.getPosition().y, o.r()); +        this.i = 0; +    } + +    @Override +    public boolean hasNext() +    { +        if ((tile == to) && (o == lastO)) +            return false; +        return true; +    } + +    @Override +    public Vector3 next() +    { +        if (tile == to) { +            v.z = lastO.r(); +            o = lastO; +            return v; +        } +        Tile nextTile; +        if (i < path.size()) +            nextTile = path.get(i); +        else +            nextTile = to; +        Orientation nextO = Orientation.fromMove(tile.col, tile.row, nextTile.col, nextTile.row); +        if (nextO != o) { +            v.z = nextO.r(); +            o = nextO; +            return v; +        } +        pawn.getPosAt(nextTile, pos); +        v.x = pos.x; +        v.y = pos.y; +        tile = nextTile; +        i += 1; +        return v; +    } + +    @Override +    public void remove() +    { +        throw new UnsupportedOperationException(); +    } +} | 
