Skip to main content
Version: 2.0.5

(PRO) Custom input

In the free version of the asset, the input for the generator is fixed. That means that we create a level graph in the GUI and give it directly to the generator. However, there are situations where we might want to alter the level graph. For example, we may want to add a secret room that is connected to a random room in the level.

In this tutorial, we will learn how to implement custom inputs in order to have more control over the input for the generator.

LevelGraph and LevelDescription​

The first thing that we need to understand is the difference between LevelGraph and LevelDescription classes. If you are reading this tutorial, you probably know what is a level graph. It is a collection of rooms and connections, and it describes the high-level structure of generated levels. With each level graph is associated an instance of the LevelGraph scriptable object.

Note: The LevelDescription class has a different suffix based on if it is used in the 2D version (LevelDescriptionGrid2D) or in the 3D version (LevelDescriptionGrid3D). The same also applies to DungeonGeneratorInputBase, while LevelGraph is the same in both versions.

However, level graphs are not directly given to the generator as an input. First, we need to convert the LevelGraph to an instance of the LevelDescription class. The reason for that is that level graphs are made primarily for the GUI editor, and we need to convert them to a real graph data structure.

Both LevelGraph and LevelDescription revolve around rooms and connections. The following code should demonstrate the basic API of both classes and how to convert one to the other one:

// [CreateAssetMenu(menuName = "Dungeon generator/Examples/Docs/My custom input task", fileName = "MyCustomInputSetup")]
public class CustomInputExample : DungeonGeneratorInputBaseGrid2D
{
public LevelGraph LevelGraph;
public bool UseCorridors;

protected override LevelDescriptionGrid2D GetLevelDescription()
{
var levelDescription = new LevelDescriptionGrid2D();

// Go through rooms in the level graph and add them to the level description
foreach (var room in LevelGraph.Rooms)
{
levelDescription.AddRoom(room, GetRoomTemplates(room));
}

// Go through connections in the level graph
foreach (var connection in LevelGraph.Connections)
{
// If corridors are enabled, add corridor connection
if (UseCorridors)
{
// Create a room for the corridor
var corridorRoom = ScriptableObject.CreateInstance<Room>();
corridorRoom.Name = "Corridor";


levelDescription.AddCorridorConnection(connection, corridorRoom, GetCorridorRoomTemplates());
}
// Else connect the rooms directly
else
{
levelDescription.AddConnection(connection);
}
}

return levelDescription;
}

/// <summary>
/// Gets room templates for a given room.
/// </summary>
private List<GameObject> GetRoomTemplates(RoomBase room)
{
// Get room templates from a given room
var roomTemplates = room.GetRoomTemplates();

// If the list is empty, we use the defaults room templates from the level graph
if (roomTemplates == null || roomTemplates.Count == 0)
{
var defaultRoomTemplates = LevelGraph.DefaultIndividualRoomTemplates;
var defaultRoomTemplatesFromSets = LevelGraph.DefaultRoomTemplateSets.SelectMany(x => x.RoomTemplates);

// Combine individual room templates with room templates from room template sets
return defaultRoomTemplates.Union(defaultRoomTemplatesFromSets).ToList();
}

return roomTemplates;
}

/// <summary>
/// Gets corridor room templates.
/// </summary>
private List<GameObject> GetCorridorRoomTemplates()
{
var defaultRoomTemplates = LevelGraph.CorridorIndividualRoomTemplates;
var defaultRoomTemplatesFromSets = LevelGraph.CorridorRoomTemplateSets.SelectMany(x => x.RoomTemplates);

return defaultRoomTemplates.Union(defaultRoomTemplatesFromSets).ToList();
}

Custom input implementation​

Custom inputs are quite similar to Custom post-processing logic. We have to create a class that inherits from DungeonGeneratorInputBase. And because the base class is a ScriptableObject, we need to add the CreateAssetMenu attribute, so we are able to create an instance of that ScriptableObject. The DungeonGeneratorInputBase class has one abstract method that we need to implement - LevelDescription GetLevelDescription():

// [CreateAssetMenu(menuName = "Dungeon generator/Examples/Docs/My custom input task", fileName = "MyCustomInputSetup")]
public class CustomInputExample2 : DungeonGeneratorInputBaseGrid2D
{
/* Create the level description here */
protected override LevelDescriptionGrid2D GetLevelDescription()
{
/* ... */
}
}

After we implement the logic, we have to create an instance of that ScriptableObject by right-clicking in the project view and Create → Edgar → Examples → Docs → My custom input. And the last step is to switch the Input Type in the generator inspector to Custom Input and drag and drop the ScriptableObject instance to the Custom Input Task field.

Typical use cases​

Add rooms to the level graph​

One typical use case is adding additional rooms (for example a random secret room) to an existing level graph. The workflow is usually as follows:

  1. Create the static part of the level graph in the GUI
  2. Create a custom input task with a public level graph field that we will assign our level graph to
  3. Convert the LevelGraph to LevelDescription (as discussed above)
  4. Create additional rooms and connect them to existing rooms in the level description

To make it easier to work with the graph of rooms and connections, LevelDescription has a IGraph<RoomBase> GetGraph() method to get the current graph of rooms. The graphs contain all the expected methods like getting all rooms or checking if two rooms are neighbours.

For an example of how can this be implemented, see the Enter the Gungeon example where we connect a secret room to a random room in the graph.

Note: The graph which is returned by the GetGraph() method currently does not get updated when you modify the level description. You need to call the method again to get a new graph.

Assign room templates automatically​

Another typical use case is implementing custom logic for assigning room templates to individual rooms. For example, if we use custom rooms, we may want to assign room templates based on the type of the room instead of manually assigning room templates to individual rooms. This can be seen both in Enter the Gungeon and Dead Cells examples.

Procedural graphs​

It is also possible to have a completely procedural structure of levels by creating the whole level description on the fly without any static parts.