- Add drawing methods to the API

- Improve tests
This commit is contained in:
Loic Guegan 2022-01-30 10:00:28 +01:00
parent 851f31e7ae
commit 159d533412
4 changed files with 123 additions and 15 deletions

View file

@ -74,14 +74,11 @@ std::string Board::GetKingLocation(bool isBlack) {
void Board::Move(std::string move) { void Board::Move(std::string move) {
std::string src(move.substr(0, 2)); std::string src(move.substr(0, 2));
std::string dst(move.substr(2, 2)); std::string dst(move.substr(2, 2));
if (!IsEmpty(src)) { for (Piece &p : pieces) {
if (!IsEmpty(dst)) { if (p.coord == src) {
RemovePiece(dst); RemovePiece(dst); // Remove piece on dst if exists
} p.coord = dst;
for (Piece &p : pieces) { break;
if (p.coord == src) {
p.coord = dst;
}
} }
} }
} }
@ -116,6 +113,9 @@ bool Board::IsMovePossible(std::string move) {
} }
// Check colors on dst square // Check colors on dst square
// Seems that checking that is empty
// instead of catching NoPieceFound exception is
// more performant
if (!IsEmpty(dst)) { if (!IsEmpty(dst)) {
Piece dstp = GetPieceAt(dst); Piece dstp = GetPieceAt(dst);
if (srcp.isBlack == dstp.isBlack) if (srcp.isBlack == dstp.isBlack)

View file

@ -5,8 +5,10 @@ ChessArbiter::ChessArbiter()
: wPawn(1), wRook(5), wKnight(3), wBishop(3), wQueen(9), wKing(0) {} : wPawn(1), wRook(5), wKnight(3), wBishop(3), wQueen(9), wKing(0) {}
void ChessArbiter::Setup(std::string fen) { void ChessArbiter::Setup(std::string fen) {
positions.clear();
SetFEN(fen); SetFEN(fen);
fen_last = this->fen; fen_last = this->fen;
positions[this->fen.board] = 1;
} }
void ChessArbiter::SetFEN(FEN fen) { SetFEN(FENParser::Serialize(fen)); } void ChessArbiter::SetFEN(FEN fen) { SetFEN(FENParser::Serialize(fen)); }
@ -114,6 +116,13 @@ bool ChessArbiter::Play(std::string move) {
return (false); return (false);
} }
// Update position map (repetitions draw)
if (positions.count(fen.board) == 0) {
positions[fen.board] = 1;
} else {
positions[fen.board] += 1;
}
return (true); return (true);
} }
return (false); return (false);
@ -292,16 +301,48 @@ bool ChessArbiter::IsPlayable() {
return (false); return (false);
} }
bool ChessArbiter::IsDrawByFiftyMoveRule() { return (fen.halfmove >= 100); }
bool ChessArbiter::IsDrawByNoMoves() {
if (!IsCheck(fen.player)) {
std::vector<std::string> moves = ListLegalMoves(fen.player);
for (std::string &move : moves) {
if (Play(move)) {
positions[fen.board]--; // If move work, remove its position
SetFEN(fen_last);
return (false);
}
}
return (true);
}
return (false);
}
bool ChessArbiter::IsDrawByRepetitions() {
for (auto const &[key, val] : positions) {
if (val >= 3) {
return (true);
}
}
return (false);
}
bool ChessArbiter::IsDraw() {
return (IsDrawByFiftyMoveRule() || IsDrawByNoMoves() ||
IsDrawByRepetitions());
}
bool ChessArbiter::IsCheckMate() { bool ChessArbiter::IsCheckMate() {
if (IsCheck(fen.player)) { if (IsCheck(fen.player)) {
std::vector<std::string> moves = ListLegalMoves(fen.player); std::vector<std::string> moves = ListLegalMoves(fen.player);
for(std::string &move: moves){ for (std::string &move : moves) {
if(Play(move)){ if (Play(move)) {
positions[fen.board]--; // If move work, remove its position
SetFEN(fen_last); SetFEN(fen_last);
return(false); return (false);
} }
} }
return(true); return (true);
} }
return (false); return (false);
} }

View file

@ -1,6 +1,7 @@
#include "Board.hpp" #include "Board.hpp"
#include "Fen.hpp" #include "Fen.hpp"
#include <iostream> #include <iostream>
#include <unordered_map>
namespace chessarbiter { namespace chessarbiter {
class ChessArbiter { class ChessArbiter {
@ -8,12 +9,15 @@ class ChessArbiter {
FEN fen; FEN fen;
FEN fen_last; // To undo a move FEN fen_last; // To undo a move
int wPawn, wRook, wKnight, wBishop, wQueen, wKing; int wPawn, wRook, wKnight, wBishop, wQueen, wKing;
/// @brief Use to compute occurences of positions
std::unordered_map<std::string, char> positions;
/// @brief FEN methods used internally
void SetFEN(std::string);
void SetFEN(FEN);
public: public:
ChessArbiter(); ChessArbiter();
void Setup(std::string); void Setup(std::string);
void SetFEN(std::string);
void SetFEN(FEN);
std::string GetFEN(); std::string GetFEN();
/// @brief Check which player is going to play /// @brief Check which player is going to play
bool IsBlackTurn(); bool IsBlackTurn();
@ -27,7 +31,7 @@ public:
std::string GetBoard(); std::string GetBoard();
/// @brief Get current position evaluation according to player's material /// @brief Get current position evaluation according to player's material
int GetMaterialScore(); int GetMaterialScore();
/// @brief Check if position is legal /// @brief Check if position is legal to be played
bool IsPlayable(); bool IsPlayable();
/// @brief Get pieces captures by a player /// @brief Get pieces captures by a player
std::string GetCaptures(bool); std::string GetCaptures(bool);
@ -36,5 +40,10 @@ public:
/// @brief Check if a specific castle is possible by a player /// @brief Check if a specific castle is possible by a player
bool IsCastlePossible(bool, bool); bool IsCastlePossible(bool, bool);
bool IsCheckMate(); bool IsCheckMate();
/// @brief Draws check
bool IsDrawByFiftyMoveRule();
bool IsDrawByNoMoves();
bool IsDrawByRepetitions();
bool IsDraw();
}; };
} // namespace chessarbiter } // namespace chessarbiter

View file

@ -300,3 +300,61 @@ TEST_CASE("IsCheckmate", "[chessarbiter/IsCheckmate]") {
CHECK(a.GetFEN() == CHECK(a.GetFEN() ==
"r3qbr1/p1p1pkp1/1p2p1p1/8/8/8/PPPPP1PP/RNBQ1RK1 b - - 1 1"); "r3qbr1/p1p1pkp1/1p2p1p1/8/8/8/PPPPP1PP/RNBQ1RK1 b - - 1 1");
} }
TEST_CASE("IsDrawByFiftyMoveRule", "[chessarbiter/IsDrawByFiftyMoveRule]") {
ChessArbiter a;
a.Setup("rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1");
CHECK_FALSE(a.IsDrawByFiftyMoveRule());
a.Setup("rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 99 1");
CHECK_FALSE(a.IsDrawByFiftyMoveRule());
a.Play("b1c3");
CHECK(a.IsDrawByFiftyMoveRule());
a.Setup("rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 100 1");
CHECK(a.IsDrawByFiftyMoveRule());
}
TEST_CASE("IsDrawByNoMoves", "[chessarbiter/IsDrawByNoMoves]") {
ChessArbiter a;
// No move for black
a.Setup("8/8/8/8/8/8/5K1p/7k b - - 0 1");
CHECK(a.IsDrawByNoMoves());
// But move for white
a.Setup("8/8/8/8/8/8/5K1p/7k w - - 0 1");
CHECK_FALSE(a.IsDrawByNoMoves());
// No move for white
a.Setup("8/8/7r/2K5/b7/2k5/6q1/8 w - - 0 1");
CHECK(a.IsDrawByNoMoves());
// But move for black
a.Setup("8/8/7r/2K5/b7/2k5/6q1/8 b - - 0 1");
CHECK_FALSE(a.IsDrawByNoMoves());
}
TEST_CASE("IsDrawByRepetitions", "[chessarbiter/IsDrawByRepetitions]") {
ChessArbiter a;
// One time
a.Setup("8/3kp3/8/8/4P3/3K4/8/8 w - - 0 1");
a.Play("d3d4");
a.Play("d7d6");
CHECK_FALSE(a.IsDrawByRepetitions());
// Two time
a.Play("d4d3");
a.Play("d6d7");
CHECK_FALSE(a.IsDrawByRepetitions());
a.Play("d3d4");
a.Play("d7d6");
CHECK_FALSE(a.IsDrawByRepetitions());
// Three time
a.Play("d4d3");
a.Play("d6d7");
CHECK(a.IsDrawByRepetitions());
}