The dependency injection system in the game engine allows for the registration, resolution, and management of dependencies within the GameObject
hierarchy. This system is designed to facilitate the decoupling of components and promote modularity by allowing dependencies to be injected into behaviors and other objects at runtime.
DependencyContainer
: Manages dependency registration and resolution using a Map
.
It be used independently of gameobject structure. Key methods:
register<T>(token, instance)
: Registers a class instance.resolve<T>(token)
: Resolves a registered instance by its constructor.Decorators:
@Inject(token, recursive?)
: Injects a dependency into a property, optionally searching up the hierarchy.@InjectGlobal(token)
: Injects a dependency from the global container.Every gameobject has a container that is used to resolve dependencies for its behaviors. When a behavior is added to a gameobject, the container will automatically resolve and inject any dependencies into the behavior. The system leverages the GameObject
hierarchy, allowing dependencies to be shared across parent and child objects.
IMPORTANT : If you want to inject a dependency into a behavior with the @Inject
or the @InjectGlobal
decorator, you will need to enable experimentalDecorators in your tsconfig.json file.
{
"compilerOptions": {
/* Meta decorators */
"experimentalDecorators": true,
"emitDecoratorMetadata": true
}
}
Registration:
DependencyContainer
associated with a GameObject
or the global container (GameEngineWindow
).Injection:
Behavior
or a child GameObject
is added to a GameObject
, the system scans its properties for @Inject
or @InjectGlobal
decorators.@Inject
:
GameObject
's container.recursive
is enabled, the system searches up the GameObject
hierarchy (parent containers) for the dependency. If enabled, it can also search the global container (GameEngineWindow
)@InjectGlobal
:
GameEngineWindow
).@Inject
used in a GameObject resolve the dependency from the parent GameObject container so you don't need to set recursive to true to access the parent GameObject dependencies.Resolution:
const container = new DependencyContainer();
const dependency = new TestDependency();
// Register the dependency
container.register(TestDependency, dependency);
// Resolve the dependency
const resolvedDependency = container.resolve(TestDependency);
console.log(resolvedDependency.value); // Output: "Hello, World!"
class LocalDependency {
public value: string = "Local Dependency";
}
class TestBehavior {
@Inject(LocalDependency)
public localDependency: LocalDependency;
}
const gameObject = new GameObject();
const localDependency = new LocalDependency();
// Register the dependency
gameObject.addBehavior(localDependency);
const behavior = new TestBehavior();
//Here the local dependency will be injected into the behavior (but it will also register TestBehavior as a dependency)
gameObject.addBehavior(behavior);
console.log(behavior.localDependency.value); // Output: "Local Dependency"
class GlobalDependency {
public value: string = "Global Dependency";
}
class TestBehaviorWithGlobalDependencies {
@InjectGlobal(GlobalDependency)
public globalDependency: GlobalDependency;
}
const window = new GameEngineWindow(new ManualTicker());
const gameObject = new GameObject();
window.root.addChild(gameObject);
const globalDependency = new GlobalDependency();
window.injectionContainer.register(GlobalDependency, globalDependency);
const behavior = new TestBehaviorWithGlobalDependencies();
gameObject.addBehavior(behavior);
console.log(behavior.globalDependency.value); // Output: "Global Dependency"