🚧 WIP - This API is a work in progress, and is subject to change.
- Adds comprehensive support for ⌨️ Keyboard, 🎮 Gamepads, and other human-interface devices
- High-performance, easy-to-use, sensible defaults
- Supports either real-time or event driven APIs
- Built-in
Navigation
API to navigate pointer/mouse based menus (optional)
npm i pixijs-input-devices
import { InputDevice } from "pixijs-input-devices"
Ticker.shared.add( () => InputDevice.update() )
(Optional) Enable the Navigation API:
import { Navigation } from "pixijs-input-devices"
// set root node
Navigation.stage = app.stage
// register mixin
registerPixiJSInputDeviceMixin( Container )
There are a few very simple themes:
- All devices are accessed through the
InputDevice
manager - There are three supported device types: ⌨️
"keyboard"
, 🎮"gamepad"
and 👻"custom"
- Inputs can be accessed directly, or configured by Named Groups
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 ) { // ...
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. |
Access global events directly through the manager:
InputDevice.on( "deviceconnected", ({ device }) => {
// a device was connected
// do additional setup here, show a dialog, etc.
})
InputDevice.off( "deviceconnected" ) // stop listening
Event | Description | Payload |
---|---|---|
"deviceconnected" |
{device} |
A device has become available. |
"devicedisconnected" |
{device} |
A device has been removed. |
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 // "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):
- Browser API (browser support)
- Keypresses
- Browser Language
You can also manually force the layout:
// force layout
InputDevice.keyboard.layout = "JCUKEN"
InputDevice.keyboard.getKeyLabel( "KeyW" ) // "Ц"
InputDevice.keyboard.layoutSource // "manual"
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. |
"group" |
{groupName,event,keyCode,keyLabel,device} |
A named input group 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. |
... | ... | ... |
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.
Use the playVibration()
method to play a haptic vibration, in supported browsers.
gamepad.playVibration()
gamepad.playVibration({
duration: 150,
weakMagnitude: 0.25,
strongMagnitude: 0.65,
})
The gamepad buttons reference Standard Controller Layout:
Button # | ButtonCode | Standard | Nintendo* | Playstation | Xbox |
---|---|---|---|---|---|
0 |
"A" |
A | A | Cross | A |
1 |
"B" |
B | X | Circle | B |
2 |
"X" |
X | B | Square | X |
3 |
"Y" |
Y | Y | Triangle | Y |
4 |
"LeftShoulder" |
Left Shoulder | L | L1 | LB |
5 |
"RightShoulder" |
Right Shoulder | R | R1 | RB |
6 |
"LeftTrigger" |
Left Trigger | L2 | ZL | LT |
7 |
"RightTrigger" |
Right Trigger | R2 | ZR | RT |
8 |
"Back" |
Back | Minus | Options | Back |
9 |
"Start" |
Start | Plus | Select | Start |
10 |
"LeftStick" |
Left Stick (click) | L3 | L3 | LSB |
11 |
"RightStick" |
Right Stick (click) | R3 | R3 | RSB |
12 |
"DPadUp" |
D-Pad Up | ⬆️ | ⬆️ | ⬆️ |
13 |
"DPadDown" |
D-Pad Down | ⬇️ | ⬇️ | ⬇️ |
14 |
"DPadLeft" |
D-Pad Left | ⬅️ | ⬅️ | ⬅️ |
15 |
"DPadRight" |
D-Pad Right | ➡️ | ➡️ | ➡️ |
*See Nintendo Layout Remapping for more context
gamepad.layout // "nintendo" | "xbox" | "playstation" | "logitech" | "steam" | "generic"
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.
Caution
*Nintendo: Both the labels and physical positions of the A,B,X,Y buttons are different on Nintendo controllers.
Set GamepadDevice.defaultOptions.remapNintendoMode
to apply the remapping as required.
"physical"
(default) – The A,B,X,Y button codes will refer the physical layout of a standard controller (Left=X, Top=Y, Bottom=A, Right=B)."accurate"
– The A,B,X,Y button codes will correspond 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:
gamepad.options.remapNintendoMode = "none"
GamepadDevice.defaultOptions.remapNintendoMode = "none"
Event | Description | Payload |
---|---|---|
"group" |
{groupName,button,buttonCode,device} |
A named input group 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. |
... | ... | ... |
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 )
Use named "groups" to create named inputs that can be referenced.
This allows you to change the keys/buttons later (e.g. allow users to override inputs).
// keyboard:
InputDevice.keyboard.options.namedGroups = {
jump: [ "ArrowUp", "Space", "KeyW" ],
crouch: [ "ArrowDown", "KeyS" ],
toggleGraphics: [ "KeyB" ],
}
// all gamepads:
GamepadDevice.defaultOptions.namedGroups = {
jump: [ "A" ],
crouch: [ "B", "X", "RightTrigger" ],
toggleGraphics: [ "RightStick" ],
}
These can then be used with either the real-time and event-based APIs.
// listen to all devices:
InputDevice.onGroup( "toggleGraphics", ( e ) => toggleGraphics() )
// listen to specific devices:
InputDevice.keyboard.onGroup( "jump", ( e ) => doJump() )
InputDevice.gamepads[0].onGroup( "jump", ( e ) => doJump() )
let jump = false, crouch = false, moveX = 0
const keyboard = InputDevice.keyboard
if ( keyboard.groupPressed( "jump" ) ) jump = true
if ( keyboard.groupPressed( "crouch" ) ) crouch = true
if ( keyboard.key.ArrowLeft ) moveX = -1
else if ( keyboard.key.ArrowRight ) moveX = 1
for ( const gamepad of InputDevice.gamepads ) {
if ( gamepad.groupPressed( "jump" ) ) jump = true
if ( gamepad.groupPressed( "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 )
}
Automatically traverse existing pointer/mouse based menus using the Navigation
API.
Navigation.stage = app.stage
const button = new ButtonSprite()
button.on( "mousedown", () => button.run( clickAnimation ) )
button.on( "mouseout", () => button.run( resetAnimation ) )
button.on( "mouseover", () => button.run( hoverAnimation ) )
app.stage.addChild( button )
button.isNavigatable // true
Note
isNavigatable: By default, any element with "mousedown"
or "pointerdown"
handlers is navigatable.
Warning
Fallback Hover Effect: If there is no "pointerover"
or "mouseover"
handler detected on a container, Navigation
will apply abasic alpha effect to the selected item to indicate which container is currently the navigation target. This
can be disabled by setting Navigation.options.useFallbackHoverEffect
to false
.
You can disable the navigation API - either permanently or temporarily - like so:
Navigation.options.enabled = false
UIs can be complex! The Navigation API allows you to take over some - or all - of the navigation elements.
class MyVerticalMenu implements NavigationResponder
{
becameFirstResponder() {
console.log( "I'm in charge now!" )
}
resignedAsFirstResponder() {
console.log( "Nooo! My power is gone!" )
}
handledNavigationIntent( intent, device ): boolean {
if ( intent === "navigateUp" ) this.moveCursorUp()
else if ( intent === "navigateDown" ) this.moveCursorDown()
else if ( intent === "navigateBack" ) this.loseFocus()
else if ( intent === "trigger" ) this.clickCursorItem()
// we are going to return false here, which will propagates unhandled
// intents ("navigateLeft", "navigateRight") up to the next responder
// in the stack - which could be a parent view, etc.
return false
}
}
const myMenu = new MyVerticalMenu()
Navigation.pushResponder( myMenu )
In a game, you might use this to disable navigation outside of menus:
class GameScene implements NavigationResponder
{
handledNavigationIntent( intent, device ) {
// ignore navigation intents, but allow other navigatable
// views to be pushed on top of me - e.g. a dialog window:
return true
}
}
Keyboard and gamepad devices are configured with a few default binds for navigation.
The default binds are below:
Navigation Intent | Keyboard | Gamepad |
---|---|---|
"navigateLeft" |
ArrowLeft , KeyA |
Left Joystick (Left), DPadLeft |
"navigateRight" |
ArrowRight , KeyD |
Left Joystick (Right), DPadRight |
"navigateUp" |
ArrowUp , KeyW |
Left Joystick (Up), DPadDown |
"navigateDown" |
ArrowDown , KeyS |
Left Joystick (Down), DPadUp |
"navigateBack" |
Escape , Backspace |
B , Back |
"trigger" |
Enter, Space |
A |
These can be manually configured in <device>.options.navigation.binds
.
Container properties | type | default | description |
---|---|---|---|
isNavigatable |
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. |
Container events | description |
---|---|
focus |
Target became focused. |
blur |
Target lost focus. |
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!
}
}
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 )
You could set up multiple named inputs:
InputDevice.keyboard.options.namedGroups = {
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 === "2p" )
{
// multiplayer
player1.jump = device.pressedGroup( "p1_jump" )
player1.defend = device.pressedGroup( "p1_defend" )
player1.moveX += device.pressedGroup( "p1_left" ) ? -1 : 0
player1.moveX += device.pressedGroup( "p1_right" ) ? 1 : 0
player2.jump = device.pressedGroup( "p2_jump" )
player2.defend = device.pressedGroup( "p2_defend" )
player2.moveX += device.pressedGroup( "p2_left" ) ? -1 : 0
player2.moveX += device.pressedGroup( "p2_right" ) ? 1 : 0
}
else
{
// single player
player1.jump = device.pressedGroup( "jump" )
player1.defend = device.pressedGroup( "defend" )
player1.moveX += device.pressedGroup( "left" ) ? -1 : 0
player1.moveX += device.pressedGroup( "right" ) ? 1 : 0
player2.updateComputerPlayer()
}