Action dispatching is an important part in Flux and Redux, to ensure any "data store" could change its data without caring about anything else other than user's action. Unit testing paradise! Just dispatch actions and see what the data have become.
I am bringing this workflow to Unity, but erasing the "string action label" pain point of actions in JavaScript by using C#'s enum
instead. Redux's designer said that, he advised using string
as an action's key rather than JS Symbol
because it is easily serializable and allows time travel. In Unity I care more about C# tooling provided by enum
.
- Every possible action by user must be represented by action. So in effect, an action cannot cause another action by itself. (So != public methods, which is kinda the "verb" of programming world.)
- Data can only mutate in response to an action. Then that mutation will cause a presentation change.
The "classic" way of handling action in Unity is something like EventTrigger
will call a public
method connected in the inspector, as the starting point. Then that object may further connected to other object via exposed variable, it ask the required objects to do something in chain. This chain is troubling when it goes on for about 2-3 layers, you started wondering who called this or change the data, and that "who" is not your user.
It is equivalent to "cascade update problem" that Facebook mentioned in their Flux documentation.
By only do something in itself based on solely action, you rarely need to chain (public) method calls to tell others to be in sync because others are also handling the action in themselves. Receiving a broadcasted action thanks to callback magic may seems cheating at first, but Unity's connected fields are not that strong either. I think it even became missing and cause bigger problem than callbacks.
- Action : This is one
enum
number inside anenum
type.enum
with the same underlyingint
value but from differentenum
type is considered different. - Category : The
enum
type serves as action's category. You can check if an action is in a category or not. - Payload : One action can be attached with
object
payload. It is so that you can vary an action's detail instead of defining many more granular actions. - Payload Key : You can attach multiple payloads to each action. Because C# do not have dynamic dot notation like JavaScript and I don't want to mess with
dynamic
, you instead use payload key to get the correct payload from an action. Payload Key is also anenum
. - Flag : Each action is strictly in one Category, however it could be added multiple Flags. An action from a different Category can be assigned the same Flag. For example you have the action
BackButtonPressed
in multiple Category describing pressing the top corner back button of each scene. You could assignBack
flag to all of them, then have an action receiver do something whenever anyBack
was dispatched. (Like unloading things, etc.). Flags are instead based onstring
and not an anotherenum
. You can defineconst string
for them.
Your action declaration may looks like this, along with some simplified usages :
public class MainMenu
{
public enum Action
{
QuitGame,
ToModeSelect,
ToCredits,
TouchedEmptyArea,
JoystickMoved,
[F(Navigation)] LeftButton, //F attribute is short for flags.
[F(Navigation)] RightButton,
}
public enum PayloadKey
{
TouchCoordinate,
DirectionVector,
Weight,
}
public const string Navigation = nameof(Navigation);
public void ToModeSelectButtonOnClick()
{
//Without payload
Dispatcher.Dispatch(MainMenu.Action.ToModeSelect);
}
public void EmptyAreaOnPointerDown()
{
//With payload (as a tuple of the key and `object`)
Dispatcher.Dispatch(MainMenu.Action.TouchedEmptyArea,
(MainMenu.PayloadKey.TouchCoordinate, new Vector2(100,150))
);
}
public void JoystickOnMove()
{
//Recommended style to enhance readability with multiple payload is to explicitly type `payload:` parameter name.
Dispatcher.Dispatch(MainMenu.Action.JobstickMoved,
payload:
(MainMenu.PayloadKey.DirectionVector, new Vector2(1,0)),
(MainMenu.PayloadKey.Weight, 13)
);
}
}
How to directly check for that exact action with if
: Use .Is
.
private void OnAction(DispatchAction action)
{
if (action.Is(MusicSelect.Action.SelectSong))
{
...
}
else if(action.Is(MusicSelect.Action.SelectDifficulty))
{
...
}
}
How to handle 2 categories at once with if
and switch case
: Use .Category<T>
then use the generic-typed out
variable with switch
.
private void OnAction(DispatchAction action)
{
if (action.Category<MusicSelect.Action>(out var actMs)) switch (actMs)
{
case MusicSelect.Action.SelectSong:
...
}
else if (action.Category<MusicStart.Action>(out var act)) switch (act)
{
case MusicStart.Action.Begin:
...
case MusicStart.Action.BeginEditor:
...
case MusicStart.Action.ToggleRivalView:
...
case MusicStart.Action.ChangeChartDifficulty:
...
}
...
}
For how to do it in C# Jobs, please see the Tests
folder.
- Strings are brittle and annoying.
- Enums can auto complete.
- Enums are easier to define than
const string
. You don't even have to name the variable. - Mass-rename by your IDE tooling.
- You can use your IDE to easily find all places that dispatch a certain event by searching enum references.
- Enum can be nested in the class so that dot notation looks nice. It allows you to for example, always name your enum as
Action
, so you don't have to worry about naming conflict. When used, it will looks likeMainMenu.Action.Back
,ModeSelect.Action.Back
which is quite readable. - There is an optimization at compiler level that make it fast with
switch case
. It does not require equality comparing case by case but a jump table instead. If theseenum
were just normalint
it would generate comparison assembly per case, same goes forstring
. This may matter if your action handling code path is hot. (And maybe being Burst compiled for even better assembly.)
- Different enum may have an equal underlying
int
value. This makes naive enum-as-label implementation wrong as action in one category replacable by action in an another category. Enum Dispatcher can detect that the sameenum
value are coming from a differentenum
type by also including/caching type information. - Check for action by
==
is fine, but you can't doswitch case
if the receiving side doesn't contains enum typing information. If the receiving side contains the action type information, then it is not capable of handling action across multiple categories. Enum Dispatcher contains an action wrapper namedDispatchAction
instead of theenum
. It contains various methods to help to determine the exact action while keeping the receiving side just know aboutDispatchAction
. - Action category via
enum
requires bookkeeping the type. Enum Dispatcher cacheenum
types on-the-go inspired by Unity's Entities package'sTypeManager
. - Can you all that in C# Jobs? So you could check on action type and act all inside a single job instead of checking on the main thread and having to relay information to the job what to do. Yes, Enum Dispatcher can! With support from
JobDispatchAction
it brings together all its category and flags data to the job. On converting fromDispatchAction
toJobDispatchAction
, it because allstruct
andNativeArray
based. This bridges the whole thing to ECS as well. Unfortunately action payload cannot go to the job as it is based onobject
type.
- C# 7.0, it uses tuples extensively too.
- Entities UPM packages and friends.
I could design it as a static
enum dispatcher where anyone can receive the action. However I decided to bring ECS into play :
- Avoid using
static
, dispatched actions are nowWorld
-bound. (Though tecnicallyWorld
arestatic
beings) - Allows me to design a
System
which automatically subscribe/unsubscribe to Enum Dispatcher's action because it knows to look for "Dispatching System" in the same world. It works together withJobDispatchAction
support, so you are not limited to just C# Jobs but use them withJobComponentSystem
-based action handling. - (Real reason : Actually I pulled this out from my other hybrid ECS library for dealing with uGUI, so I need it to be compatible with ECS and jobs.)
And so Enum Dispatcher's asmdef
requires Entity package present. Install them from Package Manager.
Also it is a good bridge from normal world to ECS. For example, Normally you connect the uGUI Button
's On Click
to some public methods. It is not possible to connect with ECS's system since they are not in the scene. With Enum Dispatcher, all uGUI Button
in the game no longer ever have to contains any logic other than dispatching an enum action. ECS system is now able to respond to button press, also your MonoBehaviour
things can subscribe as an action receiver as well. Also it is awesome for unit testing now that you can mock user's behaviour by just dispatching actions over and over.
- An ECS system
DispatchingSystem
holds C#event
. You can subscribe or dispatch by getting this system's reference from yourWorld
and call its public method. You can declare the callback method anywhere, inMonoBehaviour
, etc. - Call
dispatchingSystem.Dispatch
on the system instance will invoke all subscribers with that action immediately. You call it with yourenum
, but action handlers will receiveDispatchAction
. Alternatively, an easy utilitystatic
methodDispatcher.Dispatch
will getDispatchingSystem
in yourWorld.Active
first then do the same thing. Notice that to this point nothing is related to ECS yet. It didn't create any event entity. Just.Invoke()
. At this point it is already usable as a general purpose enum-based event system. - On each dispatch call, there is one more system which bookkeep enum types of the action. Each
enum
will get its own index. Bothenum
type index and theenum
integer value will be used together to represent one unique action.DispatchAction
is an object containing those information. This bookkeeping system does so by using native containers, so this entire "type dictionary" they could be referenced safely from a job, allowing you to check action type on thread. - Any ECS system inherited from
ActionHandlerSystem
is automatically subscribed/unsubscribe toDispatchingSystem
of theWorld
it is currently in.ActionHandlerSystem
receives actions immediately like manual subscribers, but you cannot respond to them just yet. They will all be queued, then on itsOnUpdate
you can respond to them with an opportunity to schedule a job since it is a subclass ofJobComponentSystem
. You shouldoverride
thevirtual
methodOnAction
where it will give you actions one by one in order.JobHandle
is provided in that respond context so all your jobs are hooked up to ECS job pipeline. This is why you can't respond to action immediately. You can check the action first then schedule appropriate jobs, or bring action into the job and check them inside so you could offload main thread. (If the check and respond is complicated) ActionHandlerSystem
is preconfigured to update in an update group calledActionHandlerSystem.ActionHandlerGroup
. If you want to make sure your system updates after all actions are handled you can use[UpdateAfter(typeof(ActionHandlerSystem.ActionHandlerGroup))]
, so you can use the result from scheduled jobs that was a response to an action this frame.
Please see usage examples from the Tests
folder, where you will witness an epic fight with monsters.