mastodon.gamedev.place is one of the many independent Mastodon servers you can use to participate in the fediverse.
Mastodon server focused on game development and related topics.

Server stats:

5.4K
active users

Bamboy

1/15
on how I made a style procedural level generator, with visual examples and names of image processing algorithms being used for each step.

While I am using , the level generator code is standalone and I hope to write this in a way that is useful for any language. My focus is going to be on the actual steps being taken, and less on the programming.

2/15
First off, a small bit about the 2D games. They feature destructible levels comprised of bitmaps, on which characters take turns trying to eliminate the enemy team(s) with a variety of weapons.

For speed reasons, it is desirable to separate the visual aspects of terrain from the collision data. For this reason we will generate the collision data first and then make the visuals later. (Which I still have yet to do)

Here is an example of a Worms: Armageddon (PC Release 1999) level:

3/15
Our final collision data will be made of a 2D boolean array. (C# bool[,]) Each x/y coordinate in our array will represent a single pixel of our level being solid (true), or not (false).

But, before we get to dealing with huge arrays, we are going to start with a 2D boolean array that is much smaller, but will have the same aspect ratio as our final array.

Let's do 40 x 20, but really this can be any positive size.
(Pretend the red dot in the image isn't there, lol)

4/15
Now, lets randomly place some pixels and store the coordinates of those pixels in a separate array. We want to avoid pixels being placed on the edges of our array, otherwise some algorithms we run later will have errors.

(Placed pixels will be shown as darker cyan from here on out.)

The pixels we place in this step, random or otherwise, will greatly modify the shape of our final land. I have found the best results use a mix of random and not random.

6/15
Prim's Algorithm doesn't know how to actually convey its information onto our array of pixels though

Bresenham's Line Algorithm has probably been the most critical algorithm in my entire project. What does it do? Draws lines of single pixel thickness! I modified it to write to boolean arrays instead of writing colors.

en.wikipedia.org/wiki/Bresenha
codeproject.com/Articles/15604

New color keys from here on out:
Black lines: Prim's output
Cyan: Bresenham's drawing Prim's output onto our boolean array

7/15
That's more interesting! It's still a bit too thin.
Time for the growth phase! I just kinda did this part with my own code.

For X growth iterations, do the following for each pixel NOT on the edge of our array:

Count cardinal neighbors with true values. If greater than one, add pixel coordinate to a new array.
Once all pixels have been checked, iterate through our coordinate array and set the value in our boolean array to true.

Green = grown pixel

That results in something like this:

8/15
You might have noticed a decrease in green pixels at the top of the grid. That's because after a certain height, I only grow pixels that meet the requirements with a 50% chance. I also have a variable that will randomly grow pixels with only one solid neighbor, becoming less likely the higher the pixel is.

This is a bit too... floaty. Lets set some pixels at the bottom BEFORE our growth step. Also lets add bigger chunks of pixels (I prefer one) to give a bit more "beef":

9/15
That's coming together! As far as the smaller boolean array goes, we're done with messing with it. Time to start on the much bigger boolean array that will represent our level's collision data.

Before that, we are going to create another 2D array that represents the positions of shared corners, multiplied by an integer upscaleFactor. The size of our final level will be the size of our smaller boolean array multiplied by upscaleFactor.

10/15
Next, an ugly part of my code, an IntVector2[,][]
That monstrosity is a 2D array where each X/Y index holds another array of IntVector2, specifically of length 4, to represent all four corners of any given pixel from our smaller boolean array.

I pass this data into our next algorithm: Theo Pavlidis' Contour tracing algorithm.
imageprocessingplace.com/downl
gist.github.com/Smurf-IV/45236

...which after given the edge of a shape will return a list of positions of that shape's edges.

www.imageprocessingplace.comContour Tracing

11/15
Can you guess what we do next? Yup, we feed all those positions into Bresenham's line algorithm, but this time we draw on the larger boolean array.

12/15
That outline is a bit too blocky, so we're going to introduce yet another algorithm: Chaikin's Algorithm

This algo takes a set of points and makes a curve out of them, doubling the number of points it was given for each iteration.

Lots of resources on this algo, including an interactive web version at observablehq.com/@infowantstob

cs.unc.edu/~dm/UNC/COMP258/LEC
sighack.com/post/chaikin-curve
keithmaggio.wordpress.com/2018

Smoothing the shape given to us by TheoPavlidis' gets us this:

13/15
Now for the real magic step that I left out of post #9 of this thread, for clarity purposes.

When calculating shared corners of pixels, we are going to apply a random offset to each shared corner.

The magnitude of the random offset MUST be small enough such that cell walls will not overlap, otherwise we are going to run into problems. For this reason I opted for a zero to one value for this setting.

14/15
The last step before we have something that can be used for gameplay: Flood fill inside of our shape's outline.

15/15
The results look even better at larger values of upscaleFactor. So far in this tutorial I have been using an upscaleFactor of 10 (400 x 200 image) but 30 (1200 x 600) is much closer to something that would be playable.

The last step that I will reveal for all of you here: Chop off the bottom most upscaleFactor pixels from the resulting level so that the bottom of the level looks more natural.

Hope yall found this helpful/interesting!

@bamboy That's pretty cool stuff. But for next time consider making the follow-up toots "unlisted" rather than "public" to limit the local TL spam a little bit =P

@bamboy thank you for posting this - it’s really useful!