mirror of
https://gitlab.com/manzerbredes/chess-move-interface.git
synced 2025-04-05 17:46:30 +02:00
Improve overall interface
This commit is contained in:
parent
31d28a6cda
commit
720c394c50
3 changed files with 297 additions and 216 deletions
308
src/CMI.cpp
308
src/CMI.cpp
|
@ -1,149 +1,183 @@
|
|||
#include "CMI.hpp"
|
||||
|
||||
namespace CMI{
|
||||
namespace CMI {
|
||||
|
||||
HalfMove::HalfMove(): number(1), isBlack(false){
|
||||
HalfMove::HalfMove()
|
||||
: number(1), isBlack(false), NAG(0), parent(nullptr), mainline(nullptr) {}
|
||||
|
||||
HalfMove::HalfMove(const std::string &SAN)
|
||||
: number(1), isBlack(false), parent(nullptr), mainline(nullptr) {}
|
||||
|
||||
HalfMove::HalfMove(const std::string &SAN, const std::string &comment,
|
||||
std::uint16_t number, std::uint8_t NAG, bool isBlack)
|
||||
: SAN(SAN), comment(comment), number(number), NAG(NAG), isBlack(isBlack),
|
||||
parent(nullptr), mainline(nullptr) {}
|
||||
|
||||
HalfMove::~HalfMove() {
|
||||
if (mainline != nullptr)
|
||||
delete mainline;
|
||||
for (HalfMove *v : variations)
|
||||
delete v;
|
||||
}
|
||||
|
||||
void HalfMove::SetParent(HalfMove *m) { parent = m; }
|
||||
|
||||
std::vector<HalfMove *> HalfMove::GetVariations() const { return variations; }
|
||||
|
||||
HalfMove *HalfMove::GetMainline() const { return mainline; };
|
||||
|
||||
HalfMove *HalfMove::GetParent() const { return parent; };
|
||||
|
||||
std::string HalfMove::GetSAN() const { return SAN; };
|
||||
|
||||
void HalfMove::SetSAN(const std::string &newSAN) { SAN = newSAN; };
|
||||
|
||||
std::uint16_t HalfMove::GetNumber() const { return number; };
|
||||
|
||||
void HalfMove::SetNumber(std::uint16_t n) { number = n; };
|
||||
|
||||
std::uint8_t HalfMove::GetNAG() const { return NAG; };
|
||||
|
||||
void HalfMove::SetNAG(std::uint8_t n) { NAG = n; };
|
||||
|
||||
std::string HalfMove::GetComment() const { return comment; };
|
||||
|
||||
void HalfMove::SetComment(const std::string &c) { comment = c; };
|
||||
|
||||
bool HalfMove::IsBlack() const { return isBlack; };
|
||||
|
||||
void HalfMove::SetIsBlack(bool b) { isBlack = b; };
|
||||
|
||||
void HalfMove::ClearVariations() { variations.clear(); }
|
||||
|
||||
void HalfMove::SetMainline(HalfMove *m) {
|
||||
mainline = m;
|
||||
if (m != nullptr) {
|
||||
if (!this->isBlack) {
|
||||
m->SetIsBlack(true);
|
||||
m->SetNumber(this->number);
|
||||
} else {
|
||||
m->SetIsBlack(false);
|
||||
m->SetNumber(this->number + 1);
|
||||
}
|
||||
m->SetParent(this);
|
||||
}
|
||||
}
|
||||
|
||||
HalfMove::~HalfMove() {
|
||||
if(mainline!=nullptr)
|
||||
delete mainline;
|
||||
for(HalfMove *v:variations)
|
||||
delete v;
|
||||
}
|
||||
|
||||
void HalfMove::SetParent(CMI::HalfMove* m){
|
||||
parent=static_cast<HalfMove*>(m);
|
||||
}
|
||||
|
||||
|
||||
std::vector<CMI::HalfMove*> HalfMove::GetVariations() const {
|
||||
std::vector<CMI::HalfMove*> vars;
|
||||
for(HalfMove *v:variations){
|
||||
vars.push_back(static_cast<CMI::HalfMove*>(v));
|
||||
void HalfMove::Promote() {
|
||||
HalfMove *broot = GetBranchRoot();
|
||||
if (broot != nullptr) {
|
||||
HalfMove *parent = broot->GetParent();
|
||||
if (parent != nullptr) {
|
||||
if (parent->GetMainline() != broot) {
|
||||
HalfMove *pparent = parent->GetParent();
|
||||
// First update parent of parent:
|
||||
if (pparent != nullptr) {
|
||||
if (pparent->GetMainline() == parent)
|
||||
pparent->SetMainline(broot);
|
||||
else {
|
||||
pparent->AddVariation(broot);
|
||||
pparent->RemoveChild(parent);
|
||||
}
|
||||
}
|
||||
return vars;
|
||||
// Now update parent:
|
||||
parent->RemoveChild(broot);
|
||||
broot->AddVariation(parent);
|
||||
}
|
||||
}
|
||||
void HalfMove::SetVariations(std::vector<CMI::HalfMove*> vars){
|
||||
variations.clear();
|
||||
for(auto *v: vars){
|
||||
variations.push_back(static_cast<HalfMove*>(v));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void HalfMove::SetAsMainline() {
|
||||
HalfMove *broot = GetBranchRoot();
|
||||
HalfMove *lastRoot;
|
||||
// Just promote until we cannot anymore
|
||||
do {
|
||||
lastRoot = broot;
|
||||
broot->Promote();
|
||||
broot = GetBranchRoot();
|
||||
} while (broot != lastRoot);
|
||||
}
|
||||
|
||||
HalfMove *HalfMove::GetBranchRoot() {
|
||||
HalfMove *m = this;
|
||||
HalfMove *p = parent;
|
||||
while (p != nullptr) {
|
||||
if (p->GetMainline() != m) {
|
||||
return (m);
|
||||
}
|
||||
m = p;
|
||||
p = m->GetParent();
|
||||
}
|
||||
return m;
|
||||
}
|
||||
|
||||
void HalfMove::AddVariation(HalfMove *m) {
|
||||
m->SetIsBlack(IsBlack());
|
||||
m->SetNumber(GetNumber());
|
||||
m->SetParent(this);
|
||||
variations.push_back(m);
|
||||
}
|
||||
|
||||
void HalfMove::SetMainline(CMI::HalfMove* m) {
|
||||
mainline = static_cast<HalfMove*>(m);
|
||||
if(m!=nullptr){
|
||||
if (!this->isBlack) {
|
||||
m->SetIsBlack(true);
|
||||
m->SetNumber(this->number);
|
||||
} else {
|
||||
m->SetIsBlack(false);
|
||||
m->SetNumber(this->number + 1);
|
||||
}
|
||||
m->SetParent(static_cast<CMI::HalfMove*>(this));
|
||||
}
|
||||
bool HalfMove::RemoveVariation(HalfMove *m) {
|
||||
std::vector<HalfMove *> vars;
|
||||
bool removed = false;
|
||||
for (HalfMove *v : GetVariations()) {
|
||||
if (m != v)
|
||||
vars.push_back(v);
|
||||
else
|
||||
removed = true;
|
||||
}
|
||||
if (removed)
|
||||
variations = vars;
|
||||
return removed;
|
||||
}
|
||||
|
||||
bool HalfMove::RemoveChild(HalfMove *m) {
|
||||
if (GetMainline() == m) {
|
||||
mainline = nullptr;
|
||||
return true;
|
||||
}
|
||||
return RemoveVariation(m);
|
||||
}
|
||||
|
||||
bool HalfMove::Contains(HalfMove *m) const {
|
||||
if (m == nullptr)
|
||||
return false;
|
||||
else if (this == m)
|
||||
return true;
|
||||
else if (mainline != nullptr && mainline->Contains(m)) {
|
||||
return true;
|
||||
} else {
|
||||
for (HalfMove *v : variations) {
|
||||
if (v->Contains(m))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
|
||||
CMI::HalfMove* HalfMove::GetMainline() const {return mainline;};
|
||||
CMI::HalfMove* HalfMove::GetParent() const {return parent;};
|
||||
std::string HalfMove::GetSAN() const {return SAN;};
|
||||
void HalfMove::SetSAN(std::string newSAN) {SAN=newSAN;};
|
||||
std::uint16_t HalfMove::GetNumber() const {return number;};
|
||||
void HalfMove::SetNumber(std::uint16_t n) {number=n;};
|
||||
std::uint8_t HalfMove::GetNAG() const {return NAG;};
|
||||
void HalfMove::SetNAG(std::uint8_t n) {NAG=n;};
|
||||
std::string HalfMove::GetComment() const {return comment;};
|
||||
void HalfMove::SetComment(std::string c) { comment=c;};
|
||||
bool HalfMove::IsBlack() const {return isBlack;};
|
||||
void HalfMove::SetIsBlack(bool b) {isBlack=b;};
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
// ---------- Implementation of various common operations ----------
|
||||
|
||||
void HalfMove::Promote(){
|
||||
HalfMove *broot=GetBranchRoot();
|
||||
if(broot!=nullptr){
|
||||
HalfMove *parent=broot->GetParent();
|
||||
if (parent != nullptr) {
|
||||
if (parent->GetMainline() != broot) {
|
||||
HalfMove *pparent=parent->GetParent();
|
||||
// First update parent of parent:
|
||||
if (pparent != nullptr) {
|
||||
if (pparent->GetMainline() == parent)
|
||||
pparent->SetMainline(broot);
|
||||
else {
|
||||
pparent->AddVariation(broot);
|
||||
pparent->RemoveChild(parent);
|
||||
}
|
||||
}
|
||||
// Now update parent:
|
||||
parent->RemoveChild(broot);
|
||||
broot->AddVariation(parent);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void HalfMove::SetAsMainline(){
|
||||
HalfMove *broot = GetBranchRoot();
|
||||
HalfMove *lastRoot;
|
||||
// Just promote until we cannot anymore
|
||||
do {
|
||||
lastRoot = broot;
|
||||
broot->Promote();
|
||||
broot = GetBranchRoot();
|
||||
} while (broot != lastRoot);
|
||||
}
|
||||
|
||||
HalfMove* HalfMove::GetBranchRoot(){
|
||||
HalfMove *m = this;
|
||||
HalfMove *p = GetParent();
|
||||
while (p != nullptr) {
|
||||
if (p->GetMainline() != m) {
|
||||
return (m);
|
||||
}
|
||||
m = p;
|
||||
p = m->GetParent();
|
||||
}
|
||||
return m;
|
||||
}
|
||||
|
||||
void HalfMove::AddVariation(HalfMove* m){
|
||||
m->SetIsBlack(IsBlack());
|
||||
m->SetNumber(GetNumber());
|
||||
m->SetParent(this);
|
||||
auto vars=GetVariations();
|
||||
vars.push_back(m);
|
||||
SetVariations(vars);
|
||||
}
|
||||
|
||||
bool HalfMove::RemoveVariation(HalfMove* m){
|
||||
std::vector<HalfMove*> vars;
|
||||
bool removed=false;
|
||||
for(HalfMove *v: GetVariations()){
|
||||
if(m!=v)
|
||||
vars.push_back(v);
|
||||
else
|
||||
removed=true;
|
||||
}
|
||||
if(removed)
|
||||
SetVariations(vars);
|
||||
return removed;
|
||||
}
|
||||
|
||||
bool HalfMove::RemoveChild(HalfMove* m){
|
||||
if(GetMainline()==m){
|
||||
SetMainline(nullptr);
|
||||
return true;
|
||||
}
|
||||
return RemoveVariation(m);
|
||||
}
|
||||
}
|
||||
bool HalfMove::IsConsistent() const {
|
||||
// Check mainline
|
||||
if (mainline != nullptr) {
|
||||
if (isBlack == mainline->isBlack)
|
||||
return false;
|
||||
if (isBlack && number + 1 != mainline->number)
|
||||
return false;
|
||||
if (!isBlack && number != mainline->number)
|
||||
return false;
|
||||
if (mainline->parent != this)
|
||||
return false;
|
||||
if (!mainline->IsConsistent())
|
||||
return false;
|
||||
}
|
||||
// Check variations
|
||||
for (HalfMove *v : variations) {
|
||||
if (number != v->number || isBlack != v->isBlack || v->parent != this)
|
||||
return false;
|
||||
if (!v->IsConsistent())
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
} // namespace CMI
|
124
src/CMI.hpp
124
src/CMI.hpp
|
@ -1,61 +1,53 @@
|
|||
#pragma once
|
||||
|
||||
#include <cstdint>
|
||||
#include <vector>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
namespace CMI {
|
||||
/**
|
||||
* @brief Chess Move Interface
|
||||
* A standard chess half move interface for improving chess libraries
|
||||
* interoperability.
|
||||
* NONE OF THESE METHODS IMPLEMENTATIONS MUST DELETE A CMI::HalfMove.
|
||||
* It is up to the user of the object instance to do it after ensuring that no references
|
||||
* to the CMI::HalfMove remains in the move tree.
|
||||
* NONE OF THESE METHODS IMPLEMENTATIONS DELETE A CMI::HalfMove EXCEPT
|
||||
* CMI::HalfMove::~HalfMove. It is up to the user of the object instance to do
|
||||
* it after ensuring that no references to the CMI::HalfMove remains in the move
|
||||
* tree.
|
||||
*/
|
||||
class HalfMove {
|
||||
protected:
|
||||
HalfMove *parent = nullptr;
|
||||
HalfMove *mainline = nullptr;
|
||||
|
||||
HalfMove *parent;
|
||||
HalfMove *mainline;
|
||||
std::vector<HalfMove *> variations;
|
||||
std::string SAN,comment;
|
||||
std::string SAN, comment;
|
||||
std::uint16_t number;
|
||||
std::uint8_t NAG;
|
||||
bool isBlack;
|
||||
|
||||
/// @brief Set the parent of current CMI::HalfMove, only used internally
|
||||
void SetParent(HalfMove *);
|
||||
|
||||
public:
|
||||
HalfMove();
|
||||
HalfMove(const std::string &SAN);
|
||||
HalfMove(const std::string &SAN, const std::string &comment,
|
||||
std::uint16_t number, std::uint8_t NAG, bool isBlack);
|
||||
/// @brief Ensure that the destructor of the child class is called
|
||||
virtual ~HalfMove();
|
||||
/**
|
||||
* @brief Return a pointer to the next CMI::HalfMove
|
||||
*
|
||||
* @return HalfMove* if any and nullptr otherwise
|
||||
*/
|
||||
virtual HalfMove* GetMainline() const;
|
||||
/**
|
||||
* @brief Set the next CMI::HalfMove
|
||||
* Existing main line pointer will be overriten (NOT DELETED) and the internal state (Number, IsBlack) of the new move
|
||||
* must be ajusted in the implementation of this method.
|
||||
*/
|
||||
virtual void SetMainline(HalfMove*);
|
||||
/**
|
||||
* @brief Get the previous CMI::HalfMove
|
||||
*
|
||||
* @return HalfMove* if any and nullptr otherwise
|
||||
*/
|
||||
virtual HalfMove* GetParent() const;
|
||||
/**
|
||||
* @brief Set the parent of current CMI::HalfMove
|
||||
*
|
||||
*/
|
||||
virtual void SetParent(HalfMove*);
|
||||
/// @brief Return a pointer to the next CMI::HalfMove
|
||||
virtual HalfMove *GetMainline() const;
|
||||
/// @brief Bind two moves together
|
||||
virtual void SetMainline(HalfMove *);
|
||||
/// @brief Get the previous CMI::HalfMove
|
||||
virtual HalfMove *GetParent() const;
|
||||
/// @brief Return the current move using the SAN notation e.g: "Qxc5+" or "a4"
|
||||
virtual std::string GetSAN() const;
|
||||
/// @brief Setter to replace current SAN
|
||||
virtual void SetSAN(std::string);
|
||||
/// @brief Return the HalfMove move number e.g 1 for the first white's and black's move
|
||||
/// @brief Setter to replace current SAN
|
||||
virtual void SetSAN(const std::string &);
|
||||
/// @brief Return the HalfMove move number e.g 1 for the first white's and
|
||||
/// black's move
|
||||
virtual std::uint16_t GetNumber() const;
|
||||
/// @brief Setter to replace current Number
|
||||
/// @brief Setter to replace current Number
|
||||
virtual void SetNumber(std::uint16_t);
|
||||
/// @brief Return the Numeric Annotation Glyphs code
|
||||
virtual std::uint8_t GetNAG() const;
|
||||
|
@ -64,44 +56,40 @@ public:
|
|||
/// @brief Return the comment linked to the current move or empty string
|
||||
virtual std::string GetComment() const;
|
||||
/// @brief Setter to replace current comment
|
||||
virtual void SetComment(std::string);
|
||||
virtual void SetComment(const std::string &);
|
||||
/// @brief Return true if the current HalfMove was played by black
|
||||
virtual bool IsBlack() const;
|
||||
/// @brief Setter to replace that determined the return values of HalfMove::IsBlack()
|
||||
/// @brief Setter to replace that determined the return values of
|
||||
/// HalfMove::IsBlack()
|
||||
virtual void SetIsBlack(bool);
|
||||
/// @brief All the variations of the current move
|
||||
virtual std::vector<HalfMove*> GetVariations() const;
|
||||
/// @brief Setter to replace current variations
|
||||
virtual void SetVariations(std::vector<HalfMove*>);
|
||||
|
||||
|
||||
// ---------- Implementation of various common operations ----------
|
||||
|
||||
/// @brief Promote the current variation if any
|
||||
void Promote();
|
||||
/// @brief Make the current variation the main line
|
||||
void SetAsMainline();
|
||||
virtual std::vector<HalfMove *> GetVariations() const;
|
||||
/// @brief Add a variation to the variations list
|
||||
void AddVariation(HalfMove*);
|
||||
virtual void AddVariation(HalfMove *);
|
||||
/**
|
||||
* @brief Remove the given CMI::HalfMove from the variations list
|
||||
*
|
||||
* @return true if found and deleted
|
||||
* @return false otherwise
|
||||
*/
|
||||
bool RemoveVariation(HalfMove*);
|
||||
* @brief Remove the given CMI::HalfMove from the variations list
|
||||
*
|
||||
* @return true if found and deleted
|
||||
* @return false otherwise
|
||||
*/
|
||||
virtual bool RemoveVariation(HalfMove *);
|
||||
virtual void ClearVariations();
|
||||
/**
|
||||
* @brief Remove the given CMI::HalfMove from mainline and variations list
|
||||
*
|
||||
* @return true if found and deleted
|
||||
* @return false otherwise
|
||||
*/
|
||||
bool RemoveChild(HalfMove*);
|
||||
/**
|
||||
* @brief Return the CMI::HalfMove root node of the current move branch
|
||||
*
|
||||
* @return HalfMove* the branch root (might be nullptr)
|
||||
*/
|
||||
HalfMove* GetBranchRoot();
|
||||
* @brief Remove the given CMI::HalfMove from mainline and variations list
|
||||
*
|
||||
* @return true if found and deleted
|
||||
* @return false otherwise
|
||||
*/
|
||||
virtual bool RemoveChild(HalfMove *);
|
||||
/// @brief Return the CMI::HalfMove root node of the current move branch
|
||||
virtual HalfMove *GetBranchRoot();
|
||||
/// @brief Promote the current variation if any
|
||||
virtual void Promote();
|
||||
/// @brief Make the current variation the main line
|
||||
virtual void SetAsMainline();
|
||||
/// @brief Check if the tree move contains a pointer to the given move
|
||||
virtual bool Contains(HalfMove *) const;
|
||||
/// @brief Check if the current tree of move is consistent
|
||||
virtual bool IsConsistent() const;
|
||||
};
|
||||
}
|
||||
} // namespace CMI
|
|
@ -3,21 +3,80 @@
|
|||
|
||||
using namespace CMI;
|
||||
|
||||
#define NEW_MOVE(VAR,SAN) HalfMove *(VAR)=new HalfMove(); (VAR)->SetSAN((SAN));
|
||||
#define NEW_MOVE(VAR, SAN) \
|
||||
HalfMove *(VAR) = new HalfMove(); \
|
||||
(VAR)->SetSAN((SAN));
|
||||
|
||||
|
||||
HalfMove *BuildTree(){
|
||||
NEW_MOVE(m1,"e4");
|
||||
NEW_MOVE(m2,"e5");
|
||||
m1->SetMainline(m2);
|
||||
return m1;
|
||||
HalfMove *BuildTree() {
|
||||
// Move 1
|
||||
NEW_MOVE(m1, "e4");
|
||||
NEW_MOVE(m2, "e5");
|
||||
m1->SetMainline(m2);
|
||||
// Move 2
|
||||
NEW_MOVE(m3, "Nf3");
|
||||
NEW_MOVE(m4, "Nc6");
|
||||
m2->SetMainline(m3);
|
||||
m3->SetMainline(m4);
|
||||
// Move 3 Ponziani :D
|
||||
NEW_MOVE(m5, "c3");
|
||||
NEW_MOVE(m6, "Nf6");
|
||||
m4->SetMainline(m5);
|
||||
m5->SetMainline(m6);
|
||||
// Move 4 Ponziani :D
|
||||
NEW_MOVE(m7, "d4");
|
||||
NEW_MOVE(m8, "exd4");
|
||||
m5->SetMainline(m7);
|
||||
m7->SetMainline(m8);
|
||||
// Move 4 Variation
|
||||
NEW_MOVE(m8bis, "Nxe4");
|
||||
m8->AddVariation(m8bis);
|
||||
return m1;
|
||||
}
|
||||
|
||||
TEST_CASE("CMI Tests Numbers", "[numbers]") {
|
||||
HalfMove *m = BuildTree();
|
||||
CHECK(m->IsConsistent());
|
||||
|
||||
TEST_CASE("CMI Tests", "[valid]") {
|
||||
HalfMove *m=BuildTree();
|
||||
CHECK(m->GetNumber() == 1);
|
||||
CHECK(m->GetMainline()->GetNumber() == 1);
|
||||
CHECK(m->GetMainline()->GetMainline()->GetNumber() == 2);
|
||||
}
|
||||
|
||||
CHECK(m->GetNumber()==1);
|
||||
CHECK(m->GetMainline()->GetNumber()==1);
|
||||
TEST_CASE("CMI Tests Consistancy", "[IsConsistent/bindorder]") {
|
||||
// In mainline
|
||||
NEW_MOVE(m1, "e4");
|
||||
NEW_MOVE(m2, "e5");
|
||||
m1->SetMainline(m2);
|
||||
NEW_MOVE(m3, "Nf3");
|
||||
NEW_MOVE(m4, "Nc6");
|
||||
m3->SetMainline(m4); // Should normally be done after next line
|
||||
m2->SetMainline(m3);
|
||||
CHECK(!m1->IsConsistent());
|
||||
|
||||
// In variations
|
||||
NEW_MOVE(n1, "e4");
|
||||
NEW_MOVE(n2, "e5");
|
||||
n1->SetMainline(n2);
|
||||
NEW_MOVE(n3, "c6");
|
||||
n2->AddVariation(n3);
|
||||
n3->SetNumber(4);
|
||||
CHECK(!n1->IsConsistent());
|
||||
}
|
||||
|
||||
TEST_CASE("CMI Tests Consistancy", "[IsConsistent/isBlack]") {
|
||||
// In mainline
|
||||
NEW_MOVE(m1, "e4");
|
||||
NEW_MOVE(m2, "e5");
|
||||
m1->SetMainline(m2);
|
||||
m2->SetIsBlack(false);
|
||||
CHECK(!m1->IsConsistent());
|
||||
|
||||
// In variation
|
||||
NEW_MOVE(n1, "e4");
|
||||
NEW_MOVE(n2, "e5");
|
||||
n1->SetMainline(n2);
|
||||
NEW_MOVE(n3, "c6");
|
||||
n2->AddVariation(n3);
|
||||
n3->SetIsBlack(false);
|
||||
CHECK(!n1->IsConsistent());
|
||||
}
|
Loading…
Add table
Reference in a new issue