Compare commits
8 commits
5e1553aaf6
...
9fbc09ea07
Author | SHA1 | Date | |
---|---|---|---|
9fbc09ea07 | |||
101054091d | |||
851c2f401d | |||
acb9e05e07 | |||
2c5d9079cd | |||
e08132376a | |||
3c6db20f06 | |||
e2ca1c2f38 |
5 changed files with 58 additions and 15 deletions
|
@ -17,11 +17,15 @@ public class Board {
|
||||||
// new Color() takes in an integer representing the color
|
// new Color() takes in an integer representing the color
|
||||||
// Colors are represented in hexadecimal, so we can write the hex literal by prefixing the color code with 0x
|
// Colors are represented in hexadecimal, so we can write the hex literal by prefixing the color code with 0x
|
||||||
static final Color BLACK = new Color(0x6c595c);
|
static final Color BLACK = new Color(0x6c595c);
|
||||||
static final Color WHITE = new Color(0xab9b8e);
|
static final Color WHITE = new Color(0x847875);
|
||||||
|
static final Color HIGHLIGHT = new Color(0xab9b8e7f, true);
|
||||||
|
static final Color MOVE_HIGHLIGHT = new Color(0xafb381);
|
||||||
|
static final Color CAPTURE_HIGHLIGHT = new Color(0xb65c5f);
|
||||||
King blackKing;
|
King blackKing;
|
||||||
King whiteKing;
|
King whiteKing;
|
||||||
final DrawingPanel panel;
|
final DrawingPanel panel;
|
||||||
final Graphics graphics;
|
final Graphics graphics;
|
||||||
|
public boolean aiThinking = false;
|
||||||
|
|
||||||
// The board is a two-dimensional array of nullable pieces
|
// The board is a two-dimensional array of nullable pieces
|
||||||
Piece[][] board;
|
Piece[][] board;
|
||||||
|
@ -59,6 +63,9 @@ public class Board {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setup() {
|
public void setup() {
|
||||||
|
// Clear move history
|
||||||
|
moveHistory.clear();
|
||||||
|
|
||||||
// Initialize board
|
// Initialize board
|
||||||
board = new Piece[BOARD_SIZE][BOARD_SIZE];
|
board = new Piece[BOARD_SIZE][BOARD_SIZE];
|
||||||
|
|
||||||
|
@ -167,6 +174,7 @@ public class Board {
|
||||||
setup();
|
setup();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (aiThinking) return;
|
||||||
// Get board coordinate of mouse click
|
// Get board coordinate of mouse click
|
||||||
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
|
||||||
|
@ -184,6 +192,7 @@ public class Board {
|
||||||
}
|
}
|
||||||
|
|
||||||
void handleMouseUp(int x, int y) {
|
void handleMouseUp(int x, int y) {
|
||||||
|
if (aiThinking) return;
|
||||||
// Get board coordinate of mouse release
|
// Get board coordinate of mouse release
|
||||||
BoardCoordinate newCoordinate = new ScreenCoordinate(x, y).toBoard();
|
BoardCoordinate newCoordinate = new ScreenCoordinate(x, y).toBoard();
|
||||||
// Only do something if new coordinate is different from the originating coordinate
|
// Only do something if new coordinate is different from the originating coordinate
|
||||||
|
@ -196,12 +205,18 @@ public class Board {
|
||||||
move(legalMove);
|
move(legalMove);
|
||||||
setLastMovedPieceAsMoved();
|
setLastMovedPieceAsMoved();
|
||||||
checkForCheckmate();
|
checkForCheckmate();
|
||||||
|
// Clear dragging
|
||||||
|
dragging = null;
|
||||||
|
// Redraw without dragging
|
||||||
|
draw();
|
||||||
if (!isGameOver) {
|
if (!isGameOver) {
|
||||||
move(ChessAI.findBestMove(this));
|
try {
|
||||||
setLastMovedPieceAsMoved();
|
ChessAI.move(this);
|
||||||
checkForCheckmate();
|
} catch (Exception e) {
|
||||||
|
System.out.println(e);
|
||||||
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -232,7 +247,7 @@ public class Board {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (getAllLegalMoves(movedPiece.black).isEmpty()) {
|
} else if (getAllLegalMoves(!movedPiece.black).isEmpty()) {
|
||||||
isGameOver = true;
|
isGameOver = true;
|
||||||
isStalemate = true;
|
isStalemate = true;
|
||||||
}
|
}
|
||||||
|
@ -264,14 +279,14 @@ public class Board {
|
||||||
for (int x = y % 2; x < BOARD_SIZE; x += 2)
|
for (int x = y % 2; x < BOARD_SIZE; x += 2)
|
||||||
drawRect(x, y);
|
drawRect(x, y);
|
||||||
if (dragging != null) {
|
if (dragging != null) {
|
||||||
graphics.setColor(new Color(0, 128, 0, 128));
|
graphics.setColor(HIGHLIGHT);
|
||||||
for (Move legalMove : legalMoves)
|
for (Move legalMove : legalMoves)
|
||||||
drawRect(legalMove.to);
|
drawRect(legalMove.to);
|
||||||
if (mousePosition != null) {
|
if (mousePosition != null) {
|
||||||
BoardCoordinate hovering = mousePosition.toBoard();
|
BoardCoordinate hovering = mousePosition.toBoard();
|
||||||
for (Move legalMove : legalMoves) {
|
for (Move legalMove : legalMoves) {
|
||||||
if (legalMove.to.equals(hovering)) {
|
if (legalMove.to.equals(hovering)) {
|
||||||
graphics.setColor(get(hovering) == null ? new Color(0, 0, 255, 128) : new Color(255, 0, 0, 128));
|
graphics.setColor(get(hovering) == null ? MOVE_HIGHLIGHT : CAPTURE_HIGHLIGHT);
|
||||||
drawRect(mousePosition.toBoard());
|
drawRect(mousePosition.toBoard());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -283,6 +298,10 @@ public class Board {
|
||||||
forEachPiece((boardCoordinate, piece) -> {
|
forEachPiece((boardCoordinate, piece) -> {
|
||||||
// If piece is the one being dragged, render it at the mouse position
|
// If piece is the one being dragged, render it at the mouse position
|
||||||
// Otherwise, render it at the center of the board tile
|
// Otherwise, render it at the center of the board tile
|
||||||
|
if (piece instanceof King && piece.isInCheck(this)) {
|
||||||
|
graphics.setColor(CAPTURE_HIGHLIGHT);
|
||||||
|
drawRect(boardCoordinate);
|
||||||
|
}
|
||||||
piece.draw(graphics, panel, boardCoordinate.equals(dragging) ? mousePosition : boardCoordinate.toScreen());
|
piece.draw(graphics, panel, boardCoordinate.equals(dragging) ? mousePosition : boardCoordinate.toScreen());
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -1,15 +1,39 @@
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
import java.util.concurrent.Callable;
|
||||||
|
import java.util.concurrent.Future;
|
||||||
|
import java.util.concurrent.FutureTask;
|
||||||
|
|
||||||
public class ChessAI {
|
public class ChessAI {
|
||||||
private static final int MAX_DEPTH = 3;
|
private static final int MAX_DEPTH = 3;
|
||||||
|
|
||||||
|
public static FutureTask<Void> move(Board board) {
|
||||||
|
Callable<Void> callable = new Callable<>() {
|
||||||
|
@Override
|
||||||
|
public Void call() throws Exception {
|
||||||
|
board.aiThinking = true;
|
||||||
|
Move move = findBestMove(board);
|
||||||
|
board.move(move);
|
||||||
|
board.setLastMovedPieceAsMoved();
|
||||||
|
board.checkForCheckmate();
|
||||||
|
board.draw();
|
||||||
|
board.aiThinking = false;
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
FutureTask<Void> future = new FutureTask<>(callable);
|
||||||
|
Thread thread = new Thread(future);
|
||||||
|
thread.start();
|
||||||
|
return future;
|
||||||
|
}
|
||||||
|
|
||||||
public static Move findBestMove(Board board) {
|
public static Move findBestMove(Board board) {
|
||||||
int bestScore = Integer.MIN_VALUE;
|
int bestScore = Integer.MIN_VALUE;
|
||||||
Move bestMove = null;
|
|
||||||
|
|
||||||
ArrayList<Move> legalMoves = board.getAllLegalMoves(true);
|
ArrayList<Move> legalMoves = board.getAllLegalMoves(true);
|
||||||
|
|
||||||
Collections.shuffle(legalMoves);
|
Collections.shuffle(legalMoves);
|
||||||
|
Move bestMove = legalMoves.get(0);
|
||||||
for (Move move : legalMoves) {
|
for (Move move : legalMoves) {
|
||||||
board.move(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);
|
||||||
|
|
|
@ -28,8 +28,8 @@ public class King extends Piece {
|
||||||
rightRook instanceof Rook &&
|
rightRook instanceof Rook &&
|
||||||
!rightRook.moved &&
|
!rightRook.moved &&
|
||||||
board.get(position.x + 1, position.y) == null &&
|
board.get(position.x + 1, position.y) == null &&
|
||||||
board.get(position.x + 2, position.y) == null &&
|
board.get(position.x + 2, position.y) == null // &&
|
||||||
!isInCheck(board)
|
// !isInCheck(board)
|
||||||
) {
|
) {
|
||||||
// TODO: Does not take into account squares in castling path being threatened
|
// TODO: Does not take into account squares in castling path being threatened
|
||||||
Move rightCastle = new Move(position, new BoardCoordinate(position.x + 2, position.y));
|
Move rightCastle = new Move(position, new BoardCoordinate(position.x + 2, position.y));
|
||||||
|
@ -46,8 +46,8 @@ public class King extends Piece {
|
||||||
!leftRook.moved &&
|
!leftRook.moved &&
|
||||||
board.get(position.x - 1, position.y) == null &&
|
board.get(position.x - 1, position.y) == null &&
|
||||||
board.get(position.x - 2, position.y) == null &&
|
board.get(position.x - 2, position.y) == null &&
|
||||||
board.get(position.x - 3, position.y) == null &&
|
board.get(position.x - 3, position.y) == null // &&
|
||||||
!isInCheck(board)
|
// !isInCheck(board)
|
||||||
) {
|
) {
|
||||||
// TODO: Does not take into account squares in castling path being threatened
|
// TODO: Does not take into account squares in castling path being threatened
|
||||||
Move leftCastle = new Move(position, new BoardCoordinate(position.x - 2, position.y));
|
Move leftCastle = new Move(position, new BoardCoordinate(position.x - 2, position.y));
|
||||||
|
|
|
@ -24,7 +24,7 @@ public class Pawn extends Piece {
|
||||||
if (board.get(position.x + 1, position.y + 1) != null) {
|
if (board.get(position.x + 1, position.y + 1) != null) {
|
||||||
possibleMoves.add(new Move(position, new BoardCoordinate(position.x + 1,position.y + 1)));
|
possibleMoves.add(new Move(position, new BoardCoordinate(position.x + 1,position.y + 1)));
|
||||||
}
|
}
|
||||||
for (Move move : possibleMoves) move.isPromotion = position.y + 1 == 0;
|
for (Move move : possibleMoves) move.isPromotion = position.y + 1 == Board.BOARD_SIZE - 1;
|
||||||
} else {
|
} else {
|
||||||
if (board.get(position.x, position.y - 1) == null) {
|
if (board.get(position.x, position.y - 1) == null) {
|
||||||
possibleMoves.add(new Move(position, new BoardCoordinate(position.x, position.y - 1)));
|
possibleMoves.add(new Move(position, new BoardCoordinate(position.x, position.y - 1)));
|
||||||
|
|
|
@ -61,7 +61,7 @@ public abstract class Piece {
|
||||||
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);
|
||||||
if (piece == null || piece.black == black || piece instanceof King) continue;
|
if (piece == null || piece.black == black) continue;
|
||||||
ArrayList<Move> legalMoves = piece.getLegalMoves(new BoardCoordinate(x, y), board, false);
|
ArrayList<Move> legalMoves = piece.getLegalMoves(new BoardCoordinate(x, y), board, false);
|
||||||
for (Move legalMove : legalMoves) {
|
for (Move legalMove : legalMoves) {
|
||||||
Piece pieceAtMove = board.get(legalMove.to);
|
Piece pieceAtMove = board.get(legalMove.to);
|
||||||
|
|
Reference in a new issue