aframe
Version:
A web framework for building virtual reality experiences.
529 lines (467 loc) • 20.2 kB
JavaScript
/* global ImageData, Map, Set */
import * as THREE from 'three';
import { registerComponent as register } from '../../core/component.js';
var arrowURL = '';
var CAM_LAYER = 21;
var applyPose = (function () {
var tempQuaternion = new THREE.Quaternion();
var tempVec3 = new THREE.Vector3();
function applyPose (pose, object3D, offset) {
object3D.position.copy(pose.transform.position);
object3D.quaternion.copy(pose.transform.orientation);
tempVec3.copy(offset);
tempQuaternion.copy(pose.transform.orientation);
tempVec3.applyQuaternion(tempQuaternion);
object3D.position.sub(tempVec3);
}
return applyPose;
}());
applyPose.tempFakePose = {
transform: {
orientation: new THREE.Quaternion(),
position: new THREE.Vector3()
}
};
/**
* Class to handle hit-test from a single source
*
* For a normal space provide it as a space option
* new HitTest(renderer, {
* space: viewerSpace
* });
*
* this is also useful for the targetRaySpace of an XRInputSource
*
* It can also describe a transient input source like so:
*
* var profileToSupport = 'generic-touchscreen';
* var transientHitTest = new HitTest(renderer, {
* profile: profileToSupport
* });
*
* Where the profile matches an item in a type of controller, profiles matching 'generic-touchscreen'
* will always be a transient input and as of 08/2021 all transient inputs are 'generic-touchscreen'
*
* @param {WebGLRenderer} renderer THREE.JS Renderer
* @param {object} hitTestSourceDetails The source information either as the information for a transient hit-test or a regular hit-test
*/
function HitTest (renderer, hitTestSourceDetails) {
this.renderer = renderer;
this.xrHitTestSource = null;
renderer.xr.addEventListener('sessionend', function () {
this.xrHitTestSource = null;
}.bind(this));
renderer.xr.addEventListener('sessionstart', function () {
this.sessionStart(hitTestSourceDetails);
}.bind(this));
if (this.renderer.xr.isPresenting) {
this.sessionStart(hitTestSourceDetails);
}
}
HitTest.prototype.previousFrameAnchors = new Set();
HitTest.prototype.anchorToObject3D = new Map();
function warnAboutHitTest (e) {
console.warn(e.message);
console.warn('Cannot requestHitTestSource Are you missing: webxr="optionalFeatures: hit-test;" from <a-scene>?');
}
HitTest.prototype.sessionStart = function sessionStart (hitTestSourceDetails) {
this.session = this.renderer.xr.getSession();
if (!('requestHitTestSource' in this.session)) {
warnAboutHitTest({message: 'No requestHitTestSource on the session.'});
return;
}
if (hitTestSourceDetails.space) {
this.session.requestHitTestSource(hitTestSourceDetails)
.then(function (xrHitTestSource) {
this.xrHitTestSource = xrHitTestSource;
}.bind(this))
.catch(warnAboutHitTest);
} else if (hitTestSourceDetails.profile) {
this.session.requestHitTestSourceForTransientInput(hitTestSourceDetails)
.then(function (xrHitTestSource) {
this.xrHitTestSource = xrHitTestSource;
this.transient = true;
}.bind(this))
.catch(warnAboutHitTest);
}
};
/**
* Turns the last hit test into an anchor, the provided Object3D will have its
* position update to track the anchor.
*
* @param {Object3D} object3D object to track
* @param {Vector3} offset offset of the object from the origin that gets subtracted
*/
HitTest.prototype.anchorFromLastHitTestResult = function (object3D, offset) {
var hitTest = this.lastHitTest;
if (!hitTest) { return; }
var object3DOptions = {
object3D: object3D,
offset: offset
};
Array.from(this.anchorToObject3D.entries())
.forEach(function (entry) {
var entryObject = entry[1].object3D;
var anchor = entry[0];
if (entryObject === object3D) {
this.anchorToObject3D.delete(anchor);
anchor.delete();
}
}.bind(this));
if (hitTest.createAnchor) {
hitTest.createAnchor()
.then(function (anchor) {
this.anchorToObject3D.set(anchor, object3DOptions);
}.bind(this))
.catch(function (e) {
console.warn(e.message);
console.warn('Cannot create anchor, are you missing: webxr="optionalFeatures: anchors;" from <a-scene>?');
});
}
};
HitTest.prototype.doHit = function doHit (frame) {
if (!this.renderer.xr.isPresenting) { return; }
var refSpace = this.renderer.xr.getReferenceSpace();
var xrViewerPose = frame.getViewerPose(refSpace);
var hitTestResults;
var results;
if (this.xrHitTestSource && xrViewerPose) {
if (this.transient) {
hitTestResults = frame.getHitTestResultsForTransientInput(this.xrHitTestSource);
if (hitTestResults.length > 0) {
results = hitTestResults[0].results;
if (results.length > 0) {
this.lastHitTest = results[0];
return results[0].getPose(refSpace);
} else {
return false;
}
} else {
return false;
}
} else {
hitTestResults = frame.getHitTestResults(this.xrHitTestSource);
if (hitTestResults.length > 0) {
this.lastHitTest = hitTestResults[0];
return hitTestResults[0].getPose(refSpace);
} else {
return false;
}
}
}
};
// static function
HitTest.updateAnchorPoses = function (frame, refSpace) {
// If tracked anchors isn't defined because it's not supported then just use the empty set
var trackedAnchors = frame.trackedAnchors || HitTest.prototype.previousFrameAnchors;
HitTest.prototype.previousFrameAnchors.forEach(function (anchor) {
// Handle anchor tracking loss - `anchor` was present
// in the present frame but is no longer tracked.
if (!trackedAnchors.has(anchor)) {
HitTest.prototype.anchorToObject3D.delete(anchor);
}
});
trackedAnchors.forEach(function (anchor) {
var anchorPose;
var object3DOptions;
var offset;
var object3D;
try {
// Query most recent pose of the anchor relative to some reference space:
anchorPose = frame.getPose(anchor.anchorSpace, refSpace);
if (anchorPose) {
object3DOptions = HitTest.prototype.anchorToObject3D.get(anchor);
if (!object3DOptions) { return; }
offset = object3DOptions.offset;
object3D = object3DOptions.object3D;
applyPose(anchorPose, object3D, offset);
}
} catch (e) {
console.error('while updating anchor poses:', e);
}
});
HitTest.prototype.previousFrameAnchors = trackedAnchors;
};
var hitTestCache;
export var Component = register('ar-hit-test', {
schema: {
target: { type: 'selector' },
enabled: { default: true },
src: {
default: arrowURL,
type: 'map'
},
type: {
default: 'footprint',
oneOf: ['footprint', 'map']
},
footprintDepth: {
default: 0.1
},
mapSize: {
type: 'vec2',
default: {
x: 0.5,
y: 0.5
}
}
},
sceneOnly: true,
init: function () {
this.hitTest = null;
this.imageDataArray = new Uint8ClampedArray(512 * 512 * 4);
this.imageData = new ImageData(this.imageDataArray, 512, 512);
this.textureCache = new Map();
this.orthoCam = new THREE.OrthographicCamera();
this.orthoCam.layers.set(CAM_LAYER);
this.textureTarget = new THREE.WebGLRenderTarget(512, 512, {});
this.basicMaterial = new THREE.MeshBasicMaterial({
color: 0x000000,
side: THREE.DoubleSide
});
this.canvas = document.createElement('canvas');
this.context = this.canvas.getContext('2d');
this.context.imageSmoothingEnabled = false;
this.canvas.width = 512;
this.canvas.height = 512;
this.canvasTexture = new THREE.CanvasTexture(this.canvas, {
alpha: true
});
this.canvasTexture.flipY = false;
// Update WebXR to support hit-test and anchors
var webxrData = this.el.getAttribute('webxr');
var optionalFeaturesArray = webxrData.optionalFeatures;
if (
!optionalFeaturesArray.includes('hit-test') ||
!optionalFeaturesArray.includes('anchors')
) {
optionalFeaturesArray.push('hit-test');
optionalFeaturesArray.push('anchors');
this.el.setAttribute('webxr', webxrData);
}
this.el.sceneEl.renderer.xr.addEventListener('sessionend', function () {
this.hitTest = null;
}.bind(this));
this.el.sceneEl.renderer.xr.addEventListener('sessionstart', function () {
// Don't request Hit Test unless AR (breaks WebXR Emulator)
if (!this.el.is('ar-mode')) { return; }
var renderer = this.el.sceneEl.renderer;
var session = this.session = renderer.xr.getSession();
this.hasPosedOnce = false;
this.bboxMesh.visible = false;
if (!hitTestCache) { hitTestCache = new Map(); }
// Default to selecting through the face
session.requestReferenceSpace('viewer')
.then(function (viewerSpace) {
this.viewerHitTest = this.hitTest = new HitTest(renderer, {
space: viewerSpace
});
this.el.emit('ar-hit-test-start');
}.bind(this));
// If a tracked controller is available, selects via that instead of the headset
var arHitTestComp = this;
this.el.sceneEl.addEventListener('controllersupdated', function () {
var sceneEl = this;
var inputSources = sceneEl.xrSession && sceneEl.xrSession.inputSources;
if (!inputSources) { return; }
for (var i = 0; i < inputSources.length; ++i) {
if (inputSources[i].targetRayMode === 'tracked-pointer') {
arHitTestComp.hitTest = new HitTest(renderer, {
space: inputSources[i].targetRaySpace
});
hitTestCache.set(inputSources[i], arHitTestComp.hitTest);
if (arHitTestComp.viewerHitTest && typeof arHitTestComp.viewerHitTest.cancel === 'function') {
arHitTestComp.viewerHitTest.cancel();
arHitTestComp.viewerHitTest = null;
}
break; // only uses first tracked controller
}
}
});
// These are transient inputs so need to be handled separately
var profileToSupport = 'generic-touchscreen';
var transientHitTest = new HitTest(renderer, {
profile: profileToSupport
});
session.addEventListener('selectstart', function (e) {
if (this.data.enabled !== true) { return; }
var inputSource = e.inputSource;
this.bboxMesh.visible = true;
if (this.hasPosedOnce === true) {
this.el.emit('ar-hit-test-select-start', {
inputSource: inputSource,
position: this.bboxMesh.position,
orientation: this.bboxMesh.quaternion
});
if (inputSource.profiles[0] === profileToSupport) {
this.hitTest = transientHitTest;
} else {
this.hitTest = hitTestCache.get(inputSource) || new HitTest(renderer, {
space: inputSource.targetRaySpace
});
hitTestCache.set(inputSource, this.hitTest);
}
}
}.bind(this));
session.addEventListener('selectend', function (e) {
if (!this.hitTest || this.data.enabled !== true) {
this.hitTest = null;
return;
}
var inputSource = e.inputSource;
var object;
if (this.hasPosedOnce === true) {
this.bboxMesh.visible = false;
// if we have a target with a 3D object then automatically generate an anchor for it.
if (this.data.target) {
object = this.data.target.object3D;
if (object) {
applyPose.tempFakePose.transform.position.copy(this.bboxMesh.position);
applyPose.tempFakePose.transform.orientation.copy(this.bboxMesh.quaternion);
applyPose(applyPose.tempFakePose, object, this.bboxOffset);
object.visible = true;
// create an anchor attached to the object
this.hitTest.anchorFromLastHitTestResult(object, this.bboxOffset);
}
}
this.el.emit('ar-hit-test-select', {
inputSource: inputSource,
position: this.bboxMesh.position,
orientation: this.bboxMesh.quaternion
});
this.hitTest = null;
}
}.bind(this));
}.bind(this));
this.bboxOffset = new THREE.Vector3();
this.update = this.update.bind(this);
this.makeBBox();
},
update: function () {
// If it is disabled it's cleaned up
if (this.data.enabled === false) {
this.hitTest = null;
this.bboxMesh.visible = false;
}
if (this.data.target) {
if (this.data.target.object3D) {
this.data.target.addEventListener('model-loaded', this.update);
this.data.target.object3D.layers.enable(CAM_LAYER);
this.data.target.object3D.traverse(function (child) {
child.layers.enable(CAM_LAYER);
});
} else {
this.data.target.addEventListener('loaded', this.update, {once: true});
}
}
this.bboxNeedsUpdate = true;
},
makeBBox: function () {
var geometry = new THREE.PlaneGeometry(1, 1);
var material = new THREE.MeshBasicMaterial({
transparent: true,
color: 0xffffff
});
geometry.rotateX(-Math.PI / 2);
geometry.rotateY(-Math.PI / 2);
this.bbox = new THREE.Box3();
this.bboxMesh = new THREE.Mesh(geometry, material);
this.el.setObject3D('ar-hit-test', this.bboxMesh);
this.bboxMesh.visible = false;
},
updateFootprint: function () {
var tempImageData;
var renderer = this.el.sceneEl.renderer;
var oldRenderTarget, oldBackground;
var isXREnabled = renderer.xr.enabled;
this.bboxMesh.material.map = this.canvasTexture;
this.bboxMesh.material.needsUpdate = true;
this.orthoCam.rotation.set(-Math.PI / 2, 0, -Math.PI / 2);
this.orthoCam.position.copy(this.bboxMesh.position);
this.orthoCam.position.y -= this.bboxMesh.scale.y / 2;
this.orthoCam.near = 0.1;
this.orthoCam.far = this.orthoCam.near + (this.data.footprintDepth * this.bboxMesh.scale.y);
this.orthoCam.position.y += this.orthoCam.far;
this.orthoCam.right = this.bboxMesh.scale.z / 2;
this.orthoCam.left = -this.bboxMesh.scale.z / 2;
this.orthoCam.top = this.bboxMesh.scale.x / 2;
this.orthoCam.bottom = -this.bboxMesh.scale.x / 2;
this.orthoCam.updateProjectionMatrix();
oldRenderTarget = renderer.getRenderTarget();
renderer.setRenderTarget(this.textureTarget);
renderer.xr.enabled = false;
oldBackground = this.el.object3D.background;
this.el.object3D.overrideMaterial = this.basicMaterial;
this.el.object3D.background = null;
renderer.render(this.el.object3D, this.orthoCam);
this.el.object3D.background = oldBackground;
this.el.object3D.overrideMaterial = null;
renderer.xr.enabled = isXREnabled;
renderer.setRenderTarget(oldRenderTarget);
renderer.readRenderTargetPixels(this.textureTarget, 0, 0, 512, 512, this.imageDataArray);
this.context.putImageData(this.imageData, 0, 0);
this.context.shadowColor = 'white';
this.context.shadowBlur = 10;
this.context.drawImage(this.canvas, 0, 0);
tempImageData = this.context.getImageData(0, 0, 512, 512);
for (var i = 0; i < 512 * 512; i++) {
// if it's a little bit transparent but not opaque make it middle transparent
if (tempImageData.data[i * 4 + 3] !== 0 && tempImageData.data[i * 4 + 3] !== 255) {
tempImageData.data[i * 4 + 3] = 128;
}
}
this.context.putImageData(tempImageData, 0, 0);
this.canvasTexture.needsUpdate = true;
},
tick: function () {
var pose;
var frame = this.el.sceneEl.frame;
var renderer = this.el.sceneEl.renderer;
if (frame) {
// if we are in XR then update the positions of the objects attached to anchors
HitTest.updateAnchorPoses(frame, renderer.xr.getReferenceSpace());
}
if (this.bboxNeedsUpdate) {
this.bboxNeedsUpdate = false;
if (!this.data.target || this.data.type === 'map') {
var texture;
if (this.textureCache.has(this.data.src)) {
texture = this.textureCache.get(this.data.src);
} else {
texture = new THREE.TextureLoader().load(this.data.src);
this.textureCache.set(this.data.src, texture);
}
this.bboxMesh.material.map = texture;
this.bboxMesh.material.needsUpdate = true;
}
if (this.data.target && this.data.target.object3D) {
this.bbox.setFromObject(this.data.target.object3D);
this.bbox.getCenter(this.bboxMesh.position);
this.bbox.getSize(this.bboxMesh.scale);
if (this.data.type === 'footprint') {
// Add a little buffer for the footprint border
this.bboxMesh.scale.x *= 1.04;
this.bboxMesh.scale.z *= 1.04;
this.updateFootprint();
}
this.bboxMesh.position.y -= this.bboxMesh.scale.y / 2;
this.bboxOffset.copy(this.bboxMesh.position);
this.bboxOffset.sub(this.data.target.object3D.position);
} else {
this.bboxMesh.scale.set(this.data.mapSize.x, 1, this.data.mapSize.y);
}
}
if (this.hitTest) {
pose = this.hitTest.doHit(frame);
if (pose) {
if (this.hasPosedOnce !== true) {
this.hasPosedOnce = true;
this.el.emit('ar-hit-test-achieved');
}
this.bboxMesh.visible = true;
this.bboxMesh.position.copy(pose.transform.position);
this.bboxMesh.quaternion.copy(pose.transform.orientation);
}
}
}
});