Dungeon Building
Introduction
Welcome
Welcome to the world of IVAN dungeon building. It is a part of game development that is as interesting and intricate as the game itself. Designing and building a dungeon for IVAN presents many challenges and it can feel like a steep learning curve, however, the satisfaction of creating a world in which many enthusiastic IVANers can play in and feel part of is very rewarding. This guide is aimed at those with little or no coding skills, but who are comfortably computer literate. Those who are already familiar with C++ (or any other language) will be advantaged by their programming experience, and will hopefully find the collection of examples on dungeon building suitably novel and interesting.
What we will look at
Tools
A suitable text editor and font
When editing the dungeon file, it pays to have a good text editor. A nice program in Windows might be, for example, Programmer’s Notepad 2, but any program with suitable syntax highlighting will do. The script file text is based on C/C++ syntax, so you can set the syntax highlighting to this language. It is also highly prudent to select a mono-space typeface. A mono-space typeface means each character has the same horizontal length in the editor. Because we are also going to “draw” with ASCII characters in the script files, it would be good if the characters can properly align in a grid-like fashion, making it easy to identify the geometrical patterns. This is indispensable when it comes to laying out large rooms and even particular dungeons, helping to avoid confusion along the way. Suitable typefaces to consider for this purpose might include Consolas, Courier New or Lucida Console fonts.
Backing up your work
This is really the most important step, and should be the first thing you organize. Make a copy of your dungeon file and store the copy somewhere else, that way, you can always revert to the original if you make changes all over the place. No matter if you lose original files, you can usually pick up an old file from a repository somewhere and revert back to it.
A more guru way to do this, would be to set up your own IVAN development branch, by cloning the attnam/ivan
git repository and making changes in your own branch. You can make pull requests to merge you work into the main line of IVAN development! By using a version control system like GIT, you enable for yourself a great tool for maintaining your dungeon script file.
Philosophy
Order and Chaos
IVAN has the possibility for building both persistently generated content (order) and procedurally generated content (chaos). It is possible to create very rigidly designed places, with every terrain and item having its proper place. The city of Attnam might be considered a highly persistently generated place. Some places are also generated in a way that produces unique dungeon areas - never before visited - with random layout and loot. The Gloomy Caves might be considered such a place. This is only partly true, however, since we can also find things in the Gloomy Caves that can be encountered in each game, like the Gloomy Caves shop or the Enner Beast. We will examine both philosphies in detail, using the Gloomy Caves as an example of a balance between the tools for producing persistent elements and procedural elements in a dungeon.
A first dungeon
Dungeon script hierarchy
Examples from the original game
Gloomy Caves Shop
The shop in the gloomy caves is a good example of how to manually lay out a specific room in the script. The shop is declared as a simple room. The size is set to 5 tiles wide by 9 tiles high. Because it is a shop, and Mellis is the god of trade and politics, the divine master is set to Mellis. No altars, fountains, locked doors or booby trapped doors will generate here. This will generate a bare room (see adjacent image). It generates doors and lanterns by itself, because this is specified in the default parameters for rooms in the dungeon.
Script | Result |
---|---|
Room { Size = 5,9; AltarPossible = false; DivineMaster = MELLIS; Type = ROOM_SHOP; GenerateFountains = false; AllowLockedDoors = false; AllowBoobyTrappedDoors = false; |
Next we can fill the room with characters. We can place each character in a precise way, by writing a little ascii map showing where everyone will go. The map - a character map - has a size, a position relative to the room's origin (the top-left corner is x = 0; y = 0;
), and a key showing all the character types to be found on the character map. The character map can be thought of as a little sub-map belonging to the room. We need to show where the character map is positioned relative to the room it is in. We see the top-left corner of the room Pos = 0, 0;
is occupied by a wall. We want to have a map that covers the floor area. The total floor area available in the shop is 3 tiles wide by 7 tiles tall, so this will be the size of or character map. The top-left tile in the character map will be the top-left tile in the room's available floor space. Therefore we will position the character map at Pos = 1, 1;
, relative to the co-ordinates of the room's total space.
We need a shop keeper, and some strong guards to protect the fine wares. Therefore we will add the types g
for a guard
character with the config SHOP
, and s
for a shopkeeper
with the config ELPURI_CAVE
. We'll put them both on the same team, and we'll make the shopkeeper the room's master, which is relevant to the room type being a shop. So when the player character wishes to make a trade in the shop, the person he will deal with is the shopkeeper.
We then lay the types out in a 3x7 ascii map. White spaces are ignored, full-stops represent no character, and g and s represent the guard and the shop keeper. We can place the guard in the shop many times over, and we will place one of them in each corner. We probably only want one shop keeper, and so he is placed in the middle of the map.
Script | Result |
---|---|
CharacterMap { Pos = 1, 1; Size = 3, 7; Types { g = guard(SHOP) { Team = 5; } s = shopkeeper(ELPURI_CAVE) { Team = 5; Flags = IS_MASTER; } } } { g.g ... ... .s. ... ... g.g } |
We need to put items in the room so the shopkeeper has something to sell. We can do this in the same way as with the character map, only this time with an item map. The item map will have the same footprint as the character map, so this will be the same as before. We want to fill the remaining available spaces in the room with random items of varying categories and worth. In the script, random items are created, but within the given parameters of minimum and maximum prices, and item category flags. This gives a good balance between high and low value items, preventing the game from creating a room full of broken leather boots, or for that matter, a room full of magic lamps! If we want we can generate specific items, and so by looking at this script you will see a wand of striking will generate in every shop, for every game of IVAN. This is an example of a so-called "guaranteed item", and is a design element I hope you will choose to use sparingly.
Dungeon default control variables
Dungeon
Keyword | Values | Description |
---|---|---|
Dungeon
|
0, 1, 2, ... , 64 | This is the dungeon enumerator variable. Enumerations 0 to 4 are occupied by the original dungeons with predefined variables like ATTNAM or ELPURI_CAVE. |
Levels
|
0, 1, 2, ... | This determines the total number of levels in the dungeon. |
Description
|
"string" | This is a short name for the dungeon that shows up in the high score list explaining which dungeon the player died in. |
ShortDescription
|
"string" | This is an even shorter name for the dungeon usually limited to two or three characters, for example "GC" for gloomy caves. It shows up on the side bar. |
LevelDefault
Keyword | Values | Description |
---|---|---|
Size
|
x, y;
|
This is the size of the level in squares. x and y are the horizontal and vertical size respectively with x, y ∈ {1, 2, 3, ... }. |
Rooms
|
n; or p:q;
|
Total number of rooms possible on the level. Can be a fixed number, n, or a number of generic rooms between p and q with p < q. n, p and q ∈ {0, 1, 2, 3, ... }. |
Items
|
n; or p:q;
|
Total number of items that will generate on the level. Can be a fixed number, n, or a random number of items between p and q with p < q. n, p and q ∈ {0, 1, 2, 3, ... }. |
TunnelSquare
|
solidterrain(FLOOR_TYPE), 0;
|
TunnelSquare determines the appearance of the tunnel floor. First parameter determines the floor appearance, second parameter should be zero. |
FLOOR_TYPE
|
Choose from PARQUET, FLOOR, GROUND, GRASS_TERRAIN etcetera | |
0;
|
To make a tunnel, this ought to remain zero. Non-zero will cause the tunnel to fill with material! | |
FillSquare
|
solidterrain(FLOOR_TYPE), MATERIAL earth
|
FillSquare determines what to fill the level's walls and filling material with, and what type of ground appears under those walls. First parameter determines the floor, second parameter determines the walls. |
FLOOR_TYPE
|
Choose from PARQUET, FLOOR, GROUND, GRASS_TERRAIN etcetera. Good to choose from the same type as TunnelSquare. | |
MATERIAL
|
Choose from GRAVEL, MORAINE, GOLD etcetera | |
BackGroundType
|
m;
|
Defines the default background type. Not sure what it precisely affects. m can be GRAY_FRACTAL , RED_FRACTAL , GREEN_FRACTAL etcetera.
|
EarthquakesAffectTunnels
|
true or false
|
Tells the game whether an earthquake on the level can rearrange the layout of tunnels. Open levels will crash if this is not set to false (because there are no tunnels to shift). Set to true if the level has tunnels, set to false if it is open. See also IsOnGround .
|
IsOnGround
|
true or false
|
Tells the game whether the level is above ground. This prevents earthquakes from happening on the level. Silva will summon wolves instead. |
IgnoreDefaultSpecialSquares
|
true or false
|
If specific squares containing items/traps etc are specified in LevelDefault, then setting this to true for a particular level will cause these random squares not to appear. Accordingly, in LevelDefault this should be set to false. See Square (DefaultSpecialSquares). |
CanGenerateBone
|
true or false
|
Tells the game whether a bone file can be generated on this level, or not. Best to set the default value to true. Typically set this to false if the level will generate a shop, or a victory boss like Oree (not Enner). |
LOSModifier
|
16, 24, 32, 48, ...
|
Line of sight modifier. Typical value for underwater tunnel or gloomy caves is 16. Zombie level is 24. Sumo area is 32. Towns are typically 48. Higher value means player character can see further. |
DifficultyBase
|
0, 5, 10, ...
|
Sets the base difficulty for the level. In LevelDefault, this is the base difficulty for all the levels. |
DifficultyDelta
|
0, 5, 10, ...
|
This sets the value added to the DifficultyBase for each consecutive level. In LevelDefault, this is the change in difficulty per level (dD), without needing to continually specify DifficultyBase (DB) for each level. The difficulty, D, is therefore set to D = DB + dD * L , where L is the current level number.
|
GenerateMonsters
|
true or false
|
Whether monsters can be generated on the level or not. Set to false in peaceful towns; typically set to true in caves. |
TeamDefault
|
TEAM_NAME
|
Sets the team to which spawned monsters are aligned. Typically set to MONSTER_TEAM to generate hostile enemies.
|
MonsterAmountBase
|
0, 10, 20, ...
|
Sets the maximum number of monsters for the level for the generator. While the number of monsters is less than this value, the monster generator adds a new monster. Above this value, monsters are not generated. |
MonsterAmountDelta
|
0, 5, 10, ...
|
This sets the value added to the MonsterAmountBase for each consecutive level. In LevelDefault, this is the change in monster setpoint per level (dM), without needing to continually specify MonsterAmountBase (MB) for each level. The number of monsters (M) setpoint for any given level, is set to M = MB + dM * L , where L is the current level number.
|
MonsterGenerationIntervalBase
|
100, 120, ...
|
The time taken between monster generations, in tens of ticks |
MonsterGenerationIntervalDelta
|
..., -10, 0, 10, ...
|
The increment per level in the time taken between monster generations, in tens of ticks. A negative number means that that the generation interval becomes shorter by that amount, for each consecutive level. The implication being, negative numbers increase the frequency of monster generation with increasing level depth. |
ItemMinPriceBase
|
0, 20, 40, ...
|
The minimum price for items generated on the level, in gold coins. Lower minimum price causes items to generate from low-value materials. |
ItemMinPriceDelta
|
0, 10, 20, ...
|
Incremental increase or decrease in generated item value per level, in gold coins. |
EnchantmentMinusChanceBase
|
0, 5, 10, ...
|
Make a finite chance that weapons generate with negative enchantment. High values increase the odds the player will come acrossterrible equipment. |
EnchantmentMinusChanceDelta
|
..., -5, 0, 5, ...
|
Increments the EnchantmentMinusChance with increasing level depth.
|
EnchantmentPlusChanceBase
|
0, 5, 10, ...
|
Make a finite chance that weapons generate with positive enchantment. High values increase the odds the player will come across good equipment. |
EnchantmentPlusChanceDelta
|
..., -5, 0, 5, ...
|
Increments the EnchantmentPlusChance with increasing level depth.
|
IsCatacomb
|
true or false
|
Normally set to false. This variable causes a level to spawn only ghosts, skeletons and zombies. Used in the Attnamese Catacombs. |
RoomDefault
Keyword | Values | Description |
---|---|---|
Size
|
x, y; or m:n, p:q;
|
|
Pos
|
x, y; or a:XSize-b, c:YSize-d;
|
|
XSize, YSize
|
These are special keywords that return the horizontal length (XSize) and vertical length (YSize) of the level, in squares. | |
Type
|
ROOM_TYPE
|
Tells IVAN what type of room. Choose from ROOM_NORMAL, ROOM_SHOP, ROOM_CATHEDRAL, ROOM_LIBRARY, ROOM_BANANA_DROP_AREA or ROOM_SUMO_ARENA. Most common are ROOM_NORMAL and ROOM_SHOP. |
Shape
|
1, 2
|
Determines whether room has rounded corners, or rectangular corners. Choose from either RECTANGLE or ROUND_CORNERS .
|
GenerateTunnel
|
true or false
|
Determines whether the room is to be connected to the rest of the level via a tunnel. Normally set to true. Set to false to create a secret vault... |
UseFillSquareWalls
|
true or false
|
|
WallSquare
|
||
FloorSquare
|
||
DoorSquare
|
||
GenerateDoor
|
true or false
|
Tell IVAN whether to generate doors for this room. |
GenerateWindows
|
true or false
|
Tell IVAN whether to generate windows for this room. Normally set to true in cities above ground. Normally false underground. |
GenerateLanterns
|
true or false
|
Whether lanterns are to be generated in the room. |
GenerateFountains
|
true or false
|
Whether to generate fountains in the room. |
AllowLockedDoors
|
true or false
|
Allows the possibility for doors to be spawned in a locked state. Only makes sense if doors can be generated for the room. |
AllowBoobyTrappedDoors
|
true or false
|
Allows the possibility for doors to spawn with booby traps. A door does not need to be locked in order to be booby trapped. |
DivineMaster
|
0, 1, ... , 15, 16(?)
|
Restricts range of gods player can pray to, to only this god. Select from VALPURUS , LEGIFER etc. 0 (zero) is all gods, so set this as the default value for your rooms in general.
|
AltarPossible
|
true or false
|
Allows the possibility of spawning an altar. If DivineMaster is set to zero, then it will be an arbitrary god. If DivineMaster is non-zero up to the number of gods, then it will spawn an altar of that deity. If DivineMaster is ATHEIST then it spawns a blank altar (???)
|
IsInside
|
true or false
|
|
Flags
|