/Godot-2D-RogueLike

This repo is of a 2D Top Down RogueLike made in Godot where the player will delve into a randomly generated dungeon to gather materials and experience to upgrade their skills and weapons.

Primary LanguageC#

Godot 2D RogueLike

This example game is 2D Top Down RogueLike Dungeon Crawler made in Godot where the player will delve into a randomly generated dungeon to gather materials to craft their own weapons.

The game engine is Godot using C# scripting and I am doing everything myself besides most of the assets, which are from various licensed sources. I use this as a playground for various ideas of gameplay systems that I would like to implement.

Table of Contents

World Generation

One major part of this game is the world generation. The first iteration the generator randomly selects if each tile is alive or dead based on a initial chance to create a good starting point. It then uses modified rules from Conway's Game of Life for many iterations to generate somewhat natural looking caves. Specifically based on this article. I plan to add later world generation methods as it is something that interests me.

After the initial world generation the generator finds the largest cave and coordinates to each tile inside that cave to select as the playable area. In the first iteration I used an algorithm that set every single tile its own set and then merge adjacent sets to give me the largest cave and use that as the playable area. This was immensely slow and would take several minutes for a 128x128 map which is ridiculous. A flood fill would not work because I have an N number of caves and do not know their starting positions or the minimum size of these caves. I could try to prematurely optimize and generate a sparse grid every 5 pixels or so as the starting location for the flood fill and then merge adjacent sets which would potentially ignore the smaller caves but I would prefer a complete algorithm. In the second iteration I used a Connected Component Labeling Algorithm, which is supposed to be O(NxM), was significantly faster, but also more complicated. Currently a 128x128 level takes around 1-2 seconds to run the entire CCL algorithm, Implemented here, on and I plan on optimizing this at a later date but currently 1-2 seconds is fast enough considering I am not planning to have levels much larger than 128x128 considering the size of the tiles.

Next I am looking to figure out the best way to place the level starting location and levelending location in addition to figuring out some way to differentiate rooms vs hallways in reguards to spawning enemies and items in the world. Currently my method to figure out larger rooms is to figure out the distance from each ground tile to the wall. I am doing this somewhat effectively by using the Midpoint Circle alg to pregen a series of growing radii to then run simple collision to determin the distance from the edge of a room any tile is and save that data.

I added K-Means to see if I can get some solid room differentiation and it seems to be working but there are still islands across the wall that belong to other caves. I am going to add a post step to remove them and then floodfill to fix the corners.

A* has been added and works, I am considering rewriting it in C++ and recompiling Godot as it will be faster than managed C# but its not needed currently.

Level World Generation Screenshots

Map with no overlay

DirGraph_CCL_NoOverlay

Map with Adjacency Overlay

Map01_Adjacency

Map with K-Means

Map01_K-Means

Map A* Pathfinding Heuristic is Octile distance weighted 1x

Map01_AStar

A* Debug Viewer, Question mark is open list, yellow cicles are closed list then finished path

2021-09-10.22-58-08.mp4

Set Combination Example 1

FloodFill1

CCL Example 1 128x128 @ 40% initially alive change for Game of Life

WorkingCCL_128x128

CCL Example 2 128x128 @ 45% initially alive change for Game of Life

WorkingCCL_128x128_45initial

CCL Example 3 128x128 @ 46% initially alive change for Game of Life

WorkingCCL_128x128_46initial

CCL Example 4 256x256 @ 40% initially alive change for Game of Life

WorkingCCL_256x256_40initial

CCL Example 5 200x100 @ 44% initial with only largest cave visible

CompleteCCLWithLargestCave

Adjacency overlay of 200x100

Adjacency

Map Generation UI

UpdatedLevelGenUI

Weapon Crafting System

Another major portion of the game is the weapon creation. The goal of the weapon creation is the player is able to create a weapon that suits not only their playstyle but with combinations of parts the weapon would be effective in combat. I decided on a part and material system where the materials will affect how statistically strong and effective the weapon would be, such as amount of damage on hit or applying a fire debug on enemies that are hit. The parts of the weapon would change how it swings, the swing speed, and all of the other properties of the weapon. ssss

In the first iteration of the Weapon Crafting System I decided that weapons would be based upon blueprints. For example I had a Rapier blade that required a small handle, rapier blade, and rapier handguard. These would affect the swing speed, damage, and special ability of the weapon respectively. After testing the weapon creation I found it very limiting and not allowing the player to create a new weapon but forcing them to pick an archetype of a weapon, such as big sword, rapier, dagger, and only change slight parts.

In the second iteration of the Weapon Crafting System I completely changed not only the internal data structure of the weapons, and streamlined the entire art asset loading pipeline, but also decided to adopt a more freeform creation process. The process starts with the player choosing the length of handle that they want. This affects the reach of the weapon along with the swing speed and a damage modifier to the entire weapon. The player then has nodes along the handle, currently just the top and bottom, and they can click on a node to bring up a UI to select a new part to add such as a pommel, blade, pickaxe head, etc. These parts will add to and modify the stats of the weapon. I found thiss system significantly more fun to create with.

The change in the core mechanics of the builder I decided to completely refactor the underlying system. The system before used a combination of various Maps and data structures to track the current blueprint and which parts were selected and the materials for those parts but the new system was more agnostic. It is simply a tree where the root node is the handle. This new system was much easier to design but still difficult to implement due to the UI callbacks needed to correctly change and add to the weapons. I am currently facing issues positioning the UI for the parts themselves on the weapon building screen but the underlying tree and how it's updated and changes with the UI works as intended. Next I need to fix the positioning problems and then add the material selection.

The best part of the system is that the entire thing is driven by a single json file now instead of multiple. I have a single file that registers the icon and statistics of the part along with the node connection points and the type of part that can be attached to it such as a blade or a pommel or guard. This more abstract design allows for very different weapons to be created.

After refactoring the UI placement of the weapons it works as expected. I have also added the material selection menu, this is visible in the Milestone 1 Video above. The internals of the system work and the player can mine resources and use them to craft weapons. The weapons are then placed in their inventory and they can click on them to "equip" them and gain their stats.

Currently working on loads of player feedback for this as well as figuring out a way to composite the sprites of the weapons in the crafting menu to create a single sprite for the player's inventory icon as well as how to somehow create an in game model for the custom weapons without creating tens or even hundreds of animations for each part in each swing/stab motion and compositing those.

The next goal is to update the weapon stats to not be broken and make the materials have an effect on the weapons stats and not just be the color in the crafting menu.

Weapon Crafting System Screenshots

Crafting Weapon Video

2021-09-11.16-10-16.mp4

Example Weapons

Example Weapon: Basic Sword

ModularWeaponsUI_Iter6_ExampleWeapon

Example Weapon: Large Blade

ModularWeaponsUI_Iter6_ExampleWeapon_02

Example Weapon: Large Mace

ModularWeaponsUI_Iter6_ExampleWeapon_03

Example Weapon: Tiny Axe

ModularWeaponsUI_Iter6_ExampleWeapon_04

Weapon Builder UI Iteration 4

ModularWeaponsUI_Iter4_2 ModularWeaponsUI_Iter4_1

Weapon Builder UI Iteration 3

ModularWeaponsUI_Iter3_2 ModularWeaponsUI_Iter3_1

Weapon Builder UI Iteration 2

ModularWeaponsUI_Iter2_1

Weapon Builder UI Iteration 1

ModularWeaponsUI_Iter1_1

Overworld Level Generation

The other algorithm that I am going to implement is Wave Function Collapse, specifically the Simple Tiled Model. This would be used to generate an Overworld for the player to traverse with cities, lakes, etc. The basic WFC algorithm is that every single tile exists in a state of being all tiles, similar to the Wave Function in Quantum Physics which this algorithm is named after. The goal of the algorithm is to observe, select a specific tile, the tiles with the lowest entropy, chance to change or in this case lowest amount of tiles in the possibility set, and then propagate those changes throughout the map to create a stable state where the entire map is observed. Once this is finished a map is generated. There are rules for the tiles, such as water is only adjacent to the coast etc, and these rules determine the possible set of tiles that any position can be observed as. There are some problems with this algorithm, specifically with observing on larger maps and how depending on if the algorithm is implemented recursively, as would be most natural, may cause stack issues when destroying a branch of possibility with the tile choice.

There is a paper called Automatic Generation of Game Levels Based on Controllable Wave Function Collapse Algorithm that delves deeper into using the WFC Simple Tiled Model for game level generation. They introduce the idea of constraints and layering to the model along with the idea of symmetry of tiles and the ways to transform the tile images to generate more effective tiles from a single art asset. What I think was most interesting is their usage of a multi staged level generation system where they would constrain and generate the ground level of a world and then on top of it add other levels such as the road, chests, keys, and enemies in different stages and used the stages beforehand as additional constraints for the later stages.

My goal is to have a multi staged WFC Simple Tiled Model level generator that builds upon the stages before and leverage this with the AutoTile system in Godot to automatically tile the entire map.

Currently working on several Top Down room generation methods. Right now I have a Voronoi Diagram generated from Octile and exact distance and plan to add Manhattan distance as well as a method to create rooms. The main goal is to have a BVH abstract generation so I can swap out the initial room and map generation systems to create different types of levels.

Other great sources for WFC: Wave Function Collapse Boris the Brave's Explanation of WFC Caves of Qud using WFC as a stage of Level Gen Oskar Stalberg WFC in Bad North

Generated World Goal

example_overworld

Initial Lighting Testing

For this game I wanted to have a unique art style but sadly I am not a skilled artist I decided to leverage my knowledge of 3D graphics and PBR, IBL, etc to make the game look pretty without beautiful assets. While researching 2D platformer/adventure/metroidvania games for a different project I came across this very beautiful game Seige and Sandfox. I appreciated their lighting and art style, they managed to make the game feel like an old school 8bit game while having very pretty lighting. Watching their talk Beyond Authentic Pixel Art in Unreal Engine online and seeing the upgrade and depth that normal maps added to their world inspired me to try it myself. I decided to add normal mapping with lighting to the 2D sprites and see what that type of 2D/3D depth effect would look like.

It took me a day or two but after digging through Godot's documentation found out that they have a 2D/3D lighting pipeline and after plugging in a normal map I found online and one I created for some ingots I got decently good looking lighting and plan to create a larger test scene with walls and props to see if I want to continue down this path.

Initial Lighting example

GodotNormalLighting_Test01.mp4