-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathEngine.cpp
123 lines (98 loc) · 2.68 KB
/
Engine.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
//
// Created by Brazhenko Andrew on 2/4/21.
//
#include "Engine.h"
#include <chrono>
int Gomoku::Engine::internal_(const Gomoku::Board &bs)
{
int ret = 0;
// Edge cases
{
// Game ended
if (bs.GetLastMoveResult() == Board::MoveResult::Draw)
return 0;
if (bs.GetLastMoveResult() == Board::MoveResult::WhiteWin)
return +100;
if (bs.GetLastMoveResult() == Board::MoveResult::BlackWin)
return -100;
}
{
// completed fives
auto b1 = bs.IsThereFigureOnBoard(Gomoku::Board::figure_five_w);
auto b2 = bs.IsThereFigureOnBoard(Gomoku::Board::figure_five_b);
if (b1) return +50;
if (b2) return -50;
}
{
// cheat shapes
auto t1 = bs.IsThereFigureOnBoard(Gomoku::Board::figure_dots_w);
auto t2 = bs.IsThereFigureOnBoard(Gomoku::Board::figure_dots_b);
if (t1 && bs.WhiteMove()) return +30;
if (t2 && !bs.WhiteMove()) return -30;
}
{
// free fours
auto b1 = bs.IsThereFigureOnBoard(Gomoku::Board::figure_free_four_w);
auto b2 = bs.IsThereFigureOnBoard(Gomoku::Board::figure_free_four_b);
// half free and flanked fours
auto t1 = bs.CountHalfFreeFours(Board::Side::White);
auto t2 = bs.CountHalfFreeFours(Board::Side::Black);
if (b1 || (bs.WhiteMove() && t1))
return +40;
if (b2 || (!bs.WhiteMove() && t2))
return -40;
if (t1 > 1) return +35;
if (t2 > 1) return -35;
ret += (t1 - t2) * halfFreeFourCoef;
}
{
// free threes
auto t1 = bs.CountFreeThrees(Board::Side::White);
auto t2 = bs.CountFreeThrees(Board::Side::Black);
if (t1 && bs.WhiteMove()) return 15;
if (t2 && !bs.WhiteMove()) return -15;
if (t1 > 1 && t2 == 0) return 13;
if (t2 > 1 && t1 == 0) return -13;
ret += (t1 - t2) * freeThreeCoef;
}
{
// half free and flanked threes
auto t1 = bs.CountHalfFreeThrees(Board::Side::White);
auto t2 = bs.CountHalfFreeThrees(Board::Side::Black);
ret += (t1 - t2) * halfFreeThreeCoef;
}
{
// Potential captures
int potentalWhiteCaptures = 0;
int potentalBlackCaptures = 0;
const auto& avm = bs.GetAvailableMoves();
for (const auto &move : avm)
{
bool t1 = bs.IsMoveCapture(move, 0b01);
bool t2 = bs.IsMoveCapture(move, 0b10);
if (t1 && t2)
{
if (bs.WhiteMove())
potentalWhiteCaptures++;
else
potentalBlackCaptures++;
}
else if (t1)
potentalWhiteCaptures++;
else if (t2)
potentalBlackCaptures++;
}
ret += (potentalWhiteCaptures - potentalBlackCaptures) * potentialCaptureCoef;
}
{
// Captures
auto t1 = bs.GetCapturePoints(Gomoku::Board::Side::White);
auto t2 = bs.GetCapturePoints(Gomoku::Board::Side::Black);
ret += (t1 - t2) * captureCoef;
}
return ret;
}
int Gomoku::Engine::StaticPositionAnalize(const Gomoku::Board &bs)
{
return internal_(bs);
}