Spite Eclipse
About the project
In this project we developed a Diablo 3 like RPG in a top down view. The project spanned 3 months at part time and we were sixteen people including all disciplines. As this was our first project using our own home made game-engine this page will document some of my vital contributions to the functions and features of it. The engine is based on an entity/component system (ECS) which makes accessing and iterating over objects in the game simple and efficient.
Collision System
At the start of our engine development I recieved responsibility for the collision system that I have been developing since. At first I wanted a very basic collision detection with only axis aligned sphere- and box colliders since the math and logic of these are both very simple to implement and cost efficient in terms of performance. After this I decided to implement both a grid- and octree data structure to optimize colliders even further.
Iteration
In the code snippet below all the entities with dynamic colliders are accessed and looped over in a while-loop. These are then sent into both the grid- and octree data structures which returns a set of entities to check collisions against. To avoid two entities colliding with eachother sending a collision callback each they are compared so that only the entity with a larger entity ID integer checks and sends a collision callback.
Click image to view code
Callback
In the next step below the two entities that were found to have a possible collision by the grid- and octree data structures are checked for intersection. Then if intersecting they are checked for a collision callback that if exists is called with the two colliding entities as function paramaters.
Click image to view code
Intersection
To check if two colliders are intersecting in the previous step they are sent into the function below where they are checked for which type of collider their variant type contains. After the corresponding collider-types have been accessed a simple intersection function is called and it’s result returned.
Click image to view code
Grid Data Structure
The grid data structure below is used to optimize which colliders are checked for intersection against each other. This way only the ones who are in the same grid-box are checked. This means of course that the grid has to be rebuilt every logic frame since the colliders are dynamic and can move from one grid-box to another. This however still isn’t as performance taxing as the alternative since the actuall rid is a very simple and efficient 2D-Array with entity ID integers inside of it.
Rebuild
In the code snippet below the grid data structure is created depending on the grid size which level designers define depending on where collision may ocur in the game world. Then all dynamic colliders are iterated over and sent to an insertion function to be inserted into the grid.
Click image to view code
Insertion
The insertion function mentioned in the previous step retrieves the corners of the collider bounds to be inserted. These corners are then used to determine the corresponding indicies in the 2D-Array to add the entity ID to.
Click image to view code
Query
After rebuilding and inserting all dynamic colliders into the grid structure we can query it for all the colliders to check against given a collider as parameter. This optimizes the whole collision loop by checking for a smaller amount of intersections each frame.
Click image to view code
Octree Data Structure
The octree data structure below is used similarly to the grid to optimize which colliders are checked for intersection against each other. The big difference between them is the structure. Instead of a uniform grid the octree is a recurring tree which dynamically changes and optimizes itself based on where the colliders are inserted. It is built up of nodes that have 8 child nodes. The child nodes have a volume that each covers one octant of the parent node’s volume. This makes the queries for entities to check collisions against logarithmic and thus very efficient when handling large amount of colliders. Since the tree is relatively complicated and takes a noticeable amount of performance to rebuild it is only for static colliders which do not move to have the tree not rebuild.
Insertion
In the code snippet below the octree structure is filled with all static colliders recursively. The function checks if the current node is full and needs new children nodes. If true, children nodes are created and objects moved from parent to corresponding children. It also checks if there are children nodes and then inserts the object in the corresponding child node.
Click image to view code
Query
In the core collision update loop all the dynamic colliders are sent into the function in the code snippet below. The function then searches through the octree recursively for all colliders in the same node as the one sent as a parameter. These are then returned to the collision system so that proper intersection queries can be made.
Click image to view code