123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609 |
- /******/ (function(modules) { // webpackBootstrap
- /******/ // The module cache
- /******/ var installedModules = {};
- /******/ // The require function
- /******/ function __webpack_require__(moduleId) {
- /******/ // Check if module is in cache
- /******/ if(installedModules[moduleId])
- /******/ return installedModules[moduleId].exports;
- /******/ // Create a new module (and put it into the cache)
- /******/ var module = installedModules[moduleId] = {
- /******/ exports: {},
- /******/ id: moduleId,
- /******/ loaded: false
- /******/ };
- /******/ // Execute the module function
- /******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
- /******/ // Flag the module as loaded
- /******/ module.loaded = true;
- /******/ // Return the exports of the module
- /******/ return module.exports;
- /******/ }
- /******/ // expose the modules object (__webpack_modules__)
- /******/ __webpack_require__.m = modules;
- /******/ // expose the module cache
- /******/ __webpack_require__.c = installedModules;
- /******/ // __webpack_public_path__
- /******/ __webpack_require__.p = "";
- /******/ // Load entry module and return exports
- /******/ return __webpack_require__(0);
- /******/ })
- /************************************************************************/
- /******/ ([
- /* 0 */
- /***/ (function(module, exports, __webpack_require__) {
- /* global THREE, AFRAME, Element */
- var cylinderTexture = __webpack_require__(1);
- var parabolicCurve = __webpack_require__(2);
- var RayCurve = __webpack_require__(3);
- if (typeof AFRAME === 'undefined') {
- throw new Error('Component attempted to register before AFRAME was available.');
- }
- if (!Element.prototype.matches) {
- Element.prototype.matches =
- Element.prototype.matchesSelector ||
- Element.prototype.mozMatchesSelector ||
- Element.prototype.msMatchesSelector ||
- Element.prototype.oMatchesSelector ||
- Element.prototype.webkitMatchesSelector ||
- function (s) {
- var matches = (this.document || this.ownerDocument).querySelectorAll(s);
- var i = matches.length;
- while (--i >= 0 && matches.item(i) !== this) { /* no-op */ }
- return i > -1;
- };
- }
- AFRAME.registerComponent('teleport-controls', {
- schema: {
- type: {default: 'parabolic', oneOf: ['parabolic', 'line']},
- button: {default: 'trackpad', oneOf: ['trackpad', 'trigger', 'grip', 'menu']},
- startEvents: {type: 'array'},
- endEvents: {type: 'array'},
- collisionEntities: {default: ''},
- hitEntity: {type: 'selector'},
- cameraRig: {type: 'selector'},
- teleportOrigin: {type: 'selector'},
- hitCylinderColor: {type: 'color', default: '#99ff99'},
- hitCylinderRadius: {default: 0.25, min: 0},
- hitCylinderHeight: {default: 0.3, min: 0},
- interval: {default: 0},
- maxLength: {default: 10, min: 0, if: {type: ['line']}},
- curveNumberPoints: {default: 30, min: 2, if: {type: ['parabolic']}},
- curveLineWidth: {default: 0.025},
- curveHitColor: {type: 'color', default: '#99ff99'},
- curveMissColor: {type: 'color', default: '#ff0000'},
- curveShootingSpeed: {default: 5, min: 0, if: {type: ['parabolic']}},
- defaultPlaneSize: { default: 100 },
- landingNormal: {type: 'vec3', default: '0 1 0'},
- landingMaxAngle: {default: '45', min: 0, max: 360},
- drawIncrementally: {default: false},
- incrementalDrawMs: {default: 700},
- missOpacity: {default: 1.0},
- hitOpacity: {default: 1.0}
- },
- init: function () {
- var data = this.data;
- var el = this.el;
- var teleportEntity;
- var i;
- this.active = false;
- this.obj = el.object3D;
- this.hitPoint = new THREE.Vector3();
- this.rigWorldPosition = new THREE.Vector3();
- this.newRigWorldPosition = new THREE.Vector3();
- this.teleportEventDetail = {
- oldPosition: this.rigWorldPosition,
- newPosition: this.newRigWorldPosition,
- hitPoint: this.hitPoint
- };
- this.hit = false;
- this.prevCheckTime = undefined;
- this.prevHitHeight = 0;
- this.referenceNormal = new THREE.Vector3();
- this.curveMissColor = new THREE.Color();
- this.curveHitColor = new THREE.Color();
- this.raycaster = new THREE.Raycaster();
- this.defaultPlane = createDefaultPlane(this.data.defaultPlaneSize);
- this.defaultCollisionMeshes = [this.defaultPlane];
- teleportEntity = this.teleportEntity = document.createElement('a-entity');
- teleportEntity.classList.add('teleportRay');
- teleportEntity.setAttribute('visible', false);
- el.sceneEl.appendChild(this.teleportEntity);
- this.onButtonDown = this.onButtonDown.bind(this);
- this.onButtonUp = this.onButtonUp.bind(this);
- if (this.data.startEvents.length && this.data.endEvents.length) {
- for (i = 0; i < this.data.startEvents.length; i++) {
- el.addEventListener(this.data.startEvents[i], this.onButtonDown);
- }
- for (i = 0; i < this.data.endEvents.length; i++) {
- el.addEventListener(this.data.endEvents[i], this.onButtonUp);
- }
- } else {
- el.addEventListener(data.button + 'down', this.onButtonDown);
- el.addEventListener(data.button + 'up', this.onButtonUp);
- }
- this.queryCollisionEntities();
- },
- update: function (oldData) {
- var data = this.data;
- var diff = AFRAME.utils.diff(data, oldData);
- // Update normal.
- this.referenceNormal.copy(data.landingNormal);
- // Update colors.
- this.curveMissColor.set(data.curveMissColor);
- this.curveHitColor.set(data.curveHitColor);
- // Create or update line mesh.
- if (!this.line ||
- 'curveLineWidth' in diff || 'curveNumberPoints' in diff || 'type' in diff) {
- this.line = createLine(data);
- this.line.material.opacity = this.data.hitOpacity;
- this.line.material.transparent = this.data.hitOpacity < 1;
- this.numActivePoints = data.curveNumberPoints;
- this.teleportEntity.setObject3D('mesh', this.line.mesh);
- }
- // Create or update hit entity.
- if (data.hitEntity) {
- this.hitEntity = data.hitEntity;
- } else if (!this.hitEntity || 'hitCylinderColor' in diff || 'hitCylinderHeight' in diff ||
- 'hitCylinderRadius' in diff) {
- // Remove previous entity, create new entity (could be more performant).
- if (this.hitEntity) { this.hitEntity.parentNode.removeChild(this.hitEntity); }
- this.hitEntity = createHitEntity(data);
- this.el.sceneEl.appendChild(this.hitEntity);
- }
- this.hitEntity.setAttribute('visible', false);
- if ('collisionEntities' in diff) { this.queryCollisionEntities(); }
- },
- remove: function () {
- var el = this.el;
- var hitEntity = this.hitEntity;
- var teleportEntity = this.teleportEntity;
- if (hitEntity) { hitEntity.parentNode.removeChild(hitEntity); }
- if (teleportEntity) { teleportEntity.parentNode.removeChild(teleportEntity); }
- el.sceneEl.removeEventListener('child-attached', this.childAttachHandler);
- el.sceneEl.removeEventListener('child-detached', this.childDetachHandler);
- },
- tick: (function () {
- var p0 = new THREE.Vector3();
- var v0 = new THREE.Vector3();
- var g = -9.8;
- var a = new THREE.Vector3(0, g, 0);
- var next = new THREE.Vector3();
- var last = new THREE.Vector3();
- var quaternion = new THREE.Quaternion();
- var translation = new THREE.Vector3();
- var scale = new THREE.Vector3();
- var shootAngle = new THREE.Vector3();
- var lastNext = new THREE.Vector3();
- var auxDirection = new THREE.Vector3();
- var timeSinceDrawStart = 0;
- return function (time, delta) {
- if (!this.active) { return; }
- if (this.data.drawIncrementally && this.redrawLine){
- this.redrawLine = false;
- timeSinceDrawStart = 0;
- }
- timeSinceDrawStart += delta;
- this.numActivePoints = this.data.curveNumberPoints*timeSinceDrawStart/this.data.incrementalDrawMs;
- if (this.numActivePoints > this.data.curveNumberPoints){
- this.numActivePoints = this.data.curveNumberPoints;
- }
- // Only check for intersection if interval time has passed.
- if (this.prevCheckTime && (time - this.prevCheckTime < this.data.interval)) { return; }
- // Update check time.
- this.prevCheckTime = time;
- var matrixWorld = this.obj.matrixWorld;
- matrixWorld.decompose(translation, quaternion, scale);
- var direction = shootAngle.set(0, 0, -1)
- .applyQuaternion(quaternion).normalize();
- this.line.setDirection(auxDirection.copy(direction));
- this.obj.getWorldPosition(p0);
- last.copy(p0);
- // Set default status as non-hit
- this.teleportEntity.setAttribute('visible', true);
- this.line.material.color.set(this.curveMissColor);
- this.line.material.opacity = this.data.missOpacity;
- this.line.material.transparent = this.data.missOpacity < 1;
- this.hitEntity.setAttribute('visible', false);
- this.hit = false;
- if (this.data.type === 'parabolic') {
- v0.copy(direction).multiplyScalar(this.data.curveShootingSpeed);
- this.lastDrawnIndex = 0;
- const numPoints = this.data.drawIncrementally ? this.numActivePoints : this.line.numPoints;
- for (var i = 0; i < numPoints+1; i++) {
- var t;
- if (i == Math.floor(numPoints+1)){
- t = numPoints / (this.line.numPoints - 1);
- }
- else {
- t = i / (this.line.numPoints - 1);
- }
- parabolicCurve(p0, v0, a, t, next);
- // Update the raycaster with the length of the current segment last->next
- var dirLastNext = lastNext.copy(next).sub(last).normalize();
- this.raycaster.far = dirLastNext.length();
- this.raycaster.set(last, dirLastNext);
- this.lastDrawnPoint = next;
- this.lastDrawnIndex = i;
- if (this.checkMeshCollisions(i, next)) { break; }
- last.copy(next);
- }
- for (var j = this.lastDrawnIndex+1; j < this.line.numPoints; j++) {
- this.line.setPoint(j, this.lastDrawnPoint);
- }
- } else if (this.data.type === 'line') {
- next.copy(last).add(auxDirection.copy(direction).multiplyScalar(this.data.maxLength));
- this.raycaster.far = this.data.maxLength;
- this.raycaster.set(p0, direction);
- this.line.setPoint(0, p0);
- this.checkMeshCollisions(1, next);
- }
- };
- })(),
- /**
- * Run `querySelectorAll` for `collisionEntities` and maintain it with `child-attached`
- * and `child-detached` events.
- */
- queryCollisionEntities: function () {
- var collisionEntities;
- var data = this.data;
- var el = this.el;
- if (!data.collisionEntities) {
- this.collisionEntities = [];
- return;
- }
- collisionEntities = [].slice.call(el.sceneEl.querySelectorAll(data.collisionEntities));
- this.collisionEntities = collisionEntities;
- // Update entity list on attach.
- this.childAttachHandler = function childAttachHandler (evt) {
- if (!evt.detail.el.matches(data.collisionEntities)) { return; }
- collisionEntities.push(evt.detail.el);
- };
- el.sceneEl.addEventListener('child-attached', this.childAttachHandler);
- // Update entity list on detach.
- this.childDetachHandler = function childDetachHandler (evt) {
- var index;
- if (!evt.detail.el.matches(data.collisionEntities)) { return; }
- index = collisionEntities.indexOf(evt.detail.el);
- if (index === -1) { return; }
- collisionEntities.splice(index, 1);
- };
- el.sceneEl.addEventListener('child-detached', this.childDetachHandler);
- },
- onButtonDown: function () {
- this.active = true;
- this.redrawLine = true;
- },
- /**
- * Jump!
- */
- onButtonUp: (function () {
- const teleportOriginWorldPosition = new THREE.Vector3();
- const newRigLocalPosition = new THREE.Vector3();
- const newHandPosition = [new THREE.Vector3(), new THREE.Vector3()]; // Left and right
- const handPosition = new THREE.Vector3();
- return function (evt) {
- if (!this.active) { return; }
- // Hide the hit point and the curve
- this.active = false;
- this.hitEntity.setAttribute('visible', false);
- this.teleportEntity.setAttribute('visible', false);
- if (!this.hit) {
- // Button released but not hit point
- return;
- }
- const rig = this.data.cameraRig || this.el.sceneEl.camera.el;
- rig.object3D.getWorldPosition(this.rigWorldPosition);
- this.newRigWorldPosition.copy(this.hitPoint);
- // If a teleportOrigin exists, offset the rig such that the teleportOrigin is above the hitPoint
- const teleportOrigin = this.data.teleportOrigin;
- if (teleportOrigin) {
- teleportOrigin.object3D.getWorldPosition(teleportOriginWorldPosition);
- this.newRigWorldPosition.sub(teleportOriginWorldPosition).add(this.rigWorldPosition);
- }
- // Always keep the rig at the same offset off the ground after teleporting
- this.newRigWorldPosition.y = this.rigWorldPosition.y + this.hitPoint.y - this.prevHitHeight;
- this.prevHitHeight = this.hitPoint.y;
- // Finally update the rigs position
- newRigLocalPosition.copy(this.newRigWorldPosition);
- if (rig.object3D.parent) {
- rig.object3D.parent.worldToLocal(newRigLocalPosition);
- }
- rig.setAttribute('position', newRigLocalPosition);
- // If a rig was not explicitly declared, look for hands and mvoe them proportionally as well
- if (!this.data.cameraRig) {
- var hands = document.querySelectorAll('a-entity[tracked-controls]');
- for (var i = 0; i < hands.length; i++) {
- hands[i].object3D.getWorldPosition(handPosition);
- // diff = rigWorldPosition - handPosition
- // newPos = newRigWorldPosition - diff
- newHandPosition[i].copy(this.newRigWorldPosition).sub(this.rigWorldPosition).add(handPosition);
- hands[i].setAttribute('position', newHandPosition[i]);
- }
- }
- this.el.emit('teleported', this.teleportEventDetail);
- };
- })(),
- /**
- * Check for raycaster intersection.
- *
- * @param {number} Line fragment point index.
- * @param {number} Next line fragment point index.
- * @returns {boolean} true if there's an intersection.
- */
- checkMeshCollisions: function (i, next) {
- // @todo We should add a property to define if the collisionEntity is dynamic or static
- // If static we should do the map just once, otherwise we're recreating the array in every
- // loop when aiming.
- var meshes;
- if (!this.data.collisionEntities) {
- meshes = this.defaultCollisionMeshes;
- } else {
- meshes = this.collisionEntities.map(function (entity) {
- return entity.getObject3D('mesh');
- }).filter(function (n) { return n; });
- meshes = meshes.length ? meshes : this.defaultCollisionMeshes;
- }
- var intersects = this.raycaster.intersectObjects(meshes, true);
- if (intersects.length > 0 && !this.hit &&
- this.isValidNormalsAngle(intersects[0].face.normal)) {
- var point = intersects[0].point;
- this.line.material.color.set(this.curveHitColor);
- this.line.material.opacity = this.data.hitOpacity;
- this.line.material.transparent= this.data.hitOpacity < 1;
- this.hitEntity.setAttribute('position', point);
- this.hitEntity.setAttribute('visible', true);
- this.hit = true;
- this.hitPoint.copy(intersects[0].point);
- // If hit, just fill the rest of the points with the hit point and break the loop
- for (var j = i; j < this.line.numPoints; j++) {
- this.line.setPoint(j, this.hitPoint);
- }
- return true;
- } else {
- this.line.setPoint(i, next);
- return false;
- }
- },
- isValidNormalsAngle: function (collisionNormal) {
- var angleNormals = this.referenceNormal.angleTo(collisionNormal);
- return (THREE.Math.RAD2DEG * angleNormals <= this.data.landingMaxAngle);
- },
- });
- function createLine (data) {
- var numPoints = data.type === 'line' ? 2 : data.curveNumberPoints;
- return new RayCurve(numPoints, data.curveLineWidth);
- }
- /**
- * Create mesh to represent the area of intersection.
- * Default to a combination of torus and cylinder.
- */
- function createHitEntity (data) {
- var cylinder;
- var hitEntity;
- var torus;
- // Parent.
- hitEntity = document.createElement('a-entity');
- hitEntity.className = 'hitEntity';
- // Torus.
- torus = document.createElement('a-entity');
- torus.setAttribute('geometry', {
- primitive: 'torus',
- radius: data.hitCylinderRadius,
- radiusTubular: 0.01
- });
- torus.setAttribute('rotation', {x: 90, y: 0, z: 0});
- torus.setAttribute('material', {
- shader: 'flat',
- color: data.hitCylinderColor,
- side: 'double',
- depthTest: false
- });
- hitEntity.appendChild(torus);
- // Cylinder.
- cylinder = document.createElement('a-entity');
- cylinder.setAttribute('position', {x: 0, y: data.hitCylinderHeight / 2, z: 0});
- cylinder.setAttribute('geometry', {
- primitive: 'cylinder',
- segmentsHeight: 1,
- radius: data.hitCylinderRadius,
- height: data.hitCylinderHeight,
- openEnded: true
- });
- cylinder.setAttribute('material', {
- shader: 'flat',
- color: data.hitCylinderColor,
- side: 'double',
- src: cylinderTexture,
- transparent: true,
- depthTest: false
- });
- hitEntity.appendChild(cylinder);
- return hitEntity;
- }
- function createDefaultPlane (size) {
- var geometry;
- var material;
- geometry = new THREE.PlaneBufferGeometry(100, 100);
- geometry.rotateX(-Math.PI / 2);
- material = new THREE.MeshBasicMaterial({color: 0xffff00});
- return new THREE.Mesh(geometry, material);
- }
- /***/ }),
- /* 1 */
- /***/ (function(module, exports) {
- module.exports = 'url()';
- /***/ }),
- /* 2 */
- /***/ (function(module, exports) {
- /* global THREE */
- // Parabolic motion equation, y = p0 + v0*t + 1/2at^2
- function parabolicCurveScalar (p0, v0, a, t) {
- return p0 + v0 * t + 0.5 * a * t * t;
- }
- // Parabolic motion equation applied to 3 dimensions
- function parabolicCurve (p0, v0, a, t, out) {
- out.x = parabolicCurveScalar(p0.x, v0.x, a.x, t);
- out.y = parabolicCurveScalar(p0.y, v0.y, a.y, t);
- out.z = parabolicCurveScalar(p0.z, v0.z, a.z, t);
- return out;
- }
- module.exports = parabolicCurve;
- /***/ }),
- /* 3 */
- /***/ (function(module, exports) {
- /* global THREE */
- var RayCurve = function (numPoints, width) {
- this.geometry = new THREE.BufferGeometry();
- this.vertices = new Float32Array(numPoints * 3 * 2);
- this.uvs = new Float32Array(numPoints * 2 * 2);
- this.width = width;
- this.geometry.setAttribute('position', new THREE.BufferAttribute(this.vertices, 3).setUsage(THREE.DynamicDrawUsage));
- this.material = new THREE.MeshBasicMaterial({
- side: THREE.DoubleSide,
- color: 0xff0000
- });
- this.mesh = new THREE.Mesh(this.geometry, this.material);
- THREE.BufferGeometryUtils.toTrianglesDrawMode(this.mesh.geometry, THREE.TriangleStripDrawMode);
- //this.mesh.drawMode = THREE.TriangleStripDrawMode;
- this.mesh.frustumCulled = false;
- this.mesh.vertices = this.vertices;
- this.direction = new THREE.Vector3();
- this.numPoints = numPoints;
- };
- RayCurve.prototype = {
- setDirection: function (direction) {
- var UP = new THREE.Vector3(0, 1, 0);
- this.direction
- .copy(direction)
- .cross(UP)
- .normalize()
- .multiplyScalar(this.width / 2);
- },
- setWidth: function (width) {
- this.width = width;
- },
- setPoint: (function () {
- var posA = new THREE.Vector3();
- var posB = new THREE.Vector3();
- return function (i, point) {
- posA.copy(point).add(this.direction);
- posB.copy(point).sub(this.direction);
- var idx = 2 * 3 * i;
- this.vertices[idx++] = posA.x;
- this.vertices[idx++] = posA.y;
- this.vertices[idx++] = posA.z;
- this.vertices[idx++] = posB.x;
- this.vertices[idx++] = posB.y;
- this.vertices[idx++] = posB.z;
- this.geometry.attributes.position.needsUpdate = true;
- };
- })()
- };
- module.exports = RayCurve;
- /***/ })
- /******/ ]);
|