Building a Gamestate with Zenject in Unity - Sun, Jul 23, 2023

Single Source of Truth and Inversion of Control

Unity provides countless mechanisms by which one can shoot themself in the foot. It’s a common and popular entrypoint to picking up C# for newcomers to the language. However, there’s a common gotcha individuals will quickly begin to encounter as they familiarize themselves with the basics of the language via Unity’s MonoBehavior classes.

Unfortunately this comes with a common and harsh pitfall: by default, Unity really encourages individuals to produce countless individual batches of code that they tightly couple to their game objects, which results in sooner than usual confronting the big bad wolf that is the Byzantine Generals Problem.

For those not in the know, the crux of the issue is the fact that by breaking up their game state into numerous individual “shards”, where every individual entity in their game has its own pool of statefulness, it produces a very easy pile of spaghetti one can find themself tripping and falling into.

Eventually you find yourself naturally wanting to sort the mess out and unify all this logic. Rather than countless individual objects having little pieces of the game state pie, you wonder to yourself, “Is there a better way?”. Is there perhaps some easier, cleaner way I can bring all the game state into one single place, one single source of truth, and everything else orbits around it?

Well, there absolutely is, and in fact there are countless ways to do it. In this post I will go over the solution I came up with that takes a bit of inspiration from a few other common forms of C# projects.

Centralizing the Engine through Dependency Injection

To begin we will need to do a bit of legwork. I am a massive proponent of Dependency Injection and I strongly encourage folks to get familiar with it. Unity has a fairly popular Dependency Injection library called Zenject that we will be using today. However keep in mind it is not the only option, nor is it even potentially the best! I’m merely using it in this example because it is popular and the first result on the unity asset store. Also, it’s free.

For the purpose of this project we will start things off by creating a new 2d core project, and installing the latest Zenject unity package which you can download from here: https://github.com/modesttree/Zenject/releases/latest

We will need to do some initial bootstrapping of the project to get all the basics working. Let’s start by bootstrapping up the DI Engine.

  1. Add a folder named Scripts inside of our Assets folder

  2. Inside of the scripts folder, create a new Mono Installer and name it StartupInstaller (Right Click > Create > Zenject > Mono Installer)

  3. Add a Zenject Scene Context to the root of the project and name is Startup (Right Click Scene > Zenject > Scene Context)

  4. Drag and Drop the StartupInstaller script onto our Startup to add it as a component.

  5. Drag the Startup Installer script component up to the Mono Installers list in the Scene Context component. (You may need to hit the + button, and you can just drag and drop the whole Startup object onto the field if you like, Unity will figure it out)

Your Startup object should now look like this: Startup Example one

And your StartupInstaller script like so:

1
2
3
4
5
6
public class StartupInstaller : MonoInstaller
{
    public override void InstallBindings()
    {
    }
}

This is now the bare bones basic of registering for Dependency Injection, lets now move on to:

Utility Files

Let’s put together a few pieces to demonstrate the Game State system in action. To Begin with, we need a couple cs files that provide handy utility for the purpose of this project. You can download and save them to the following path (you’ll need to make the folder for it): Assets/Scripts/Utilities

PropertyWatcherBase.cs

This file has a very handy base class that implements the INotifyPropertyChanged event, which is going to be the key lynchpin used to apply Single Source of Truth logistics to our project. Our Game State class we make later will make heavy use of this class.

I currently have it hosted up as a gist on my github, you can feel free to check it out. It has a bunch of utility classes and methods that culminate in the functionality needed for this project. You can use them as you wish!

Source code: here

Raw Download: here

You can just Right Click -> Save as… the second link, and it should work.

There are multiple important methods on this base class, let’s dig into how they work. To begin with, the class implements the event delegate PropertyChanged to exist, which is our Pub/Sub model that our services can publish and subscribe to. This will be our main mechanism for how we inform services that changes have occurred to the Game state.

The first method, Mutate, is our “writer” that will automatically invoke the aforementioned event when we write changes. I’ll show how we hook that up to Full Properties on an implementing class in a moment below.

The second method, BindChild is a handy tool to let us nest PropertyWatcherBase models, so we can have children of the parent GameState (and further nested down, grandchildren and etc), this method will listen to a child’s event and append the parent’s name to it in the form of Parent.Child, which gives it a unique signature.

The third exposed method, BindTo, is a handy method subscribers utilize to subscribe to a given event. We will show how to use this farther down.

The fourth method, OpenTransaction is a bit more advanced and won’t be covered by this tutorial, but you can see examples of how to utilize it on my gist comment here:

Building our GameState

Alright, time to show it in action! Create a new script file, let’s call it GameState.cs to keep it simple, and put it in the folder Assets/Scripts/PropertyWatchers. Lets also make a second file called ChildGameState.cs and put it in that folder too.

Both will inherit from PropertyWatcherBase<TSelf>, where TSelf is the class itself (this needs to be done for some internal code specific reasons, we wont dig into the details here).

We also want to put a full property on the child, let’s make it an int and just call it Counter with the backing field _counter.

The parent GameState class will get its own full property, let’s make that one a ChildGameState called Child, backed by the field _child

Next, we utilize the Mutate method on Counter’s Set method to hook in the event, and BindChild on Child’s setter (in GameState) to bubble the event up. We can keep Child’s setter private, because though we want the properties of our classes to be mutable, we don’t want the classes themselves to be.

Since we are utilizing a Dependency Injection Engine, we will want to do Constructor Injection to hand GameState its copy of the ChildGameState

You’re two files should now roughly look like this:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
public class GameState : PropertyWatcherBase<GameState>
{
    public GameState(ChildGameState child)
    {
        Child = child;
    }

    private ChildGameState _child;
    public ChildGameState Child { get => _child; private set => BindChild(ref _child, value); }
}

public class ChildGameState : PropertyWatcherBase<ChildGameState>
{
    private int _counter;
    public int Counter { get => _counter; set => Mutate(ref _counter, value); }
}

Now let’s finish up by registering these classes in our StartupInstaller as singletons.

StartupInstaller.cs:

1
2
Container.Bind<GameState>().AsSingle();
Container.Bind<ChildGameState>().AsSingle();

Basic Project Setup

With our DI Engine up and running, Game State built, let’s do the basic legwork to make a scene that we can actually interact with now!

To start, lets add a Canvas + EventSystem to our scene (Scene Right Click > UI > Canvas)

On the Canvas we will want to modify Render Mode to Screen Space - Camera and drag and drop our camera onto the Render Camera field. Then we will want to set UI Scale Mode to Scale With Screen Size and set Screen Match Mode to Match Width or Height, then finally set Match to 0.5 (slider dead in the middle)

This will make our UI work smoothly regardless of what resolution the game runs at.

Next up lets add a Vertical Layout Group component to the canvas, so we can split it in half. Check Control Child Size to both True and True for Width and Height respectively. Do the same for Child Force Expand

Add a Text (TextMeshPro)(UI > Text - TextMeshPro) (to the Canvas as a child (this will cause a TMP Importer prompt to appear, click Import TMP Essentials, this is very important), as well as a Button (UI > Button - TextMeshPro)

If all went well, your button and text box should equally fill the Canvas up 50/50, with the text at the top and button at the bottom. If the order is reversed its not a big deal, but if you wanna fix that just change the order the two are as children of the Canvas object.

To finish off, modify the Text (TMP)’s Main Settings fields to make the text easier to read. You prolly will want to modify Font Size, and set Alignment to Center + Middle to make it dead center.

You can also do the same for the Text (TMP) child of the Button to make it’s text bigger and easier to read, if you like.

Theoretically you should have something that sort of looks like this when you run your game: Game visual example 1

Clicking the button doesnt do anything yet, but let’s start wiring up our services!

Event pushing a MonoBehavior

Now, theoretically we could directly try and access the GameState from any given MonoBehavior, but that is the way to spaghetti land. Instead, we want to utilize Inversion of Control to keep all of our MonoBehavior objects as incredibly simple, lightweight, and “glass box” as possible. For all intents and purposes, each of our MonoBehavior scripts should have no concept of any other MonoBehaviors,nor the Game state or engine or anything else. They should purely have two things they do:

  1. Expose methods to manipulate their own fields, like transform, sprites, text, etc.

  2. Expose agnostic “This thing happened” events that can be subscribed to by the GameEngine and any of its descendants.

To distinguish these “simple” MonoBehavior’s, I like to refer to them as Entities, as they represent actual “things” in the scene. Objects, Sprites, Text, Buttons, etc etc.

Let’s create two of these as an example, with the TextMeshPro we made on the top being a prime example of #1, and the Button we made as an example of #2.

Start off by adding a new script component to both the Text (TMP) game object (the big one, not the one inside the button), and name it TextEntity. We will also add a new on to the Button and call it ButtonEntity

Make sure both of these end up in Assets/Scripts/Entities, to stay organized.

TextEntity needs a serialized TMP_Text field, let’s call it Text, as well as a public method called SetText that takes in a string and assigns the value of it to Text.text

ButtonEntity needs a public event called Clicked (of untyped EventHandler), and a method called OnClick that null coalesce invokes it (as per typical event style). If you’d like to read up on C# event delegates, check this link out!

Your two MonoBehaviors should now look like this:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
public class TextEntity : MonoBehaviour
{
    [SerializeField]
    TMP_Text Text;
    public void SetText(string text) => Text.text = text;
}

public class ButtonEntity : MonoBehaviour
{
    public event EventHandler Clicked;
    public void OnClicked() => Clicked?.Invoke(this, null);
}

Finally we just need to wire these two up in the editor. For the Text (TMP) we just repeat our same trick as before and drag and drop itself onto the Text field that should now show up for the Text Entity component on it.

For the Button, we want to hit the + button on its On Click () list in the Button component, then drag and drop the Button Entity component just below onto its Object field, and select ButtonEntity > OnClicked() for the Function dropdown (which currently says No Function)

Your Text (TMP) and Button should now look like these, respectively:

TMP and Button Component Examples

Finally, let’s register both of these on our Dependency Injection Engine, to make sure we can inject references to them for the next step. Our StartupInstaller will want them as Field References, ideally.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
public class StartupInstaller : MonoInstaller
{
    [SerializeField]
    private TextEntity text;

    [SerializeField]
    private ButtonEntity button;

    public override void InstallBindings()
    {
        // Game State
        Container.Bind<GameState>().AsSingle();
        Container.Bind<ChildGameState>().AsSingle();

        // Entities
        Container.Bind<TextEntity>().FromInstance(text);
        Container.Bind<ButtonEntity>().FromInstance(button);
    }
}

IMPORTANT! Don’t forget to drag and drop the TMP Pro and the Button onto the two new fields we added to Startup’s StartupInstaller component!

We are now prepared to wire our Scene up to our GameEngine!

Binding the UI to the Engine and GameState

To start, let’s make ourselves a Service to delegate out the task of handling these UI objects. Services will be what we refer to individual “modules” of our stack, and they should be POCOs (Plain ole classes), not MonoBehaviors. They can be injected into each other (but try and avoid circular dependency loops!), and make up the majority of our dependency tree.

Largely speaking, nearly all of your code should be inside of various Services, and they should act as your way of sorting out your code to different areas that make sense.

Let’s make a folder for these, at Assets/Scripts/Services, and make our first service in there. Lets call it EntityService, and it will have the job of managing our Entities.

We will want to inject the GameState, TextEntity, and ButtonEntity into it via Constructor Injection (which is as simple as just using it as a param in the public constructor)

Next, we can have this service wired up to handle delegation of mutating the Text Entity, and listening for the Button entity’s click event. At the core of it, the EntityService serves the role of delegating information back and forth between Game State <-> Entities. This isn’t completely mandatory as you could just inject the Game State directly into the Entities, but in actual practice there likely would be much more complicated logic than what we have here, and the service should act as the space to put all that logic to keep it compartmentalized off.

We will implement Zenjects IInitializable interface to declare this as a “root” service that has initial startup logic to begin running.

You should end up with something a bit like this:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public class EntityService : IInitializable
{
    private GameState GameState { get; }
    private TextEntity Text { get; }
    private ButtonEntity Button { get; }

    [Inject]
    public EntityService(GameState gameState, TextEntity text, ButtonEntity button)
    {
        GameState = gameState ?? throw new ArgumentNullException(nameof(gameState));
        Text = text ?? throw new ArgumentNullException(nameof(text));
        Button = button ?? throw new ArgumentNullException(nameof(button));
    }

    public void Initialize()
    {
        GameState.BindTo(g => g.Child.Counter, OnChildCounter);
        Button.Clicked += OnButtonClicked;
    }

    private void OnChildCounter(in int count)
    {
        Text.SetText($"Count: {count}");
    }

    private void OnButtonClicked(object _, EventArgs e)
    {
        GameState.Child.Counter++;
    }
}

And we once again register it as a single on our StartupInstaller:

1
2
// Services
Container.BindInterfacesTo<EntityService>().AsSingle();

If you run your program and click the button, you should see the count increase! Congrats, you now have a working single source of truth GameState!

TMP and Button Component Working

Okay but why all the theatrics if we could wire it directly?

Simple, because now if any of your services modifies GameState.Child.Counter, it will also update the counter text without any extra effort.

Let’s do an example with an async void to make a sort of background timer that will also automatically increment the count every 5 seconds, on top of any button clicks you do.

To start, let’s make a new service for this called BackgroundTimerService, and this time we only need to inject the GameState, and this one will also be an IInitializable

We will make a background async method via async void, and simply just have an infinite loop that increments the counter every 5 seconds via Task.Delay(...)

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
public class BackgroundTimerService : IInitializable
{
    private GameState GameState { get; }
    public BackgroundTimerService(GameState gameState)
    {
        GameState = gameState ?? throw new ArgumentNullException(nameof(gameState));
    }

    public void Initialize()
    {
        Start(5000);
    }

    private async void Start(int millisecondsDelay)
    {
        while (true)
        {
            await Task.Delay(millisecondsDelay);
            GameState.Child.Counter++;
        }
    }
}

And register it on our DI Engine (you know the drill by now!)

1
2
3
// Services
...
Container.BindInterfacesTo<BackgroundTimerService>().AsSingle();

And if you run the game now you should see that not only does clicking the button increment our counter, but every 5 seconds it gets incremented by the background service as well!

Now imagine in a way more complex game, where you may have dozens of services all capable of mutating values on the GameState at once, hopefully this sheds some light on how by unifying all your game state down into a single source of truth, it can get a lot easier to maintain all that data flying all over the place at once!

Reference Project

If you are having any issues, please check out the link below for a repo I setup with a working example of the project detailed above. It should theoretically be working fine, and demonstrate everything discussed above. You can compare your project against it to perhaps sus out any issues.

https://repo.technically.fun/sblake/INotifyPropertyChangedExample