Skip to content

Files

Latest commit

 

History

History
74 lines (38 loc) · 8.03 KB

README.md

File metadata and controls

74 lines (38 loc) · 8.03 KB

Alizarin: Overview

Intro Screen

This was my senior design project for Fall 2016 for UPenn's digital media design program. This is a Diablo-inspired dungeon crawler created in Unreal Engine 4. The main task was to create an art-directable level generator, and the gameplay is very simple. Development has paused since then. This readme is my postmortem and will show off the highlights of the project.

High Level Generation Algorithm

This is a tileset generator made to allow artists (especially me) space to make interesting tiles, so it is broken up into large modules. These modules occupy a grid, with objective / hero modules generalized to multiple grid spaces. The algorithm is:

  • Scatter objective rooms and remove collisions
  • BFS from each objective to the next, and mark the best path as a required path
  • Along each required path, recursively generate a hallway / maze area from the modules (pick some compatible number of doors, pick a room with that many doors, place it and recurse)

There are a few twists to this algorithm described in "Generation Parameters and Effects" section.

This means that in the Unreal editor window, none of the level exists before runtime. Unfortunately this means no light baking or nav-mesh baking. Still, with some tweaking the updating nav-mesh was responsive enough and the level still looks good with two shadow-casting lights. It runs at 60FPS on my laptop with a GTX 970M. Generating the level itself and spawning assets is on the order of milliseconds.

Generator Tool

The generator is written in C++ and each 'tileset' would be a blueprint (Unreal's visual scripting system) subclass of the base. This provided an easy frontend editor for someone to use while working in engine. For the project I created a single tileset, a space platform, and its editor looks like this:

The generator modules section is where I define what rooms are included in the generator. If I develop a new room that I want to add, I find the list with the appropriate number of doorways (1, 2, 3, or 4) and click the (+) button. This will add a new drop-down menu that allows me to pick out my room. Here is what a single, ordinary room module looks like:

The same is true for the objective rooms, where there are more specific gameplay encounters. However, those rooms can cover more than one grid cell. I also must specify a start and end room. These can also be generalized across multiple cells and even ordinary objective rooms if I want. If I developed a story mode, for example, I could make a big setpiece area for the start and end to fit the plot of the level if I wanted. And if I wanted to make another tileset altogether, I could just make another blueprint class and fill in the same lists with new modules.

Generation Parameters and Effects

I implemented three simple parameters to control the randomness of my generator. I will break down the results of testing each extensively.

Target Density

Treat every room module as a vertex in a graph, and the number of doorways as its degree. The Target Density (TD) is the desired average degree of all vertices. Here are some sample outputs:

And here are the points of interest in the same outputs:

And lastly the paths found by the generation algorithm that made 'required' routes:

Target Density of 1, of course, produces (usually) completely linear levels. In the top-middle example you can see that the generator placed the exit of objective 1 fairly far away from the entrance of objective 2, so the path between them has generated a cycle. But really, the only interesting thing about TD 1 is that it shows my pathfinding working. TD of 4 is also boring: it will connect everything to everything else and become a grid. Target Density of 2.5 is an even distribution because it is the expected value of a randomly chosen degree. This is probably higher than we would want if I didn't add other parameters, because it will usually fill up the allotted grid. But while the shapes are looking more interesting, it did happen to generate a pretty unfun level:

This is the middle-right example in the table. I've marked each of the required cells (on connecting paths) with dots. Blue dots are in one path. Green are in two, which means backtracking. There is even a cell where all three intersect, with the red dot. What makes this egregiously bad is the objectives must be completed in order, so the far one must be completed before the near one. This backtracking / path intersection is a real issue because it won't become evident until a lot of work has already been done. I will have to find a better strategy of placing rooms, or perhaps I will alternate between generating rooms and paths.

The other issue is labelled with the yellow numbers. They show the distance in cells from the nearest required / path cells. If you explore this path in game, you'll have to walk all the way back. These kinds of large branches are mostly gone with the addition of my other parameters, but if I allow something like this to exist then there needs to be some quicker way to get back, like a teleporter.

Conformity

I honestly couldn't decide whether I should call this 'conformity' or 'strictness', so I mixed it up in the diagrams. Oops. Whatever it's called, it is the percent chance that each cell will adjust to the TD. All of the examples above have conformity 1, or 100%. Conformity of 0 means that TD does not affect the generation at all. Here are some basic examples with conformity on:

Now we're seeing some branches on the TD=1 levels, and some gaps in the TD=4 levels. But that still isn't quite enough control, and that's where the third parameter comes in:

Soft Limit Cell Ratio

Wordy title here, unfortunately. SLCR is used to influence the number of regular rooms that show up in the level. As the level generates paths and branches, it will keep a running estimate of the projected number of rooms. The number of absolutely required rooms is the number of cells along the paths between objectives. If the generator is running and SLCR <= (# projected) / (# required), then it will snap subsequent rooms to the minimum number of doorways, hence "soft limit." I also distributed this limit along the paths found between objectives; if the SLCR did not account for that then levels would be very dense at the beginning and branch-less paths at the end. This also accounts for the possibility of intersecting paths.

Now we have an interesting way to use our other parameters. With the example on the right, each room will be forced to maximize its number of doorways by TD=4, but it's stopped by SLCR=2. This also leads to the interesting layout with some small area to explore before objective 1 and a larger one after. So a smaller SLCR will create a more directed level, even with a bigger (or maximum) TD.