Game.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404
  1. #include <iostream>
  2. #include "Game.h"
  3. #include "Types.h"
  4. std::array<Vector, 8> Game::neighbours;
  5. std::array<Vector, 9 * 5> Game::fieldVectors;
  6. Game::Game() : active(-1, -1), mustTake(false) {
  7. std::cout << "0\n1\n0\n";
  8. neighbours[0].set(1, 0);
  9. neighbours[1].set(0, 1);
  10. neighbours[2].set(-1, 0);
  11. neighbours[3].set(0, -1);
  12. neighbours[4].set(1, 1);
  13. neighbours[5].set(-1, 1);
  14. neighbours[6].set(1, -1);
  15. neighbours[7].set(-1, -1);
  16. uint index = 0;
  17. for(uint y = 0; y < 5; y++) {
  18. for(uint x = 0; x < 9; x++) {
  19. fieldVectors[index++].set(x, y);
  20. }
  21. }
  22. }
  23. char Game::getChar(int x, int y) const {
  24. if(active.compare(x, y)) {
  25. return '*';
  26. }
  27. static const char map[] = {'#', 'O', '.'};
  28. return map[fields[x][y]];
  29. }
  30. void Game::toString(String& s) const {
  31. s.append(" 0 1 2 3 4 5 6 7 8\n");
  32. lineToString(s, 0);
  33. s.append(" |\\|/|\\|/|\\|/|\\|/|\n");
  34. lineToString(s, 1);
  35. s.append(" |/|\\|/|\\|/|\\|/|\\|\n");
  36. lineToString(s, 2);
  37. s.append(" |\\|/|\\|/|\\|/|\\|/|\n");
  38. lineToString(s, 3);
  39. s.append(" |/|\\|/|\\|/|\\|/|\\|\n");
  40. lineToString(s, 4);
  41. s.append("\n");
  42. }
  43. void Game::lineToString(String& s, int y) const {
  44. s.append(y + '0').append(' ');
  45. for(int x = 0; x < 8; x++) {
  46. s.append(getChar(x, y)).append('-');
  47. }
  48. s.append(getChar(8, y)).append("\n\r");
  49. }
  50. void Game::readLine() {
  51. String line;
  52. while(true) {
  53. char c = std::cin.get();
  54. if(c == '\n') {
  55. if(parseLine(line)) {
  56. return;
  57. }
  58. line.clear();
  59. } else if(c != '\r') {
  60. line.append(c);
  61. }
  62. }
  63. }
  64. bool Game::parseLine(const String& line) {
  65. if(line == " 0 1 2 3 4 5 6 7 8") {
  66. readFields();
  67. printFields();
  68. } else if(line == "Please enter origin x-axis") {
  69. makeSelection();
  70. } else if(line == "Please enter destination x-axis") {
  71. makeMove();
  72. } else if(line == "Please enter wether you want to Withdraw or Approach [W/A]") {
  73. takeStone();
  74. } else if(line == "Do you want to continue with your turn [Y/N]?") {
  75. std::cout << "Y\n";
  76. } else if(line == "************************Player 1 won!**********************" ||
  77. line == "************************Player 2 won!**********************") {
  78. std::cerr << line << "\n";
  79. return true;
  80. }
  81. return false;
  82. }
  83. void Game::consumeLine() const {
  84. // skip until newline
  85. while(std::cin.get() != '\n');
  86. // skip any trailing line carriage
  87. while(true) {
  88. char c = std::cin.peek();
  89. if(c != '\r') {
  90. break;
  91. }
  92. std::cin.get();
  93. }
  94. }
  95. void Game::readLine(uint y) {
  96. // skip line heading
  97. std::cin.get();
  98. std::cin.get();
  99. std::cin.get();
  100. for(uint x = 0; x < 9; x++) {
  101. // read field state
  102. char c = std::cin.get();
  103. if(c == '2') {
  104. fields[x][y] = BLACK;
  105. } else if(c == '0') {
  106. fields[x][y] = EMPTY;
  107. } else if(c == '1') {
  108. fields[x][y] = WHITE;
  109. } else {
  110. std::cerr << "game field parsing error\n";
  111. }
  112. // skip " - " until the next field (or the next line for x = 8)
  113. std::cin.get();
  114. std::cin.get();
  115. std::cin.get();
  116. }
  117. }
  118. void Game::readFields() {
  119. for(uint i = 0; i < 5; i++) {
  120. consumeLine();
  121. readLine(i);
  122. }
  123. }
  124. void Game::printFields() const {
  125. String s;
  126. toString(s);
  127. std::cerr << s;
  128. }
  129. bool Game::isInRange(const Vector& v) const {
  130. return v.x >= 0 && v.x <= 8 && v.y >= 0 && v.y <= 4;
  131. }
  132. bool Game::areNeighbours(const Vector& from, const Vector& to) const {
  133. if(from == to) {
  134. return false;
  135. }
  136. Vector diff = from - to;
  137. if(diff.x < -1 || diff.x > 1 || diff.y < -1 || diff.y > 1) {
  138. return false;
  139. }
  140. if(diff.x == 0 || diff.y == 0) {
  141. return true;
  142. }
  143. return !((from.x & 1) ^ (from.y & 1));
  144. }
  145. void Game::removeLine(const Vector& from, const Vector& to, FieldState state) {
  146. Vector current = to;
  147. Vector diff = to - from;
  148. while(true) {
  149. current += diff;
  150. if(!isInRange(current) || !hasState(current, state)) {
  151. return;
  152. }
  153. setState(current, EMPTY);
  154. }
  155. }
  156. uint Game::getRank(const Vector& from, const Vector& to, FieldState state) const {
  157. Vector current = to;
  158. Vector diff = to - from;
  159. uint rank = 0;
  160. while(true) {
  161. current += diff;
  162. if(!isInRange(current) || !hasState(current, state)) {
  163. return rank;
  164. }
  165. rank++;
  166. }
  167. }
  168. uint Game::getStoneTakes(const Vector& from, const Vector& to) const {
  169. if(!isInRange(to) || !hasState(to, EMPTY) || !areNeighbours(from, to) || lastLocations.contains(to) || (to - from) == direction) {
  170. return 0;
  171. }
  172. FieldState enemy = (hasState(from, BLACK) ? WHITE : BLACK);
  173. uint rankA = getRank(from, to, enemy);
  174. uint rankB = getRank(to, from, enemy);
  175. if(rankA == 0 && rankB == 0) {
  176. return 0;
  177. }
  178. Game copy = *this;
  179. uint rank = 0;
  180. if(rankA >= rankB) {
  181. copy.removeLine(from, to, enemy);
  182. rank = rankA;
  183. } else {
  184. copy.removeLine(to, from, enemy);
  185. rank = rankB;
  186. }
  187. copy.lastLocations.add(from);
  188. copy.move(from, to);
  189. uint maxRank = 0;
  190. for(Vector& m : neighbours) {
  191. uint rank = copy.getStoneTakes(to, to + m);
  192. if(rank > maxRank) {
  193. maxRank = rank;
  194. }
  195. }
  196. return rank + maxRank;
  197. }
  198. uint Game::getStoneFreedom(const Vector& from) const {
  199. if(hasState(from, EMPTY)) {
  200. return 0;
  201. }
  202. uint counter = 0;
  203. for(const Vector& m : neighbours) {
  204. Vector to = from + m;
  205. if(!isInRange(to) || !areNeighbours(from, to) || !hasState(to, EMPTY)) {
  206. continue;
  207. }
  208. counter++;
  209. }
  210. return counter;
  211. }
  212. uint Game::getFreedom(FieldState state) const {
  213. int sum = 0;
  214. for(const Vector& v : fieldVectors) {
  215. if(hasState(v, state)) {
  216. sum += getStoneFreedom(v);
  217. }
  218. }
  219. return sum;
  220. }
  221. bool Game::hasState(const Vector& v, FieldState state) const {
  222. return fields[v.x][v.y] == state;
  223. }
  224. Game::FieldState Game::getState(const Vector& v) const {
  225. return fields[v.x][v.y];
  226. }
  227. void Game::setState(const Vector& v, FieldState state) {
  228. fields[v.x][v.y] = state;
  229. }
  230. void Game::move(const Vector& from, const Vector& to) {
  231. direction = to - from;
  232. setState(to, getState(from));
  233. setState(from, EMPTY);
  234. }
  235. void Game::makeSelection() {
  236. lastLocations.clear();
  237. direction.set(0, 0);
  238. mustTake = isTakingPossible();
  239. Vector selection(0, 0);
  240. int maxRank = -100000;
  241. for(Vector& to : fieldVectors) {
  242. if(!hasState(to, EMPTY)) {
  243. continue;
  244. }
  245. for(Vector& m : neighbours) {
  246. Vector from = to + m;
  247. if(!isInRange(from) || !hasState(from, WHITE) || !areNeighbours(from, to)) {
  248. continue;
  249. }
  250. if(mustTake && getRank(from, to, BLACK) + getRank(to, from, BLACK) == 0) {
  251. continue;
  252. }
  253. int rank = getQuality(from, to);
  254. if(rank > maxRank) {
  255. maxRank = rank;
  256. selection = from;
  257. }
  258. }
  259. }
  260. active = selection;
  261. std::cerr << "Selecting (" << selection.x << ", " << selection.y << ")\n";
  262. String s;
  263. toString(s);
  264. std::cerr << s;
  265. std::cout << selection.x << "\n" << selection.y << "\n";
  266. }
  267. void Game::makeMove() {
  268. Vector maxTo(0, 0);
  269. int maxRank = -100000;
  270. for(Vector& m : neighbours) {
  271. Vector to = active + m;
  272. if(!isInRange(to) || !hasState(to, EMPTY) || lastLocations.contains(to) || !areNeighbours(active, to) || m == direction) {
  273. continue;
  274. }
  275. if(!lastLocations.isEmpty() || mustTake) {
  276. int rank = getRank(active, to, BLACK) + getRank(to, active, BLACK);
  277. if(rank == 0) {
  278. continue;
  279. }
  280. }
  281. int rank = getQuality(active, to);
  282. if(rank > maxRank) {
  283. maxRank = rank;
  284. maxTo = to;
  285. }
  286. }
  287. lastLocations.add(active);
  288. direction = maxTo - active;
  289. active = maxTo;
  290. std::cerr << "Move to (" << maxTo.x << ", " << maxTo.y << ")\n";
  291. std::cout << maxTo.x << "\n" << maxTo.y << "\n";
  292. }
  293. void Game::takeStone() {
  294. uint rankA = getRank(active, active + direction, BLACK);
  295. uint rankB = getRank(active + direction, active, BLACK);
  296. if(rankA >= rankB) {
  297. std::cout << "W\n";
  298. } else {
  299. std::cout << "A\n";
  300. }
  301. }
  302. int Game::getQuality(const Vector& from, const Vector& to) const {
  303. int quality = getStoneTakes(from, to);
  304. Game copy = *this;
  305. copy.move(from, to);
  306. uint rankA = copy.getRank(from, to, BLACK);
  307. uint rankB = copy.getRank(to, from, BLACK);
  308. if(rankA >= rankB) {
  309. copy.removeLine(from, to, BLACK);
  310. } else {
  311. copy.removeLine(to, from, BLACK);
  312. }
  313. copy.lastLocations.clear();
  314. copy.direction.set(0, 0);
  315. int whiteFreedom = copy.getFreedom(WHITE);
  316. int blackFreedom = copy.getFreedom(BLACK);
  317. int maxRank = -1;
  318. for(Vector& to : fieldVectors) {
  319. if(!copy.hasState(to, EMPTY)) {
  320. continue;
  321. }
  322. for(Vector& m : neighbours) {
  323. Vector from = to + m;
  324. if(!copy.isInRange(from) || !copy.hasState(from, BLACK)) {
  325. continue;
  326. }
  327. int rank = copy.getStoneTakes(from, to);
  328. if(rank > maxRank) {
  329. maxRank = rank;
  330. }
  331. }
  332. }
  333. int black = countBlack();
  334. if(black >= 8) {
  335. black = 1;
  336. }
  337. return quality - maxRank * black + (whiteFreedom - blackFreedom) / 4;
  338. }
  339. int Game::countBlack() const {
  340. int blackCounter = 0;
  341. for(Vector& v : fieldVectors) {
  342. blackCounter += hasState(v, BLACK);
  343. }
  344. return blackCounter;
  345. }
  346. bool Game::isTakingPossible() const {
  347. for(Vector& to : fieldVectors) {
  348. if(!hasState(to, EMPTY)) {
  349. continue;
  350. }
  351. for(Vector& m : neighbours) {
  352. Vector from = to + m;
  353. if(!isInRange(from) || !areNeighbours(from, to) || !hasState(from, WHITE)) {
  354. continue;
  355. }
  356. if(getRank(from, to, BLACK) + getRank(to, from, BLACK) > 0) {
  357. return true;
  358. }
  359. }
  360. }
  361. return false;
  362. }