From 75f0a026c941d1b7a71899c699be0bdb13a19eb8 Mon Sep 17 00:00:00 2001 From: kpal Date: Wed, 8 Jan 2025 14:56:58 +0000 Subject: [PATCH 1/8] Rewrote FPS scripts into ESM --- .../examples/camera/first-person.example.mjs | 18 +- scripts/camera/first-person-camera.js | 905 ------------------ scripts/esm/first-person-controls.mjs | 834 ++++++++++++++++ 3 files changed, 846 insertions(+), 911 deletions(-) delete mode 100644 scripts/camera/first-person-camera.js create mode 100644 scripts/esm/first-person-controls.mjs diff --git a/examples/src/examples/camera/first-person.example.mjs b/examples/src/examples/camera/first-person.example.mjs index e8e7eee742f..bb6b5156f13 100644 --- a/examples/src/examples/camera/first-person.example.mjs +++ b/examples/src/examples/camera/first-person.example.mjs @@ -1,7 +1,14 @@ // @config DESCRIPTION
(WASD) Move
(Space) Jump
(Mouse) Look
-import { deviceType, rootPath } from 'examples/utils'; +import { deviceType, fileImport, rootPath } from 'examples/utils'; import * as pc from 'playcanvas'; +const { + DesktopInput, + MobileInput, + GamePadInput, + CharacterController +} = await fileImport(`${rootPath}/static/scripts/esm/first-person-controls.mjs`); + const canvas = /** @type {HTMLCanvasElement} */ (document.getElementById('application-canvas')); window.focus(); @@ -23,7 +30,6 @@ const gfxOptions = { const assets = { map: new pc.Asset('map', 'container', { url: `${rootPath}/static/assets/models/fps-map.glb` }), - script: new pc.Asset('script', 'script', { url: `${rootPath}/static/scripts/camera/first-person-camera.js` }), helipad: new pc.Asset( 'helipad-env-atlas', 'texture', @@ -112,15 +118,15 @@ function createCharacterController(camera) { restitution: 0 }); entity.addComponent('script'); - entity.script.create('characterController', { + entity.script.create(CharacterController, { attributes: { camera: camera, jumpForce: 850 } }); - entity.script.create('desktopInput'); - entity.script.create('mobileInput'); - entity.script.create('gamePadInput'); + entity.script.create(DesktopInput); + entity.script.create(MobileInput); + entity.script.create(GamePadInput); return entity; } diff --git a/scripts/camera/first-person-camera.js b/scripts/camera/first-person-camera.js deleted file mode 100644 index cd8d20d6fa5..00000000000 --- a/scripts/camera/first-person-camera.js +++ /dev/null @@ -1,905 +0,0 @@ -(() => { - const { createScript, math, Vec2, Vec3, Mat4 } = pc; - - const LOOK_MAX_ANGLE = 90; - - const tmpV1 = new Vec3(); - const tmpV2 = new Vec3(); - const tmpM1 = new Mat4(); - - /** - * Utility function for both touch and gamepad handling of deadzones. Takes a 2-axis joystick - * position in the range -1 to 1 and applies an upper and lower radial deadzone, remapping values in - * the legal range from 0 to 1. - * - * @param {Vec2} pos - The joystick position. - * @param {Vec2} remappedPos - The remapped joystick position. - * @param {number} deadZoneLow - The lower dead zone. - * @param {number} deadZoneHigh - The upper dead zone. - */ - function applyRadialDeadZone(pos, remappedPos, deadZoneLow, deadZoneHigh) { - const magnitude = pos.length(); - - if (magnitude > deadZoneLow) { - const legalRange = 1 - deadZoneHigh - deadZoneLow; - const normalizedMag = Math.min(1, (magnitude - deadZoneLow) / legalRange); - remappedPos.copy(pos).scale(normalizedMag / magnitude); - } else { - remappedPos.set(0, 0); - } - } - - class DesktopInput { - /** - * @type {HTMLCanvasElement} - * @private - */ - _canvas; - - /** - * @type {boolean} - * @private - */ - _enabled = true; - - /** - * @type {AppBase} - */ - app; - - /** - * @param {AppBase} app - The application. - */ - constructor(app) { - this.app = app; - this._canvas = app.graphicsDevice.canvas; - - this._onKeyDown = this._onKeyDown.bind(this); - this._onKeyUp = this._onKeyUp.bind(this); - this._onMouseDown = this._onMouseDown.bind(this); - this._onMouseMove = this._onMouseMove.bind(this); - - this.enabled = true; - } - - set enabled(val) { - this._enabled = val; - - if (val) { - window.addEventListener('keydown', this._onKeyDown); - window.addEventListener('keyup', this._onKeyUp); - window.addEventListener('mousedown', this._onMouseDown); - window.addEventListener('mousemove', this._onMouseMove); - } else { - window.removeEventListener('keydown', this._onKeyDown); - window.removeEventListener('keyup', this._onKeyUp); - window.removeEventListener('mousedown', this._onMouseDown); - window.removeEventListener('mousemove', this._onMouseMove); - } - } - - get enabled() { - return this._enabled; - } - - /** - * @param {string} key - The key pressed. - * @param {number} val - The key value. - * @private - */ - _handleKey(key, val) { - switch (key.toLowerCase()) { - case 'w': - case 'arrowup': - this.app.fire('cc:move:forward', val); - break; - case 's': - case 'arrowdown': - this.app.fire('cc:move:backward', val); - break; - case 'a': - case 'arrowleft': - this.app.fire('cc:move:left', val); - break; - case 'd': - case 'arrowright': - this.app.fire('cc:move:right', val); - break; - case ' ': - this.app.fire('cc:jump', !!val); - break; - case 'shift': - this.app.fire('cc:sprint', !!val); - break; - } - } - - /** - * @param {KeyboardEvent} e - The keyboard event. - * @private - */ - _onKeyDown(e) { - if (document.pointerLockElement !== this._canvas) { - return; - } - - if (e.repeat) { - return; - } - this._handleKey(e.key, 1); - } - - /** - * @param {KeyboardEvent} e - The keyboard event. - * @private - */ - _onKeyUp(e) { - if (e.repeat) { - return; - } - this._handleKey(e.key, 0); - } - - _onMouseDown(e) { - if (document.pointerLockElement !== this._canvas) { - this._canvas.requestPointerLock(); - } - } - - /** - * @param {MouseEvent} e - The mouse event. - * @private - */ - _onMouseMove(e) { - if (document.pointerLockElement !== this._canvas) { - return; - } - - const movementX = e.movementX || e.mozMovementX || e.webkitMovementX || 0; - const movementY = e.movementY || e.mozMovementY || e.webkitMovementY || 0; - - this.app.fire('cc:look', movementX, movementY); - } - - destroy() { - this.enabled = false; - } - } - - class MobileInput { - /** - * @type {GraphicsDevice} - * @private - */ - _device; - - /** - * @type {HTMLCanvasElement} - * @private - */ - _canvas; - - /** - * @type {boolean} - * @private - */ - _enabled = true; - - /** - * @type {number} - * @private - */ - _lastRightTap = 0; - - /** - * @type {number} - * @private - */ - _jumpTimeout; - - /** - * @type {Vec2} - * @private - */ - _remappedPos = new Vec2(); - - /** - * @type {{ identifier: number, center: Vec2; pos: Vec2 }} - * @private - */ - _leftStick = { - identifier: -1, - center: new Vec2(), - pos: new Vec2() - }; - - /** - * @type {{ identifier: number, center: Vec2; pos: Vec2 }} - * @private - */ - _rightStick = { - identifier: -1, - center: new Vec2(), - pos: new Vec2() - }; - - /** - * @type {AppBase} - */ - app; - - /** - * @type {number} - */ - deadZone = 0.3; - - /** - * @type {number} - */ - turnSpeed = 30; - - /** - * @type {number} - */ - radius = 50; - - /** - * @type {number} - */ - _doubleTapInterval = 300; - - /** - * @param {AppBase} app - The application. - */ - constructor(app) { - this.app = app; - this._device = app.graphicsDevice; - this._canvas = app.graphicsDevice.canvas; - - this._onTouchStart = this._onTouchStart.bind(this); - this._onTouchMove = this._onTouchMove.bind(this); - this._onTouchEnd = this._onTouchEnd.bind(this); - - this.enabled = true; - } - - set enabled(val) { - this._enabled = val; - if (val) { - this._canvas.addEventListener('touchstart', this._onTouchStart, false); - this._canvas.addEventListener('touchmove', this._onTouchMove, false); - this._canvas.addEventListener('touchend', this._onTouchEnd, false); - } else { - this._canvas.removeEventListener('touchstart', this._onTouchStart, false); - this._canvas.removeEventListener('touchmove', this._onTouchMove, false); - this._canvas.removeEventListener('touchend', this._onTouchEnd, false); - } - } - - get enabled() { - return this._enabled; - } - - /** - * @private - * @param {TouchEvent} e - The touch event. - */ - _onTouchStart(e) { - e.preventDefault(); - - const xFactor = this._device.width / this._canvas.clientWidth; - const yFactor = this._device.height / this._canvas.clientHeight; - - const touches = e.changedTouches; - for (let i = 0; i < touches.length; i++) { - const touch = touches[i]; - - if (touch.pageX <= this._canvas.clientWidth / 2 && this._leftStick.identifier === -1) { - // If the user touches the left half of the screen, create a left virtual joystick... - this._leftStick.identifier = touch.identifier; - this._leftStick.center.set(touch.pageX, touch.pageY); - this._leftStick.pos.set(0, 0); - this.app.fire('leftjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); - } else if (touch.pageX > this._canvas.clientWidth / 2 && this._rightStick.identifier === -1) { - // ...otherwise create a right virtual joystick - this._rightStick.identifier = touch.identifier; - this._rightStick.center.set(touch.pageX, touch.pageY); - this._rightStick.pos.set(0, 0); - this.app.fire('rightjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); - - // See how long since the last tap of the right virtual joystick to detect a double tap (jump) - const now = Date.now(); - if (now - this._lastRightTap < this._doubleTapInterval) { - if (this._jumpTimeout) { - clearTimeout(this._jumpTimeout); - } - this.app.fire('cc:jump', true); - this._jumpTimeout = setTimeout(() => this.app.fire('cc:jump', false), 50); - } - this._lastRightTap = now; - } - } - } - - /** - * @private - * @param {TouchEvent} e - The touch event. - */ - _onTouchMove(e) { - e.preventDefault(); - - const xFactor = this._device.width / this._canvas.clientWidth; - const yFactor = this._device.height / this._canvas.clientHeight; - - const touches = e.changedTouches; - for (let i = 0; i < touches.length; i++) { - const touch = touches[i]; - - // Update the current positions of the two virtual joysticks - if (touch.identifier === this._leftStick.identifier) { - this._leftStick.pos.set(touch.pageX, touch.pageY); - this._leftStick.pos.sub(this._leftStick.center); - this._leftStick.pos.scale(1 / this.radius); - this.app.fire('leftjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); - } else if (touch.identifier === this._rightStick.identifier) { - this._rightStick.pos.set(touch.pageX, touch.pageY); - this._rightStick.pos.sub(this._rightStick.center); - this._rightStick.pos.scale(1 / this.radius); - this.app.fire('rightjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); - } - } - } - - /** - * @private - * @param {TouchEvent} e - The touch event. - */ - _onTouchEnd(e) { - e.preventDefault(); - - var touches = e.changedTouches; - for (var i = 0; i < touches.length; i++) { - var touch = touches[i]; - - // If this touch is one of the sticks, get rid of it... - if (touch.identifier === this._leftStick.identifier) { - this._leftStick.identifier = -1; - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', 0); - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', 0); - this.app.fire('leftjoystick:disable'); - } else if (touch.identifier === this._rightStick.identifier) { - this._rightStick.identifier = -1; - this.app.fire('rightjoystick:disable'); - } - } - } - - /** - * @param {number} dt - The delta time. - */ - update(dt) { - // Moving - if (this._leftStick.identifier !== -1) { - // Apply a lower radial dead zone. We don't need an upper zone like with a real joypad - applyRadialDeadZone(this._leftStick.pos, this._remappedPos, this.deadZone, 0); - - const forward = -this._remappedPos.y; - if (this._lastForward !== forward) { - if (forward > 0) { - this.app.fire('cc:move:forward', Math.abs(forward)); - this.app.fire('cc:move:backward', 0); - } - if (forward < 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', Math.abs(forward)); - } - if (forward === 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', 0); - } - this._lastForward = forward; - } - - const strafe = this._remappedPos.x; - if (this._lastStrafe !== strafe) { - if (strafe > 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', Math.abs(strafe)); - } - if (strafe < 0) { - this.app.fire('cc:move:left', Math.abs(strafe)); - this.app.fire('cc:move:right', 0); - } - if (strafe === 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', 0); - } - this._lastStrafe = strafe; - } - } - - // Looking - if (this._rightStick.identifier !== -1) { - // Apply a lower radial dead zone. We don't need an upper zone like with a real joypad - applyRadialDeadZone(this._rightStick.pos, this._remappedPos, this.deadZone, 0); - - const movX = this._remappedPos.x * this.turnSpeed; - const movY = this._remappedPos.y * this.turnSpeed; - this.app.fire('cc:look', movX, movY); - } - } - - destroy() { - this.enabled = false; - } - } - - class GamePadInput { - /** - * @type {number} - * @private - */ - _jumpTimeout; - - /** - * @type {number} - * @private - */ - _lastForward = 0; - - /** - * @type {number} - * @private - */ - _lastStrafe = 0; - - /** - * @type {boolean} - * @private - */ - _lastJump = false; - - /** - * @type {Vec2} - * @private - */ - _remappedPos = new Vec2(); - - /** - * @type {{ center: Vec2; pos: Vec2 }} - * @private - */ - _leftStick = { - center: new Vec2(), - pos: new Vec2() - }; - - /** - * @type {{ center: Vec2; pos: Vec2 }} - * @private - */ - _rightStick = { - center: new Vec2(), - pos: new Vec2() - }; - - /** - * @type {AppBase} - */ - app; - - /** - * @type {number} - */ - deadZoneLow = 0.1; - - /** - * @type {number} - */ - deadZoneHigh = 0.1; - - /** - * @type {number} - */ - turnSpeed = 30; - - /** - * @param {AppBase} app - The application. - */ - constructor(app) { - this.app = app; - } - - /** - * @param {number} dt - The delta time. - */ - update(dt) { - const gamepads = navigator.getGamepads ? navigator.getGamepads() : []; - - for (let i = 0; i < gamepads.length; i++) { - const gamepad = gamepads[i]; - - // Only proceed if we have at least 2 sticks - if (gamepad && gamepad.mapping === 'standard' && gamepad.axes.length >= 4) { - // Moving (left stick) - this._leftStick.pos.set(gamepad.axes[0], gamepad.axes[1]); - applyRadialDeadZone(this._leftStick.pos, this._remappedPos, this.deadZoneLow, this.deadZoneHigh); - - const forward = -this._remappedPos.y; - if (this._lastForward !== forward) { - if (forward > 0) { - this.app.fire('cc:move:forward', Math.abs(forward)); - this.app.fire('cc:move:backward', 0); - } - if (forward < 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', Math.abs(forward)); - } - if (forward === 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', 0); - } - this._lastForward = forward; - } - - const strafe = this._remappedPos.x; - if (this._lastStrafe !== strafe) { - if (strafe > 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', Math.abs(strafe)); - } - if (strafe < 0) { - this.app.fire('cc:move:left', Math.abs(strafe)); - this.app.fire('cc:move:right', 0); - } - if (strafe === 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', 0); - } - this._lastStrafe = strafe; - } - - // Looking (right stick) - this._rightStick.pos.set(gamepad.axes[2], gamepad.axes[3]); - applyRadialDeadZone(this._rightStick.pos, this._remappedPos, this.deadZoneLow, this.deadZoneHigh); - - const movX = this._remappedPos.x * this.turnSpeed; - const movY = this._remappedPos.y * this.turnSpeed; - this.app.fire('cc:look', movX, movY); - - // Jumping (bottom button of right cluster) - if (gamepad.buttons[0].pressed && !this._lastJump) { - if (this._jumpTimeout) { - clearTimeout(this._jumpTimeout); - } - this.app.fire('cc:jump', true); - this._jumpTimeout = setTimeout(() => this.app.fire('cc:jump', false), 50); - } - this._lastJump = gamepad.buttons[0].pressed; - } - } - } - - destroy() {} - } - - class CharacterController { - /** - * @type {Entity} - * @private - */ - _camera; - - /** - * @type {RigidBodyComponent} - * @private - */ - _rigidbody; - - /** - * @type {boolean} - * @private - */ - _jumping = false; - - /** - * @type {AppBase} - */ - app; - - /** - * @type {Entity} - */ - entity; - - /** - * @type {Vec2} - */ - look = new Vec2(); - - /** - * @type {Record} - */ - controls = { - forward: 0, - backward: 0, - left: 0, - right: 0, - jump: false, - sprint: false - }; - - /** - * @type {number} - */ - lookSens = 0.08; - - /** - * @type {number} - */ - speedGround = 50; - - /** - * @type {number} - */ - speedAir = 5; - - /** - * @type {number} - */ - sprintMult = 1.5; - - /** - * @type {number} - */ - velocityDampingGround = 0.99; - - /** - * @type {number} - */ - velocityDampingAir = 0.99925; - - /** - * @type {number} - */ - jumpForce = 600; - - /** - * @param {AppBase} app - The application. - * @param {Entity} camera - The camera entity. - * @param {Entity} entity - The controller entity. - */ - constructor(app, camera, entity) { - this.app = app; - this.entity = entity; - - if (!camera) { - throw new Error('No camera entity found'); - } - this._camera = camera; - if (!entity.rigidbody) { - throw new Error('No rigidbody component found'); - } - this._rigidbody = entity.rigidbody; - - this.app.on('cc:look', (movX, movY) => { - this.look.x = math.clamp(this.look.x - movY * this.lookSens, -LOOK_MAX_ANGLE, LOOK_MAX_ANGLE); - this.look.y -= movX * this.lookSens; - }); - this.app.on('cc:move:forward', (val) => { - this.controls.forward = val; - }); - this.app.on('cc:move:backward', (val) => { - this.controls.backward = val; - }); - this.app.on('cc:move:left', (val) => { - this.controls.left = val; - }); - this.app.on('cc:move:right', (val) => { - this.controls.right = val; - }); - this.app.on('cc:jump', (state) => { - this.controls.jump = state; - }); - this.app.on('cc:sprint', (state) => { - this.controls.sprint = state; - }); - } - - /** - * @private - */ - _checkIfGrounded() { - const start = this.entity.getPosition(); - const end = tmpV1.copy(start).add(Vec3.DOWN); - end.y -= 0.1; - this._grounded = !!this._rigidbody.system.raycastFirst(start, end); - } - - /** - * @private - */ - _jump() { - if (this._rigidbody.linearVelocity.y < 0) { - this._jumping = false; - } - if (this.controls.jump && !this._jumping && this._grounded) { - this._jumping = true; - this._rigidbody.applyImpulse(0, this.jumpForce, 0); - } - } - - /** - * @private - */ - _look() { - this._camera.setLocalEulerAngles(this.look.x, this.look.y, 0); - } - - /** - * @param {number} dt - The delta time. - */ - _move(dt) { - tmpM1.setFromAxisAngle(Vec3.UP, this.look.y); - const dir = tmpV1.set(0, 0, 0); - if (this.controls.forward) { - dir.add(tmpV2.set(0, 0, -this.controls.forward)); - } - if (this.controls.backward) { - dir.add(tmpV2.set(0, 0, this.controls.backward)); - } - if (this.controls.left) { - dir.add(tmpV2.set(-this.controls.left, 0, 0)); - } - if (this.controls.right) { - dir.add(tmpV2.set(this.controls.right, 0, 0)); - } - tmpM1.transformVector(dir, dir); - - let speed = this._grounded ? this.speedGround : this.speedAir; - if (this.controls.sprint) { - speed *= this.sprintMult; - } - - const accel = dir.mulScalar(speed * dt); - const velocity = this._rigidbody.linearVelocity.add(accel); - - const damping = this._grounded ? this.velocityDampingGround : this.velocityDampingAir; - const mult = Math.pow(damping, dt * 1e3); - velocity.x *= mult; - velocity.z *= mult; - - this._rigidbody.linearVelocity = velocity; - } - - /** - * @param {number} dt - The delta time. - */ - update(dt) { - this._checkIfGrounded(); - this._jump(); - this._look(); - this._move(dt); - } - } - - // SCRIPTS - - const DesktopInputScript = createScript('desktopInput'); - - DesktopInputScript.prototype.initialize = function () { - this.input = new DesktopInput(this.app); - this.on('enable', () => (this.input.enabled = true)); - this.on('disable', () => (this.input.enabled = false)); - this.on('destroy', () => this.input.destroy()); - }; - - const MobileInputScript = createScript('mobileInput'); - - MobileInputScript.attributes.add('deadZone', { - title: 'Dead Zone', - description: 'Radial thickness of inner dead zone of the virtual joysticks. This dead zone ensures the virtual joysticks report a value of 0 even if a touch deviates a small amount from the initial touch.', - type: 'number', - min: 0, - max: 0.4, - default: 0.3 - }); - MobileInputScript.attributes.add('turnSpeed', { - title: 'Turn Speed', - description: 'Maximum turn speed in degrees per second', - type: 'number', - default: 30 - }); - MobileInputScript.attributes.add('radius', { - title: 'Radius', - description: 'The radius of the virtual joystick in CSS pixels.', - type: 'number', - default: 50 - }); - MobileInputScript.attributes.add('_doubleTapInterval', { - title: 'Double Tap Interval', - description: 'The time in milliseconds between two taps of the right virtual joystick for a double tap to register. A double tap will trigger a cc:jump.', - type: 'number', - default: 300 - }); - - MobileInputScript.prototype.initialize = function () { - this.input = new MobileInput(this.app); - this.input.deadZone = this.deadZone; - this.input.turnSpeed = this.turnSpeed; - this.input.radius = this.radius; - this.input._doubleTapInterval = this._doubleTapInterval; - this.on('enable', () => (this.input.enabled = true)); - this.on('disable', () => (this.input.enabled = false)); - this.on('destroy', () => this.input.destroy()); - }; - - MobileInputScript.prototype.update = function (dt) { - this.input.update(dt); - }; - - const GamePadInputScript = createScript('gamePadInput'); - - GamePadInputScript.attributes.add('deadZoneLow', { - title: 'Low Dead Zone', - description: 'Radial thickness of inner dead zone of pad\'s joysticks. This dead zone ensures that all pads report a value of 0 for each joystick axis when untouched.', - type: 'number', - min: 0, - max: 0.4, - default: 0.1 - }); - GamePadInputScript.attributes.add('deadZoneHigh', { - title: 'High Dead Zone', - description: 'Radial thickness of outer dead zone of pad\'s joysticks. This dead zone ensures that all pads can reach the -1 and 1 limits of each joystick axis.', - type: 'number', - min: 0, - max: 0.4, - default: 0.1 - }); - GamePadInputScript.attributes.add('turnSpeed', { - title: 'Turn Speed', - description: 'Maximum turn speed in degrees per second', - type: 'number', - default: 30 - }); - - GamePadInputScript.prototype.initialize = function () { - this.input = new GamePadInput(this.app); - this.input.deadZoneLow = this.deadZoneLow; - this.input.deadZoneHigh = this.deadZoneHigh; - this.input.turnSpeed = this.turnSpeed; - this.on('destroy', () => this.input.destroy()); - }; - - GamePadInputScript.prototype.update = function (dt) { - this.input.update(dt); - }; - - const CharacterControllerScript = createScript('characterController'); - - CharacterControllerScript.attributes.add('camera', { type: 'entity' }); - CharacterControllerScript.attributes.add('lookSens', { type: 'number', default: 0.08 }); - CharacterControllerScript.attributes.add('speedGround', { type: 'number', default: 50 }); - CharacterControllerScript.attributes.add('speedAir', { type: 'number', default: 5 }); - CharacterControllerScript.attributes.add('sprintMult', { type: 'number', default: 1.5 }); - CharacterControllerScript.attributes.add('velocityDampingGround', { type: 'number', default: 0.99 }); - CharacterControllerScript.attributes.add('velocityDampingAir', { type: 'number', default: 0.99925 }); - CharacterControllerScript.attributes.add('jumpForce', { type: 'number', default: 600 }); - - CharacterControllerScript.prototype.initialize = function () { - this.controller = new CharacterController(this.app, this.camera, this.entity); - this.controller.lookSens = this.lookSens; - this.controller.speedGround = this.speedGround; - this.controller.speedAir = this.speedAir; - this.controller.sprintMult = this.sprintMult; - this.controller.velocityDampingGround = this.velocityDampingGround; - this.controller.velocityDampingAir = this.velocityDampingAir; - this.controller.jumpForce = this.jumpForce; - }; - - CharacterControllerScript.prototype.update = function (dt) { - this.controller.update(dt); - }; -})(); diff --git a/scripts/esm/first-person-controls.mjs b/scripts/esm/first-person-controls.mjs new file mode 100644 index 00000000000..e9f1a739bf4 --- /dev/null +++ b/scripts/esm/first-person-controls.mjs @@ -0,0 +1,834 @@ +/* eslint-disable-next-line import/no-unresolved */ +import { math, Script, Vec2, Vec3, Mat4 } from 'playcanvas'; + +/** @import { GraphicsDevice, Entity, RigidBodyComponent } from 'playcanvas' */ + +const LOOK_MAX_ANGLE = 90; + +const tmpV1 = new Vec3(); +const tmpV2 = new Vec3(); +const tmpM1 = new Mat4(); + +/** + * Utility function for both touch and gamepad handling of deadzones. Takes a 2-axis joystick + * position in the range -1 to 1 and applies an upper and lower radial deadzone, remapping values in + * the legal range from 0 to 1. + * + * @param {Vec2} pos - The joystick position. + * @param {Vec2} remappedPos - The remapped joystick position. + * @param {number} deadZoneLow - The lower dead zone. + * @param {number} deadZoneHigh - The upper dead zone. + */ +const applyRadialDeadZone = (pos, remappedPos, deadZoneLow, deadZoneHigh) => { + const magnitude = pos.length(); + + if (magnitude > deadZoneLow) { + const legalRange = 1 - deadZoneHigh - deadZoneLow; + const normalizedMag = Math.min(1, (magnitude - deadZoneLow) / legalRange); + remappedPos.copy(pos).scale(normalizedMag / magnitude); + } else { + remappedPos.set(0, 0); + } +}; + +class DesktopInput extends Script { + /** + * @type {HTMLCanvasElement} + * @private + */ + _canvas; + + /** + * @param {object} args - The script arguments. + */ + constructor(args) { + super(args); + this._canvas = this.app.graphicsDevice.canvas; + + this._onKeyDown = this._onKeyDown.bind(this); + this._onKeyUp = this._onKeyUp.bind(this); + this._onMouseDown = this._onMouseDown.bind(this); + this._onMouseMove = this._onMouseMove.bind(this); + + this.on('enable', () => this._bind()); + this.on('disable', () => this._unbind()); + this._bind(); + } + + /** + * @private + */ + _bind() { + window.addEventListener('keydown', this._onKeyDown); + window.addEventListener('keyup', this._onKeyUp); + window.addEventListener('mousedown', this._onMouseDown); + window.addEventListener('mousemove', this._onMouseMove); + } + + /** + * @private + */ + _unbind() { + window.removeEventListener('keydown', this._onKeyDown); + window.removeEventListener('keyup', this._onKeyUp); + window.removeEventListener('mousedown', this._onMouseDown); + window.removeEventListener('mousemove', this._onMouseMove); + } + + /** + * @param {string} key - The key pressed. + * @param {number} val - The key value. + * @private + */ + _handleKey(key, val) { + switch (key.toLowerCase()) { + case 'w': + case 'arrowup': + this.app.fire('cc:move:forward', val); + break; + case 's': + case 'arrowdown': + this.app.fire('cc:move:backward', val); + break; + case 'a': + case 'arrowleft': + this.app.fire('cc:move:left', val); + break; + case 'd': + case 'arrowright': + this.app.fire('cc:move:right', val); + break; + case ' ': + this.app.fire('cc:jump', !!val); + break; + case 'shift': + this.app.fire('cc:sprint', !!val); + break; + } + } + + /** + * @param {KeyboardEvent} e - The keyboard event. + * @private + */ + _onKeyDown(e) { + if (document.pointerLockElement !== this._canvas) { + return; + } + + if (e.repeat) { + return; + } + this._handleKey(e.key, 1); + } + + /** + * @param {KeyboardEvent} e - The keyboard event. + * @private + */ + _onKeyUp(e) { + if (e.repeat) { + return; + } + this._handleKey(e.key, 0); + } + + _onMouseDown(e) { + if (document.pointerLockElement !== this._canvas) { + this._canvas.requestPointerLock(); + } + } + + /** + * @param {MouseEvent} e - The mouse event. + * @private + */ + _onMouseMove(e) { + if (document.pointerLockElement !== this._canvas) { + return; + } + + const movementX = e.movementX || 0; + const movementY = e.movementY || 0; + + this.app.fire('cc:look', movementX, movementY); + } + + destroy() { + this._unbind(); + } +} + +class MobileInput extends Script { + /** + * @type {GraphicsDevice} + * @private + */ + _device; + + /** + * @type {HTMLCanvasElement} + * @private + */ + _canvas; + + /** + * @type {number} + * @private + */ + _lastRightTap = 0; + + /** + * @type {ReturnType | null} + * @private + */ + _jumpTimeout = null; + + /** + * @type {Vec2} + * @private + */ + _remappedPos = new Vec2(); + + /** + * @type {{ identifier: number, center: Vec2; pos: Vec2 }} + * @private + */ + _leftStick = { + identifier: -1, + center: new Vec2(), + pos: new Vec2() + }; + + /** + * @type {{ identifier: number, center: Vec2; pos: Vec2 }} + * @private + */ + _rightStick = { + identifier: -1, + center: new Vec2(), + pos: new Vec2() + }; + + /** + * @attribute + * @title Dead Zone + * @description Radial thickness of inner dead zone of the virtual joysticks. This dead zone ensures the virtual joysticks report a value of 0 even if a touch deviates a small amount from the initial touch. + * @type {number} + * @range [0, 0.4] + */ + deadZone = 0.3; + + /** + * @attribute + * @title Turn Speed + * @description Maximum turn speed in degrees per second + * @type {number} + */ + turnSpeed = 30; + + /** + * @attribute + * @title Radius + * @description The radius of the virtual joystick in CSS pixels. + * @type {number} + */ + radius = 50; + + /** + * @attribute + * @title Double Tap Interval + * @description The time in milliseconds between two taps of the right virtual joystick for a double tap to register. A double tap will trigger a cc:jump. + * @type {number} + */ + doubleTapInterval = 300; + + /** + * @param {object} args - The script arguments. + */ + constructor(args) { + super(args); + const { + deadZone, + turnSpeed, + radius, + doubleTapInterval + } = args.attributes; + + this.deadZone = deadZone ?? this.deadZone; + this.turnSpeed = turnSpeed ?? this.turnSpeed; + this.radius = radius ?? this.radius; + this.doubleTapInterval = doubleTapInterval ?? this.doubleTapInterval; + + this._device = this.app.graphicsDevice; + this._canvas = this._device.canvas; + + this._onTouchStart = this._onTouchStart.bind(this); + this._onTouchMove = this._onTouchMove.bind(this); + this._onTouchEnd = this._onTouchEnd.bind(this); + + this.on('enable', () => this._bind()); + this.on('disable', () => this._unbind()); + this._bind(); + } + + /** + * @private + */ + _bind() { + this._canvas.addEventListener('touchstart', this._onTouchStart, false); + this._canvas.addEventListener('touchmove', this._onTouchMove, false); + this._canvas.addEventListener('touchend', this._onTouchEnd, false); + } + + /** + * @private + */ + _unbind() { + this._canvas.removeEventListener('touchstart', this._onTouchStart, false); + this._canvas.removeEventListener('touchmove', this._onTouchMove, false); + this._canvas.removeEventListener('touchend', this._onTouchEnd, false); + } + + /** + * @private + * @param {TouchEvent} e - The touch event. + */ + _onTouchStart(e) { + e.preventDefault(); + + const xFactor = this._device.width / this._canvas.clientWidth; + const yFactor = this._device.height / this._canvas.clientHeight; + + const touches = e.changedTouches; + for (let i = 0; i < touches.length; i++) { + const touch = touches[i]; + + if (touch.pageX <= this._canvas.clientWidth / 2 && this._leftStick.identifier === -1) { + // If the user touches the left half of the screen, create a left virtual joystick... + this._leftStick.identifier = touch.identifier; + this._leftStick.center.set(touch.pageX, touch.pageY); + this._leftStick.pos.set(0, 0); + this.app.fire('leftjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); + } else if (touch.pageX > this._canvas.clientWidth / 2 && this._rightStick.identifier === -1) { + // ...otherwise create a right virtual joystick + this._rightStick.identifier = touch.identifier; + this._rightStick.center.set(touch.pageX, touch.pageY); + this._rightStick.pos.set(0, 0); + this.app.fire('rightjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); + + // See how long since the last tap of the right virtual joystick to detect a double tap (jump) + const now = Date.now(); + if (now - this._lastRightTap < this.doubleTapInterval) { + if (this._jumpTimeout) { + clearTimeout(this._jumpTimeout); + } + this.app.fire('cc:jump', true); + this._jumpTimeout = setTimeout(() => this.app.fire('cc:jump', false), 50); + } + this._lastRightTap = now; + } + } + } + + /** + * @private + * @param {TouchEvent} e - The touch event. + */ + _onTouchMove(e) { + e.preventDefault(); + + const xFactor = this._device.width / this._canvas.clientWidth; + const yFactor = this._device.height / this._canvas.clientHeight; + + const touches = e.changedTouches; + for (let i = 0; i < touches.length; i++) { + const touch = touches[i]; + + // Update the current positions of the two virtual joysticks + if (touch.identifier === this._leftStick.identifier) { + this._leftStick.pos.set(touch.pageX, touch.pageY); + this._leftStick.pos.sub(this._leftStick.center); + this._leftStick.pos.scale(1 / this.radius); + this.app.fire('leftjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); + } else if (touch.identifier === this._rightStick.identifier) { + this._rightStick.pos.set(touch.pageX, touch.pageY); + this._rightStick.pos.sub(this._rightStick.center); + this._rightStick.pos.scale(1 / this.radius); + this.app.fire('rightjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); + } + } + } + + /** + * @private + * @param {TouchEvent} e - The touch event. + */ + _onTouchEnd(e) { + e.preventDefault(); + + const touches = e.changedTouches; + for (let i = 0; i < touches.length; i++) { + const touch = touches[i]; + + // If this touch is one of the sticks, get rid of it... + if (touch.identifier === this._leftStick.identifier) { + this._leftStick.identifier = -1; + this.app.fire('cc:move:forward', 0); + this.app.fire('cc:move:backward', 0); + this.app.fire('cc:move:left', 0); + this.app.fire('cc:move:right', 0); + this.app.fire('leftjoystick:disable'); + } else if (touch.identifier === this._rightStick.identifier) { + this._rightStick.identifier = -1; + this.app.fire('rightjoystick:disable'); + } + } + } + + update() { + // Moving + if (this._leftStick.identifier !== -1) { + // Apply a lower radial dead zone. We don't need an upper zone like with a real joypad + applyRadialDeadZone(this._leftStick.pos, this._remappedPos, this.deadZone, 0); + + const forward = -this._remappedPos.y; + if (this._lastForward !== forward) { + if (forward > 0) { + this.app.fire('cc:move:forward', Math.abs(forward)); + this.app.fire('cc:move:backward', 0); + } + if (forward < 0) { + this.app.fire('cc:move:forward', 0); + this.app.fire('cc:move:backward', Math.abs(forward)); + } + if (forward === 0) { + this.app.fire('cc:move:forward', 0); + this.app.fire('cc:move:backward', 0); + } + this._lastForward = forward; + } + + const strafe = this._remappedPos.x; + if (this._lastStrafe !== strafe) { + if (strafe > 0) { + this.app.fire('cc:move:left', 0); + this.app.fire('cc:move:right', Math.abs(strafe)); + } + if (strafe < 0) { + this.app.fire('cc:move:left', Math.abs(strafe)); + this.app.fire('cc:move:right', 0); + } + if (strafe === 0) { + this.app.fire('cc:move:left', 0); + this.app.fire('cc:move:right', 0); + } + this._lastStrafe = strafe; + } + } + + // Looking + if (this._rightStick.identifier !== -1) { + // Apply a lower radial dead zone. We don't need an upper zone like with a real joypad + applyRadialDeadZone(this._rightStick.pos, this._remappedPos, this.deadZone, 0); + + const movX = this._remappedPos.x * this.turnSpeed; + const movY = this._remappedPos.y * this.turnSpeed; + this.app.fire('cc:look', movX, movY); + } + } + + destroy() { + this._unbind(); + } +} + +class GamePadInput extends Script { + /** + * @type {ReturnType | null} + * @private + */ + _jumpTimeout = null; + + /** + * @type {number} + * @private + */ + _lastForward = 0; + + /** + * @type {number} + * @private + */ + _lastStrafe = 0; + + /** + * @type {boolean} + * @private + */ + _lastJump = false; + + /** + * @type {Vec2} + * @private + */ + _remappedPos = new Vec2(); + + /** + * @type {{ center: Vec2; pos: Vec2 }} + * @private + */ + _leftStick = { + center: new Vec2(), + pos: new Vec2() + }; + + /** + * @type {{ center: Vec2; pos: Vec2 }} + * @private + */ + _rightStick = { + center: new Vec2(), + pos: new Vec2() + }; + + /** + * @attribute + * @title Dead Zone Low + * @description Radial thickness of inner dead zone of pad's joysticks. This dead zone ensures that all pads report a value of 0 for each joystick axis when untouched. + * @type {number} + * @range [0, 0.4] + */ + deadZoneLow = 0.1; + + /** + * @attribute + * @title Dead Zone High + * @description Radial thickness of outer dead zone of pad's joysticks. This dead zone ensures that all pads can reach the -1 and 1 limits of each joystick axis. + * @type {number} + * @range [0, 0.4] + */ + deadZoneHigh = 0.1; + + /** + * @attribute + * @title Turn Speed + * @description Maximum turn speed in degrees per second + * @type {number} + */ + turnSpeed = 30; + + /** + * @param {object} args - The script arguments. + */ + constructor(args) { + super(args); + const { + deadZoneLow, + deadZoneHigh, + turnSpeed + } = args.attributes; + + this.deadZoneLow = deadZoneLow ?? this.deadZoneLow; + this.deadZoneHigh = deadZoneHigh ?? this.deadZoneHigh; + this.turnSpeed = turnSpeed ?? this.turnSpeed; + } + + update() { + const gamepads = navigator.getGamepads ? navigator.getGamepads() : []; + + for (let i = 0; i < gamepads.length; i++) { + const gamepad = gamepads[i]; + + // Only proceed if we have at least 2 sticks + if (gamepad && gamepad.mapping === 'standard' && gamepad.axes.length >= 4) { + // Moving (left stick) + this._leftStick.pos.set(gamepad.axes[0], gamepad.axes[1]); + applyRadialDeadZone(this._leftStick.pos, this._remappedPos, this.deadZoneLow, this.deadZoneHigh); + + const forward = -this._remappedPos.y; + if (this._lastForward !== forward) { + if (forward > 0) { + this.app.fire('cc:move:forward', Math.abs(forward)); + this.app.fire('cc:move:backward', 0); + } + if (forward < 0) { + this.app.fire('cc:move:forward', 0); + this.app.fire('cc:move:backward', Math.abs(forward)); + } + if (forward === 0) { + this.app.fire('cc:move:forward', 0); + this.app.fire('cc:move:backward', 0); + } + this._lastForward = forward; + } + + const strafe = this._remappedPos.x; + if (this._lastStrafe !== strafe) { + if (strafe > 0) { + this.app.fire('cc:move:left', 0); + this.app.fire('cc:move:right', Math.abs(strafe)); + } + if (strafe < 0) { + this.app.fire('cc:move:left', Math.abs(strafe)); + this.app.fire('cc:move:right', 0); + } + if (strafe === 0) { + this.app.fire('cc:move:left', 0); + this.app.fire('cc:move:right', 0); + } + this._lastStrafe = strafe; + } + + // Looking (right stick) + this._rightStick.pos.set(gamepad.axes[2], gamepad.axes[3]); + applyRadialDeadZone(this._rightStick.pos, this._remappedPos, this.deadZoneLow, this.deadZoneHigh); + + const movX = this._remappedPos.x * this.turnSpeed; + const movY = this._remappedPos.y * this.turnSpeed; + this.app.fire('cc:look', movX, movY); + + // Jumping (bottom button of right cluster) + if (gamepad.buttons[0].pressed && !this._lastJump) { + if (this._jumpTimeout) { + clearTimeout(this._jumpTimeout); + } + this.app.fire('cc:jump', true); + this._jumpTimeout = setTimeout(() => this.app.fire('cc:jump', false), 50); + } + this._lastJump = gamepad.buttons[0].pressed; + } + } + } +} + +class CharacterController extends Script { + /** + * @type {RigidBodyComponent} + * @private + */ + _rigidbody; + + /** + * @type {boolean} + * @private + */ + _jumping = false; + + /** + * @type {Vec2} + */ + look = new Vec2(); + + /** + * @type {Record} + */ + controls = { + forward: 0, + backward: 0, + left: 0, + right: 0, + jump: false, + sprint: false + }; + + /** + * @attribute + * @type {Entity} + */ + camera; + + /** + * @attribute + * @type {number} + */ + lookSens = 0.08; + + /** + * @attribute + * @title Ground Speed + * @description The speed of the character when on the ground. + * @type {number} + */ + speedGround = 50; + + /** + * @attribute + * @title Air Speed + * @description The speed of the character when in the air. + * @type {number} + */ + speedAir = 5; + + /** + * @attribute + * @title Sprint Multiplier + * @description The multiplier applied to the speed when sprinting. + * @type {number} + */ + sprintMult = 1.5; + + /** + * @attribute + * @title Velocity Damping Ground + * @description The damping applied to the velocity when on the ground. + * @type {number} + */ + velocityDampingGround = 0.99; + + /** + * @attribute + * @title Velocity Damping Air + * @description The damping applied to the velocity when in the air. + * @type {number} + */ + velocityDampingAir = 0.99925; + + /** + * @attribute + * @title Jump Force + * @description The force applied when jumping. + * @type {number} + */ + jumpForce = 600; + + /** + * @param {object} args - The script arguments. + */ + constructor(args) { + super(args); + const { + camera, + lookSens, + speedGround, + speedAir, + sprintMult, + velocityDampingGround, + velocityDampingAir, + jumpForce + } = args.attributes; + + this.camera = camera; + this.lookSens = lookSens ?? this.lookSens; + this.speedGround = speedGround ?? this.speedGround; + this.speedAir = speedAir ?? this.speedAir; + this.sprintMult = sprintMult ?? this.sprintMult; + this.velocityDampingGround = velocityDampingGround ?? this.velocityDampingGround; + this.velocityDampingAir = velocityDampingAir ?? this.velocityDampingAir; + this.jumpForce = jumpForce ?? this.jumpForce; + + if (!camera) { + throw new Error('No camera entity found'); + } + if (!this.entity.rigidbody) { + throw new Error('No rigidbody component found'); + } + this._rigidbody = this.entity.rigidbody; + + this.app.on('cc:look', (movX, movY) => { + this.look.x = math.clamp(this.look.x - movY * this.lookSens, -LOOK_MAX_ANGLE, LOOK_MAX_ANGLE); + this.look.y -= movX * this.lookSens; + }); + this.app.on('cc:move:forward', (val) => { + this.controls.forward = val; + }); + this.app.on('cc:move:backward', (val) => { + this.controls.backward = val; + }); + this.app.on('cc:move:left', (val) => { + this.controls.left = val; + }); + this.app.on('cc:move:right', (val) => { + this.controls.right = val; + }); + this.app.on('cc:jump', (state) => { + this.controls.jump = state; + }); + this.app.on('cc:sprint', (state) => { + this.controls.sprint = state; + }); + } + + /** + * @private + */ + _checkIfGrounded() { + const start = this.entity.getPosition(); + const end = tmpV1.copy(start).add(Vec3.DOWN); + end.y -= 0.1; + this._grounded = !!this._rigidbody.system.raycastFirst(start, end); + } + + /** + * @private + */ + _jump() { + if (this._rigidbody.linearVelocity.y < 0) { + this._jumping = false; + } + if (this.controls.jump && !this._jumping && this._grounded) { + this._jumping = true; + this._rigidbody.applyImpulse(0, this.jumpForce, 0); + } + } + + /** + * @private + */ + _look() { + this.camera.setLocalEulerAngles(this.look.x, this.look.y, 0); + } + + /** + * @param {number} dt - The delta time. + */ + _move(dt) { + tmpM1.setFromAxisAngle(Vec3.UP, this.look.y); + const dir = tmpV1.set(0, 0, 0); + if (this.controls.forward) { + dir.add(tmpV2.set(0, 0, -this.controls.forward)); + } + if (this.controls.backward) { + dir.add(tmpV2.set(0, 0, this.controls.backward)); + } + if (this.controls.left) { + dir.add(tmpV2.set(-this.controls.left, 0, 0)); + } + if (this.controls.right) { + dir.add(tmpV2.set(this.controls.right, 0, 0)); + } + tmpM1.transformVector(dir, dir); + + let speed = this._grounded ? this.speedGround : this.speedAir; + if (this.controls.sprint) { + speed *= this.sprintMult; + } + + const accel = dir.mulScalar(speed * dt); + const velocity = this._rigidbody.linearVelocity.add(accel); + + const damping = this._grounded ? this.velocityDampingGround : this.velocityDampingAir; + const mult = Math.pow(damping, dt * 1e3); + velocity.x *= mult; + velocity.z *= mult; + + this._rigidbody.linearVelocity = velocity; + } + + /** + * @param {number} dt - The delta time. + */ + update(dt) { + this._checkIfGrounded(); + this._jump(); + this._look(); + this._move(dt); + } +} + +export { + DesktopInput, + MobileInput, + GamePadInput, + CharacterController +}; From 7abc8f4c9aea9b4cd3b3032ff3e14315eedb5a10 Mon Sep 17 00:00:00 2001 From: kpal Date: Wed, 8 Jan 2025 15:27:47 +0000 Subject: [PATCH 2/8] Renamed to character controls and script to first person controller --- examples/src/examples/camera/first-person.example.mjs | 8 ++++---- .../{first-person-controls.mjs => character-controls.mjs} | 4 ++-- 2 files changed, 6 insertions(+), 6 deletions(-) rename scripts/esm/{first-person-controls.mjs => character-controls.mjs} (99%) diff --git a/examples/src/examples/camera/first-person.example.mjs b/examples/src/examples/camera/first-person.example.mjs index bb6b5156f13..1cb8749b743 100644 --- a/examples/src/examples/camera/first-person.example.mjs +++ b/examples/src/examples/camera/first-person.example.mjs @@ -6,8 +6,8 @@ const { DesktopInput, MobileInput, GamePadInput, - CharacterController -} = await fileImport(`${rootPath}/static/scripts/esm/first-person-controls.mjs`); + FirstPersonController, +} = await fileImport(`${rootPath}/static/scripts/esm/character-controls.mjs`); const canvas = /** @type {HTMLCanvasElement} */ (document.getElementById('application-canvas')); window.focus(); @@ -118,9 +118,9 @@ function createCharacterController(camera) { restitution: 0 }); entity.addComponent('script'); - entity.script.create(CharacterController, { + entity.script.create(FirstPersonController, { attributes: { - camera: camera, + camera, jumpForce: 850 } }); diff --git a/scripts/esm/first-person-controls.mjs b/scripts/esm/character-controls.mjs similarity index 99% rename from scripts/esm/first-person-controls.mjs rename to scripts/esm/character-controls.mjs index e9f1a739bf4..dfccde1dc11 100644 --- a/scripts/esm/first-person-controls.mjs +++ b/scripts/esm/character-controls.mjs @@ -602,7 +602,7 @@ class GamePadInput extends Script { } } -class CharacterController extends Script { +class FirstPersonController extends Script { /** * @type {RigidBodyComponent} * @private @@ -830,5 +830,5 @@ export { DesktopInput, MobileInput, GamePadInput, - CharacterController + FirstPersonController }; From 1628a9e0ee08b01137a7a77ba539e242e58be511 Mon Sep 17 00:00:00 2001 From: kpal Date: Wed, 8 Jan 2025 17:45:07 +0000 Subject: [PATCH 3/8] Removed comma --- examples/src/examples/camera/first-person.example.mjs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/src/examples/camera/first-person.example.mjs b/examples/src/examples/camera/first-person.example.mjs index 1cb8749b743..8c3d06fc0e3 100644 --- a/examples/src/examples/camera/first-person.example.mjs +++ b/examples/src/examples/camera/first-person.example.mjs @@ -6,7 +6,7 @@ const { DesktopInput, MobileInput, GamePadInput, - FirstPersonController, + FirstPersonController } = await fileImport(`${rootPath}/static/scripts/esm/character-controls.mjs`); const canvas = /** @type {HTMLCanvasElement} */ (document.getElementById('application-canvas')); From 290ac5b078eaf4cae6430ef9a023c1107f449215 Mon Sep 17 00:00:00 2001 From: kpal Date: Mon, 13 Jan 2025 11:43:36 +0000 Subject: [PATCH 4/8] Renamed character controls to first person controller --- examples/src/examples/camera/first-person.example.mjs | 2 +- .../{character-controls.mjs => first-person-controller.mjs} | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) rename scripts/esm/{character-controls.mjs => first-person-controller.mjs} (99%) diff --git a/examples/src/examples/camera/first-person.example.mjs b/examples/src/examples/camera/first-person.example.mjs index 8c3d06fc0e3..29c8c59a98c 100644 --- a/examples/src/examples/camera/first-person.example.mjs +++ b/examples/src/examples/camera/first-person.example.mjs @@ -7,7 +7,7 @@ const { MobileInput, GamePadInput, FirstPersonController -} = await fileImport(`${rootPath}/static/scripts/esm/character-controls.mjs`); +} = await fileImport(`${rootPath}/static/scripts/esm/first-person-controller.mjs`); const canvas = /** @type {HTMLCanvasElement} */ (document.getElementById('application-canvas')); window.focus(); diff --git a/scripts/esm/character-controls.mjs b/scripts/esm/first-person-controller.mjs similarity index 99% rename from scripts/esm/character-controls.mjs rename to scripts/esm/first-person-controller.mjs index dfccde1dc11..240ace6adeb 100644 --- a/scripts/esm/character-controls.mjs +++ b/scripts/esm/first-person-controller.mjs @@ -31,7 +31,7 @@ const applyRadialDeadZone = (pos, remappedPos, deadZoneLow, deadZoneHigh) => { } }; -class DesktopInput extends Script { +class KeyboardMouseInput extends Script { /** * @type {HTMLCanvasElement} * @private @@ -827,7 +827,7 @@ class FirstPersonController extends Script { } export { - DesktopInput, + KeyboardMouseInput, MobileInput, GamePadInput, FirstPersonController From 68f50918a5055d67a2edaf18f3bae564794fec36 Mon Sep 17 00:00:00 2001 From: kpal Date: Mon, 13 Jan 2025 14:47:57 +0000 Subject: [PATCH 5/8] Updated keyboard mouse input --- examples/src/examples/camera/first-person.example.mjs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/src/examples/camera/first-person.example.mjs b/examples/src/examples/camera/first-person.example.mjs index 29c8c59a98c..0417da77c9d 100644 --- a/examples/src/examples/camera/first-person.example.mjs +++ b/examples/src/examples/camera/first-person.example.mjs @@ -3,7 +3,7 @@ import { deviceType, fileImport, rootPath } from 'examples/utils'; import * as pc from 'playcanvas'; const { - DesktopInput, + KeyboardMouseInput, MobileInput, GamePadInput, FirstPersonController @@ -124,7 +124,7 @@ function createCharacterController(camera) { jumpForce: 850 } }); - entity.script.create(DesktopInput); + entity.script.create(KeyboardMouseInput); entity.script.create(MobileInput); entity.script.create(GamePadInput); From 2f69205cfc4e36e7ebc92a3a985850e34e86d4ef Mon Sep 17 00:00:00 2001 From: kpal Date: Mon, 13 Jan 2025 15:49:14 +0000 Subject: [PATCH 6/8] Combined inputs inside fps controller class --- .../examples/camera/first-person.example.mjs | 10 +- scripts/esm/first-person-controller.mjs | 277 +++++++++++++----- 2 files changed, 204 insertions(+), 83 deletions(-) diff --git a/examples/src/examples/camera/first-person.example.mjs b/examples/src/examples/camera/first-person.example.mjs index 0417da77c9d..8ded0debf74 100644 --- a/examples/src/examples/camera/first-person.example.mjs +++ b/examples/src/examples/camera/first-person.example.mjs @@ -2,12 +2,7 @@ import { deviceType, fileImport, rootPath } from 'examples/utils'; import * as pc from 'playcanvas'; -const { - KeyboardMouseInput, - MobileInput, - GamePadInput, - FirstPersonController -} = await fileImport(`${rootPath}/static/scripts/esm/first-person-controller.mjs`); +const { FirstPersonController } = await fileImport(`${rootPath}/static/scripts/esm/first-person-controller.mjs`); const canvas = /** @type {HTMLCanvasElement} */ (document.getElementById('application-canvas')); window.focus(); @@ -124,9 +119,6 @@ function createCharacterController(camera) { jumpForce: 850 } }); - entity.script.create(KeyboardMouseInput); - entity.script.create(MobileInput); - entity.script.create(GamePadInput); return entity; } diff --git a/scripts/esm/first-person-controller.mjs b/scripts/esm/first-person-controller.mjs index 240ace6adeb..b66a386f569 100644 --- a/scripts/esm/first-person-controller.mjs +++ b/scripts/esm/first-person-controller.mjs @@ -1,7 +1,7 @@ /* eslint-disable-next-line import/no-unresolved */ import { math, Script, Vec2, Vec3, Mat4 } from 'playcanvas'; -/** @import { GraphicsDevice, Entity, RigidBodyComponent } from 'playcanvas' */ +/** @import { AppBase, GraphicsDevice, Entity, RigidBodyComponent } from 'playcanvas' */ const LOOK_MAX_ANGLE = 90; @@ -31,7 +31,13 @@ const applyRadialDeadZone = (pos, remappedPos, deadZoneLow, deadZoneHigh) => { } }; -class KeyboardMouseInput extends Script { +class KeyboardMouseInput { + /** + * @private + * @type {AppBase} + */ + _app; + /** * @type {HTMLCanvasElement} * @private @@ -39,22 +45,43 @@ class KeyboardMouseInput extends Script { _canvas; /** - * @param {object} args - The script arguments. + * @type {boolean} + * @private */ - constructor(args) { - super(args); - this._canvas = this.app.graphicsDevice.canvas; + _enabled = true; + + /** + * @param {AppBase} app - The application. + */ + constructor(app) { + this._app = app; + this._canvas = app.graphicsDevice.canvas; this._onKeyDown = this._onKeyDown.bind(this); this._onKeyUp = this._onKeyUp.bind(this); this._onMouseDown = this._onMouseDown.bind(this); this._onMouseMove = this._onMouseMove.bind(this); - this.on('enable', () => this._bind()); - this.on('disable', () => this._unbind()); this._bind(); } + set enabled(value) { + if (value === this._enabled) { + return; + } + this._enabled = value; + + if (value) { + this._bind(); + } else { + this._unbind(); + } + } + + get enabled() { + return this._enabled; + } + /** * @private */ @@ -84,25 +111,25 @@ class KeyboardMouseInput extends Script { switch (key.toLowerCase()) { case 'w': case 'arrowup': - this.app.fire('cc:move:forward', val); + this._app.fire('cc:move:forward', val); break; case 's': case 'arrowdown': - this.app.fire('cc:move:backward', val); + this._app.fire('cc:move:backward', val); break; case 'a': case 'arrowleft': - this.app.fire('cc:move:left', val); + this._app.fire('cc:move:left', val); break; case 'd': case 'arrowright': - this.app.fire('cc:move:right', val); + this._app.fire('cc:move:right', val); break; case ' ': - this.app.fire('cc:jump', !!val); + this._app.fire('cc:jump', !!val); break; case 'shift': - this.app.fire('cc:sprint', !!val); + this._app.fire('cc:sprint', !!val); break; } } @@ -151,7 +178,7 @@ class KeyboardMouseInput extends Script { const movementX = e.movementX || 0; const movementY = e.movementY || 0; - this.app.fire('cc:look', movementX, movementY); + this._app.fire('cc:look', movementX, movementY); } destroy() { @@ -159,7 +186,13 @@ class KeyboardMouseInput extends Script { } } -class MobileInput extends Script { +class MobileInput { + /** + * @type {AppBase} + * @private + */ + _app; + /** * @type {GraphicsDevice} * @private @@ -210,6 +243,12 @@ class MobileInput extends Script { pos: new Vec2() }; + /** + * @type {boolean} + * @private + */ + _enabled = true; + /** * @attribute * @title Dead Zone @@ -244,34 +283,50 @@ class MobileInput extends Script { doubleTapInterval = 300; /** - * @param {object} args - The script arguments. + * @param {AppBase} app - The application. + * @param {object} args - The arguments. */ - constructor(args) { - super(args); + constructor(app, args = {}) { + this._app = app; const { deadZone, turnSpeed, radius, doubleTapInterval - } = args.attributes; + } = args; this.deadZone = deadZone ?? this.deadZone; this.turnSpeed = turnSpeed ?? this.turnSpeed; this.radius = radius ?? this.radius; this.doubleTapInterval = doubleTapInterval ?? this.doubleTapInterval; - this._device = this.app.graphicsDevice; + this._device = this._app.graphicsDevice; this._canvas = this._device.canvas; this._onTouchStart = this._onTouchStart.bind(this); this._onTouchMove = this._onTouchMove.bind(this); this._onTouchEnd = this._onTouchEnd.bind(this); - this.on('enable', () => this._bind()); - this.on('disable', () => this._unbind()); this._bind(); } + set enabled(value) { + if (value === this._enabled) { + return; + } + this._enabled = value; + + if (value) { + this._bind(); + } else { + this._unbind(); + } + } + + get enabled() { + return this._enabled; + } + /** * @private */ @@ -309,13 +364,13 @@ class MobileInput extends Script { this._leftStick.identifier = touch.identifier; this._leftStick.center.set(touch.pageX, touch.pageY); this._leftStick.pos.set(0, 0); - this.app.fire('leftjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); + this._app.fire('leftjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); } else if (touch.pageX > this._canvas.clientWidth / 2 && this._rightStick.identifier === -1) { // ...otherwise create a right virtual joystick this._rightStick.identifier = touch.identifier; this._rightStick.center.set(touch.pageX, touch.pageY); this._rightStick.pos.set(0, 0); - this.app.fire('rightjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); + this._app.fire('rightjoystick:enable', touch.pageX * xFactor, touch.pageY * yFactor); // See how long since the last tap of the right virtual joystick to detect a double tap (jump) const now = Date.now(); @@ -323,8 +378,8 @@ class MobileInput extends Script { if (this._jumpTimeout) { clearTimeout(this._jumpTimeout); } - this.app.fire('cc:jump', true); - this._jumpTimeout = setTimeout(() => this.app.fire('cc:jump', false), 50); + this._app.fire('cc:jump', true); + this._jumpTimeout = setTimeout(() => this._app.fire('cc:jump', false), 50); } this._lastRightTap = now; } @@ -350,12 +405,12 @@ class MobileInput extends Script { this._leftStick.pos.set(touch.pageX, touch.pageY); this._leftStick.pos.sub(this._leftStick.center); this._leftStick.pos.scale(1 / this.radius); - this.app.fire('leftjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); + this._app.fire('leftjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); } else if (touch.identifier === this._rightStick.identifier) { this._rightStick.pos.set(touch.pageX, touch.pageY); this._rightStick.pos.sub(this._rightStick.center); this._rightStick.pos.scale(1 / this.radius); - this.app.fire('rightjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); + this._app.fire('rightjoystick:move', touch.pageX * xFactor, touch.pageY * yFactor); } } } @@ -374,19 +429,23 @@ class MobileInput extends Script { // If this touch is one of the sticks, get rid of it... if (touch.identifier === this._leftStick.identifier) { this._leftStick.identifier = -1; - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', 0); - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', 0); - this.app.fire('leftjoystick:disable'); + this._app.fire('cc:move:forward', 0); + this._app.fire('cc:move:backward', 0); + this._app.fire('cc:move:left', 0); + this._app.fire('cc:move:right', 0); + this._app.fire('leftjoystick:disable'); } else if (touch.identifier === this._rightStick.identifier) { this._rightStick.identifier = -1; - this.app.fire('rightjoystick:disable'); + this._app.fire('rightjoystick:disable'); } } } update() { + if (!this.enabled) { + return; + } + // Moving if (this._leftStick.identifier !== -1) { // Apply a lower radial dead zone. We don't need an upper zone like with a real joypad @@ -395,16 +454,16 @@ class MobileInput extends Script { const forward = -this._remappedPos.y; if (this._lastForward !== forward) { if (forward > 0) { - this.app.fire('cc:move:forward', Math.abs(forward)); - this.app.fire('cc:move:backward', 0); + this._app.fire('cc:move:forward', Math.abs(forward)); + this._app.fire('cc:move:backward', 0); } if (forward < 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', Math.abs(forward)); + this._app.fire('cc:move:forward', 0); + this._app.fire('cc:move:backward', Math.abs(forward)); } if (forward === 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', 0); + this._app.fire('cc:move:forward', 0); + this._app.fire('cc:move:backward', 0); } this._lastForward = forward; } @@ -412,16 +471,16 @@ class MobileInput extends Script { const strafe = this._remappedPos.x; if (this._lastStrafe !== strafe) { if (strafe > 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', Math.abs(strafe)); + this._app.fire('cc:move:left', 0); + this._app.fire('cc:move:right', Math.abs(strafe)); } if (strafe < 0) { - this.app.fire('cc:move:left', Math.abs(strafe)); - this.app.fire('cc:move:right', 0); + this._app.fire('cc:move:left', Math.abs(strafe)); + this._app.fire('cc:move:right', 0); } if (strafe === 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', 0); + this._app.fire('cc:move:left', 0); + this._app.fire('cc:move:right', 0); } this._lastStrafe = strafe; } @@ -434,7 +493,7 @@ class MobileInput extends Script { const movX = this._remappedPos.x * this.turnSpeed; const movY = this._remappedPos.y * this.turnSpeed; - this.app.fire('cc:look', movX, movY); + this._app.fire('cc:look', movX, movY); } } @@ -443,7 +502,13 @@ class MobileInput extends Script { } } -class GamePadInput extends Script { +class GamePadInput { + /** + * @type {AppBase} + * @private + */ + _app; + /** * @type {ReturnType | null} * @private @@ -492,6 +557,12 @@ class GamePadInput extends Script { pos: new Vec2() }; + /** + * @type {boolean} + * @private + */ + _enabled = true; + /** * @attribute * @title Dead Zone Low @@ -519,22 +590,38 @@ class GamePadInput extends Script { turnSpeed = 30; /** - * @param {object} args - The script arguments. + * @param {AppBase} app - The application. + * @param {object} args - The arguments. */ - constructor(args) { - super(args); + constructor(app, args = {}) { + this.app = app; const { deadZoneLow, deadZoneHigh, turnSpeed - } = args.attributes; + } = args; this.deadZoneLow = deadZoneLow ?? this.deadZoneLow; this.deadZoneHigh = deadZoneHigh ?? this.deadZoneHigh; this.turnSpeed = turnSpeed ?? this.turnSpeed; } + set enabled(value) { + if (value === this._enabled) { + return; + } + this._enabled = value; + } + + get enabled() { + return this._enabled; + } + update() { + if (!this.enabled) { + return; + } + const gamepads = navigator.getGamepads ? navigator.getGamepads() : []; for (let i = 0; i < gamepads.length; i++) { @@ -549,16 +636,16 @@ class GamePadInput extends Script { const forward = -this._remappedPos.y; if (this._lastForward !== forward) { if (forward > 0) { - this.app.fire('cc:move:forward', Math.abs(forward)); - this.app.fire('cc:move:backward', 0); + this._app.fire('cc:move:forward', Math.abs(forward)); + this._app.fire('cc:move:backward', 0); } if (forward < 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', Math.abs(forward)); + this._app.fire('cc:move:forward', 0); + this._app.fire('cc:move:backward', Math.abs(forward)); } if (forward === 0) { - this.app.fire('cc:move:forward', 0); - this.app.fire('cc:move:backward', 0); + this._app.fire('cc:move:forward', 0); + this._app.fire('cc:move:backward', 0); } this._lastForward = forward; } @@ -566,16 +653,16 @@ class GamePadInput extends Script { const strafe = this._remappedPos.x; if (this._lastStrafe !== strafe) { if (strafe > 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', Math.abs(strafe)); + this._app.fire('cc:move:left', 0); + this._app.fire('cc:move:right', Math.abs(strafe)); } if (strafe < 0) { - this.app.fire('cc:move:left', Math.abs(strafe)); - this.app.fire('cc:move:right', 0); + this._app.fire('cc:move:left', Math.abs(strafe)); + this._app.fire('cc:move:right', 0); } if (strafe === 0) { - this.app.fire('cc:move:left', 0); - this.app.fire('cc:move:right', 0); + this._app.fire('cc:move:left', 0); + this._app.fire('cc:move:right', 0); } this._lastStrafe = strafe; } @@ -586,20 +673,24 @@ class GamePadInput extends Script { const movX = this._remappedPos.x * this.turnSpeed; const movY = this._remappedPos.y * this.turnSpeed; - this.app.fire('cc:look', movX, movY); + this._app.fire('cc:look', movX, movY); // Jumping (bottom button of right cluster) if (gamepad.buttons[0].pressed && !this._lastJump) { if (this._jumpTimeout) { clearTimeout(this._jumpTimeout); } - this.app.fire('cc:jump', true); - this._jumpTimeout = setTimeout(() => this.app.fire('cc:jump', false), 50); + this._app.fire('cc:jump', true); + this._jumpTimeout = setTimeout(() => this._app.fire('cc:jump', false), 50); } this._lastJump = gamepad.buttons[0].pressed; } } } + + destroy() { + this.enabled = false; + } } class FirstPersonController extends Script { @@ -615,6 +706,24 @@ class FirstPersonController extends Script { */ _jumping = false; + /** + * @type {KeyboardMouseInput} + * @private + */ + _keyboardMouseInput; + + /** + * @type {MobileInput} + * @private + */ + _mobileInput; + + /** + * @type {GamePadInput} + * @private + */ + _gamepadInput; + /** * @type {Vec2} */ @@ -747,6 +856,22 @@ class FirstPersonController extends Script { this.app.on('cc:sprint', (state) => { this.controls.sprint = state; }); + + // input + this._keyboardMouseInput = new KeyboardMouseInput(this.app); + this._mobileInput = new MobileInput(this.app); + this._gamepadInput = new GamePadInput(this.app); + + this.on('enable', () => { + this._keyboardMouseInput.enabled = true; + this._mobileInput.enabled = true; + this._gamepadInput.enabled = true; + }); + this.on('disable', () => { + this._keyboardMouseInput.enabled = false; + this._mobileInput.enabled = false; + this._gamepadInput.enabled = false; + }); } /** @@ -819,16 +944,20 @@ class FirstPersonController extends Script { * @param {number} dt - The delta time. */ update(dt) { + this._mobileInput.update(); + this._gamepadInput.update(); + this._checkIfGrounded(); this._jump(); this._look(); this._move(dt); } + + destroy() { + this._keyboardMouseInput.destroy(); + this._mobileInput.destroy(); + this._gamepadInput.destroy(); + } } -export { - KeyboardMouseInput, - MobileInput, - GamePadInput, - FirstPersonController -}; +export { FirstPersonController }; From 2812c7bff9039c8a9d506da92b65a9d60a37def5 Mon Sep 17 00:00:00 2001 From: kpal Date: Mon, 13 Jan 2025 16:04:49 +0000 Subject: [PATCH 7/8] Re-exposed all attributes for each input --- scripts/esm/first-person-controller.mjs | 289 ++++++++++++++++++------ 1 file changed, 215 insertions(+), 74 deletions(-) diff --git a/scripts/esm/first-person-controller.mjs b/scripts/esm/first-person-controller.mjs index b66a386f569..18a59e4ff0f 100644 --- a/scripts/esm/first-person-controller.mjs +++ b/scripts/esm/first-person-controller.mjs @@ -250,55 +250,30 @@ class MobileInput { _enabled = true; /** - * @attribute - * @title Dead Zone - * @description Radial thickness of inner dead zone of the virtual joysticks. This dead zone ensures the virtual joysticks report a value of 0 even if a touch deviates a small amount from the initial touch. * @type {number} - * @range [0, 0.4] */ deadZone = 0.3; /** - * @attribute - * @title Turn Speed - * @description Maximum turn speed in degrees per second * @type {number} */ turnSpeed = 30; /** - * @attribute - * @title Radius - * @description The radius of the virtual joystick in CSS pixels. * @type {number} */ radius = 50; /** - * @attribute - * @title Double Tap Interval - * @description The time in milliseconds between two taps of the right virtual joystick for a double tap to register. A double tap will trigger a cc:jump. * @type {number} */ doubleTapInterval = 300; /** * @param {AppBase} app - The application. - * @param {object} args - The arguments. */ - constructor(app, args = {}) { + constructor(app) { this._app = app; - const { - deadZone, - turnSpeed, - radius, - doubleTapInterval - } = args; - - this.deadZone = deadZone ?? this.deadZone; - this.turnSpeed = turnSpeed ?? this.turnSpeed; - this.radius = radius ?? this.radius; - this.doubleTapInterval = doubleTapInterval ?? this.doubleTapInterval; this._device = this._app.graphicsDevice; this._canvas = this._device.canvas; @@ -564,46 +539,25 @@ class GamePadInput { _enabled = true; /** - * @attribute - * @title Dead Zone Low - * @description Radial thickness of inner dead zone of pad's joysticks. This dead zone ensures that all pads report a value of 0 for each joystick axis when untouched. * @type {number} - * @range [0, 0.4] */ deadZoneLow = 0.1; /** - * @attribute - * @title Dead Zone High - * @description Radial thickness of outer dead zone of pad's joysticks. This dead zone ensures that all pads can reach the -1 and 1 limits of each joystick axis. * @type {number} - * @range [0, 0.4] */ deadZoneHigh = 0.1; /** - * @attribute - * @title Turn Speed - * @description Maximum turn speed in degrees per second * @type {number} */ turnSpeed = 30; /** * @param {AppBase} app - The application. - * @param {object} args - The arguments. */ - constructor(app, args = {}) { + constructor(app) { this.app = app; - const { - deadZoneLow, - deadZoneHigh, - turnSpeed - } = args; - - this.deadZoneLow = deadZoneLow ?? this.deadZoneLow; - this.deadZoneHigh = deadZoneHigh ?? this.deadZoneHigh; - this.turnSpeed = turnSpeed ?? this.turnSpeed; } set enabled(value) { @@ -718,11 +672,54 @@ class FirstPersonController extends Script { */ _mobileInput; + /** + * @type {number} + * @private + */ + _mobileDeadZone = 0.3; + + /** + * @type {number} + * @private + */ + _mobileTurnSpeed = 30; + + /** + * @type {number} + * @private + */ + _mobileRadius = 50; + + /** + * @type {number} + * @private + */ + _mobileDoubleTapInterval = 300; + /** * @type {GamePadInput} * @private */ - _gamepadInput; + _gamePadInput; + + + /** + * @type {number} + * @private + */ + _gamePadDeadZoneLow = 0.1; + + /** + * @type {number} + * @private + */ + _gamePadDeadZoneHigh = 0.1; + + /** + * @type {number} + * @private + */ + _gamePadTurnSpeed = 30; /** * @type {Vec2} @@ -749,6 +746,8 @@ class FirstPersonController extends Script { /** * @attribute + * @title Look Sensitivity + * @description The sensitivity of the look controls. * @type {number} */ lookSens = 0.08; @@ -806,6 +805,23 @@ class FirstPersonController extends Script { */ constructor(args) { super(args); + + // input + this._keyboardMouseInput = new KeyboardMouseInput(this.app); + this._mobileInput = new MobileInput(this.app); + this._gamePadInput = new GamePadInput(this.app); + + this.on('enable', () => { + this._keyboardMouseInput.enabled = true; + this._mobileInput.enabled = true; + this._gamePadInput.enabled = true; + }); + this.on('disable', () => { + this._keyboardMouseInput.enabled = false; + this._mobileInput.enabled = false; + this._gamePadInput.enabled = false; + }); + const { camera, lookSens, @@ -814,9 +830,24 @@ class FirstPersonController extends Script { sprintMult, velocityDampingGround, velocityDampingAir, - jumpForce + jumpForce, + mobileDeadZone, + mobileTurnSpeed, + mobileRadius, + mobileDoubleTapInterval, + gamePadDeadZoneLow, + gamePadDeadZoneHigh, + gamePadTurnSpeed } = args.attributes; + if (!camera) { + throw new Error('No camera entity found'); + } + if (!this.entity.rigidbody) { + throw new Error('No rigidbody component found'); + } + this._rigidbody = this.entity.rigidbody; + this.camera = camera; this.lookSens = lookSens ?? this.lookSens; this.speedGround = speedGround ?? this.speedGround; @@ -825,14 +856,13 @@ class FirstPersonController extends Script { this.velocityDampingGround = velocityDampingGround ?? this.velocityDampingGround; this.velocityDampingAir = velocityDampingAir ?? this.velocityDampingAir; this.jumpForce = jumpForce ?? this.jumpForce; - - if (!camera) { - throw new Error('No camera entity found'); - } - if (!this.entity.rigidbody) { - throw new Error('No rigidbody component found'); - } - this._rigidbody = this.entity.rigidbody; + this.mobileDeadZone = mobileDeadZone ?? this.mobileDeadZone; + this.mobileTurnSpeed = mobileTurnSpeed ?? this.mobileTurnSpeed; + this.mobileRadius = mobileRadius ?? this.mobileRadius; + this.mobileDoubleTapInterval = mobileDoubleTapInterval ?? this.mobileDoubleTapInterval; + this.gamePadDeadZoneLow = gamePadDeadZoneLow ?? this.gamePadDeadZoneLow; + this.gamePadDeadZoneHigh = gamePadDeadZoneHigh ?? this.gamePadDeadZoneHigh; + this.gamePadTurnSpeed = gamePadTurnSpeed ?? this.gamePadTurnSpeed; this.app.on('cc:look', (movX, movY) => { this.look.x = math.clamp(this.look.x - movY * this.lookSens, -LOOK_MAX_ANGLE, LOOK_MAX_ANGLE); @@ -856,22 +886,133 @@ class FirstPersonController extends Script { this.app.on('cc:sprint', (state) => { this.controls.sprint = state; }); + } - // input - this._keyboardMouseInput = new KeyboardMouseInput(this.app); - this._mobileInput = new MobileInput(this.app); - this._gamepadInput = new GamePadInput(this.app); + /** + * @attribute + * @title Mobile Dead Zone + * @description Radial thickness of inner dead zone of the virtual joysticks. This dead zone ensures the virtual joysticks report a value of 0 even if a touch deviates a small amount from the initial touch. + * @type {number} + * @range [0, 0.4] + */ + set mobileDeadZone(value) { + if (value === this._mobileDeadZone) { + return; + } + this._mobileDeadZone = value; + this._mobileInput.deadZone = value; + } - this.on('enable', () => { - this._keyboardMouseInput.enabled = true; - this._mobileInput.enabled = true; - this._gamepadInput.enabled = true; - }); - this.on('disable', () => { - this._keyboardMouseInput.enabled = false; - this._mobileInput.enabled = false; - this._gamepadInput.enabled = false; - }); + get mobileDeadZone() { + return this._mobileDeadZone; + } + + /** + * @attribute + * @title Mobile Turn Speed + * @description Maximum turn speed in degrees per second + * @type {number} + */ + set mobileTurnSpeed(value) { + if (value === this._mobileTurnSpeed) { + return; + } + this._mobileTurnSpeed = value; + this._mobileInput.turnSpeed = value; + } + + get mobileTurnSpeed() { + return this._mobileTurnSpeed; + } + + /** + * @attribute + * @title Mobile Radius + * @description The radius of the virtual joystick in CSS pixels. + * @type {number} + */ + set mobileRadius(value) { + if (value === this._mobileRadius) { + return; + } + this._mobileRadius = value; + } + + get mobileRadius() { + return this._mobileRadius; + } + + /** + * @attribute + * @title Mobile Double Tap Interval + * @description The time in milliseconds between two taps of the right virtual joystick for a double tap to register. A double tap will trigger a cc:jump. + * @type {number} + */ + set mobileDoubleTapInterval(value) { + if (value === this._mobileDoubleTapInterval) { + return; + } + this._mobileDoubleTapInterval = value; + } + + get mobileDoubleTapInterval() { + return this._mobileDoubleTapInterval; + } + + /** + * @attribute + * @title GamePad Dead Zone Low + * @description Radial thickness of inner dead zone of pad's joysticks. This dead zone ensures that all pads report a value of 0 for each joystick axis when untouched. + * @type {number} + * @range [0, 0.4] + */ + set gamePadDeadZoneLow(value) { + if (value === this._gamePadDeadZoneLow) { + return; + } + this._gamePadDeadZoneLow = value; + this._gamePadInput.deadZoneLow = value; + } + + get gamePadDeadZoneLow() { + return this._gamePadDeadZoneLow; + } + + /** + * @attribute + * @title GamePad Dead Zone High + * @description Radial thickness of outer dead zone of pad's joysticks. This dead zone ensures that all pads can reach the -1 and 1 limits of each joystick axis. + * @type {number} + * @range [0, 0.4] + */ + set gamePadDeadZoneHigh(value) { + if (value === this._gamePadDeadZoneHigh) { + return; + } + this._gamePadDeadZoneHigh = value; + this._gamePadInput.deadZoneHigh = value; + } + + get gamePadDeadZoneHigh() { + return this._gamePadDeadZoneHigh; + } + + /** + * @attribute + * @title GamePad Turn Speed + * @description Maximum turn speed in degrees per second + * @type {number} + */ + set gamePadTurnSpeed(value) { + if (value === this._gamePadTurnSpeed) { + return; + } + this._gamePadTurnSpeed = value; + this._gamePadInput.turnSpeed = value; + } + + get gamePadTurnSpeed() { + return this._gamePadTurnSpeed; } /** @@ -945,7 +1086,7 @@ class FirstPersonController extends Script { */ update(dt) { this._mobileInput.update(); - this._gamepadInput.update(); + this._gamePadInput.update(); this._checkIfGrounded(); this._jump(); @@ -956,7 +1097,7 @@ class FirstPersonController extends Script { destroy() { this._keyboardMouseInput.destroy(); this._mobileInput.destroy(); - this._gamepadInput.destroy(); + this._gamePadInput.destroy(); } } From 7d774731164fb71fd305f731e02fd8e24c29fe3a Mon Sep 17 00:00:00 2001 From: kpal Date: Mon, 13 Jan 2025 16:09:08 +0000 Subject: [PATCH 8/8] Added missing definitions of lastFoward and lastStrafe back in --- scripts/esm/first-person-controller.mjs | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/scripts/esm/first-person-controller.mjs b/scripts/esm/first-person-controller.mjs index 18a59e4ff0f..bae758a29ee 100644 --- a/scripts/esm/first-person-controller.mjs +++ b/scripts/esm/first-person-controller.mjs @@ -217,6 +217,18 @@ class MobileInput { */ _jumpTimeout = null; + /** + * @type {number} + * @private + */ + _lastForward = 0; + + /** + * @type {number} + * @private + */ + _lastStrafe = 0; + /** * @type {Vec2} * @private