Controls
Introduction
Controls
in A-Frame Extras by Don McCurdy (GitHub Source)
Allows us to declare custom-controls
html
<script>
AFRAME.registerComponent('custom-controls', {
isVelocityActive: function () {
return Math.random() < 0.25;
},
getPositionDelta: function () {
return new THREE.Vector3(1, 0, 0);
}
});
</script>
Listen for the Oculus Touch X button
javascript
AFRAME.registerComponent('x-button-listener', {
init: function () {
var el = this.el;
el.addEventListener('xbuttondown', function (evt) {
el.setAttribute('visible', !el.getAttribute('visible'));
});
}
});
Then attach the component to the entity
html
<a-entity oculus-touch-controls x-button-listener></a-entity>
gamepad-controls
(GitHub Source)
Includes a rotate function
javascript
updateRotation: function (dt) {
if (!this.isRotationActive()) return;
const data = this.data;
const yaw = this.yaw;
const pitch = this.pitch;
const lookControls = data.camera.components['look-controls'];
const hasLookControls = lookControls && lookControls.pitchObject && lookControls.yawObject;
// Sync with look-controls pitch/yaw if available.
if (hasLookControls) {
pitch.rotation.copy(lookControls.pitchObject.rotation);
yaw.rotation.copy(lookControls.yawObject.rotation);
}
const lookVector = this.getJoystick(1);
if (Math.abs(lookVector.x) <= JOYSTICK_EPS) lookVector.x = 0;
if (Math.abs(lookVector.y) <= JOYSTICK_EPS) lookVector.y = 0;
lookVector.multiplyScalar(data.rotationSensitivity * dt / 1000);
yaw.rotation.y -= lookVector.x;
pitch.rotation.x -= lookVector.y;
pitch.rotation.x = Math.max(- Math.PI / 2, Math.min(Math.PI / 2, pitch.rotation.x));
data.camera.object3D.rotation.set(pitch.rotation.x, yaw.rotation.y, 0);
// Sync with look-controls pitch/yaw if available.
if (hasLookControls) {
lookControls.pitchObject.rotation.copy(pitch.rotation);
lookControls.yawObject.rotation.copy(yaw.rotation);
}
},
keyboard-controls
(GitHub Source)
Declares keypress values
javascript
getVelocityDelta: function () {
const data = this.data,
keys = this.getKeys();
this.dVelocity.set(0, 0, 0);
if (data.enabled) {
if (keys.KeyW || keys.ArrowUp) { this.dVelocity.z -= 1; }
if (keys.KeyA || keys.ArrowLeft) { this.dVelocity.x -= 1; }
if (keys.KeyS || keys.ArrowDown) { this.dVelocity.z += 1; }
if (keys.KeyD || keys.ArrowRight) { this.dVelocity.x += 1; }
}
return this.dVelocity.clone();
},
A-Frame Touch Rotation Controls by Allan Weir (GitHub Source)
Extends Controls
to add touch-rotation-controls
javascript
AFRAME.registerComponent('touch-rotation-controls', {
/**
* Touch Rotation controls.
*
* Based on: https://github.com/aframevr/aframe/pull/1056
*/
schema: {
enabled: { default: true },
sensitivity: { default: 1 / 25 }
},
init: function () {
this.touchDown = false;
this.lookVector = new THREE.Vector2();
this.bindMethods();
},
play: function () {
this.addEventListeners();
},
pause: function () {
this.removeEventListeners();
this.lookVector.set(0, 0);
},
remove: function () {
this.pause();
},
bindMethods: function () {
this.onTouchStart = this.onTouchStart.bind(this);
this.onTouchMove = this.onTouchMove.bind(this);
this.onTouchEnd = this.onTouchEnd.bind(this);
this.onTouchCancel = this.onTouchCancel.bind(this);
},
addEventListeners: function () {
var sceneEl = this.el.sceneEl;
var canvasEl = sceneEl.canvas;
if (!canvasEl) {
sceneEl.addEventListener('render-target-loaded', this.addEventListeners.bind(this));
return;
}
canvasEl.addEventListener('touchstart', this.onTouchStart, false);
canvasEl.addEventListener('touchmove', this.onTouchMove, false);
canvasEl.addEventListener('touchend', this.onTouchEnd, false);
canvasEl.addEventListener('touchcancel', this.onTouchCancel, false);
},
removeEventListeners: function () {
var canvasEl = this.el.sceneEl && this.el.sceneEl.canvas;
if (canvasEl) {
canvasEl.removeEventListener('touchstart', this.onTouchStart, false);
canvasEl.removeEventListener('touchmove', this.onTouchMove, false);
canvasEl.removeEventListener('touchend', this.onTouchEnd, false);
canvasEl.removeEventListener('touchcancel', this.onTouchCancel, false);
}
},
isRotationActive: function () {
return this.data.enabled && this.touchDown;
},
/**
* Returns the sum the touch movement since last call.
*/
getRotationDelta: function () {
var dRotation = this.lookVector.clone().multiplyScalar(this.data.sensitivity);
this.lookVector.set(0, 0);
return dRotation;
},
onTouchMove: function (event) {
if (!this.data.enabled || !this.touchDown) {
return;
}
var touch = event.touches[0];
var movementX = touch.screenX - this.previousTouchX;
var movementY = touch.screenY - this.previousTouchY;
this.lookVector.x += movementX;
this.lookVector.y += movementY;
this.previousTouchX = touch.screenX;
this.previousTouchY = touch.screenY;
},
onTouchStart: function (event) {
this.touchDown = true;
this.previousTouchX = event.touches[0].x;
this.previousTouchY = event.touches[0].y;
},
onTouchEnd: function () {
this.touchDown = false;
},
onTouchCancel: function () {
this.touchDown = false;
}
});