-
Notifications
You must be signed in to change notification settings - Fork 425
Setting up key binding containers
Once you have completed setting up your first project, we are ready to start building our game! In addition to having the ability to handle input events of various types, you can also set up a KeyBindingContainer
which allows you to specify default bindings for actions, as well as choose which key handling mode should be used in the container.
First, we should name every input we want to be a part of this KeyBindingContainer
. This lets us use the type field when inheriting the KeyBindingContainer<T>
class so that it knows what action data it should map input actions to.
For this example, we will map our W, A, and D keys to our Jump, Left, and Right actions respectively. We will create our enum type like so:
public enum InputAction
{
Left,
Right,
Jump
}
It is recommended you create this new type inside the same file where you're creating the new KeyBindingContainer
itself in step 2. That way, it will be easier to keep track of where it is if someone were to try to find them, as well as guarantee that both exist in the same namespace.
We will then take the enum type we just created and use it to create our KeyBindingContainer.
public partial class AwesomeKeyBindingContainer : KeyBindingContainer<InputAction>
{
}
We can now map key presses to the action we wish to perform! To do so, we should first create a default set of keybinds. In our newly created KeyBindingContainer, override the existing DefaultKeyBindings
property with your own:
public partial class AwesomeKeyBindingContainer : KeyBindingContainer<InputAction>
{
public override IEnumerable<KeyBinding> DefaultKeyBindings => new[]
{
new KeyBinding(new[] { InputKey.A }, InputAction.Left),
new KeyBinding(new[] { InputKey.W }, InputAction.Jump),
new KeyBinding(new[] { InputKey.D }, InputAction.Right)
}
}
If you wish to have multiple alternative key combinations, you will need to specify multiple KeyBinding
instances associated with a common InputAction
.
There are two modes that can be specified on the creation of a new keybinding container via the parameters. One is the SimultaneousBindingMode
, and the other is the KeyCombinationMatchingMode
. The SimultaneousBindingMode
specifies whether or not an input should be valid based on whether or not other buttons are pressed simultaneously, whereas the KeyCombinationMatchingMode
specifies how strict the keybinding should be in regards to whether or not the exact buttons are pressed.
For a better visual representation of these modes and how they restrict multiple key press events, please see TestCaseKeyBindingsGrid
.
Now that we have created our KeyBindingContainer
, we can add behavior that triggers when the key-bound action has been fired. The IKeyBindingHandler<T>
interface provides a way for bound input events to be handled by any drawable, as long as it is a child of our KeyBindingContainer
.
For example, to create a player class that inherits RigidBodyContainer
and implements IKeybindingHandler
, we would do this:
public partial class Player : RigidBodyContainer<Drawable>, IKeyBindingHandler<InputAction>
{
}
The IKeyBindingHandler
interface provides access to the following:
-
OnPressed(KeyBindingPressEvent<T>)
is triggered when any action is pressed. The action type depends on the type specified in the type specifier when you inheritIKeyBindingHandler<T>
. Returntrue
if this action should block the action from traversing the scene graph up to its parent. -
OnReleased(KeyBindingReleaseEvent<T>)
is triggered when any action is released. This handler cannot be blocked, and will always be fired on the drawable that handled the previous press.
Our completed player RigidBodyContainer
with keybinding handling would then look something like this:
public partial class Player : RigidBodyContainer<Drawable>, IKeyBindingHandler<InputAction>
{
private float constantXForce;
[BackgroundDependencyLoader]
private void load(TextureStore textureStore)
{
Child = new Box
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Size = new Vector2(150, 150),
};
Position = new Vector2(500, 500);
Size = new Vector2(200, 200);
Rotation = 45;
Masking = true;
}
private const float PLAYER_VELOCITY = 500f;
public bool OnPressed(KeyBindingPressEvent<InputAction> e)
{
switch (e.Action)
{
case InputAction.Jump:
Velocity = new Vector2(constantXForce, Velocity.Y - 500);
break;
case InputAction.Right:
constantXForce += PLAYER_VELOCITY;
break;
case InputAction.Left:
constantXForce -= PLAYER_VELOCITY;
break;
}
return true;
}
public void OnReleased(KeyBindingReleaseEvent<InputAction> e)
{
switch (e.Action)
{
case InputAction.Left:
constantXForce += PLAYER_VELOCITY;
break;
case InputAction.Right:
constantXForce -= PLAYER_VELOCITY;
break;
}
}
protected override void Update()
{
base.Update();
Velocity = new Vector2(constantXForce, Velocity.Y);
}
}
For this example to work as intended, we will need to change the simultaneous binding mode in our keybinding container to allow multiple bindings to be pressed at the same time. We can specify a constructor for this:
public AwesomeKeyBindingContainer(KeyCombinationMatchingMode keyCombinationMatchingMode = KeyCombinationMatchingMode.Any, SimultaneousBindingMode simultaneousBindingMode = SimultaneousBindingMode.All)
: base(simultaneousBindingMode, keyCombinationMatchingMode)
{
}
- Create your first project
- Learning framework key bindings
- Adding resource stores
- Adding custom key bindings
- Adding custom fonts