Want to contribute? Fork me on Codeberg.org!
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
This repo is archived. You can view files and clone it, but cannot push or open issues/pull-requests.

301 lines
10 KiB

import java.awt.*;
import java.awt.image.ImageObserver;
import java.util.ArrayList;
2 years ago
import java.util.LinkedList;
import java.util.Queue;
import java.util.Stack;
2 years ago
public class Board {
// Width and height of each board tile
2 years ago
static final int TILE_SIZE = 64;
// Width and height of board in tiles
2 years ago
static final int BOARD_SIZE = 8;
// Width and height of window in pixels
2 years ago
static final int DIMENSION = TILE_SIZE * BOARD_SIZE;
// Colors from Lost Century 24
// https://lospec.com/palette-list/lost-century-24
// 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
2 years ago
static final Color BLACK = new Color(0x6c595c);
static final Color WHITE = new Color(0xab9b8e);
King blackKing;
King whiteKing;
2 years ago
final DrawingPanel panel;
final Graphics graphics;
// The board is a two-dimensional array of nullable pieces
2 years ago
Piece[][] board;
// The current board coordinate that's being dragged
2 years ago
BoardCoordinate dragging = null;
ArrayList<BoardCoordinate> legalMoves = null;
2 years ago
Stack<Move> moveHistory;
public boolean isGameOver;
public boolean victor;
Image youWin;
Image youLose;
2 years ago
public Board() {
2 years ago
moveHistory = new Stack<>();
// Initialize DrawingPanel
2 years ago
panel = new DrawingPanel(DIMENSION, DIMENSION);
youWin = panel.loadImage("you-win.png");
youLose = panel.loadImage("you-lose.png");
graphics = panel.getGraphics();
// Connect up event handlers
2 years ago
panel.onMouseDown(this::handleMouseDown);
panel.onMouseDrag((x, y) -> {
// We want to re-render with new mouse position when dragging pieces
if (dragging != null) draw(x, y);
});
2 years ago
panel.onMouseUp(this::handleMouseUp);
// Initialize board
setup();
}
public void setup() {
// Initialize board
2 years ago
board = new Piece[BOARD_SIZE][BOARD_SIZE];
// Initialize pieces
for(int i = 0; i <= 7; i++){
if(i == 0 || i == 7){
for(int j = 0; j <= 7; j++){
if(j == 0 || j == 7){
set(j, i, new Rook(i==0, panel));
} else if(j == 1 || j == 6){
set(j, i, new Knight(i==0, panel));
} else if(j == 2 || j == 5){
set(j, i, new Bishop(i==0, panel));
} else if(j == 4){
if (i == 0) {
blackKing = new King(true, panel);
set(j, i, blackKing);
} else {
whiteKing = new King(false, panel);
set(j, i, whiteKing);
}
} else {
set(j, i, new Queen(i==0, panel));
}
}
}
if(i == 1 || i == 6){
for(int j = 0; j <= 7; j++){
set(j, i, new Pawn(i==1, panel));
}
}
}
2 years ago
}
public boolean outOfBounds(int x, int y) {
return x < 0 || y < 0 || x >= BOARD_SIZE || y >= BOARD_SIZE;
}
public boolean outOfBounds(BoardCoordinate coordinate) {
return outOfBounds(coordinate.x, coordinate.y);
}
2 years ago
public Piece get(int x, int y) {
if (outOfBounds(x, y)) return null;
2 years ago
return board[y][x];
}
public Piece get(BoardCoordinate coordinate) {
if (coordinate == null) return null;
2 years ago
return get(coordinate.x, coordinate.y);
}
public void set(int x, int y, Piece piece) {
board[y][x] = piece;
}
public void set(BoardCoordinate coordinate, Piece piece) {
set(coordinate.x, coordinate.y, piece);
}
public void move(int fromX, int fromY, int toX, int toY) {
2 years ago
moveHistory.add(new Move(new BoardCoordinate(fromX, fromY), new BoardCoordinate(toX, toY), get(toX, toY)));
2 years ago
set(toX, toY, get(fromX, fromY));
set(fromX, fromY, null);
}
public void move(BoardCoordinate from, BoardCoordinate to) {
move(from.x, from.y, to.x, to.y);
}
public void move(Move move) {
2 years ago
if (move == null) return;
move(move.from, move.to);
}
public void undoMove() {
2 years ago
if (moveHistory.isEmpty()) return;
Move lastMove = moveHistory.pop();
set(lastMove.from, get(lastMove.to));
2 years ago
set(lastMove.to, lastMove.captured);
2 years ago
}
// Mouse down event handler
// This sets the currently dragging piece
2 years ago
void handleMouseDown(int x, int y) {
if (isGameOver) {
isGameOver = false;
setup();
return;
}
// Get board coordinate of mouse click
2 years ago
BoardCoordinate coordinate = new ScreenCoordinate(x, y).toBoard();
// If there's no piece there, return
Piece piece = get(coordinate);
if (piece == null || piece.black) return;
// Set currently dragging piece to that coordinate
2 years ago
dragging = coordinate;
legalMoves = piece.getLegalMoves(coordinate, this);
// Redraw with dragging
2 years ago
draw(x, y);
}
void handleMouseUp(int x, int y) {
// Get board coordinate of mouse release
2 years ago
BoardCoordinate newCoordinate = new ScreenCoordinate(x, y).toBoard();
// Only do something if new coordinate is different from the originating coordinate
if (dragging != null && !newCoordinate.equals(dragging)) {
// dragging is BoardCoordinate of piece being dragged
Piece piece = get(dragging);
ArrayList<BoardCoordinate> legalMoves = piece.getLegalMoves(dragging, this);
for (BoardCoordinate legalMove : legalMoves) {
if (newCoordinate.equals(legalMove)) {
move(dragging, newCoordinate);
checkForCheckmate();
if (!isGameOver) {
move(ChessAI.findBestMove(this));
checkForCheckmate();
}
break;
}
2 years ago
}
}
// Clear dragging
2 years ago
dragging = null;
// Redraw without dragging
2 years ago
draw();
}
public void checkForCheckmate() {
BoardCoordinate movedCoordinate = moveHistory.peek().to;
Piece movedPiece = get(movedCoordinate);
King oppositeKing = movedPiece.black ? whiteKing : blackKing;
BoardCoordinate oppositeKingPosition = null;
boolean inCheck = false;
for (BoardCoordinate move : movedPiece.getLegalMoves(movedCoordinate, this)) {
if (get(move) == oppositeKing) {
oppositeKingPosition = move;
inCheck = true;
break;
}
}
isGameOver = inCheck && oppositeKing.getLegalMoves(oppositeKingPosition, this).size() == 0;
victor = movedPiece.black;
}
2 years ago
public void draw() {
draw(null);
2 years ago
}
public void draw(int mouseX, int mouseY) {
draw(new ScreenCoordinate(mouseX, mouseY));
}
void drawRect(int x, int y) {
graphics.fillRect(x * TILE_SIZE, y * TILE_SIZE, TILE_SIZE, TILE_SIZE);
}
void drawRect(BoardCoordinate coordinate) {
drawRect(coordinate.x, coordinate.y);
}
public void draw(ScreenCoordinate mousePosition) {
2 years ago
// Draw board
graphics.setColor(BLACK);
graphics.fillRect(0, 0, DIMENSION, DIMENSION);
graphics.setColor(WHITE);
2 years ago
for (int y = 0; y < BOARD_SIZE; y++)
for (int x = y % 2; x < BOARD_SIZE; x += 2)
drawRect(x, y);
if (dragging != null) {
graphics.setColor(new Color(0, 128, 0, 128));
for (BoardCoordinate legalMove : legalMoves)
drawRect(legalMove);
if (mousePosition != null) {
BoardCoordinate hovering = mousePosition.toBoard();
if (legalMoves.contains(hovering)) {
graphics.setColor(get(hovering) == null ? new Color(0, 0, 255, 128) : new Color(255, 0, 0, 128));
drawRect(mousePosition.toBoard());
}
}
}
2 years ago
// Draw pieces
forEachPiece((boardCoordinate, piece) -> {
// If piece is the one being dragged, render it at the mouse position
// Otherwise, render it at the center of the board tile
piece.draw(graphics, panel, boardCoordinate.equals(dragging) ? mousePosition : boardCoordinate.toScreen());
2 years ago
});
// Draw game over text
if (isGameOver) {
graphics.drawImage(victor ? youLose : youWin, 0, 0, panel);
}
2 years ago
}
// Functional interfaces for forEachPiece
// Provide ability to run code for each piece on board without having to duplicate nested for loop boilerplate
2 years ago
@FunctionalInterface
interface PieceActionXY {
void forEachTile(int x, int y, Piece piece);
}
@FunctionalInterface
interface PieceActionCoordinate {
void forEachTile(BoardCoordinate coordinate, Piece piece);
}
// Run code on each tile on board
// Usage:
// forEachPiece((x, y, piece) -> {
// // do something with piece
// });
2 years ago
public void forEachPiece(PieceActionXY tileAction) {
for (int y = 0; y < BOARD_SIZE; y++)
for (int x = 0; x < BOARD_SIZE; x++) {
Piece piece = get(x, y);
2 years ago
if (piece == null) continue;
tileAction.forEachTile(x, y, piece);
}
}
// Run code on each tile on board
// Usage:
// forEachPiece((coordinate, piece) -> {
// // do something with piece
// });
2 years ago
public void forEachPiece(PieceActionCoordinate tileAction) {
forEachPiece((x, y, piece) -> tileAction.forEachTile(new BoardCoordinate(x, y), get(x, y)));
2 years ago
}
public ArrayList<Move> getAllLegalMoves() {
ArrayList<Move> allLegalMoves = new ArrayList<>();
forEachPiece((from, piece) -> {
if (!piece.black) return;
ArrayList<BoardCoordinate> legalTiles = piece.getLegalMoves(from, this);
for (BoardCoordinate to : legalTiles) {
allLegalMoves.add(new Move(from, to));
}
});
return allLegalMoves;
}
}