2024-12-18 20:26:28 +01:00
|
|
|
//
|
|
|
|
// Created by hamac on 18.12.2024.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "Chessboard.h"
|
|
|
|
#include "Utils.cpp"
|
|
|
|
#include <iostream>
|
2024-12-18 23:12:48 +01:00
|
|
|
#include <queue>
|
2024-12-18 20:26:28 +01:00
|
|
|
#include <string>
|
2024-12-18 23:12:48 +01:00
|
|
|
#include <unordered_map>
|
2024-12-18 20:26:28 +01:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
class Chessboard {
|
|
|
|
private:
|
|
|
|
/* Constants */
|
|
|
|
// Chars to display board
|
|
|
|
// Corners
|
|
|
|
const std::string topLeft = "┌";
|
|
|
|
const std::string topRight = "┐";
|
|
|
|
const std::string bottomLeft = "└";
|
|
|
|
const std::string bottomRight = "┘";
|
|
|
|
|
|
|
|
// Line chars
|
|
|
|
const std::string horizontal = "─";
|
|
|
|
const std::string vertical = "│";
|
|
|
|
const std::string topIntersection = "┬";
|
|
|
|
const std::string bottomIntersection = "┴";
|
|
|
|
const std::string middleIntersection = "┼";
|
|
|
|
|
|
|
|
// White pieces
|
|
|
|
const std::string whiteSquare = "\u25A1";
|
|
|
|
const std::string whiteKing = "\u2654";
|
|
|
|
const std::string whiteQueen = "\u2655";
|
|
|
|
const std::string whiteRook = "\u2656";
|
|
|
|
const std::string whiteBischop = "\u2657";
|
|
|
|
const std::string whiteKnight = "\u2658";
|
|
|
|
const std::string whitePawn = "\u2659";
|
|
|
|
|
|
|
|
// Black pieces
|
|
|
|
const std::string blackSquare = "\u25A0";
|
|
|
|
const std::string blackKing = "\u265A";
|
|
|
|
const std::string blackQueen = "\u265B";
|
|
|
|
const std::string blackRook = "\u265C";
|
|
|
|
const std::string blackBischop = "\u265D";
|
|
|
|
const std::string blackKnight = "\u265E";
|
|
|
|
const std::string blackPawn = "\u265F";
|
|
|
|
|
|
|
|
/* class fields */
|
2024-12-19 00:05:13 +01:00
|
|
|
std::vector<std::vector<char>> currentBoard;
|
2024-12-18 20:26:28 +01:00
|
|
|
|
|
|
|
// Starting formatting
|
2024-12-19 00:05:13 +01:00
|
|
|
std::vector<std::vector<char>> startBoard = {
|
|
|
|
{'R', 'N', 'B', 'Q', 'K', 'B', 'N', 'R'},
|
|
|
|
{'P', 'P', 'P', 'P', 'P', 'P', 'P', 'P'},
|
|
|
|
{'x', 'w', 'x', 'w', 'x', 'w', 'x', 'w'},
|
|
|
|
{'w', 'x', 'w', 'x' ,'w', 'x', 'w', 'x'},
|
|
|
|
{'x', 'w', 'x', 'w', 'x', 'w', 'x', 'w'},
|
|
|
|
{'w', 'x', 'w', 'x' ,'w', 'x', 'w', 'x'},
|
|
|
|
{'p', 'p', 'p', 'p', 'p', 'p', 'p', 'p'},
|
|
|
|
{'r', 'n', 'b', 'q', 'k', 'b', 'n', 'r'},
|
2024-12-18 20:26:28 +01:00
|
|
|
};
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
std::vector<std::vector<char>> emptyBoard = {
|
|
|
|
{'x', 'w', 'x', 'w', 'x', 'w', 'x', 'w'},
|
|
|
|
{'w', 'x', 'w', 'x' ,'w', 'x', 'w', 'x'},
|
|
|
|
{'x', 'w', 'x', 'w', 'x', 'w', 'x', 'w'},
|
|
|
|
{'w', 'x', 'w', 'x' ,'w', 'x', 'w', 'x'},
|
|
|
|
{'x', 'w', 'x', 'w', 'x', 'w', 'x', 'w'},
|
|
|
|
{'w', 'x', 'w', 'x' ,'w', 'x', 'w', 'x'},
|
|
|
|
{'x', 'w', 'x', 'w', 'x', 'w', 'x', 'w'},
|
|
|
|
{'w', 'x', 'w', 'x' ,'w', 'x', 'w', 'x'}
|
2024-12-18 23:12:48 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// Saves turn order; true = white and false = black
|
|
|
|
bool turnOrder = true;
|
|
|
|
|
|
|
|
// ToDo: Max Size definieren?
|
|
|
|
std::queue<std::string> queue;
|
|
|
|
|
2024-12-18 20:26:28 +01:00
|
|
|
/* methods */
|
|
|
|
// Method returns unicode for chess icon depending on the identifier
|
2024-12-19 00:05:13 +01:00
|
|
|
std::string getChessIcon(char identifier) {
|
2024-12-18 20:26:28 +01:00
|
|
|
switch (identifier) {
|
2024-12-19 00:05:13 +01:00
|
|
|
case 'x': return blackSquare;
|
|
|
|
case 'r': return blackRook;
|
|
|
|
case 'n': return blackKnight;
|
|
|
|
case 'b': return blackBischop;
|
|
|
|
case 'q': return blackQueen;
|
|
|
|
case 'k': return blackKing;
|
|
|
|
case 'p': return blackPawn;
|
|
|
|
case 'w': return whiteSquare;
|
|
|
|
case 'R': return whiteRook;
|
|
|
|
case 'N': return whiteKnight;
|
|
|
|
case 'B': return whiteBischop;
|
|
|
|
case 'Q': return whiteQueen;
|
|
|
|
case 'K': return whiteKing;
|
|
|
|
case 'P': return whitePawn;
|
2024-12-18 20:26:28 +01:00
|
|
|
default: return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void generateTopLine() {
|
2024-12-18 23:12:48 +01:00
|
|
|
display += " " + horizontal + " " + topLeft;
|
2024-12-18 20:26:28 +01:00
|
|
|
for (int col = 0; col < boardSize; ++col) {
|
|
|
|
display += horizontal + horizontal + horizontal;
|
|
|
|
if (col < boardSize - 1) display += topIntersection;
|
|
|
|
}
|
|
|
|
display += topRight + "\n";
|
|
|
|
}
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
void generatePlayingField(const std::vector<std::vector<char>>& chessboard) {
|
2024-12-18 23:12:48 +01:00
|
|
|
char desc = 56;
|
2024-12-18 20:26:28 +01:00
|
|
|
for (int row = 0; row < boardSize; ++row) {
|
2024-12-18 23:12:48 +01:00
|
|
|
display += " ";
|
|
|
|
display += desc--;
|
|
|
|
display += " " + vertical;
|
2024-12-18 20:26:28 +01:00
|
|
|
for (int col = 0; col < boardSize; ++col) {
|
|
|
|
display += " " + getChessIcon(chessboard[row][col]) + " " + vertical;
|
|
|
|
}
|
|
|
|
display += "\n";
|
|
|
|
|
|
|
|
// Horizontale Trennlinie (außer nach der letzten Zeile)
|
|
|
|
if (row < boardSize - 1) {
|
2024-12-18 23:12:48 +01:00
|
|
|
display += " " + horizontal + " " + vertical;
|
2024-12-18 20:26:28 +01:00
|
|
|
for (int col = 0; col < boardSize; ++col) {
|
|
|
|
display += horizontal + horizontal + horizontal;
|
|
|
|
if (col < boardSize - 1) display += middleIntersection;
|
|
|
|
}
|
|
|
|
display += vertical + "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void generateBottomLine() {
|
2024-12-18 23:12:48 +01:00
|
|
|
char desc = 65;
|
|
|
|
display += " " + horizontal + " " + bottomLeft;
|
2024-12-18 20:26:28 +01:00
|
|
|
for (int col = 0; col < boardSize; ++col) {
|
|
|
|
display += horizontal + horizontal + horizontal;
|
|
|
|
if (col < boardSize - 1) display += bottomIntersection;
|
|
|
|
}
|
|
|
|
display += bottomRight + "\n";
|
2024-12-18 23:12:48 +01:00
|
|
|
display += " " + vertical;
|
|
|
|
for (int col = 0; col < boardSize; ++col) {
|
|
|
|
display += " ";
|
|
|
|
display += (desc++);
|
|
|
|
display += " " + vertical;
|
|
|
|
}
|
2024-12-18 20:26:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
/* fields */
|
|
|
|
std::string display;
|
|
|
|
int boardSize = 8;
|
|
|
|
|
2024-12-18 23:12:48 +01:00
|
|
|
bool getTurnOrder() {
|
|
|
|
return this->turnOrder;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setTurnOrder(bool turnOrder) {
|
|
|
|
this->turnOrder = turnOrder;
|
|
|
|
}
|
|
|
|
|
2024-12-18 20:26:28 +01:00
|
|
|
/* methods */
|
2024-12-19 00:05:13 +01:00
|
|
|
void setBoard(std::vector<std::vector<char>> board) {
|
2024-12-18 20:26:28 +01:00
|
|
|
this->currentBoard = board;
|
|
|
|
}
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
std::vector<std::vector<char>> getBoard() {
|
2024-12-18 20:26:28 +01:00
|
|
|
return this->currentBoard;
|
|
|
|
}
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
std::vector<std::vector<char>> getStartBoard() {
|
2024-12-18 20:26:28 +01:00
|
|
|
return this->startBoard;
|
|
|
|
}
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
std::vector<std::vector<char>> getEmptyBoard() {
|
2024-12-18 23:12:48 +01:00
|
|
|
return this->emptyBoard;
|
|
|
|
}
|
|
|
|
|
2024-12-18 20:26:28 +01:00
|
|
|
void draw () {
|
|
|
|
draw(getStartBoard());
|
|
|
|
}
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
void draw (const std::vector<std::vector<char>>& chessboard) {
|
2024-12-18 20:26:28 +01:00
|
|
|
// Obere Rahmenlinie
|
|
|
|
generateTopLine();
|
|
|
|
|
|
|
|
// Schachbrett mit vertikalen Linien
|
|
|
|
generatePlayingField(chessboard);
|
|
|
|
|
|
|
|
// Untere Rahmenlinie
|
|
|
|
generateBottomLine();
|
|
|
|
|
2024-12-18 23:12:48 +01:00
|
|
|
setBoard(chessboard);
|
|
|
|
|
2024-12-18 20:26:28 +01:00
|
|
|
std::cout << display << std::endl;
|
|
|
|
}
|
|
|
|
|
2024-12-18 23:12:48 +01:00
|
|
|
// ToDo:
|
|
|
|
// Methode um mögliche Züge anzuzeigen
|
|
|
|
// Rochade
|
|
|
|
// Mate
|
|
|
|
// Schachmate
|
|
|
|
// En Passond
|
|
|
|
// Restlichen moves
|
|
|
|
// Angabe mit Menü Optionen als Zahlen
|
|
|
|
// Figuren wählen mit Schachnotation
|
|
|
|
|
|
|
|
int getCorrectPiece(char pieceIdentifier) {
|
|
|
|
bool isWhite = getTurnOrder();
|
|
|
|
|
|
|
|
static const std::unordered_map<char, int> pieceMap = {
|
|
|
|
{'P', 200},
|
|
|
|
{'K', 205},
|
|
|
|
{'Q', 204},
|
|
|
|
{'N', 202},
|
|
|
|
{'R', 201},
|
|
|
|
{'B', 203}
|
|
|
|
};
|
|
|
|
|
|
|
|
int piece = pieceMap.find(pieceIdentifier)->second;
|
|
|
|
|
|
|
|
if (isWhite) {
|
|
|
|
return piece - 100;
|
|
|
|
}
|
|
|
|
return piece;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<int, int> convertPosition(const std::string &pos) {
|
2024-12-19 00:05:13 +01:00
|
|
|
static const std::unordered_map<int, int> fileConvertion = {
|
|
|
|
{0, 7},
|
|
|
|
{1, 6},
|
|
|
|
{2, 5},
|
|
|
|
{3, 4},
|
|
|
|
{4, 3},
|
|
|
|
{5, 2},
|
|
|
|
{6, 1},
|
|
|
|
{7, 0}
|
|
|
|
};
|
|
|
|
|
2024-12-18 23:12:48 +01:00
|
|
|
int y = pos[pos.size()-2] - 'a';
|
|
|
|
int x = fileConvertion.find((pos[pos.size()-1] - '0')-1)->second;
|
|
|
|
return std::make_pair(x, y);
|
|
|
|
}
|
|
|
|
|
2024-12-18 20:26:28 +01:00
|
|
|
//
|
|
|
|
void move(std::string move) {
|
2024-12-18 23:12:48 +01:00
|
|
|
// ToDo:
|
|
|
|
// add move to history queue
|
|
|
|
|
|
|
|
std::vector<std::string> splitMove;
|
|
|
|
|
|
|
|
if (move.find('#') != std::string::npos) {
|
|
|
|
// Schachmate
|
|
|
|
// Finish game
|
|
|
|
} else if (move.rfind("0-0", 0) == 0) { // Kleine Rochade
|
|
|
|
// Check for Rochade
|
|
|
|
} else if (move.rfind("0-0-0", 0) == 0) {
|
|
|
|
// Große Rochade
|
|
|
|
}
|
|
|
|
|
|
|
|
// Standard move
|
|
|
|
if (move.find('-')) {
|
|
|
|
splitMove = Utils::split(move, '-');
|
|
|
|
} else if (move.find('x')) {
|
|
|
|
splitMove = Utils::split(move, 'x');
|
|
|
|
}
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
std::pair oldCoords = convertPosition(splitMove[0]);
|
2024-12-18 23:12:48 +01:00
|
|
|
std::pair newCoords = convertPosition(splitMove[1]);
|
|
|
|
|
2024-12-19 00:05:13 +01:00
|
|
|
std::vector<std::vector<char>> board = getBoard();
|
|
|
|
|
|
|
|
getCorrectPiece(board[oldCoords.first][oldCoords.second]);
|
2024-12-18 23:12:48 +01:00
|
|
|
|
|
|
|
board[oldCoords.first][oldCoords.second] = getEmptyBoard()[oldCoords.first][oldCoords.second];
|
|
|
|
board[newCoords.first][newCoords.second] = getCorrectPiece(splitMove[0][0]);
|
2024-12-18 20:26:28 +01:00
|
|
|
|
2024-12-18 23:12:48 +01:00
|
|
|
draw(board);
|
2024-12-18 20:26:28 +01:00
|
|
|
|
|
|
|
// Notation
|
|
|
|
// Start with current position - dash - new position
|
|
|
|
// eg.: b1-c3
|
|
|
|
// Letter first than number
|
|
|
|
// eg.: a5
|
|
|
|
// Pawn: e4 or p e4 (pawn)
|
|
|
|
// R for rook
|
|
|
|
// N for Knight
|
|
|
|
// K for King
|
|
|
|
// B for Bischop
|
|
|
|
// Q for Queen
|
|
|
|
// Special:
|
|
|
|
// 0-0 short castle
|
|
|
|
// 0-0-0 long castle
|
|
|
|
// en passond: write square where pawn lands
|
|
|
|
|
|
|
|
// capture: x
|
|
|
|
// check: +
|
|
|
|
// checkmate: #
|
|
|
|
// draw/stalemate: 1/2-1/2
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// This method saves the current board state
|
|
|
|
void saveBoard(Chessboard& chessboard) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// This method loads a save state
|
|
|
|
void loadBoard(int saveState) {
|
|
|
|
// readJSONFile
|
|
|
|
//
|
|
|
|
}
|
|
|
|
};
|