WIP broken AI integration
This commit is contained in:
parent
3eec0e7833
commit
dcfc4ce3d9
15 changed files with 110 additions and 64 deletions
|
@ -1,11 +0,0 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<module type="JAVA_MODULE" version="4">
|
|
||||||
<component name="NewModuleRootManager" inherit-compiler-output="true">
|
|
||||||
<exclude-output />
|
|
||||||
<content url="file://$MODULE_DIR$">
|
|
||||||
<sourceFolder url="file://$MODULE_DIR$/src" isTestSource="false" />
|
|
||||||
</content>
|
|
||||||
<orderEntry type="inheritedJdk" />
|
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
|
||||||
</component>
|
|
||||||
</module>
|
|
|
@ -1,18 +0,0 @@
|
||||||
public class Board {
|
|
||||||
public getAllLegalMoves(){
|
|
||||||
|
|
||||||
}
|
|
||||||
public makeMove(){
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public undoMove(){
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public isGameover(){
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public
|
|
||||||
}
|
|
|
@ -1,2 +0,0 @@
|
||||||
public class Color {
|
|
||||||
}
|
|
|
@ -1,2 +0,0 @@
|
||||||
public class Move {
|
|
||||||
}
|
|
|
@ -1,2 +0,0 @@
|
||||||
public class Piece {
|
|
||||||
}
|
|
|
@ -5,6 +5,10 @@ public class Bishop extends Piece {
|
||||||
super(black, panel, "black-bishop.png", "white-bishop.png");
|
super(black, panel, "black-bishop.png", "white-bishop.png");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
||||||
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
||||||
|
|
||||||
|
|
|
@ -25,6 +25,9 @@ public class Board {
|
||||||
// The current board coordinate that's being dragged
|
// The current board coordinate that's being dragged
|
||||||
BoardCoordinate dragging = null;
|
BoardCoordinate dragging = null;
|
||||||
ArrayList<BoardCoordinate> legalMoves = null;
|
ArrayList<BoardCoordinate> legalMoves = null;
|
||||||
|
Move lastMove;
|
||||||
|
Piece captured;
|
||||||
|
public boolean isGameOver;
|
||||||
|
|
||||||
public Board() {
|
public Board() {
|
||||||
// Initialize DrawingPanel
|
// Initialize DrawingPanel
|
||||||
|
@ -99,15 +102,32 @@ public class Board {
|
||||||
set(coordinate.x, coordinate.y, piece);
|
set(coordinate.x, coordinate.y, piece);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Piece move(int fromX, int fromY, int toX, int toY) {
|
public void move(int fromX, int fromY, int toX, int toY) {
|
||||||
Piece captured = get(toX, toY);
|
lastMove = new Move(new BoardCoordinate(fromX, fromY), new BoardCoordinate(toX, toY));
|
||||||
|
captured = get(toX, toY);
|
||||||
set(toX, toY, get(fromX, fromY));
|
set(toX, toY, get(fromX, fromY));
|
||||||
set(fromX, fromY, null);
|
set(fromX, fromY, null);
|
||||||
return captured;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public Piece move(BoardCoordinate from, BoardCoordinate to) {
|
public void move(BoardCoordinate from, BoardCoordinate to) {
|
||||||
return move(from.x, from.y, to.x, to.y);
|
move(from.x, from.y, to.x, to.y);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void move(Move move) {
|
||||||
|
move(move.from, move.to);
|
||||||
|
try {
|
||||||
|
Thread.sleep(500);
|
||||||
|
} catch(InterruptedException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void undoMove() {
|
||||||
|
if (lastMove == null) return;
|
||||||
|
set(lastMove.from, get(lastMove.to));
|
||||||
|
set(lastMove.to, captured);
|
||||||
|
lastMove = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mouse down event handler
|
// Mouse down event handler
|
||||||
|
@ -117,7 +137,7 @@ public class Board {
|
||||||
BoardCoordinate coordinate = new ScreenCoordinate(x, y).toBoard();
|
BoardCoordinate coordinate = new ScreenCoordinate(x, y).toBoard();
|
||||||
// If there's no piece there, return
|
// If there's no piece there, return
|
||||||
Piece piece = get(coordinate);
|
Piece piece = get(coordinate);
|
||||||
if (piece == null) return;
|
if (piece == null || piece.black) return;
|
||||||
// Set currently dragging piece to that coordinate
|
// Set currently dragging piece to that coordinate
|
||||||
dragging = coordinate;
|
dragging = coordinate;
|
||||||
legalMoves = piece.getLegalMoves(coordinate, this);
|
legalMoves = piece.getLegalMoves(coordinate, this);
|
||||||
|
@ -132,6 +152,9 @@ public class Board {
|
||||||
if (dragging != null && !newCoordinate.equals(dragging)) {
|
if (dragging != null && !newCoordinate.equals(dragging)) {
|
||||||
// dragging is BoardCoordinate of piece being dragged
|
// dragging is BoardCoordinate of piece being dragged
|
||||||
Piece piece = get(dragging);
|
Piece piece = get(dragging);
|
||||||
|
move(dragging, newCoordinate);
|
||||||
|
move(ChessAI.findBestMove(this));
|
||||||
|
/*
|
||||||
ArrayList<BoardCoordinate> legalMoves = piece.getLegalMoves(dragging, this);
|
ArrayList<BoardCoordinate> legalMoves = piece.getLegalMoves(dragging, this);
|
||||||
for (BoardCoordinate legalMove : legalMoves) {
|
for (BoardCoordinate legalMove : legalMoves) {
|
||||||
if (newCoordinate.equals(legalMove)) {
|
if (newCoordinate.equals(legalMove)) {
|
||||||
|
@ -148,10 +171,15 @@ public class Board {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (inCheck && oppositeKing.getLegalMoves(oppositeKingPosition, this).size() == 0) System.out.println("Checkmate");
|
if (inCheck && oppositeKing.getLegalMoves(oppositeKingPosition, this).size() == 0)
|
||||||
|
isGameOver = true;
|
||||||
|
|
||||||
|
|
||||||
|
move(ChessAI.findBestMove(this));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
// Clear dragging
|
// Clear dragging
|
||||||
dragging = null;
|
dragging = null;
|
||||||
|
@ -187,10 +215,12 @@ public class Board {
|
||||||
graphics.setColor(new Color(0, 128, 0, 128));
|
graphics.setColor(new Color(0, 128, 0, 128));
|
||||||
for (BoardCoordinate legalMove : legalMoves)
|
for (BoardCoordinate legalMove : legalMoves)
|
||||||
drawRect(legalMove);
|
drawRect(legalMove);
|
||||||
BoardCoordinate hovering = mousePosition.toBoard();
|
if (mousePosition != null) {
|
||||||
if (legalMoves.contains(hovering)) {
|
BoardCoordinate hovering = mousePosition.toBoard();
|
||||||
graphics.setColor(get(hovering) == null ? new Color(0, 0, 255, 128) : new Color(255, 0, 0, 128));
|
if (legalMoves.contains(hovering)) {
|
||||||
drawRect(mousePosition.toBoard());
|
graphics.setColor(get(hovering) == null ? new Color(0, 0, 255, 128) : new Color(255, 0, 0, 128));
|
||||||
|
drawRect(mousePosition.toBoard());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -237,4 +267,15 @@ public class Board {
|
||||||
public void forEachPiece(PieceActionCoordinate tileAction) {
|
public void forEachPiece(PieceActionCoordinate tileAction) {
|
||||||
forEachPiece((x, y, piece) -> tileAction.forEachTile(new BoardCoordinate(x, y), get(x, y)));
|
forEachPiece((x, y, piece) -> tileAction.forEachTile(new BoardCoordinate(x, y), get(x, y)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public ArrayList<Move> getAllLegalMoves() {
|
||||||
|
ArrayList<Move> allLegalMoves = new ArrayList<>();
|
||||||
|
forEachPiece((from, piece) -> {
|
||||||
|
ArrayList<BoardCoordinate> legalTiles = piece.getLegalMoves(from, this);
|
||||||
|
for (BoardCoordinate to : legalTiles) {
|
||||||
|
allLegalMoves.add(new Move(from, to));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return allLegalMoves;
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -1,12 +1,12 @@
|
||||||
public class ChessAI {
|
public class ChessAI {
|
||||||
private final int MAX_DEPTH = 4;
|
private static final int MAX_DEPTH = 4;
|
||||||
|
|
||||||
public Move findBestMove(Board board) {
|
public static Move findBestMove(Board board) {
|
||||||
int bestScore = Integer.MIN_VALUE;
|
int bestScore = Integer.MIN_VALUE;
|
||||||
Move bestMove = null;
|
Move bestMove = null;
|
||||||
|
|
||||||
for (Move move : board.getAllLegalMoves()) {
|
for (Move move : board.getAllLegalMoves()) {
|
||||||
board.makeMove(move);
|
board.move(move);
|
||||||
int score = minimax(board, MAX_DEPTH, Integer.MIN_VALUE, Integer.MAX_VALUE, false);
|
int score = minimax(board, MAX_DEPTH, Integer.MIN_VALUE, Integer.MAX_VALUE, false);
|
||||||
board.undoMove();
|
board.undoMove();
|
||||||
|
|
||||||
|
@ -19,8 +19,8 @@ public class ChessAI {
|
||||||
return bestMove;
|
return bestMove;
|
||||||
}
|
}
|
||||||
|
|
||||||
private int minimax(Board board, int depth, int alpha, int beta, boolean maximizingPlayer) {
|
private static int minimax(Board board, int depth, int alpha, int beta, boolean maximizingPlayer) {
|
||||||
if (depth == 0 || board.isGameOver()) {
|
if (depth == 0 || board.isGameOver) {
|
||||||
return evaluateBoard(board);
|
return evaluateBoard(board);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -28,7 +28,7 @@ public class ChessAI {
|
||||||
if (maximizingPlayer) {
|
if (maximizingPlayer) {
|
||||||
score = Integer.MIN_VALUE;
|
score = Integer.MIN_VALUE;
|
||||||
for (Move move : board.getAllLegalMoves()) {
|
for (Move move : board.getAllLegalMoves()) {
|
||||||
board.makeMove(move);
|
board.move(move);
|
||||||
score = Math.max(score, minimax(board, depth - 1, alpha, beta, false));
|
score = Math.max(score, minimax(board, depth - 1, alpha, beta, false));
|
||||||
board.undoMove();
|
board.undoMove();
|
||||||
alpha = Math.max(alpha, score);
|
alpha = Math.max(alpha, score);
|
||||||
|
@ -39,7 +39,7 @@ public class ChessAI {
|
||||||
} else {
|
} else {
|
||||||
score = Integer.MAX_VALUE;
|
score = Integer.MAX_VALUE;
|
||||||
for (Move move : board.getAllLegalMoves()) {
|
for (Move move : board.getAllLegalMoves()) {
|
||||||
board.makeMove(move);
|
board.move(move);
|
||||||
score = Math.min(score, minimax(board, depth - 1, alpha, beta, true));
|
score = Math.min(score, minimax(board, depth - 1, alpha, beta, true));
|
||||||
board.undoMove();
|
board.undoMove();
|
||||||
beta = Math.min(beta, score);
|
beta = Math.min(beta, score);
|
||||||
|
@ -52,16 +52,22 @@ public class ChessAI {
|
||||||
return score;
|
return score;
|
||||||
}
|
}
|
||||||
|
|
||||||
private int evaluateBoard(Board board) {
|
private static int evaluateBoard(Board board) {
|
||||||
int score = 0;
|
Score score = new Score();
|
||||||
for (Piece piece : board.getPieces()) {
|
board.forEachPiece((coordinate, piece) -> {
|
||||||
if (piece.getColor() == Color.WHITE) {
|
if (!piece.black) {
|
||||||
score += piece.getValue();
|
score.add(piece.getValue());
|
||||||
} else {
|
} else {
|
||||||
score -= piece.getValue();
|
score.add(piece.getValue());
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
return score;
|
return score.score;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class Score {
|
||||||
|
public int score;
|
||||||
|
public void add(int value) {
|
||||||
|
score += value;
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,6 +5,10 @@ public class King extends Piece {
|
||||||
super(black, panel, "black-king.png", "white-king.png");
|
super(black, panel, "black-king.png", "white-king.png");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return Integer.MAX_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
||||||
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
||||||
possibleMoves.add(new BoardCoordinate(position.x - 1, position.y - 1));
|
possibleMoves.add(new BoardCoordinate(position.x - 1, position.y - 1));
|
||||||
|
|
|
@ -6,6 +6,10 @@ public class Knight extends Piece {
|
||||||
super(black, panel, "black-knight.png", "white-knight.png");
|
super(black, panel, "black-knight.png", "white-knight.png");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
||||||
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
||||||
possibleMoves.add(new BoardCoordinate(position.x - 2, position.y - 1));
|
possibleMoves.add(new BoardCoordinate(position.x - 2, position.y - 1));
|
||||||
|
|
9
src/Move.java
Normal file
9
src/Move.java
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
public class Move {
|
||||||
|
public BoardCoordinate from;
|
||||||
|
public BoardCoordinate to;
|
||||||
|
|
||||||
|
public Move(BoardCoordinate from, BoardCoordinate to) {
|
||||||
|
this.from = from;
|
||||||
|
this.to = to;
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,6 +5,10 @@ public class Pawn extends Piece {
|
||||||
super(black, panel, "black-pawn.png", "white-pawn.png");
|
super(black, panel, "black-pawn.png", "white-pawn.png");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
||||||
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
||||||
if (this.black) {
|
if (this.black) {
|
||||||
|
|
|
@ -9,6 +9,8 @@ public abstract class Piece {
|
||||||
Image image;
|
Image image;
|
||||||
public boolean black;
|
public boolean black;
|
||||||
|
|
||||||
|
public abstract int getValue();
|
||||||
|
|
||||||
public Piece(boolean black, DrawingPanel panel, String blackImagePath, String whiteImagePath) {
|
public Piece(boolean black, DrawingPanel panel, String blackImagePath, String whiteImagePath) {
|
||||||
this.black = black;
|
this.black = black;
|
||||||
image = panel.loadImage(black ? blackImagePath : whiteImagePath);
|
image = panel.loadImage(black ? blackImagePath : whiteImagePath);
|
||||||
|
@ -48,7 +50,7 @@ public abstract class Piece {
|
||||||
|
|
||||||
boolean isInCheck(BoardCoordinate from, BoardCoordinate to, Board board) {
|
boolean isInCheck(BoardCoordinate from, BoardCoordinate to, Board board) {
|
||||||
boolean isInCheck = false;
|
boolean isInCheck = false;
|
||||||
Piece captured = board.move(from, to);
|
board.move(from, to);
|
||||||
outer: for (int y = 0; y < Board.BOARD_SIZE; y++) {
|
outer: for (int y = 0; y < Board.BOARD_SIZE; y++) {
|
||||||
for (int x = 0; x < Board.BOARD_SIZE; x++) {
|
for (int x = 0; x < Board.BOARD_SIZE; x++) {
|
||||||
Piece piece = board.get(x, y);
|
Piece piece = board.get(x, y);
|
||||||
|
@ -63,8 +65,7 @@ public abstract class Piece {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
board.move(to, from);
|
board.undoMove();
|
||||||
board.set(to, captured);
|
|
||||||
return isInCheck;
|
return isInCheck;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,10 @@ public class Queen extends Piece {
|
||||||
super(black, panel, "black-queen.png", "white-queen.png");
|
super(black, panel, "black-queen.png", "white-queen.png");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return 9;
|
||||||
|
}
|
||||||
|
|
||||||
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
||||||
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,10 @@ public class Rook extends Piece {
|
||||||
super(black, panel, "black-rook.png", "white-rook.png");
|
super(black, panel, "black-rook.png", "white-rook.png");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return 5;
|
||||||
|
}
|
||||||
|
|
||||||
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
public ArrayList<BoardCoordinate> getPossibleMoves(BoardCoordinate position, Board board) {
|
||||||
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
ArrayList<BoardCoordinate> possibleMoves = new ArrayList<>();
|
||||||
|
|
||||||
|
|
Reference in a new issue