forked from junxiaosong/AlphaZero_Gomoku
-
Notifications
You must be signed in to change notification settings - Fork 0
/
GUI_v1_4.py
362 lines (318 loc) · 13.6 KB
/
GUI_v1_4.py
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
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
# -*- coding: utf-8 -*-
"""
Created on Wed Dec 12 15:32:27 2018
@author: Tokarev-TT-33
introduction:
This is a UI for Gomoku game which does not contain the rule code.
Set self.UnitSize in __init__() to a different value can change the basic size of all elements
There are no limits for the value of board_size. So a board of any size can be created(if the system supports).
Add limits in self.reset() if necessary.
"""
import pygame
from pygame.locals import *
class GUI:
def __init__(self, board_size=11):
pygame.init()
self.score = [0, 0]
self.BoardSize = board_size
self.UnitSize = 40 # the basic size of all elements, try a different value!
self.TestSize = int(self.UnitSize * 0.625)
self.state = {} # a dictionary for pieces on board. filled with move-player pairs, such as 34:1
self.areas = {} # a dictionary for button areas. filled with name-Rect pairs
self.ScreenSize = None # save the screen size for some calculation
self.screen = None
self.last_action_player = None
self.round_counter = 0
self.messages = ''
self._background_color = (197, 227, 205)
self._board_color = (254, 185, 120)
self.reset(board_size)
# restart_game() must be called before reset_score() because restart_game() will add value to self.round_counter
self.restart_game(False)
self.reset_score()
def reset(self, bs):
"""
reset screen
:param bs: board size
"""
# # you can add limits for board size
# bs = int(bs)
# if bs < 5:
# raise ValueError('board size too small')
self.BoardSize = bs
self.ScreenSize = (self.BoardSize * self.UnitSize + 2 * self.UnitSize,
self.BoardSize * self.UnitSize + 3 * self.UnitSize)
self.screen = pygame.display.set_mode(self.ScreenSize, 0, 32)
pygame.display.set_caption('AlphaZero_Gomoku')
# button areas
self.areas['SwitchPlayer'] = Rect(self.ScreenSize[0]/2-self.UnitSize*1.5, self.ScreenSize[1] - self.UnitSize, self.UnitSize*3, self.UnitSize)
self.areas['RestartGame'] = Rect(self.ScreenSize[0] - self.UnitSize*3, self.ScreenSize[1] - self.UnitSize, self.UnitSize*3, self.UnitSize)
self.areas['ResetScore'] = Rect(0, self.ScreenSize[1] - self.UnitSize, self.UnitSize*2.5, self.UnitSize)
board_lenth = self.UnitSize * self.BoardSize
self.areas['board'] = Rect(self.UnitSize, self.UnitSize, board_lenth, board_lenth)
def restart_game(self, button_down=True):
"""
restart for a new round
:param button_down: whether the RestartGame button is pressed, used to highlight button.
"""
self.round_counter += 1
self._draw_static()
if button_down:
self._draw_button('RestartGame', 1)
self.state = {}
self.last_action_player = None
pygame.display.update()
def reset_score(self):
"""
reset score and round
"""
self.score = [0, 0]
self.round_counter = 1
self.show_messages()
def add_score(self, winner):
"""
add score for winner
:param winner: the name of the winner
"""
if winner == 1:
self.score[0] += 1
elif winner == 2:
self.score[1] += 1
else:
raise ValueError('player number error')
self.show_messages()
def render_step(self, action, player):
"""
render a step of the game
:param action: 1*2 dimension location value such as (2, 3) or an int type move value such as 34
:param player: the name of the player
"""
try:
action = int(action)
except Exception:
pass
if type(action) != int:
move = self.loc_2_move(action)
else:
move = action
for event in pygame.event.get():
if event.type == QUIT:
exit()
if self.last_action_player: # draw a cross on the last piece to mark the last move
self._draw_pieces(self.last_action_player[0], self.last_action_player[1], False)
self._draw_pieces(action, player, True)
self.state[move] = player
self.last_action_player = move, player
def move_2_loc(self, move):
"""
transfer a move value to a location value
:param move: an int type move value such as 34
:return: an 1*2 dimension location value such as (2, 3)
"""
return move % self.BoardSize, move // self.BoardSize
def loc_2_move(self, loc):
"""
transfer a move value to a location value
:param loc: an 1*2 dimension location value such as (2, 3)
:return: an int type move value such as 34
"""
return loc[0] + loc[1] * self.BoardSize
def get_input(self):
"""
get inputs from clicks
:return: variable-length array.[0] is the name. Additional information behind (maybe not exist).
"""
while True:
event = pygame.event.wait()
if event.type == QUIT:
return 'quit',
if event.type == MOUSEBUTTONDOWN: # check mouse click event
if event.button == 1:
mouse_pos = event.pos
for name, rec in self.areas.items():
if self._in_area(mouse_pos, rec):
if name != 'board':
self._draw_button(name, 2, True)
pygame.time.delay(100)
self._draw_button(name, 1, True)
return name,
else:
x = (mouse_pos[0] - self.UnitSize)//self.UnitSize
y = self.BoardSize - (mouse_pos[1] - self.UnitSize)//self.UnitSize - 1
move = self.loc_2_move((x, y))
if move not in self.state:
return 'move', move
if event.type == MOUSEMOTION: # check mouse move event to highlight buttons
mouse_pos = event.pos
for name, rec in self.areas.items():
if name != 'board':
if self._in_area(mouse_pos, rec):
self._draw_button(name, 1, True)
else:
self._draw_button(name, update=True)
def deal_with_input(self, inp, player):
"""
This is just a example to deal with inputs
:param inp: inputs from get_input()
:param player: the name of the player
"""
if inp[0] == 'RestartGame':
self.restart_game()
elif inp[0] == 'ResetScore':
self.reset_score()
elif inp[0] == 'quit':
exit()
elif inp[0] == 'move':
self.render_step(inp[1], player)
elif inp[0] == 'SwitchPlayer':
# restart_game() must be called before reset_score(). The reason is mentioned above.
UI.restart_game(False)
UI.reset_score()
# code for switch is needed
def show_messages(self, messages=None):
"""
show extra messages on screen
:param messages:
:return:
"""
if messages:
self.messages = messages
pygame.draw.rect(self.screen, self._background_color, (0, self.ScreenSize[1]-self.UnitSize*2, self.ScreenSize[0], self.UnitSize))
self._draw_round(False)
self._draw_text(self.messages, (self.ScreenSize[0]/2, self.ScreenSize[1]-self.UnitSize*1.5), text_height=self.TestSize)
self._draw_score()
def _draw_score(self, update=True):
score = 'Score: ' + str(self.score[0]) + ' : ' + str(self.score[1])
self._draw_text(score, (self.ScreenSize[0] * 0.11, self.ScreenSize[1] - self.UnitSize*1.5),
backgroud_color=self._background_color, text_height=self.TestSize)
if update:
pygame.display.update()
def _draw_round(self, update=True):
self._draw_text('Round: ' + str(self.round_counter), (self.ScreenSize[0]*0.88, self.ScreenSize[1] - self.UnitSize*1.5),
backgroud_color=self._background_color, text_height=self.TestSize)
if update:
pygame.display.update()
def _draw_pieces(self, loc, player, last_step=False):
"""
draw pieces
:param loc: 1*2 dimension location value such as (2, 3) or an int type move value such as 34
:param player: the name of the player
:param last_step: whether it is the last step
"""
try:
loc = int(loc)
except Exception:
pass
if type(loc) is int:
x, y = self.move_2_loc(loc)
else:
x, y = loc
pos = int(self.UnitSize * 1.5 + x * self.UnitSize), int(self.UnitSize * 1.5 + (self.BoardSize - y - 1) * self.UnitSize)
if player == 1:
c = (0, 0, 0)
elif player == 2:
c = (255, 255, 255)
else:
raise ValueError('num input ValueError')
pygame.draw.circle(self.screen, c, pos, int(self.UnitSize * 0.45))
if last_step:
if player == 1:
c = (255, 255, 255)
elif player == 2:
c = (0, 0, 0)
start_p1 = pos[0] - self.UnitSize * 0.3, pos[1]
end_p1 = pos[0] + self.UnitSize * 0.3, pos[1]
pygame.draw.line(self.screen, c, start_p1, end_p1)
start_p2 = pos[0], pos[1] - self.UnitSize * 0.3
end_p2 = pos[0], pos[1] + self.UnitSize * 0.3
pygame.draw.line(self.screen, c, start_p2, end_p2)
def _draw_static(self):
"""
Draw static elements that will not change in a round.
"""
# draw background
self.screen.fill(self._background_color)
# draw board
board_lenth = self.UnitSize * self.BoardSize
pygame.draw.rect(self.screen, self._board_color, self.areas['board'])
for i in range(self.BoardSize):
# draw grid lines
start = self.UnitSize * (i + 0.5)
pygame.draw.line(self.screen, (0, 0, 0), (start + self.UnitSize, self.UnitSize*1.5),
(start + self.UnitSize, board_lenth + self.UnitSize*0.5))
pygame.draw.line(self.screen, (0, 0, 0), (self.UnitSize*1.5, start + self.UnitSize),
(board_lenth + self.UnitSize*0.5, start + self.UnitSize))
pygame.draw.rect(self.screen, (0, 0, 0), (self.UnitSize, self.UnitSize, board_lenth, board_lenth), 1)
# coordinate values
self._draw_text(self.BoardSize - i - 1, (self.UnitSize / 2, start + self.UnitSize), text_height=self.TestSize) # 竖的
self._draw_text(i, (start + self.UnitSize, self.UnitSize / 2), text_height=self.TestSize) # 横的
# draw buttons
for name in self.areas.keys():
if name != 'board':
self._draw_button(name)
self.show_messages()
def _draw_text(self, text, position, text_height=25, font_color=(0, 0, 0), backgroud_color=None, pos='center',
angle=0):
"""
draw text
:param text: a string type text
:param position: the location point
:param text_height: text height
:param font_color: font color
:param backgroud_color: background color
:param pos: the location point is where in the text rectangle.
'center','top','bottom','left','right'and their combination such as 'topleft' can be selected
:param angle: the rotation angle of the text
"""
posx, posy = position
font_obj = pygame.font.Font(None, int(text_height))
text_surface_obj = font_obj.render(str(text), True, font_color, backgroud_color)
text_surface_obj = pygame.transform.rotate(text_surface_obj, angle)
text_rect_obj = text_surface_obj.get_rect()
exec('text_rect_obj.' + pos + ' = (posx, posy)')
self.screen.blit(text_surface_obj, text_rect_obj)
def _draw_button(self, name, high_light=0, update=False):
rec = self.areas[name]
if not high_light:
color = (225, 225, 225)
elif high_light == 1:
color = (245, 245, 245)
elif high_light == 2:
color = (255, 255, 255)
else:
raise ValueError('high_light value error')
pygame.draw.rect(self.screen, color, rec)
pygame.draw.rect(self.screen, (0, 0, 0), rec, 1)
self._draw_text(name, rec.center, text_height=self.TestSize)
if update:
pygame.display.update()
@staticmethod
def _in_area(loc, area):
"""
check whether the location is in area
:param loc: a 1*2 dimension location value such as (123, 45)
:param area: a Rect type value in pygame
"""
return True if area[0] < loc[0] < area[0] + area[2] and area[1] < loc[1] < area[1] + area[3] else False
if __name__ == '__main__':
# test
UI = GUI()
action = 22
player = 1
i = 1
UI.add_score(1)
while True:
if i == 1:
UI.show_messages('first player\'s turn')
else:
UI.show_messages('second player\'s turn')
inp = UI.get_input()
print(inp)
UI.deal_with_input(inp, i)
if inp[0] == 'move':
i %= 2
i += 1
elif inp[0] == 'RestartGame':
i = 1
elif inp[0] == 'SwitchPlayer':
i = 1