-
Notifications
You must be signed in to change notification settings - Fork 110
/
Copy pathprocedural_generator.ml
293 lines (268 loc) · 12.5 KB
/
procedural_generator.ml
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
open Actors
open Object
(*Note: Canvas is 512 by 256 (w*h) -> 32 by 16 blocks*)
(*Holds obj typ and its coordinates. (int, (x-coord, y-coord))*)
type obj_coord = int * (float * float)
(*Checks if the given location checkloc is already part of the list of locations
* in loclist.*)
let rec mem_loc (checkloc: float * float) (loclist: obj_coord list) : bool =
match loclist with
|[] -> false
|h::t -> if (checkloc = (snd h)) then true
else mem_loc checkloc t
(*Converts list of locations from blocksize to pixelsize by multiplying (x,y) by
* 16.*)
let rec convert_list (lst:obj_coord list) :obj_coord list =
match lst with
|[] -> []
|(h::t) -> [(fst h, ((fst (snd h))*.16.,(snd (snd h))*.16.))]@(convert_list t)
(*Chooses what type of enemy should be instantiated given typ number*)
let choose_enemy_typ (typ:int) : enemy_typ =
match typ with
|0 -> RKoopa
|1 -> GKoopa
|2 -> Goomba
|_ -> failwith "Shouldn't reach here"
(*Chooses what type of block should be instantiated given typ number*)
let choose_sblock_typ (typ:int) : block_typ =
match typ with
|0 -> Brick
|1 -> UnBBlock
|2 -> Cloud
|3 -> QBlock Mushroom
|4 -> Ground
|_ -> failwith "Shouldn't reach here"
(*Optimizes lst such that there are no two items in the list that have the same
* coordinates. If there is one, it is removed.*)
let rec avoid_overlap (lst:obj_coord list) (currentLst:obj_coord list)
: obj_coord list =
match lst with
|[] -> []
|h::t -> if(mem_loc (snd h) currentLst) then avoid_overlap t currentLst
else [h]@(avoid_overlap t currentLst)
(*Gets rid of objects with coordinates in the ending frame, within 128 pixels of
* the start, at the very top, and two blocks from the ground.*)
let rec trim_edges (lst: obj_coord list) (blockw:float) (blockh: float)
: obj_coord list =
match lst with
|[] -> []
|h::t -> let cx = fst(snd h) in
let cy = snd(snd h) in
let pixx = blockw*.16. in
let pixy = blockh*.16. in
if(cx<128. || pixx-.cx<528. || cy = 0. || pixy-.cy<48.)
then trim_edges t blockw blockh
else [h]@trim_edges t blockw blockh
(*Generates a stair formation with block typ being dependent on typ. This type
* of stair formation requires that the first step be on the ground.*)
let generate_ground_stairs cbx cby typ =
let four = [(typ, (cbx, cby));(typ, (cbx+.1., cby));(typ, (cbx+.2., cby));
(typ, (cbx+.3., cby))] in
let three = [(typ,(cbx +. 1., cby -. 1.));(typ,(cbx +. 2., cby -. 1.));
(typ,(cbx +. 3., cby -. 1.))] in
let two = [(typ,(cbx +. 2., cby -. 2.));(typ,(cbx +. 3., cby -. 2.))] in
let one = [(typ,(cbx +. 3., cby -. 3.))] in
four@three@two@one
(*Generates a stair formation going upwards.*)
let generate_airup_stairs cbx cby typ =
let one = [(typ,(cbx, cby));(typ,(cbx +. 1., cby))] in
let two = [(typ,(cbx +. 3., cby -. 1.));(typ,(cbx +. 4., cby -. 1.))] in
let three = [(typ,(cbx +. 4., cby -. 2.));(typ,(cbx +. 5., cby -. 2.));
(typ,(cbx +. 6., cby -. 2.))] in
one@two@three
(*Generates a stair formation going downwards*)
let generate_airdown_stairs cbx cby typ =
let three = [(typ,(cbx, cby));(typ,(cbx +. 1., cby));(typ,(cbx +. 2., cby))]in
let two = [(typ,(cbx +. 2., cby +. 1.));(typ,(cbx +. 3., cby +. 1.))] in
let one = [(typ,(cbx +. 5., cby +. 2.));(typ,(cbx +. 6., cby +. 2.))] in
three@two@one
(*Generates a cloud block platform with some length num.*)
let rec generate_clouds cbx cby typ num =
if(num = 0) then []
else [(typ,(cbx, cby))]@generate_clouds (cbx+.1.) cby typ (num-1)
(*Generates an obj_coord list (typ, coordinates) of coins to be placed.*)
let rec generate_coins (block_coord: obj_coord list) : obj_coord list =
let place_coin = Random.int 2 in
match block_coord with
|[] -> []
|h::t -> if(place_coin = 0) then
let xc = fst(snd h) in
let yc = snd(snd h) in
[(0,(xc,(yc-.16.)))]@generate_coins t
else generate_coins t
(*Chooses the form of the blocks to be placed.
* When called, leaves a 1 block gap from canvas size.
* 1. If current xblock or yblock is greater than canvas width or height
* respectively, return an empty list.
* 2. If current xblock or yblock is within 10 blocks of the left and right sides
* of the level map, prevent any objects from being initialized.
* 3. Else call helper methods to created block formations and return obj_coord
* list.
**)
let choose_block_pattern (blockw:float) (blockh: float) (cbx:float) (cby:float)
(prob:int) : obj_coord list=
if(cbx > blockw || cby > blockh) then []
else
let block_typ = Random.int 4 in
let stair_typ = Random.int 2 in
let life_block_chance = Random.int 5 in
let middle_block = if(life_block_chance = 0) then 3 else stair_typ in
let obj_coord =
match prob with
|0 -> if(blockw -. cbx > 2.) then [(stair_typ, (cbx, cby));
(middle_block,(cbx +. 1., cby));(stair_typ,(cbx +. 2., cby))]
else if (blockw -. cbx > 1.) then [(block_typ,(cbx, cby));
(block_typ,(cbx +. 1., cby))]
else [(block_typ,(cbx, cby))]
|1 -> let num_clouds = (Random.int 5) + 5 in
if(cby < 5.) then generate_clouds cbx cby 2 num_clouds
else []
|2 -> if(blockh-.cby = 1.) then generate_ground_stairs cbx cby stair_typ
else []
|3 -> if(stair_typ = 0 && blockh -. cby > 3.) then
generate_airdown_stairs cbx cby stair_typ
else if (blockh-.cby>2.) then generate_airup_stairs cbx cby stair_typ
else [(stair_typ,(cbx, cby))]
|4 -> if ((cby +. 3.) -. blockh = 2.) then [(stair_typ,(cbx, cby))]
else if ((cby +. 3.) -. blockh = 1.) then [(stair_typ, (cbx,cby));
(stair_typ, (cbx, cby +. 1.))]
else [(stair_typ,(cbx, cby)); (stair_typ,(cbx, cby +. 1.));
(stair_typ,(cbx, cby +. 2.))]
|5 -> [(3,(cbx, cby))]
|_ -> failwith "Shouldn't reach here" in
obj_coord
(*Generates a list of enemies to be placed on the ground.*)
let rec generate_enemies (blockw: float) (blockh: float) (cbx: float)
(cby: float) (acc: obj_coord list) =
if(cbx > (blockw-.32.)) then []
else if (cby > (blockh-. 1.) || cbx < 15.) then
generate_enemies blockw blockh (cbx +. 1.) 0. acc
else if(mem_loc (cbx, cby) acc || cby = 0.) then
generate_enemies blockw blockh cbx (cby+.1.) acc
else
let prob = Random.int 30 in
let enem_prob = 3 in
if(prob < enem_prob && (blockh -. 1. = cby)) then
let enemy = [(prob,(cbx*.16.,cby*.16.))] in
enemy@(generate_enemies blockw blockh cbx (cby+.1.) acc)
else generate_enemies blockw blockh cbx (cby+.1.) acc
(*Generates a list of enemies to be placed upon the block objects.*)
let rec generate_block_enemies (block_coord: obj_coord list) : obj_coord list =
let place_enemy = Random.int 20 in
let enemy_typ = Random.int 3 in
match block_coord with
|[] -> []
|h::t -> if(place_enemy = 0) then
let xc = fst(snd h) in
let yc = snd(snd h) in
[(enemy_typ,(xc,(yc-.16.)))]@generate_block_enemies t
else generate_block_enemies t
(*Generates an obj_coord list (typ, coordinates) of blocks to be placed.*)
let rec generate_block_locs (blockw: float) (blockh: float) (cbx: float)
(cby: float) (acc: obj_coord list) : obj_coord list =
if(blockw-.cbx<33.) then acc
else if (cby > (blockh-. 1.)) then
generate_block_locs blockw blockh (cbx+.1.) 0. acc
else if(mem_loc (cbx, cby) acc || cby = 0.) then
generate_block_locs blockw blockh cbx (cby+.1.) acc
else
let prob = Random.int 100 in
let block_prob = 5 in
if(prob < block_prob) then
let newacc = choose_block_pattern blockw blockh cbx cby prob in
let undup_lst = avoid_overlap newacc acc in
let called_acc = acc@undup_lst in
generate_block_locs blockw blockh cbx (cby+.1.) called_acc
else generate_block_locs blockw blockh cbx (cby+.1.) acc
(*Generates the ending item panel at the end of the level. Games ends upon
* collision with player.*)
let generate_panel (context:Dom_html.canvasRenderingContext2D Js.t)
(blockw: float) (blockh: float) : collidable =
let ob = Object.spawn (SBlock Panel) context
((blockw*.16.)-.256., (blockh *. 16.)*.2./.3.) in
ob
(*Generates the list of brick locations needed to display the ground.
* 1/10 chance that a ground block is skipped each call to create holes.*)
let rec generate_ground (blockw:float) (blockh:float) (inc:float)
(acc: obj_coord list) : obj_coord list =
if(inc > blockw) then acc
else
if(inc > 10.) then
let skip = Random.int 10 in
let newacc = acc@[(4, (inc*. 16.,blockh *. 16.))] in
if (skip = 7 && blockw-.inc>32.)
then generate_ground blockw blockh (inc +. 1.) acc
else generate_ground blockw blockh (inc +. 1.) newacc
else let newacc = acc@[(4, (inc*. 16.,blockh *. 16.))] in
generate_ground blockw blockh (inc +. 1.) newacc
(*Converts the obj_coord list called by generate_block_locs to a list of objects
* with the coordinates given from the obj_coord list. *)
let rec convert_to_block_obj (lst:obj_coord list)
(context:Dom_html.canvasRenderingContext2D Js.t) : collidable list =
match lst with
|[] -> []
|h::t ->
let sblock_typ = choose_sblock_typ (fst h) in
let ob = Object.spawn (SBlock sblock_typ) context (snd h) in
[ob]@(convert_to_block_obj t context)
(*Converts the obj_coord list called by generate_enemies to a list of objects
* with the coordinates given from the obj_coord list. *)
let rec convert_to_enemy_obj (lst:obj_coord list)
(context:Dom_html.canvasRenderingContext2D Js.t) : collidable list =
match lst with
|[] -> []
|h::t ->
let senemy_typ = choose_enemy_typ (fst h) in
let ob = Object.spawn (SEnemy senemy_typ) context (snd h) in
[ob]@(convert_to_enemy_obj t context)
(*Converts the list of coordinates into a list of Coin objects*)
let rec convert_to_coin_obj (lst:obj_coord list)
(context:Dom_html.canvasRenderingContext2D Js.t) : collidable list =
match lst with
|[] -> []
|h::t ->
let sitem_typ = Coin in
let ob = Object.spawn (SItem sitem_typ) context (snd h) in
[ob]@(convert_to_coin_obj t context)
(*Procedurally generates a list of collidables given canvas width, height and
* context. Arguments block width (blockw) and block height (blockh) are in
* block form, not pixels.*)
let generate_helper (blockw:float) (blockh:float) (cx:float) (cy:float)
(context:Dom_html.canvasRenderingContext2D Js.t) : collidable list =
let block_locs = generate_block_locs blockw blockh 0. 0. [] in
let converted_block_locs = trim_edges (convert_list block_locs)
blockw blockh in
let obj_converted_block_locs = convert_to_block_obj converted_block_locs
context in
let ground_blocks = generate_ground blockw blockh 0. [] in
let obj_converted_ground_blocks = convert_to_block_obj ground_blocks
context in
let block_locations = block_locs@ground_blocks in
let all_blocks = obj_converted_block_locs@obj_converted_ground_blocks in
let enemy_locs = generate_enemies blockw blockh 0. 0. block_locations in
let obj_converted_enemies = convert_to_enemy_obj enemy_locs context in
let coin_locs = generate_coins converted_block_locs in
let undup_coin_locs = trim_edges(avoid_overlap coin_locs converted_block_locs)
blockw blockh in
let converted_block_coin_locs = converted_block_locs@coin_locs in
let enemy_block_locs = generate_block_enemies converted_block_locs in
let undup_enemy_block_locs = avoid_overlap enemy_block_locs
converted_block_coin_locs in
let obj_enemy_blocks = convert_to_enemy_obj undup_enemy_block_locs context in
let coin_objects = convert_to_coin_obj undup_coin_locs context in
let obj_panel = generate_panel context blockw blockh in
all_blocks@obj_converted_enemies@coin_objects@obj_enemy_blocks@[obj_panel]
(*Main function called to procedurally generate the level map. w and h args
* are in pixel form. Converts to block form to call generate_helper. Spawns
* the list of collidables received from generate_helper to display on canvas.*)
let generate (w:float) (h:float)
(context:Dom_html.canvasRenderingContext2D Js.t) :
(collidable * collidable list) =
let blockw = w/.16. in
let blockh = (h/.16.) -. 1. in
let collide_list = generate_helper blockw blockh 0. 0. context in
let player = Object.spawn (SPlayer(SmallM,Standing)) context (100.,224.) in
(player, collide_list)
(*Makes sure level map is uniquely generated at each call.*)
let init () =
Random.self_init();