Bömberbots

Project Details

Development Time: 4 months

Team Size: 2

Engine: LÖVE

Programming language: Lua

Main contributions: Debug tools, render sorting, asset loader


Bömberbots Repo

About

Bömberbots is a Bomberman clone that I developed together with

. The goal with the project was to improve our Lua knowledge and to make a smaller game using the LÖVE framework.

Introduction

Initially we set out to make a game inspired by a Warcraft3 mod called “Enfo’s Team Survival”, which is a “tower-defense” game where you control one character that gains items and abilities as you fight waves of monsters.

This side-project ended up clashing a bit with school scheduling, so we ended up scrapping/scaling down the initial concept in favor of creating a Bomberman game. Luckily, we were able to reuse our tools and systems!

Development

We both worked on a lot of different areas of the game, with no clear designated individual focus areas. However, among other things, Tomas created the ECS structure which was integral to the project.

Some of my main contributions were the

used for our collision system, render order sorting, and some debug tools (used to check performance and more easily handle debug drawing.)

Whenever we were done with a feature we would review each other’s code before it was merged into the main branch of the project, to make sure we were aware of any changes and knew how things worked.

Spatial Hash Grid

Spatial Hash Grid

A spatial hash grid is used to store objects based on their position and size in a grid. By doing this we are able to only check AABB-collision against objects in adjacent grid cells.

The way the objects are stored in cells is by converting the size and position from world-space into grid-space, then finding the minimum and maximum indices (position - size, position + size) that they occupy and generate keys for these indices which we kept in the format “x_index;y_index”. After that we could get the objects from the cells we wanted by just passing in the cell-indices.

Render Sorting

Drawing & Render Sort

Players, bombs and tiles being drawn on top of each other

I struggled for a while with coming up with a way of sorting entities based on their position and Z-Index. I wanted a z-index in order to support giving entities different drawing priorities, but I also wanted to be able to sort using just a single “score” in order quickly sort them into a tidy array.

With my first attempts I was encountering issues where with the boundary values of the z-index due to overflow, causing entities on other rows to disappear or be drawn in the wrong order (as the stored entity ID would get replaced with another one).

In the final version the x position is stored in the fractional part of the score (position as a percentage relative to screen boundaries), the z-index is stored in the first 4 digits of the integer part, and the y position is stored in the 5th and greater digits of the integer part. The score is inserted into an array using binary insertion, and the entity ID is stored in a table using the score as key. With this setup we can easily loop through the array of scores, and get the ID from the entity table matching to each score in order to draw the corresponding entity.

Sorting score calculation examples
z_offset = 5000
z_index = [-5000..4999]
x = [0,1]
y = [-inf..inf]

score = 10000 * y + z + z_offset + x

Example: y = 1, z = 2345, x = .67, score = 12345.67
Example 2: y = 76, z = 5432, x = .10, score = 765432.10

Debug Tools

Performance

Performance Debug

We wanted a way to easily verify whether a change to code would result in a performance improvement or not, and to measure the time it took for functions to execute so that we could find out where we improvements could be made.

I created a get_execution_time function which accepted a function and it’s parameters as a variable argument, which meant we could use it for any type of function. The get_execution_time would execute the function and return the amount of time it took in seconds accurate to the microsecond. I additionally added a few print methods which would first get the time and then print out the time it took in a specified time format (milliseconds, microseconds, nanoseconds etc.)

Debug Drawing

Debug Gizmos

The drawing methods supplied by the LÖVE framework requires you to call them from inside the love.draw() function, which was inconvenient as we often wanted to draw from inside update(). To get around this I created few functions based on the LÖVE drawing functions, that create the shape and store the needed arguments in a table, as well as the amount of time it should be drawn for. The shapes stored in the table will then get drawn in the draw() function, but the shapes can be added from anywhere.

Learning Outcomes

This is one of the largest projects developed outside of school that I have finished. Prior to this project I had only used Lua for modding other games, and using it with the LÖVE framework helped me learn a lot more about its intricacies. I also learned a lot from working with Tomas who is a really talented programmer and helped me become more organized and consistent in my code structure. Finally, I learned about working in ECS, what a spatial hash grid is and how to implement one, and lots of other smaller things.

Despite the project not becoming the game that we originally planned, I am very happy with it!

Webpage made by me using SvelteKit, Tailwind CSS and daisyUI. Updated February 1st 2023.