Design-Focused Entity Component System for a Survival City Builder
Start Date: July 2022
Description: A Survival City Builder game for Windows, where the player manages their city’s citizens to collect resources and develop the city as the environment becomes increasingly hot and inhospitable. The game and all its major systems are built within a new Entity Component System scheme whose primary goal is to simplify game code architecture.
Title screen for “SandPunk”
Context
In the year of custom engine development prior to the start of this project, I made projects which explored various programming paradigms for game code. In this project, I built an Entity Component System (ECS) for my custom game engine alongside a game which could benefit from an ECS paradigm. For the uninitiated, Entity Component Systems are made up of three parts:
Entities - Contains a set of components. Generally, it’s something in the game world.
Components - Typed structures to hold game data. Does not have functionality. For example, a Transform Component holds an entity’s position and rotation.
Component-Systems - Iterates over sets of components. Game functionality are implemented through systems. For example, a Renderer System might render all entities with a Model Component.
There is no standard way to make an ECS, but there are commonly cited benefits for implementing one:
Data Oriented Design - Carefully implemented component allocation caters to data locality.
Flexible Behavior Design - New behaviors can be designed by adding to or removing components from entities.
Clean Code - Component Systems are coupled to as few other components and systems as is necessary. This quality makes ECS games potentially easier to refactor and maintain.
This project implemented an ECS that focuses on #2 and #3. In other words, this was an ECS focused on the design benefits of ECS rather than performance gains from multithreading or cache coherency, for example.
Major Takeaways
“Components have no functionality, Systems have no state.”
Pure ECS systems follow this rule strictly, but when there was a sufficient pro-design reason to break these rules, I did so. Here were some common patterns between times I broke the ECS rules:
Transient State - state which was relevant only to a particular frame was convenient to throw between systems. For example, my DrawSystem queues up draw calls in the RenderSystem, which draws them later in the frame.
Hierarchical State - some features, like UI, are stateful and hierarchical. There was no benefit to forcing those solutions to be ECS friendly if it made the code harder to understand or maintain.
Unique State - Some kind of data was unique. Information about the dimensions of the game’s grid or the player are examples of these. A decision was made in a case by case basis for these, agnostic of ECS, regarding code quality and effectiveness.
Noteworthy Problem Solves
Data Driven UI System
While a survival city builder game has many features which made it well suited for an ECS, the genre is one with a number of unique UI challenges. My custom engine had no preexisting UI framework (prior projects had small uniquely fit UI solutions), and one needed to be developed from the ground up. I designed my UI systems and classes in a way that didn’t require ECS so it could be used immediately on non-ECS projects. The end result had features valuable for this game:
Data Driven Layouts and Styles - the heirarchy, layout and appearance of UI Elements are driven entirely through XML.
User-Defined Custom Widgets - game-specific widgets can be designed by extending the UIWidget class. This survival game used a special weather forecast widget, whose appearance depended on weather data, for one example.
Reusable UI Scripts - The UI system is highly event driven. With little extra effort, I could program a generic button which modified other canvas element properties.
Survival City Builder UI in this project. A timeline and forecast bar, resource bar, various modes and a build menu were implemented on the new UI system.
Screen-Space Selection of Arbitrary Silhouettes
As a first pass for determining what entity was under the player’s cursor, I performed a raycast against the game’s polar grid and determined what entity, if any, was contained at that position. This created noticeable precision issues when entities occluded each other or had unusual silhouettes. In order to fix this, I wrote a ComponentSystem which encoded an entity’s ID in a render pass to a screen-sized texture. When the player selects an entity, the entity’s ID extracted from the texture created earlier. Below is a visualization of this feature in action.
Visualization of the screen selection shader. Each of the entities on the left render their ID’s to the texture on the right.
Component Pool Custom Memory Allocators
During development of a game of this scope, substantial performance challenges were rarely problematic enough to warrant optimization, but optimizability was a goal of the project. As evidence that this ECS solution could be optimized or made more Data Oriented Design friendly, I created a custom pool allocator that components can opt into with a macro. This allocator stores all components by value in contiguous memory alongside other components of the same type.
In the future, when required, this solution could be extended to store copies of component data next to other types of components that they are frequently next to (called “ECS with Archetypes”). I would only consider this if the game developed substantial performance challenges.
Macro for components replaces the default new and delete to use custom allocator.