123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520 |
- module BABYLON {
- export class Gamepads {
- private babylonGamepads: Array<Gamepad> = [];
- private oneGamepadConnected: boolean = false;
- private isMonitoring: boolean = false;
- private gamepadEventSupported: boolean = 'GamepadEvent' in window;
- private gamepadSupportAvailable: boolean = <boolean> (navigator.getGamepads ||
- !!navigator.webkitGetGamepads || !!navigator.msGetGamepads || !!navigator.webkitGamepads);
- private _callbackGamepadConnected: (gamepad: Gamepad) => void;
- private buttonADataURL: string = "";
- private static gamepadDOMInfo: HTMLElement;
- constructor(ongamedpadconnected: (gamepad: Gamepad) => void) {
- this._callbackGamepadConnected = ongamedpadconnected;
- if (this.gamepadSupportAvailable) {
- // Checking if the gamepad connected event is supported (like in Firefox)
- if (this.gamepadEventSupported) {
- window.addEventListener('gamepadconnected', (evt) => {
- this._onGamepadConnected(evt);
- }, false);
- window.addEventListener('gamepaddisconnected',
- (evt) => {
- this._onGamepadDisconnected(evt);
- }, false);
- }
- else {
- this._startMonitoringGamepads();
- }
- if (!this.oneGamepadConnected) {
- this._insertGamepadDOMInstructions();
- }
- }
- else {
- this._insertGamepadDOMNotSupported();
- }
- }
- private _insertGamepadDOMInstructions() {
- Gamepads.gamepadDOMInfo = <HTMLDivElement>document.createElement("div");
- var buttonAImage = <HTMLImageElement>document.createElement("img");
- buttonAImage.src = this.buttonADataURL;
- var spanMessage = <HTMLSpanElement>document.createElement("span");
- spanMessage.innerHTML = "<strong>to activate gamepad</strong>";
- Gamepads.gamepadDOMInfo.appendChild(buttonAImage);
- Gamepads.gamepadDOMInfo.appendChild(spanMessage);
- Gamepads.gamepadDOMInfo.style.position = "absolute";
- Gamepads.gamepadDOMInfo.style.width = "100%";
- Gamepads.gamepadDOMInfo.style.height = "48px";
- Gamepads.gamepadDOMInfo.style.bottom = "0px";
- Gamepads.gamepadDOMInfo.style.backgroundColor = "rgba(1, 1, 1, 0.15)";
- Gamepads.gamepadDOMInfo.style.textAlign = "center";
- Gamepads.gamepadDOMInfo.style.zIndex = "10";
- buttonAImage.style.position = "relative";
- buttonAImage.style.bottom = "8px";
- spanMessage.style.position = "relative";
- spanMessage.style.fontSize = "32px";
- spanMessage.style.bottom = "32px";
- spanMessage.style.color = "green";
- document.body.appendChild(Gamepads.gamepadDOMInfo);
- }
- private _insertGamepadDOMNotSupported() {
- Gamepads.gamepadDOMInfo = <HTMLDivElement>document.createElement("div");
- var spanMessage = <HTMLSpanElement>document.createElement("span");
- spanMessage.innerHTML = "<strong>gamepad not supported</strong>";
- Gamepads.gamepadDOMInfo.appendChild(spanMessage);
- Gamepads.gamepadDOMInfo.style.position = "absolute";
- Gamepads.gamepadDOMInfo.style.width = "100%";
- Gamepads.gamepadDOMInfo.style.height = "40px";
- Gamepads.gamepadDOMInfo.style.bottom = "0px";
- Gamepads.gamepadDOMInfo.style.backgroundColor = "rgba(1, 1, 1, 0.15)";
- Gamepads.gamepadDOMInfo.style.textAlign = "center";
- Gamepads.gamepadDOMInfo.style.zIndex = "10";
- spanMessage.style.position = "relative";
- spanMessage.style.fontSize = "32px";
- spanMessage.style.color = "red";
- document.body.appendChild(Gamepads.gamepadDOMInfo);
- }
- public dispose() {
- if (Gamepads.gamepadDOMInfo) {
- document.body.removeChild(Gamepads.gamepadDOMInfo);
- }
- }
- private _onGamepadConnected(evt) {
- var newGamepad = this._addNewGamepad(evt.gamepad);
- if (this._callbackGamepadConnected) this._callbackGamepadConnected(newGamepad);
- this._startMonitoringGamepads();
- }
- private _addNewGamepad(gamepad): Gamepad {
- if (!this.oneGamepadConnected) {
- this.oneGamepadConnected = true;
- if (Gamepads.gamepadDOMInfo) {
- document.body.removeChild(Gamepads.gamepadDOMInfo);
- Gamepads.gamepadDOMInfo = null;
- }
- }
- var newGamepad;
- if ((<string>gamepad.id).search("Xbox 360") !== -1 || (<string>gamepad.id).search("xinput") !== -1) {
- newGamepad = new BABYLON.Xbox360Pad(gamepad.id, gamepad.index, gamepad);
- }
- else {
- newGamepad = new BABYLON.GenericPad(gamepad.id, gamepad.index, gamepad);
- }
- this.babylonGamepads.push(newGamepad);
- return newGamepad;
- }
- private _onGamepadDisconnected(evt) {
- // Remove the gamepad from the list of gamepads to monitor.
- for (var i in this.babylonGamepads) {
- if (this.babylonGamepads[i].index == evt.gamepad.index) {
- this.babylonGamepads.splice(i, 1);
- break;
- }
- }
- // If no gamepads are left, stop the polling loop.
- if (this.babylonGamepads.length == 0) {
- this._stopMonitoringGamepads();
- }
- }
- private _startMonitoringGamepads() {
- if (!this.isMonitoring) {
- this.isMonitoring = true;
- this._checkGamepadsStatus();
- }
- }
- private _stopMonitoringGamepads() {
- this.isMonitoring = false;
- }
- private _checkGamepadsStatus() {
- // updating gamepad objects
- this._updateGamepadObjects();
- for (var i in this.babylonGamepads) {
- this.babylonGamepads[i].update();
- }
- if (this.isMonitoring) {
- if (window.requestAnimationFrame) {
- window.requestAnimationFrame(() => { this._checkGamepadsStatus(); });
- } else if (window.mozRequestAnimationFrame) {
- window.mozRequestAnimationFrame(() => { this._checkGamepadsStatus(); });
- } else if (window.webkitRequestAnimationFrame) {
- window.webkitRequestAnimationFrame(() => { this._checkGamepadsStatus(); });
- }
- }
- }
- // This function is called only on Chrome, which does not yet support
- // connection/disconnection events, but requires you to monitor
- // an array for changes.
- private _updateGamepadObjects() {
- var gamepads = navigator.getGamepads ? navigator.getGamepads() : (navigator.webkitGetGamepads ? navigator.webkitGetGamepads() : []);
- for (var i = 0; i < gamepads.length; i++) {
- if (gamepads[i]) {
- if (!(gamepads[i].index in this.babylonGamepads)) {
- var newGamepad = this._addNewGamepad(gamepads[i]);
- if (this._callbackGamepadConnected) {
- this._callbackGamepadConnected(newGamepad);
- }
- }
- else {
- this.babylonGamepads[i].browserGamepad = gamepads[i];
- }
- }
- }
- }
- }
- export class StickValues {
- constructor(public x, public y) {
- }
- }
- export class Gamepad {
- private _leftStick: StickValues;
- private _rightStick: StickValues;
- private _onleftstickchanged: (values: StickValues) => void;
- private _onrightstickchanged: (values: StickValues) => void;
- constructor(public id: string, public index: number, public browserGamepad) {
- if (this.browserGamepad.axes.length >= 2) {
- this._leftStick = { x: this.browserGamepad.axes[0], y: this.browserGamepad.axes[1] };
- }
- if (this.browserGamepad.axes.length >= 4) {
- this._rightStick = { x: this.browserGamepad.axes[2], y: this.browserGamepad.axes[3] };
- }
- }
- public onleftstickchanged(callback: (values: StickValues) => void) {
- this._onleftstickchanged = callback;
- }
- public onrightstickchanged(callback: (values: StickValues) => void) {
- this._onrightstickchanged = callback;
- }
- public get leftStick(): StickValues {
- return this._leftStick;
- }
- public set leftStick(newValues: StickValues) {
- if (this._onleftstickchanged && (this._leftStick.x !== newValues.x || this._leftStick.y !== newValues.y)) {
- this._onleftstickchanged(newValues);
- }
- this._leftStick = newValues;
- }
- public get rightStick(): StickValues {
- return this._rightStick;
- }
- public set rightStick(newValues: StickValues) {
- if (this._onrightstickchanged && (this._rightStick.x !== newValues.x || this._rightStick.y !== newValues.y)) {
- this._onrightstickchanged(newValues);
- }
- this._rightStick = newValues;
- }
- public update() {
- if (this._leftStick) {
- this.leftStick = { x: this.browserGamepad.axes[0], y: this.browserGamepad.axes[1] };
- }
- if (this._rightStick) {
- this.rightStick = { x: this.browserGamepad.axes[2], y: this.browserGamepad.axes[3] };
- }
- }
- }
- export class GenericPad extends Gamepad {
- private _buttons: Array<number>;
- private _onbuttondown: (buttonPressed: number) => void;
- private _onbuttonup: (buttonReleased: number) => void;
- public onbuttondown(callback: (buttonPressed: number) => void) {
- this._onbuttondown = callback;
- }
- public onbuttonup(callback: (buttonReleased: number) => void) {
- this._onbuttonup = callback;
- }
- constructor(public id: string, public index: number, public gamepad) {
- super(id, index, gamepad);
- this._buttons = new Array(gamepad.buttons.length);
- }
- private _setButtonValue(newValue: number, currentValue: number, buttonIndex: number): number {
- if (newValue !== currentValue) {
- if (this._onbuttondown && newValue === 1) {
- this._onbuttondown(buttonIndex);
- }
- if (this._onbuttonup && newValue === 0) {
- this._onbuttonup(buttonIndex);
- }
- }
- return newValue;
- }
- public update() {
- super.update();
- for (var index = 0; index < this._buttons.length; index++) {
- this._buttons[index] = this._setButtonValue(this.gamepad.buttons[index].value, this._buttons[index], index);
- }
- }
- }
- export enum Xbox360Button {
- A,
- B,
- X,
- Y,
- Start,
- Back,
- LB,
- RB,
- LeftStick,
- RightStick
- }
- export enum Xbox360Dpad {
- Up,
- Down,
- Left,
- Right
- }
- export class Xbox360Pad extends Gamepad {
- private _leftTrigger: number = 0;
- private _rightTrigger: number = 0;
- private _onlefttriggerchanged: (value: number) => void;
- private _onrighttriggerchanged: (value: number) => void;
- private _onbuttondown: (buttonPressed: Xbox360Button) => void;
- private _onbuttonup: (buttonReleased: Xbox360Button) => void;
- private _ondpaddown: (dPadPressed: Xbox360Dpad) => void;
- private _ondpadup: (dPadReleased: Xbox360Dpad) => void;
- private _buttonA: number = 0;
- private _buttonB: number = 0;
- private _buttonX: number = 0;
- private _buttonY: number = 0;
- private _buttonBack: number = 0;
- private _buttonStart: number = 0;
- private _buttonLB: number = 0;
- private _buttonRB: number = 0;
- private _buttonLeftStick: number = 0;
- private _buttonRightStick: number = 0;
- private _dPadUp: number = 0;
- private _dPadDown: number = 0;
- private _dPadLeft: number = 0;
- private _dPadRight: number = 0;
- public onlefttriggerchanged(callback: (value: number) => void) {
- this._onlefttriggerchanged = callback;
- }
- public onrighttriggerchanged(callback: (value: number) => void) {
- this._onrighttriggerchanged = callback;
- }
- public get leftTrigger(): number {
- return this._leftTrigger;
- }
- public set leftTrigger(newValue: number) {
- if (this._onlefttriggerchanged && this._leftTrigger !== newValue) {
- this._onlefttriggerchanged(newValue);
- }
- this._leftTrigger = newValue;
- }
- public get rightTrigger(): number {
- return this._rightTrigger;
- }
- public set rightTrigger(newValue: number) {
- if (this._onrighttriggerchanged && this._rightTrigger !== newValue) {
- this._onrighttriggerchanged(newValue);
- }
- this._rightTrigger = newValue;
- }
- public onbuttondown(callback: (buttonPressed: Xbox360Button) => void) {
- this._onbuttondown = callback;
- }
- public onbuttonup(callback: (buttonReleased: Xbox360Button) => void) {
- this._onbuttonup = callback;
- }
- public ondpaddown(callback: (dPadPressed: Xbox360Dpad) => void) {
- this._ondpaddown = callback;
- }
- public ondpadup(callback: (dPadReleased: Xbox360Dpad) => void) {
- this._ondpadup = callback;
- }
- private _setButtonValue(newValue: number, currentValue: number, buttonType: Xbox360Button): number {
- if (newValue !== currentValue) {
- if (this._onbuttondown && newValue === 1) {
- this._onbuttondown(buttonType);
- }
- if (this._onbuttonup && newValue === 0) {
- this._onbuttonup(buttonType);
- }
- }
- return newValue;
- }
- private _setDPadValue(newValue: number, currentValue: number, buttonType: Xbox360Dpad): number {
- if (newValue !== currentValue) {
- if (this._ondpaddown && newValue === 1) {
- this._ondpaddown(buttonType);
- }
- if (this._ondpadup && newValue === 0) {
- this._ondpadup(buttonType);
- }
- }
- return newValue;
- }
- public get buttonA(): number {
- return this._buttonA;
- }
- public set buttonA(value) {
- this._buttonA = this._setButtonValue(value, this._buttonA, Xbox360Button.A);
- }
- public get buttonB(): number {
- return this._buttonB;
- }
- public set buttonB(value) {
- this._buttonB = this._setButtonValue(value, this._buttonB, Xbox360Button.B);
- }
- public get buttonX(): number {
- return this._buttonX;
- }
- public set buttonX(value) {
- this._buttonX = this._setButtonValue(value, this._buttonX, Xbox360Button.X);
- }
- public get buttonY(): number {
- return this._buttonY;
- }
- public set buttonY(value) {
- this._buttonY = this._setButtonValue(value, this._buttonY, Xbox360Button.Y);
- }
- public get buttonStart(): number {
- return this._buttonStart;
- }
- public set buttonStart(value) {
- this._buttonStart = this._setButtonValue(value, this._buttonStart, Xbox360Button.Start);
- }
- public get buttonBack(): number {
- return this._buttonBack;
- }
- public set buttonBack(value) {
- this._buttonBack = this._setButtonValue(value, this._buttonBack, Xbox360Button.Back);
- }
- public get buttonLB(): number {
- return this._buttonLB;
- }
- public set buttonLB(value) {
- this._buttonLB = this._setButtonValue(value, this._buttonLB, Xbox360Button.LB);
- }
- public get buttonRB(): number {
- return this._buttonRB;
- }
- public set buttonRB(value) {
- this._buttonRB = this._setButtonValue(value, this._buttonRB, Xbox360Button.RB);
- }
- public get buttonLeftStick(): number {
- return this._buttonLeftStick;
- }
- public set buttonLeftStick(value) {
- this._buttonLeftStick = this._setButtonValue(value, this._buttonLeftStick, Xbox360Button.LeftStick);
- }
- public get buttonRightStick(): number {
- return this._buttonRightStick;
- }
- public set buttonRightStick(value) {
- this._buttonRightStick = this._setButtonValue(value, this._buttonRightStick, Xbox360Button.RightStick);
- }
- public get dPadUp(): number {
- return this._dPadUp;
- }
- public set dPadUp(value) {
- this._dPadUp = this._setDPadValue(value, this._dPadUp, Xbox360Dpad.Up);
- }
- public get dPadDown(): number {
- return this._dPadDown;
- }
- public set dPadDown(value) {
- this._dPadDown = this._setDPadValue(value, this._dPadDown, Xbox360Dpad.Down);
- }
- public get dPadLeft(): number {
- return this._dPadLeft;
- }
- public set dPadLeft(value) {
- this._dPadLeft = this._setDPadValue(value, this._dPadLeft, Xbox360Dpad.Left);
- }
- public get dPadRight(): number {
- return this._dPadRight;
- }
- public set dPadRight(value) {
- this._dPadRight = this._setDPadValue(value, this._dPadRight, Xbox360Dpad.Right);
- }
- public update() {
- super.update();
- this.buttonA = this.browserGamepad.buttons[0].value;
- this.buttonB = this.browserGamepad.buttons[1].value;
- this.buttonX = this.browserGamepad.buttons[2].value;
- this.buttonY = this.browserGamepad.buttons[3].value;
- this.buttonLB = this.browserGamepad.buttons[4].value;
- this.buttonRB = this.browserGamepad.buttons[5].value;
- this.leftTrigger = this.browserGamepad.buttons[6].value;
- this.rightTrigger = this.browserGamepad.buttons[7].value;
- this.buttonBack = this.browserGamepad.buttons[8].value;
- this.buttonStart = this.browserGamepad.buttons[9].value;
- this.buttonLeftStick = this.browserGamepad.buttons[10].value;
- this.buttonRightStick = this.browserGamepad.buttons[11].value;
- this.dPadUp = this.browserGamepad.buttons[12].value;
- this.dPadDown = this.browserGamepad.buttons[13].value;
- this.dPadLeft = this.browserGamepad.buttons[14].value;
- this.dPadRight = this.browserGamepad.buttons[15].value;
- }
- }
- }
- // Mixins
- //interface Window {
- // webkitRequestAnimationFrame(func: any): any;
- // mozRequestAnimationFrame(func: any): any;
- // oRequestAnimationFrame(func: any): any;
- // WebGLRenderingContext: WebGLRenderingContext;
- // MSGesture: MSGesture;
- // ongamepadconnected(func?: any): any;
- //}
- interface Navigator {
- getGamepads(func?: any): any;
- webkitGetGamepads(func?: any): any
- msGetGamepads(func?: any): any;
- webkitGamepads(func?: any): any;
- }
|