-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgenerator.py
129 lines (107 loc) · 5.22 KB
/
generator.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
#Oliver Temple Computer Science NEA 2022
#python version 3.10.4
#https://github.com/olivertemple/nea
#Generator class for generating mazes in the aws lambda function
import random
import sys
class Generator:
def __init__(self):
pass
#setup for recursive backtracking
def recursive_backtracking(self, Grid):
sys.setrecursionlimit(2000)#set the recursion limit to 2000
#create a list of all nodes in maze
unvisited = []
for row in Grid.grid:
for item in row:
unvisited.append(item)
#pick a random start node
start = random.choice(unvisited)
unvisited.remove(start)
#start recursive backtracking
self.recursive_backtracking_run(Grid, unvisited, start, [])
#recursive backtracking algorithm
def recursive_backtracking_run(self, Grid, unvisited, current, previous):
#work out which walls can be removed
orientation_options = []
if current.x > 0 and Grid.grid[current.y][current.x-1] in unvisited:
orientation_options.append("left")
if current.y > 0 and Grid.grid[current.y-1][current.x] in unvisited:
orientation_options.append("top")
if current.x < Grid.width - 1 and Grid.grid[current.y][current.x+1] in unvisited:
orientation_options.append("right")
if current.y < Grid.height - 1 and Grid.grid[current.y+1][current.x] in unvisited:
orientation_options.append("bottom")
#if there are no walls to remove, backtrack, else pick one and remove it
if len(orientation_options) > 0:
#pick a random wall to remove
orientation = random.choice(orientation_options)
#add the current node to the previous nodes list
previous.append(current)
#remove the wall depending on the orientation
if orientation == "left":
connecting_cell = Grid.grid[current.y][current.x - 1]
current.wallLeft = False
elif orientation == "bottom":
connecting_cell = Grid.grid[current.y + 1][current.x]
current.wallBottom = False
elif orientation == "right":
connecting_cell = Grid.grid[current.y][current.x + 1]
connecting_cell.wallLeft = False
elif orientation == "top":
connecting_cell = Grid.grid[current.y - 1][current.x]
connecting_cell.wallBottom = False
#remove the connecting node from the unvisited list
unvisited.remove(connecting_cell)
#recurse
self.recursive_backtracking_run(Grid, unvisited, connecting_cell, previous)
else:
#if not back at the start, backtrack
if len(previous) > 0:
new = previous.pop()
self.recursive_backtracking_run(Grid, unvisited, new, previous)
#generate a maze using prims algorithm
def prims(self, Grid):
#start at the top left node
inMaze = [[0, 0]]
#create a list of nodes connected to the start node
frontier = [[[0, 1],["left"]],[[1, 0],["top"]]]
#while there are still nodes to visit
while (len(frontier) > 0):
#pick a random node from the frontier
new = frontier.pop(random.randint(0, len(frontier)-1))
toAdd = new[0]
#pick a random wall from the available walls
wall = new[1][random.randint(0, len(new[1])-1)]
#add the wall to the inMaze list
inMaze.append(toAdd)
#remove the selected wall from the grid
if wall == "bottom":
Grid.grid[toAdd[0]][toAdd[1]].wallBottom = False
if wall == "left":
Grid.grid[toAdd[0]][toAdd[1]].wallLeft = False
if wall == "top" and toAdd[0] > 0:
Grid.grid[toAdd[0]-1][toAdd[1]].wallBottom = False
if wall == "right" and toAdd[1] < Grid.width:
Grid.grid[toAdd[0]][toAdd[1]+1].wallLeft = False
#calculate the possible nodes to add to the frontier
possible = [[toAdd[0]-1, toAdd[1]],[toAdd[0]+1,toAdd[1]],[toAdd[0],toAdd[1]-1],[toAdd[0], toAdd[1]+1]]
#possible walls
walls = ["bottom", "top", "right", "left"]
#iterate through the possible nodes
for i in range(4):
p = possible[i]
wall = walls[i]
#check that the wall can be removed
if 0<=p[0]<Grid.height and 0<=p[1]<Grid.width:
#check that the node is not already in the maze
if p not in inMaze:
found = False
#check if the node is already in the frontier, if it is then add the wall to the possible walls for the node in the frontier
for v in frontier:
if v[0]==p:
v[1].append(wall)
found = True
#if the node is not in the frontier, add it to the frontier
if not found:
frontier.append([p,[wall]])