Game.cpp 9.3 KB

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