|
@@ -3,7 +3,25 @@
|
|
#include "Game.h"
|
|
#include "Game.h"
|
|
#include "Types.h"
|
|
#include "Types.h"
|
|
|
|
|
|
-Game::Game() : activeX(-1), activeY(-1), directionX(0), directionY(0), lastLocation(0) {
|
|
|
|
|
|
+std::array<Vector, 8> Game::neighbours;
|
|
|
|
+std::array<Vector, 9 * 5> Game::fieldVectors;
|
|
|
|
+
|
|
|
|
+Game::Game() : active(-1, -1), direction(0, 0), lastLocation(0) {
|
|
|
|
+ neighbours[0].set(1, 0);
|
|
|
|
+ neighbours[1].set(0, 1);
|
|
|
|
+ neighbours[2].set(-1, 0);
|
|
|
|
+ neighbours[3].set(0, -1);
|
|
|
|
+ neighbours[4].set(1, 1);
|
|
|
|
+ neighbours[5].set(-1, 1);
|
|
|
|
+ neighbours[6].set(1, -1);
|
|
|
|
+ neighbours[7].set(-1, -1);
|
|
|
|
+
|
|
|
|
+ uint index = 0;
|
|
|
|
+ for(uint y = 0; y < 5; y++) {
|
|
|
|
+ for(uint x = 0; x < 9; x++) {
|
|
|
|
+ fieldVectors[index++].set(x, y);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
}
|
|
}
|
|
|
|
|
|
void Game::print(String& s) const {
|
|
void Game::print(String& s) const {
|
|
@@ -20,22 +38,22 @@ void Game::print(String& s) const {
|
|
s.append("\n\r");
|
|
s.append("\n\r");
|
|
}
|
|
}
|
|
|
|
|
|
-void Game::printLine(String& s, int index) const {
|
|
|
|
|
|
+void Game::printLine(String& s, int y) const {
|
|
static const char map[] = {
|
|
static const char map[] = {
|
|
'#', 'O', '.'
|
|
'#', 'O', '.'
|
|
};
|
|
};
|
|
- s.append(index + '0').append(' ');
|
|
|
|
|
|
+ s.append(y + '0').append(' ');
|
|
for(int x = 0; x < 8; x++) {
|
|
for(int x = 0; x < 8; x++) {
|
|
- if(x == activeX && index == activeY) {
|
|
|
|
|
|
+ if(active.compare(x, y)) {
|
|
s.append("*-");
|
|
s.append("*-");
|
|
continue;
|
|
continue;
|
|
}
|
|
}
|
|
- s.append(map[fields[x][index]]).append('-');
|
|
|
|
|
|
+ s.append(map[fields[x][y]]).append('-');
|
|
}
|
|
}
|
|
- if(8 == activeX && index == activeY) {
|
|
|
|
|
|
+ if(active.compare(8, y)) {
|
|
s.append('*');
|
|
s.append('*');
|
|
} else {
|
|
} else {
|
|
- s.append(map[fields[8][index]]);
|
|
|
|
|
|
+ s.append(map[fields[8][y]]);
|
|
}
|
|
}
|
|
s.append("\n\r");
|
|
s.append("\n\r");
|
|
}
|
|
}
|
|
@@ -57,15 +75,15 @@ void Game::parse() {
|
|
if(s == "select stone:") {
|
|
if(s == "select stone:") {
|
|
std::cerr << s;
|
|
std::cerr << s;
|
|
s.clear();
|
|
s.clear();
|
|
- selectStone();
|
|
|
|
|
|
+ makeSelection();
|
|
} else if(s == "select location to move:") {
|
|
} else if(s == "select location to move:") {
|
|
std::cerr << s;
|
|
std::cerr << s;
|
|
s.clear();
|
|
s.clear();
|
|
- selectLocationToMove();
|
|
|
|
|
|
+ makeMove();
|
|
} else if(s == "select stone to take:") {
|
|
} else if(s == "select stone to take:") {
|
|
std::cerr << s;
|
|
std::cerr << s;
|
|
s.clear();
|
|
s.clear();
|
|
- selectStoneToTake();
|
|
|
|
|
|
+ takeStone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
@@ -81,21 +99,20 @@ void Game::consumeLine() const {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-void Game::consumeLine(uint index) {
|
|
|
|
|
|
+void Game::consumeLine(uint y) {
|
|
std::cin.get();
|
|
std::cin.get();
|
|
std::cin.get();
|
|
std::cin.get();
|
|
- for(uint i = 0; i < 9; i++) {
|
|
|
|
|
|
+ for(uint x = 0; x < 9; x++) {
|
|
char c = std::cin.get();
|
|
char c = std::cin.get();
|
|
if(c == '#') {
|
|
if(c == '#') {
|
|
- fields[i][index] = BLACK;
|
|
|
|
|
|
+ fields[x][y] = BLACK;
|
|
} else if(c == '.') {
|
|
} else if(c == '.') {
|
|
- fields[i][index] = EMPTY;
|
|
|
|
|
|
+ fields[x][y] = EMPTY;
|
|
} else if(c == 'O') {
|
|
} else if(c == 'O') {
|
|
- fields[i][index] = WHITE;
|
|
|
|
|
|
+ fields[x][y] = WHITE;
|
|
} else if(c == '*') {
|
|
} else if(c == '*') {
|
|
- activeX = i;
|
|
|
|
- activeY = index;
|
|
|
|
- fields[i][index] = WHITE;
|
|
|
|
|
|
+ active.set(x, y);
|
|
|
|
+ fields[x][y] = WHITE;
|
|
} else {
|
|
} else {
|
|
std::cerr << "game field parsing error\n";
|
|
std::cerr << "game field parsing error\n";
|
|
}
|
|
}
|
|
@@ -113,129 +130,251 @@ void Game::parseFields() {
|
|
std::cerr << s;
|
|
std::cerr << s;
|
|
}
|
|
}
|
|
|
|
|
|
-void Game::selectStone() {
|
|
|
|
|
|
+bool Game::isInRange(const Vector& v) const {
|
|
|
|
+ return v.x >= 0 && v.x <= 8 && v.y >= 0 && v.y <= 4;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+bool Game::areNeighbours(const Vector& from, const Vector& to) const {
|
|
|
|
+ if(from == to) {
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+ Vector diff = from - to;
|
|
|
|
+ if(diff.x < -1 || diff.x > 1 || diff.y < -1 || diff.y > 1) {
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+ if(diff.x == 0 || diff.y == 0) {
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ return !((from.x & 1) ^ (from.y & 1));
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+void Game::removeLine(const Vector& from, const Vector& to, FieldState state) {
|
|
|
|
+ Vector current = to;
|
|
|
|
+ Vector diff = to - from;
|
|
|
|
+ while(true) {
|
|
|
|
+ current += diff;
|
|
|
|
+ if(!isInRange(current) || !hasState(current, state)) {
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+ setState(current, EMPTY);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+uint Game::getRank(const Vector& from, const Vector& to, FieldState state) const {
|
|
|
|
+ Vector current = to;
|
|
|
|
+ Vector diff = to - from;
|
|
|
|
+ uint rank = 0;
|
|
|
|
+ while(true) {
|
|
|
|
+ current += diff;
|
|
|
|
+ if(!isInRange(current) || !hasState(current, state)) {
|
|
|
|
+ return rank;
|
|
|
|
+ }
|
|
|
|
+ rank++;
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+void Game::addLocation(const Vector& v) {
|
|
|
|
+ if(lastLocation < lastLocations.size()) {
|
|
|
|
+ lastLocations[lastLocation] = v;
|
|
|
|
+ lastLocation++;
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+bool Game::isInQueue(const Vector& v) const {
|
|
|
|
+ for(uint i = 0; i < lastLocation; i++) {
|
|
|
|
+ if(lastLocations[i] == v) {
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+uint Game::getStoneTakes(const Vector& from, const Vector& to) const {
|
|
|
|
+ if(!isInRange(to) || !hasState(to, EMPTY) || !areNeighbours(from, to) || isInQueue(to) || (to - from) == direction) {
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ FieldState enemy = (hasState(from, BLACK) ? WHITE : BLACK);
|
|
|
|
+ uint rankA = getRank(from, to, enemy);
|
|
|
|
+ uint rankB = getRank(to, from, enemy);
|
|
|
|
+ if(rankA == 0 && rankB == 0) {
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Game copy = *this;
|
|
|
|
+ uint rank = 0;
|
|
|
|
+ if(rankA >= rankB) {
|
|
|
|
+ copy.removeLine(from, to, enemy);
|
|
|
|
+ rank = rankA;
|
|
|
|
+ } else {
|
|
|
|
+ copy.removeLine(to, from, enemy);
|
|
|
|
+ rank = rankB;
|
|
|
|
+ }
|
|
|
|
+ copy.addLocation(from);
|
|
|
|
+ copy.move(from, to);
|
|
|
|
+
|
|
|
|
+ uint maxRank = 0;
|
|
|
|
+ for(Vector& m : neighbours) {
|
|
|
|
+ uint rank = copy.getStoneTakes(to, to + m);
|
|
|
|
+ if(rank > maxRank) {
|
|
|
|
+ maxRank = rank;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return rank + maxRank;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+uint Game::getStoneFreedom(const Vector& from) const {
|
|
|
|
+ if(hasState(from, EMPTY)) {
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ uint counter = 0;
|
|
|
|
+ for(const Vector& m : neighbours) {
|
|
|
|
+ Vector to = from + m;
|
|
|
|
+ if(!isInRange(to) || !areNeighbours(from, to) || !hasState(to, EMPTY)) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+ counter++;
|
|
|
|
+ }
|
|
|
|
+ return counter;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+uint Game::getFreedom(FieldState state) const {
|
|
|
|
+ int sum = 0;
|
|
|
|
+ for(const Vector& v : fieldVectors) {
|
|
|
|
+ if(hasState(v, state)) {
|
|
|
|
+ sum += getStoneFreedom(v);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ return sum;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+bool Game::hasState(const Vector& v, FieldState state) const {
|
|
|
|
+ return fields[v.x][v.y] == state;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+Game::FieldState Game::getState(const Vector& v) const {
|
|
|
|
+ return fields[v.x][v.y];
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+void Game::setState(const Vector& v, FieldState state) {
|
|
|
|
+ fields[v.x][v.y] = state;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+void Game::move(const Vector& from, const Vector& to) {
|
|
|
|
+ direction = to - from;
|
|
|
|
+ setState(to, getState(from));
|
|
|
|
+ setState(from, EMPTY);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+void Game::makeSelection() {
|
|
lastLocation = 0;
|
|
lastLocation = 0;
|
|
- directionX = 0;
|
|
|
|
- directionY = 0;
|
|
|
|
-
|
|
|
|
- Location from;
|
|
|
|
- int rank = -1;
|
|
|
|
- for(int bx = 0; bx < 9; bx++) {
|
|
|
|
- for(int by = 0; by < 5; by++) {
|
|
|
|
- if(fields[bx][by] != EMPTY) {
|
|
|
|
|
|
+ direction.set(0, 0);
|
|
|
|
+
|
|
|
|
+ Vector selection(0, 0);
|
|
|
|
+ int maxRank = -100000;
|
|
|
|
+ for(Vector& to : fieldVectors) {
|
|
|
|
+ if(!hasState(to, EMPTY)) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+ for(Vector& m : neighbours) {
|
|
|
|
+ Vector from = to + m;
|
|
|
|
+ if(!isInRange(from) || !hasState(from, WHITE) || !areNeighbours(from, to)) {
|
|
continue;
|
|
continue;
|
|
}
|
|
}
|
|
- for(int mx = -1; mx <= 1; mx++) {
|
|
|
|
- for(int my = -1; my <= 1; my++) {
|
|
|
|
- int x = bx + mx;
|
|
|
|
- int y = by + my;
|
|
|
|
- if((mx == 0 && my == 0) || !isInRange(x, y) || !areNeighbours(bx, by, x, y) || fields[x][y] != WHITE) {
|
|
|
|
- continue;
|
|
|
|
- }
|
|
|
|
- int rankA = getRank(bx, by, x, y);
|
|
|
|
- int rankB = getRank(x, y, bx, by);
|
|
|
|
- if(rankA > rank) {
|
|
|
|
- rank = rankA;
|
|
|
|
- from = {x, y};
|
|
|
|
- }
|
|
|
|
- if(rankB > rank) {
|
|
|
|
- rank = rankB;
|
|
|
|
- from = {x, y};
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+ int rank = getQuality(from, to);
|
|
|
|
+ if(rank > maxRank) {
|
|
|
|
+ maxRank = rank;
|
|
|
|
+ selection = from;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- std::cout << from.x << " " << from.y << "\n";
|
|
|
|
|
|
+ std::cout << selection.x << " " << selection.y << "\n";
|
|
}
|
|
}
|
|
|
|
|
|
-void Game::selectLocationToMove() {
|
|
|
|
- Location to;
|
|
|
|
- int rank = -1;
|
|
|
|
- for(int mx = -1; mx <= 1; mx++) {
|
|
|
|
- for(int my = -1; my <= 1; my++) {
|
|
|
|
- int x = activeX + mx;
|
|
|
|
- int y = activeY + my;
|
|
|
|
- if((mx == 0 && my == 0) || (mx == directionX && my == directionY) || !isInRange(x, y) ||
|
|
|
|
- !areNeighbours(activeX, activeY, x, y) || fields[x][y] != EMPTY || isInQueue(x, y)) {
|
|
|
|
|
|
+void Game::makeMove() {
|
|
|
|
+ Vector maxTo(0, 0);
|
|
|
|
+ int maxRank = -100000;
|
|
|
|
+ for(Vector& m : neighbours) {
|
|
|
|
+ Vector to = active + m;
|
|
|
|
+ if(!isInRange(to) || !hasState(to, EMPTY) || isInQueue(to) || !areNeighbours(active, to) || m == direction) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+ if(lastLocation > 0) {
|
|
|
|
+ int rank = getRank(active, to, BLACK) + getRank(to, active, BLACK);
|
|
|
|
+ if(rank == 0) {
|
|
continue;
|
|
continue;
|
|
}
|
|
}
|
|
- int rankA = getRank(x, y, activeX, activeY);
|
|
|
|
- int rankB = getRank(activeX, activeY, x, y);
|
|
|
|
- if(rankA > rank) {
|
|
|
|
- rank = rankA;
|
|
|
|
- to = {x, y};
|
|
|
|
- }
|
|
|
|
- if(rankB > rank) {
|
|
|
|
- rank = rankB;
|
|
|
|
- to = {x, y};
|
|
|
|
- }
|
|
|
|
|
|
+ }
|
|
|
|
+ int rank = getQuality(active, to);
|
|
|
|
+ if(rank > maxRank) {
|
|
|
|
+ maxRank = rank;
|
|
|
|
+ maxTo = to;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- addLocation(activeX, activeY);
|
|
|
|
- directionX = to.x - activeX;
|
|
|
|
- directionY = to.y - activeY;
|
|
|
|
- std::cout << to.x << " " << to.y << "\n";
|
|
|
|
|
|
+ addLocation(active);
|
|
|
|
+ direction = maxTo - active;
|
|
|
|
+ std::cerr << maxTo.x << " " << maxTo.y << "\n";
|
|
|
|
+ std::cout << maxTo.x << " " << maxTo.y << "\n";
|
|
}
|
|
}
|
|
|
|
|
|
-void Game::selectStoneToTake() {
|
|
|
|
- int rankA = getRank(activeX, activeY, activeX + directionX, activeY + directionY);
|
|
|
|
- int rankB = getRank(activeX + directionX, activeY + directionY, activeX, activeY);
|
|
|
|
- if(rankA > rankB) {
|
|
|
|
- std::cout << (activeX + directionX * 2) << " " << (activeY + directionY * 2) << "\n";
|
|
|
|
|
|
+void Game::takeStone() {
|
|
|
|
+ uint rankA = getRank(active, active + direction, BLACK);
|
|
|
|
+ uint rankB = getRank(active + direction, active, BLACK);
|
|
|
|
+ Vector v;
|
|
|
|
+ if(rankA >= rankB) {
|
|
|
|
+ v = active + direction + direction;
|
|
} else {
|
|
} else {
|
|
- std::cout << (activeX - directionX) << " " << (activeY - directionY) << "\n";
|
|
|
|
|
|
+ v = active - direction;
|
|
}
|
|
}
|
|
|
|
+ std::cout << v.x << " " << v.y << "\n";
|
|
}
|
|
}
|
|
|
|
|
|
-bool Game::isInRange(int x, int y) const {
|
|
|
|
- return x >= 0 && x <= 8 && y >= 0 && y <= 4;
|
|
|
|
-}
|
|
|
|
|
|
+int Game::getQuality(const Vector& from, const Vector& to) const {
|
|
|
|
+ int quality = getStoneTakes(from, to);
|
|
|
|
|
|
-bool Game::areNeighbours(int x, int y, int x2, int y2) const {
|
|
|
|
- if(x == x2 && y == y2) {
|
|
|
|
- return false;
|
|
|
|
- }
|
|
|
|
- int diffX = x - x2;
|
|
|
|
- int diffY = y - y2;
|
|
|
|
- if(diffX < -1 || diffX > 1 || diffY < -1 || diffY > 1) {
|
|
|
|
- return false;
|
|
|
|
- }
|
|
|
|
- if(diffX == 0 || diffY == 0) {
|
|
|
|
- return true;
|
|
|
|
|
|
+ Game copy = *this;
|
|
|
|
+ copy.move(from, to);
|
|
|
|
+ uint rankA = copy.getRank(from, to, BLACK);
|
|
|
|
+ uint rankB = copy.getRank(to, from, BLACK);
|
|
|
|
+ if(rankA >= rankB) {
|
|
|
|
+ copy.removeLine(from, to, BLACK);
|
|
|
|
+ } else {
|
|
|
|
+ copy.removeLine(to, from, BLACK);
|
|
}
|
|
}
|
|
- return !((x & 1) ^ (y & 1));
|
|
|
|
-}
|
|
|
|
|
|
+ copy.lastLocation = 0;
|
|
|
|
+ copy.direction.set(0, 0);
|
|
|
|
|
|
-int Game::getRank(int x, int y, int x2, int y2) const {
|
|
|
|
- int diffX = x2 - x;
|
|
|
|
- int diffY = y2 - y;
|
|
|
|
- int rank = 0;
|
|
|
|
- while(true) {
|
|
|
|
- x2 += diffX;
|
|
|
|
- y2 += diffY;
|
|
|
|
- if(!isInRange(x2, y2) || fields[x2][y2] != BLACK) {
|
|
|
|
- return rank;
|
|
|
|
|
|
+ int whiteFreedom = copy.getFreedom(WHITE);
|
|
|
|
+ int blackFreedom = copy.getFreedom(BLACK);
|
|
|
|
+
|
|
|
|
+ int maxRank = -1;
|
|
|
|
+ for(Vector& to : fieldVectors) {
|
|
|
|
+ if(!copy.hasState(to, EMPTY)) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+ for(Vector& m : neighbours) {
|
|
|
|
+ Vector from = to + m;
|
|
|
|
+ if(!copy.isInRange(from) || !copy.hasState(from, BLACK)) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+ int rank = copy.getStoneTakes(from, to);
|
|
|
|
+ if(rank > maxRank) {
|
|
|
|
+ maxRank = rank;
|
|
|
|
+ }
|
|
}
|
|
}
|
|
- rank++;
|
|
|
|
}
|
|
}
|
|
-}
|
|
|
|
|
|
|
|
-void Game::addLocation(int x, int y) {
|
|
|
|
- if(lastLocation >= lastLocations.size()) {
|
|
|
|
- // safely to ignore, such long turns will never happen
|
|
|
|
- return;
|
|
|
|
|
|
+ int black = countBlack();
|
|
|
|
+ if(black >= 8) {
|
|
|
|
+ black = 1;
|
|
}
|
|
}
|
|
- lastLocations[lastLocation].x = x;
|
|
|
|
- lastLocations[lastLocation].y = y;
|
|
|
|
- lastLocation++;
|
|
|
|
|
|
+ return quality - maxRank * black + (whiteFreedom - blackFreedom) / 4;
|
|
}
|
|
}
|
|
|
|
|
|
-bool Game::isInQueue(int x, int y) const {
|
|
|
|
- for(uint i = 0; i < lastLocation; i++) {
|
|
|
|
- if(lastLocations[i].x == x && lastLocations[i].y == y) {
|
|
|
|
- return true;
|
|
|
|
- }
|
|
|
|
|
|
+int Game::countBlack() const {
|
|
|
|
+ int blackCounter = 0;
|
|
|
|
+ for(Vector& v : fieldVectors) {
|
|
|
|
+ blackCounter += hasState(v, BLACK);
|
|
}
|
|
}
|
|
- return false;
|
|
|
|
|
|
+ return blackCounter;
|
|
}
|
|
}
|