-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathgame.py
105 lines (91 loc) · 3.87 KB
/
game.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
import time
import numpy as np
from numpy import multiply,zeros,add,random,count_nonzero
from numpy import int as intnp
from numba import njit
class MineSweeper():
def __init__(self,width,height,bomb_no):
### grid contains the values of entire map : [0 = nothing, -1 = bomb, 1,2,3... = hints of bombs nearby]
### fog contains the things that are visible to the player/ system : [ 0 = not visible, 1 = visible ]
### grid_width, grid_height, bomb_no are self explanatory
### bomb_locs contains *flattened* locations of the bombs in the grid
self.grid_width = width
self.grid_height = height
self.bomb_no = bomb_no
self.box_count = self.grid_width*self.grid_height
self.uncovered_count = 0
self.reset()
def reset(self):
self.grid = zeros((self.grid_width,self.grid_height),dtype=intnp)
self.fog = zeros((self.grid_width,self.grid_height),dtype=intnp)
self.state = zeros((self.grid_width,self.grid_height),dtype=intnp)
self.bomb_locs = random.choice(range(self.box_count), self.bomb_no, replace=False)
self.plant_bombs()
self.hint_maker()
self.update_state()
self.uncovered_count = 0
### Updates the state after choosing decision
def update_state(self):
self.state = multiply(self.grid,self.fog)
self.state = add(self.state,(self.fog-1))
### Used during initialization to make bomb areas -1 on grid
def plant_bombs(self):
reordered_bomb_locs = []
grid_width = self.grid_width
for bomb_loc in self.bomb_locs:
row = int(bomb_loc/grid_width)
col = int(bomb_loc%grid_width)
self.grid[row][col] = -1
reordered_bomb_locs.append((row,col))
self.bomb_locs = reordered_bomb_locs
### Used after planting bombs in initialization phase to make hints 1,2,3... bombs nearby etc
def hint_maker(self):
grid_height = self.grid_height
grid_width = self.grid_width
for r,c in self.bomb_locs:
for i in range(r-1,r+2):
for j in range(c-1,c+2):
if(i>-1 and j>-1 and i<grid_height and j<grid_width and self.grid[i][j]!=-1):
self.grid[i][j]+=1
### Game logic for choosing a point in grid
def choose(self,i,j):
if(self.grid[i][j]==0):
unfog_zeros(self.grid,self.fog,i,j)
self.uncovered_count=count_nonzero(self.fog)
self.update_state()
if(self.uncovered_count==self.box_count-self.bomb_no):
return self.state,True,1
return self.state,False,0.5
elif(self.grid[i][j]>0):
self.fog[i][j]=1
self.uncovered_count=count_nonzero(self.fog)
self.update_state()
if(self.uncovered_count==self.box_count-self.bomb_no):
return self.state,True,1
return self.state,False,0.5
else:
return self.state,True,-1
### THIS is the silightly more complex logic with Breadth First Search
### Used to unfog the grid if zeros are there in nearby region and if the chosen grid is a zero cell
@njit(fastmath=True)
def unfog_zeros(grid,fog,i,j):
h,w = grid.shape
queue = []
queue.append((i,j))
while(len(queue)>0):
i,j = queue.pop()
for r in range(i-1,i+2):
for c in range(j-1,j+2):
if(r>=0 and r<h and c>=0 and c<w):
if(grid[r][c]==0 and fog[r][c]==0):
queue.append((r,c))
fog[r][c]=1
def speed_test(iterations):
start = time.perf_counter()
for i in range(iterations):
game = MineSweeper(10,10,10)
game.choose(0,0)
end = time.perf_counter()-start
return end
# iterations = 2500
# print("Time taken for "+str(iterations)+" steps is "+str(speed_test(iterations)))