Broguelike Dungeon Creation, Part 2
This is part 2 of implementing Brogue’s dungeon generation algorithm in JavaScript. Part one is here.
Loops
Last post left off with something like this:
This is fully-connected dungeon made of a few kinds of rooms. One of the problems is that it’s a tree—each room except the starting room links to exactly one “parent,” shown here:
This can create a lot of frustration during gameplay if you need to backtrack through the same rooms to explore somewhere new, or if you get trapped in a corner when a swarm of kobolds appear.
To solve this, the algorithm adds “loopiness” to the dungeons. A loop is created by putting a doorway between two rooms. Every possible cell in the dungeon is checked to see if it’s a candidate for a loop-creating door, following these rules:
- the door site must be stone (empty)
- the cells on either side of the door must be floor
- the cells on either side of the door must be more than 20 spaces apart.
If all three of these conditions are met, a door is placed, and rooms that were once distant are now connected!
All of this is straightforward to check except rule 3, which requires an algorithm for determining pathing distance.
Pathing distance is calculated using A*, a variation of Dijkstra’s algorithm that uses a heuristic function. In this case, the heuristic is Manhattan distance. Adding a heuristic function means that the search function will try searching in the general direction of the destination before trying other routes.
Here’s an example you can play with. Left click a cell (sorry mobile users), then right click another cell, and the path between them will be shown using a path of o
s, and total distance in the top**. The numbers that appear are the distance from the starting cell (left click).
You can try to identify cells that would meet the condition for loopiness. The dungeon below adds the loops it finds, but with shallow water as the doors 🙃 Some dungeons (especially at this tiny size) don’t require loops to be added, so step through if you don’t see one at first.
Lakes
Generally, lakes are cellular automata placed on top of the dungeon. The caveat is lakes cannot disrupt passibility. Any placement of a lake cannot make any two tiles in the dungeon disconnect.
The clever solution for figuring this out is a flood fill. To determine if the lake disrupts connectivity, “paint” is poured into any floor cell in the dungeon and allowed to flood as far as it can (assume for this metaphor that paint can’t travel through water). This is just the bucket tool in MS Paint. At the end of the flooding, if there’s any dry tiles, those tiles were made unreachable by the lake.
Try it here—clicking a cell will start a flood fill from that location. If there’s any untouched floor tile, the lakes disrupt passibility and should not be placed.
Here’s the total code, including code which adds “wreaths,” the shallow areas that surround the lake like a halo.
for ( | |
let lakeMaxHeight = 10, lakeMaxWidth = 10; | |
lakeMaxWidth >= 5; | |
lakeMaxHeight -= 2, lakeMaxWidth -= 2 | |
) { | |
// create the cellular automata blob | |
({ blob, minX, minY, maxX, maxY } = runCA({ | |
width: lakeMaxWidth, | |
height: lakeMaxHeight, | |
rules: CA_RULES.LAKE_GENERATION, | |
nIterations: 5, | |
startingPercent: 0.45 | |
})); | |
lakeHeight = maxY - minY; | |
lakeWidth = maxX - minX; | |
for (let k = 0; k < 20; k++) { | |
proposedLakeY = randomRange( | |
1 - minY, | |
HEIGHT - lakeHeight - minY - 2 | |
); | |
proposedLakeX = randomRange(1 - minX, WIDTH - lakeWidth - minX - 2); | |
if ( | |
!lakeDisruptsPassability({ | |
dungeon, | |
lake: blob, | |
y: proposedLakeY, | |
x: proposedLakeX | |
}) | |
) { | |
dungeon = drawContinuousShapeOnGrid( | |
blob, | |
proposedLakeY, | |
proposedLakeX, | |
dungeon, | |
cell => { | |
return cell === 1 ? CELL_TYPES.LAKE : 0; | |
} | |
); | |
dungeon = createWreath({ | |
wreathLiquid: CELL_TYPES.SHALLOW_WATER, | |
wreathWidth: 2, | |
dungeon, | |
deepLiquidValue: CELL_TYPES.LAKE | |
}); | |
break; | |
} | |
} | |
} |
And here are the final lakes, not disrupting passibility, with those wreaths added.
Part 3
Part three is now here.