EXAMPLES & DEMOS | DOCS | GETTING STARTED | CHANGELOG
All the parts needed to display and emulate interactive virtual gamepads on the web for multi-touch, mouse, keyboard, or simulated interaction.
Features:
Each module can be used separately, but they work best together!
Add emulated gamepads to the browser Gamepad API. Emulated gamepads are controlled from any source - like mouse, keyboard or multiplayer game events. Use with the GamepadDisplay module for drop in multi-touch and mouse interaction.
Display the state of any gamepad (emulated or not) in SVG or HTML in a very customizable way.
Receive the state of any gamepad as a series of customizable events.
npm install virtual-gamepad-lib
CSS
@import ("virtual-gamepad-lib/gamepad_assets/base.css");
JS
import { setupPresetInteractiveGamepad } from "virtual-gamepad-lib/helpers";
// Import the svg source code for the full on-screen gamepad SVG
import FULL_GPAD_SVG_SOURCE_CODE from "virtual-gamepad-lib/gamepad_assets/rounded/display-gamepad-full.svg?raw";
// - this could be done like with a build tool like vite(shown here) or webpack or at runtime with a fetch request
// Get the HTML container that the onscreen gamepad will go in.
const GPAD_DISPLAY_CONTAINER = document.getElementById("gpad_display_container");
// Insert the SVG contents for the onscreen gamepad into the DOM
GPAD_DISPLAY_CONTAINER.innerHTML = LEFT_GPAD_SVG_SOURCE_CODE;
// Setup the onscreen gamepad to react to the state of the emulated gamepad.
const { gpadApiWrapper, gpadDisplay, gpadEmulator } = setupPresetInteractiveGamepad(GPAD_DISPLAY_CONTAINER, {
/* Options: see interactiveGamepadPresetConfig type in helpers.ts */
});
import { GamepadApiWrapper, GamepadEmulator, GamepadDisplay } from "virtual-gamepad-lib";
// Typescript types are included
import type { GamepadApiWrapper, wrapperConfig /* etc. */ } from "virtual-gamepad-lib";
// Individual modules can be imported like this:
import { GamepadEmulator } from "virtual-gamepad-lib/GamepadEmulator";
// Common.js style also works:
const { GamepadApiWrapper /* etc. */ } = require("virtual-gamepad-lib");
See IMPORTING_GUIDE.md for help.
See the code examples which are live demos at kw-m.github.io/virtual-gamepad-lib
SVG or HTML/CSS gamepads must go inline in your website HTML.
SVG: Either manually copy paste the contents of the SVG into your HTML or add the full SVG contents to the page using your framework of choice at build time or runtime.
// 1. Get the full source of the svg:
import GAMEPAD_SVG = "virtual-gamepad-lib/premade-gamepads/full.svg?raw"
// 2. Add the full svg to your webpage html:
/* Vanilla JS */
document.body.getElementById("MY_GAMEPAD_CONTAINER").innerHTML = GAMEPAD_SVG;
/* Svelte */
<div id="MY_GAMEPAD_CONTAINER" />{@html GAMEPAD_SVG}</div>
/* Vue */
<div id="MY_GAMEPAD_CONTAINER" v-html="GAMEPAD_SVG"></div>
/* React */
<div id="MY_GAMEPAD_CONTAINER" dangerouslySetInnerHTML={createMarkup(GAMEPAD_SVG)} />
// All the above examples are safe from XSS attacks since the gamepad svg is not user-generated & the svg source is included at build time.
HTML/CSS: Write as you would any other part of your page, make sure to add IDs to every button and joystick to reference later!
We have example vector gamepads in the gamepad_asset_originals folder:
MAKE SURE TO FOLLOW THE SVG Authoring Guide IF YOU WANT TO MAKE YOUR OWN VECTOR GAMEPAD!
Making custom SVG gamepads for display on the web is tricky, there are rules that need to be followed to get a good result!
In your vector graphics program, name your layers with this format: #svgElementId
or #svgElementId.svgElementClass
to make it easier to split the id and class attributes in the exported svg with our automated SVGO config.
Use the export option flatten transforms or use absolute positioning in your vector graphics program AND wrap a group around any part of your svg that you want to more or apply css transforms to (such as joysticks) and make sure the element you wish to animate within the empty group is labeled with a recognizable id. Without an empty group, any css driven transforms may overwrite the transforms applied by your vector graphics program - see this question and this solution.
Use the centerTransformOrigin() function from virtual-gamepad-lib/utilities on all elements you wish to rotate or scale using css. This will allow you to use css transforms based on the center of the element rather than the center of the svg.
transform-origin: center center; transform-box: fill-box
on these svg elements; Note that this won't play nice in some browsers or with all vector program exports (especially matrix transforms).Make Tap zones for each button or joystick that are larger than the button/joystick for fat touchscreen fingers!
Tap zones must be single paths, not nested groups with child elements.
Make sure to have ids & classes on your tap targets (use the handy naming scheme #id.gpad-tap-target
in your vector graphics program).
The default CSS disables pointer events on everything in your svg except the tap target. If you use a custom css class you'll need to do this yourself.
/* example css to disable touches/clicks on everything but the tap targets */
svg.my-gamepad-svg *:not(.gpad-tap-target) {
pointer-events: none;
}
If you want to use a custom cursor, make sure to set the css property cursor: none
on the svg tap elements. This will prevent the browser from displaying the default cursor when hovering over the svg.
See: https://www.youtube.com/watch?v=bWcweY66DL8
Auto (recommened)
Use the pre-configured SVGO config in the source github repo. It will optimize the svgs while retaining compatiblity with css & js interaction and avoid id collisions. Use the config with the SVGO CLI. The config file is located at: svgo.config.js See the optimize:svg npm script in the package.json for an example of how to use the config with the SVGO CLI.
Manual (not recommened)
Clean IDs
to keep the id
attributes of the SVG elements from your editorRemove ViewBox
to keep the viewBox
attribute, which makes scaling the SVG easierRemove Unknowns & Defaults
as this can remove the id
and class
attributes even if Clean IDs
is offRemove unneeded group attrs
and collapse useless groups
if you used empty groups to alllow css transforms apply correctly to svg elements.Remove hidden elements
if you used transparent or hidden elements as touch targets or bounding boxesMerge Paths
if you used overlapping paths that should be separate elements on the gamepad, eg: touch targets for the d-pad or buttonsRemove title
or Remove desc
if the title or description is relavant for acessabilityIf you format your layer names in the format elementId.elementClass
, you can use the following regular expression to extract the elementId and elementClass from the layer name (The pre-configured SVGO config has a improved version of this inbuilt, but you can do this manually with find and replace):
Example:
<path id="right_highlight.gpad-highlight" />
becomes
<path id="right_highlight" class="gpad-highlight" />
Find (Regex):
/id="([^"\.]*)(?:"|(?:.([^"\.]*)"))/g
Replace:
id="$1" class="$2"
Contributions are welcome! This project is still in early development, so there are many ways to contribute.
npm run watch:docs
to view the docs locally, then edit comments in the source code or README and submit a pull request!pnpm run build
becomes npm run build
)pnpm run examples:watch
to check the examples are working locally. Run pnpm run pub-preview
to build and test the library, examples and docs, then submit a pull request - Thank You!NPM / PNPM / Yarn scripts: