Skip to content

reececomo/pixijs-input-devices

Repository files navigation

🎮 PixiJS Input Devices  License Tests Downloads per month NPM version

⚡ Simple keyboard & gamepad management for PixiJS

🎮 Interface keyboards, gamepads, and more! 🚀 Flexible update and event-driven APIs
⚡ Optimized for INP performance 🪄 Built-in named binds
🔮 Highly configurable 🌐 Built-in international keyboard support
✅ Cross-platform & mobile-friendly [1] [2] [3] 🧭 Built-in UI navigation (optional)
🍃 Zero dependencies & tree-shakeable ✨ Supports PixiJS v8, v7, v6.3+

Sample Usage

Handle device inputs with ease.

import { InputDevice, GamepadDevice } from "pixijs-input-devices";

// Set named binds
InputDevice.keyboard.configureBinds({ jump: [ "Space" ] })
GamepadDevice.configureDefaultBinds({ jump: [ "A", "LeftStickUp" ] })

// Use binds
for ( const device of InputDevice.devices ) {
    if ( device.pressedBind("jump") ) doJump()
}

// Event-driven
InputDevice.onBind( "jump", ({ device }) =>
    if ( device.type === "gamepad" ) {
        e.device.playVibration({ duration: 100 })
    }
);

Getting Started with PixiJS Input Devices

Everything you need to quickly integrate powerful device management.

PixiJS Input Devices adds first-class support for input device management and input handling. It also provides an optional navigation manager that can enable input devices to traverse pointer-based UIs.

The core concepts are:

  1. Devices: Any human interface device
  2. Binds: Custom, named input actions that can be triggered by assigned keys or buttons
  3. UINavigation: A global controller that allows non-pointer devices to navigate UIs

Note

See UINavigation API for more information.

Installation

Quick start guide.

1. Install the latest pixijs-input-devices package:

# npm
npm install pixijs-input-devices -D

# yarn
yarn add pixijs-input-devices --dev

2. Register the update loop:

import { Ticker } from 'pixi.js';
import { InputDevice } from 'pixijs-input-devices';

Ticker.shared.add(ticker => InputDevice.update());

Tip

Input polling: In the context of a video game, you may want to put the input update at the start of your game event loop instead.

3. (Optional) enable the UINavigation API

import * as PIXI from 'pixi.js';
import { UINavigation, registerPixiJSNavigationMixin } from 'pixijs-input-devices';

const app = new PIXI.Application(/*…*/)

// enable the navigation API
UINavigation.configureWithRoot( app.stage )
registerPixiJSNavigationMixin( PIXI.Container )

✨ You are now ready to use inputs!

Features

InputDevice Manager

The InputDevice singleton controls all device discovery.

InputDevice.keyboard  // KeyboardDevice
InputDevice.gamepads  // Array<GamepadDevice>
InputDevice.custom    // Array<CustomDevice>

You can access all active/connected devices using .devices:

for ( const device of InputDevice.devices ) {  // …

InputDevice - properties

Property Type Description
InputDevice.isMobile boolean Whether the context is mobile (including tablets).
InputDevice.isTouchCapable boolean Whether the context has touchscreen capability.
InputDevice.lastInteractedDevice Device? The most recently interacted device (or first if multiple).
InputDevice.devices Device[] All active, connected devices.
InputDevice.keyboard KeyboardDevice The global keyboard.
InputDevice.gamepads GamepadDevice[] Connected gamepads.
InputDevice.custom CustomDevice[] Custom devices.

InputDevice - on() Events

Access global events directly through the manager:

InputDevice.on( "deviceadded", ({ device }) => {
    // a device was connected
    // do additional setup here, show a dialog, etc.
})

InputDevice.off( "deviceadded" ) // stop listening
Event Description Payload
"deviceadded" {device} A device has been added.
"deviceremoved" {device} A device has been removed.

KeyboardDevice

Unlike gamepads & custom devices, there is a single global keyboard device.

let keyboard = InputDevice.keyboard

if ( keyboard.key.ControlLeft ) {  // …

Note

Detection: On mobiles/tablets the keyboard will not appear in InputDevice.devices until a keyboard is detected. See keyboard.detected.

Keyboard Layout - detection

keyboard.layout  // "AZERTY" | "JCUKEN" | "QWERTY" | "QWERTZ"

keyboard.getKeyLabel( "KeyZ" )  // Я

Note

Layout support: Detects the "big four" (AZERTY, JCUKEN, QWERTY and QWERTZ). Almost every keyboard is one of these four (or a regional derivative – e.g. Hangeul, Kana). There is no built-in detection for specialist or esoteric layouts (e.g. Dvorak, Colemak, BÉPO).

The keyboard.getKeyLabel( key ) uses the KeyboardLayoutMap API when available, before falling back to default AZERTY, JCUKEN, QWERTY or QWERTZ key values.

The keyboard layout is automatically detected from (in order):

  1. Browser API (browser support)
  2. Keypresses
  3. Browser Language

You can also manually force the layout:

// force layout
InputDevice.keyboard.layout = "JCUKEN"

InputDevice.keyboard.getKeyLabel( "KeyW" )  // "Ц"
InputDevice.keyboard.layoutSource  // "manual"

KeyboardDevice Events

Event Description Payload
"layoutdetected" {layout,layoutSource,device} The keyboard layout ("QWERTY", "QWERTZ", "AZERTY", or "JCUKEN") has been detected, either from the native API or from keypresses.
"bind" {name,event,keyCode,keyLabel,device} A named bind key was pressed.
Key presses:
"KeyA" {event,keyCode,keyLabel,device} The "KeyA" was pressed.
"KeyB" {event,keyCode,keyLabel,device} The "KeyB" was pressed.
"KeyC" {event,keyCode,keyLabel,device} The "KeyC" was pressed.

GamepadDevice

Gamepads are automatically detected via the browser API when first interacted with (read more).

Gamepad accessors are modelled around the "Standard Controller Layout":

let gamepad = InputDevice.gamepads[0]

if ( gamepad.button.Start ) {  // …
if ( gamepad.leftTrigger > 0.25 ) {  // …
if ( gamepad.leftJoystick.x > 0.5 ) {  // …

Tip

Special requirements? You can always access gamepad.source and reference the underlying API directly as needed.

Vibration & Haptics

Use the playVibration() method to play a haptic vibration, in supported browsers.

gamepad.playVibration({
    duration: 150,
    weakMagnitude: 0.75,
    strongMagnitude: 0.25,
    // …
})

Gamepad Button Codes

The gamepad buttons reference Standard Controller Layout:

Button Index GamepadCode Description Xbox Playstation Nintendo[?]
0 "A" Face Button 0 A Cross A
1 "B" Face Button 1 B Circle X*
2 "X" Face Button 2 X Square B*
3 "Y" Face Button 3 Y Triangle Y
4 "LeftShoulder" Left Shoulder LB L1 L
5 "RightShoulder" Right Shoulder RB R1 R
6 "LeftTrigger" Left Trigger LT L2 ZL
7 "RightTrigger" Right Trigger RT R2 ZR
8 "Back" Back Back Options Minus
9 "Start" Start Start Select Plus
10 "LeftStickClick" Left Stick Click LSB L3 L3
11 "RightStickClick" Right Stick Click RSB R3 R3
12 "DPadUp" D-Pad Up ⬆️ ⬆️ ⬆️
13 "DPadDown" D-Pad Down ⬇️ ⬇️ ⬇️
14 "DPadLeft" D-Pad Left ⬅️ ⬅️ ⬅️
15 "DPadRight" D-Pad Right ➡️ ➡️ ➡️

Gamepad Axis Codes

Bindable helpers are available for the joysticks too:

Axis # GamepadCode Standard Layout
0 "LeftStickLeft"
"LeftStickRight"
Left Stick (Left/Right) ⬅️➡️
1 "LeftStickUp"
"LeftStickDown"
Left Stick (Up/Down) ⬆️⬇️
2 "RightStickLeft"
"RightStickRight"
Right Stick (Left/Right) ⬅️➡️
3 "RightStickUp"
"RightStickDown"
Right Stick (Up/Down) ⬆️⬇️

Tip

Set the joystick.threshold option in GamepadDevice.defaultOptions to control when this is triggered.

Gamepad Layouts

gamepad.layout  // "nintendo" | "xbox" | "playstation" | "logitech" | "steam" | "standard"

Layout detection is highly non-standard across major browsers, it should generally be used for aesthetic improvements (e.g. showing device-specific icons).

There is some limited layout remapping support built-in for Nintendo controllers, which appear to be the only major brand controller that deviates from the standard.

Gamepad - Nintendo Layout Remapping

Caution

*Nintendo: Both the labels and physical positions of the A,B,X,Y buttons are different on Nintendo controllers.

Set GamepadDevice.defaultOptions.nintendoRemapMode to apply the remapping as required.

  • "physical" (default) – The A,B,X,Y button codes will refer the standard face button positions (Left=X, Top=Y, Bottom=A, Right=B).
  • "accurate" – The A,B,X,Y button codes will refer to the exact Nintendo labels (Left=Y, Top=X, Bottom=B, Right=A).
  • "none" – The A,B,X,Y button codes mapping stay at the default indices (Left=Y, Top=B, Bottom=X, Right=A).
standard       nintendo        nintendo       nintendo
 layout       "physical"      "accurate"       "none"
reference      (default)

    Y             Y              X               B
  X   B         X   B          Y   A           Y   A
    A             A              B               X

    3             3              2               1
  2   1         2   1          3   0           3   0
    0             0              1               2

You can manually override this per-gamepad, or for all gamepads:

// set default
GamepadDevice.defaultOptions.nintendoRemapMode = "none"

// set for a single gamepad
gamepad.options.nintendoRemapMode = "accurate"

GamepadDevice Events

Event Description Payload
"bind" {name,button,buttonCode,device} A named bind button was pressed.
Button presses:
"A" {button,buttonCode,device} Standard layout button "A" was pressed. Equivalent to 0.
"B" {button,buttonCode,device} Standard layout button "B" was pressed. Equivalent to 1.
"X" {button,buttonCode,device} Standard layout button "X" was pressed. Equivalent to 2.
Button presses (no label):
0 or Button.A {button,buttonCode,device} Button at offset 0 was pressed.
1 or Button.B {button,buttonCode,device} Button at offset 1 was pressed.
2 or Button.X {button,buttonCode,device} Button at offset 2 was pressed.

Custom Devices

You can add custom devices to the device manager so it will be polled togehter and included in InputDevice.devices.

import { type CustomDevice, InputDevice } from "pixijs-input-devices"

export const myDevice: CustomDevice = {
    id: "on-screen-buttons",
    type: "custom",
    meta: {},
    
    update: ( now: number ) => {
        // polling update
    }
}

InputDevice.add( myDevice )

Named Binds

Use named binds to create mappings between abstract inputs and the keys/buttons that trigger those inputs.

This allows you to change the keys/buttons later (e.g. allow users to override inputs).

// keyboard:
InputDevice.keyboard.configureBinds({
    jump: [ "ArrowUp", "Space", "KeyW" ],
    crouch: [ "ArrowDown", "KeyS" ],
    toggleGraphics: [ "KeyB" ],
})

// all gamepads:
GamepadDevice.configureDefaultBinds({
    jump: [ "A", "LeftStickUp" ],
    crouch: [ "B", "X", "RightTrigger" ],
    toggleGraphics: [ "RightStickUp", "RightStickDown" ],
})

These can then be used with either the real-time and event-based APIs.

Event-based:

// listen to all devices:
InputDevice.onBind( "toggleGraphics", ( e ) => toggleGraphics() )

// listen to specific devices:
InputDevice.keyboard.onBind( "jump", ( e ) => doJump() )
InputDevice.gamepads[0].onBind( "jump", ( e ) => doJump() )

Real-time:

let jump = false, crouch = false, moveX = 0

const keyboard = InputDevice.keyboard
if ( keyboard.pressedBind( "jump" ) ) jump = true
if ( keyboard.pressedBind( "crouch" ) ) crouch = true
if ( keyboard.key.ArrowLeft ) moveX = -1
else if ( keyboard.key.ArrowRight ) moveX = 1

for ( const gamepad of InputDevice.gamepads ) {
    if ( gamepad.pressedBind( "jump" ) ) jump = true
    if ( gamepad.pressedBind( "crouch" ) ) crouch = true

    // gamepads have additional analog inputs
    // we're going to apply these only if touched
    if ( gamepad.leftJoystick.x != 0 ) moveX = gamepad.leftJoystick.x
    if ( gamepad.leftTrigger > 0 ) moveX *= ( 1 - gamepad.leftTrigger )
}

UINavigation API

Traverse a UI using input devices.

UINavigation.configureWithRoot( app.stage )  // (or any Container)

You can manually take control of navigation using:

// take control
UINavigation.pushResponder( myModalView )

// relinquish control
UINavigation.popResponder()

The Navigation API is centered around the UINavigation manager, which receives navigation intents from devices and forwards it to the UI context.

The UINavigation manager maintains a stack of responders, which can be a Container, or any object that implements the NavigationResponder interface.

When a device sends a navigation intent, the UINavigation manager is responsible for asking the first responder whether it can handle the intent.

If it returns false, any other responders are checked (if they exist), otherwise the default global navigation behavior kicks in.

Default Global Navigation Behaviors

When a navigation intent is not handled manually by a responder, it is handled in one of the following ways:

Intent Behavior
"navigate.back"
  • No action.
"navigate.left", "navigate.right", "navigate.up", "navigate.down"
  • Looks for the nearest Container where container.isNavigatable in the direction given, and if found, receives a "deviceover" event.
  • Additionally, if the newly focused container has registered an event handler for either "pointerover" or "mouseover" (in that order), it will fire that too.
  • If we were previously focused on a container, that previous container receives a "deviceout" event.
  • If the blurred container has register an event handler for either "pointerout" or "mouseout" (in that order), that event handler will be fired too.
"navigate.trigger"
  • Checks if we are currently focused on a container, and then issue a "devicedown" event.
  • If the focused container has registered an event handler for either "pointerdown" or "mousedown" (in that order), that event handler will be fired too.

Container event | Description | Compatibility -----------------|-------------------------------------------------------- "devicedown" | Target was triggered. | "pointerdown", "mousedown" "deviceover" | Target became focused. | "pointerover", "mouseover" "deviceout" | Target lost focus. | "pointerout", "mouseout"

Container Navigatability

Containers are extended with a few properties/accessors:

Container properties type default description
isNavigatable get(): boolean false returns true if navigationMode is set to "target", or is "auto" and a "pointerdown" or "mousedown" event handler is registered.
navigationMode "auto" | "disabled" | "target" "auto" When set to "auto", a Container can be navigated to if it has a "pointerdown" or "mousedown" event handler registered.
navigationPriority number 0 The priority relative to other navigation items in this group.

Note

isNavigatable: By default, any element with "pointerdown" or "mousedown" handlers is navigatable.

Warning

Fallback Hover Effect: If there is no "pointerover" or "mouseover" handler detected on a container, UINavigation will apply abasic alpha effect to the selected item to indicate which container is currently the navigation target. This can be disabled by setting UINavigation.options.useFallbackHoverEffect to false.

Default Binds

The keyboard and gamepad devices are preconfigured with the following binds, feel free to modify them:

Navigation Intent Bind Keyboard Gamepad
"navigate.left" "ArrowLeft", "KeyA" "DPadLeft", "LeftStickLeft"
"navigate.right" "ArrowRight", "KeyD" "DPadRight", "LeftStickRight"
"navigate.up" "ArrowUp", "KeyW" "DPadUp", "LeftStickUp"
"navigate.down" "ArrowDown", "KeyS" "DPadDown", "LeftStickDown"
"navigate.trigger" "Enter", "Space" "A"
"navigate.back" "Escape", "Backspace" "B", "Back"

Advanced usage

Local Player Assignment

Use the <device>.meta property to set assorted meta data on devices as needed.

You lose TypeScript's nice strong types, but its very handy for things like user assignment in multiplayer games.

InputDevice.on("deviceconnected", ({ device }) =>
    // assign!
    device.meta.localPlayerId = 123
)

for ( const device of InputDevice.devices )
{
    if ( device.meta.localPlayerId === 123 )
    {
        // use assigned input device!
    }
}

On-Screen Inputs

You can easily map an on-screen input device using the CustomDevice interface.

export class OnScreenInputContainer extends Container implements CustomDevice {
    id = "onscreen";
    type = "custom" as const;
    meta: Record<string, any> = {};

    inputs = {
        moveX: 0.0
        jump: false,
    }

    update( now )
    {
        this.moveX = this._virtualJoystick.x
        this.jump = this._jumpButton.isTouching()
    }
}

const onscreen = new OnScreenInputContainer();

InputDevice.add( onscreen )
InputDevice.remove( onscreen )

Two Users; One Keyboard

You could set up multiple named inputs:

InputDevice.keyboard.configureBinds({
    jump: [ "ArrowUp", "KeyW" ],
    defend: [ "ArrowDown", "KeyS" ],
    left: [ "ArrowLeft", "KeyA" ],
    right: [ "ArrowRight", "KeyD" ],

    p1_jump: [ "KeyW" ],
    p1_defend: [ "KeyS" ],
    p1_left: [ "KeyA" ],
    p1_right: [ "KeyD" ],

    p2_jump: [ "ArrowUp" ],
    p2_defend: [ "ArrowDown" ],
    p2_left: [ "ArrowLeft" ],
    p2_right: [ "ArrowRight" ],
})

and then switch groups depending on the mode:

if ( gameMode === "multiplayer" )
{
    player1.jump   = device.pressedBind( "p1_jump" )
    player1.defend = device.pressedBind( "p1_defend" )
    player1.moveX += device.pressedBind( "p1_left" ) ? -1 : 0
    player1.moveX += device.pressedBind( "p1_right" ) ? 1 : 0

    player2.jump   = device.pressedBind( "p2_jump" )
    player2.defend = device.pressedBind( "p2_defend" )
    player2.moveX += device.pressedBind( "p2_left" ) ? -1 : 0
    player2.moveX += device.pressedBind( "p2_right" ) ? 1 : 0
}
else
{
    player1.jump   = device.pressedBind( "jump" )
    player1.defend = device.pressedBind( "defend" )
    player1.moveX += device.pressedBind( "left" ) ? -1 : 0
    player1.moveX += device.pressedBind( "right" ) ? 1 : 0

    updateComputerPlayerInput( player2 )
}