UNPKG

aframe

Version:

A web framework for building virtual reality experiences.

529 lines (467 loc) 20.2 kB
/* 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); } } } });