UNPKG

@laser-tracer/core

Version:

A real-time 3D engine for creating time-decaying volumetric light drawings that mimic phosphor vector monitors.

457 lines 510 kB
import * as a from "three"; import { Ray as K, Plane as q, MathUtils as V, Vector3 as w, Controls as _, MOUSE as Y, TOUCH as D, Quaternion as R, Spherical as C, Vector2 as M } from "three"; const k = { type: "change" }, B = { type: "start" }, J = { type: "end" }, T = new K(), F = new q(), $ = Math.cos(70 * V.DEG2RAD), O = new w(), v = 2 * Math.PI, j = { NONE: -1, ROTATE: 0, DOLLY: 1, PAN: 2, TOUCH_ROTATE: 3, TOUCH_PAN: 4, TOUCH_DOLLY_PAN: 5, TOUCH_DOLLY_ROTATE: 6 }, I = 1e-6; class tt extends _ { /** * Constructs a new controls instance. * * @param {Object3D} object - The object that is managed by the controls. * @param {?HTMLDOMElement} domElement - The HTML element used for event listeners. */ constructor(t, e = null) { super(t, e), this.state = j.NONE, this.target = new w(), this.cursor = new w(), this.minDistance = 0, this.maxDistance = 1 / 0, this.minZoom = 0, this.maxZoom = 1 / 0, this.minTargetRadius = 0, this.maxTargetRadius = 1 / 0, this.minPolarAngle = 0, this.maxPolarAngle = Math.PI, this.minAzimuthAngle = -1 / 0, this.maxAzimuthAngle = 1 / 0, this.enableDamping = !1, this.dampingFactor = 0.05, this.enableZoom = !0, this.zoomSpeed = 1, this.enableRotate = !0, this.rotateSpeed = 1, this.keyRotateSpeed = 1, this.enablePan = !0, this.panSpeed = 1, this.screenSpacePanning = !0, this.keyPanSpeed = 7, this.zoomToCursor = !1, this.autoRotate = !1, this.autoRotateSpeed = 2, this.keys = { LEFT: "ArrowLeft", UP: "ArrowUp", RIGHT: "ArrowRight", BOTTOM: "ArrowDown" }, this.mouseButtons = { LEFT: Y.ROTATE, MIDDLE: Y.DOLLY, RIGHT: Y.PAN }, this.touches = { ONE: D.ROTATE, TWO: D.DOLLY_PAN }, this.target0 = this.target.clone(), this.position0 = this.object.position.clone(), this.zoom0 = this.object.zoom, this._domElementKeyEvents = null, this._lastPosition = new w(), this._lastQuaternion = new R(), this._lastTargetPosition = new w(), this._quat = new R().setFromUnitVectors(t.up, new w(0, 1, 0)), this._quatInverse = this._quat.clone().invert(), this._spherical = new C(), this._sphericalDelta = new C(), this._scale = 1, this._panOffset = new w(), this._rotateStart = new M(), this._rotateEnd = new M(), this._rotateDelta = new M(), this._panStart = new M(), this._panEnd = new M(), this._panDelta = new M(), this._dollyStart = new M(), this._dollyEnd = new M(), this._dollyDelta = new M(), this._dollyDirection = new w(), this._mouse = new M(), this._performCursorZoom = !1, this._pointers = [], this._pointerPositions = {}, this._controlActive = !1, this._onPointerMove = st.bind(this), this._onPointerDown = et.bind(this), this._onPointerUp = it.bind(this), this._onContextMenu = zt.bind(this), this._onMouseWheel = xt.bind(this), this._onKeyDown = ft.bind(this), this._onTouchStart = ot.bind(this), this._onTouchMove = at.bind(this), this._onMouseDown = nt.bind(this), this._onMouseMove = ct.bind(this), this._interceptControlDown = jt.bind(this), this._interceptControlUp = Pt.bind(this), this.domElement !== null && this.connect(this.domElement), this.update(); } connect(t) { super.connect(t), this.domElement.addEventListener("pointerdown", this._onPointerDown), this.domElement.addEventListener("pointercancel", this._onPointerUp), this.domElement.addEventListener("contextmenu", this._onContextMenu), this.domElement.addEventListener("wheel", this._onMouseWheel, { passive: !1 }), this.domElement.getRootNode().addEventListener("keydown", this._interceptControlDown, { passive: !0, capture: !0 }), this.domElement.style.touchAction = "none"; } disconnect() { this.domElement.removeEventListener("pointerdown", this._onPointerDown), this.domElement.removeEventListener("pointermove", this._onPointerMove), this.domElement.removeEventListener("pointerup", this._onPointerUp), this.domElement.removeEventListener("pointercancel", this._onPointerUp), this.domElement.removeEventListener("wheel", this._onMouseWheel), this.domElement.removeEventListener("contextmenu", this._onContextMenu), this.stopListenToKeyEvents(), this.domElement.getRootNode().removeEventListener("keydown", this._interceptControlDown, { capture: !0 }), this.domElement.style.touchAction = "auto"; } dispose() { this.disconnect(); } /** * Get the current vertical rotation, in radians. * * @return {number} The current vertical rotation, in radians. */ getPolarAngle() { return this._spherical.phi; } /** * Get the current horizontal rotation, in radians. * * @return {number} The current horizontal rotation, in radians. */ getAzimuthalAngle() { return this._spherical.theta; } /** * Returns the distance from the camera to the target. * * @return {number} The distance from the camera to the target. */ getDistance() { return this.object.position.distanceTo(this.target); } /** * Adds key event listeners to the given DOM element. * `window` is a recommended argument for using this method. * * @param {HTMLDOMElement} domElement - The DOM element */ listenToKeyEvents(t) { t.addEventListener("keydown", this._onKeyDown), this._domElementKeyEvents = t; } /** * Removes the key event listener previously defined with `listenToKeyEvents()`. */ stopListenToKeyEvents() { this._domElementKeyEvents !== null && (this._domElementKeyEvents.removeEventListener("keydown", this._onKeyDown), this._domElementKeyEvents = null); } /** * Save the current state of the controls. This can later be recovered with `reset()`. */ saveState() { this.target0.copy(this.target), this.position0.copy(this.object.position), this.zoom0 = this.object.zoom; } /** * Reset the controls to their state from either the last time the `saveState()` * was called, or the initial state. */ reset() { this.target.copy(this.target0), this.object.position.copy(this.position0), this.object.zoom = this.zoom0, this.object.updateProjectionMatrix(), this.dispatchEvent(k), this.update(), this.state = j.NONE; } update(t = null) { const e = this.object.position; O.copy(e).sub(this.target), O.applyQuaternion(this._quat), this._spherical.setFromVector3(O), this.autoRotate && this.state === j.NONE && this._rotateLeft(this._getAutoRotationAngle(t)), this.enableDamping ? (this._spherical.theta += this._sphericalDelta.theta * this.dampingFactor, this._spherical.phi += this._sphericalDelta.phi * this.dampingFactor) : (this._spherical.theta += this._sphericalDelta.theta, this._spherical.phi += this._sphericalDelta.phi); let n = this.minAzimuthAngle, i = this.maxAzimuthAngle; isFinite(n) && isFinite(i) && (n < -Math.PI ? n += v : n > Math.PI && (n -= v), i < -Math.PI ? i += v : i > Math.PI && (i -= v), n <= i ? this._spherical.theta = Math.max(n, Math.min(i, this._spherical.theta)) : this._spherical.theta = this._spherical.theta > (n + i) / 2 ? Math.max(n, this._spherical.theta) : Math.min(i, this._spherical.theta)), this._spherical.phi = Math.max(this.minPolarAngle, Math.min(this.maxPolarAngle, this._spherical.phi)), this._spherical.makeSafe(), this.enableDamping === !0 ? this.target.addScaledVector(this._panOffset, this.dampingFactor) : this.target.add(this._panOffset), this.target.sub(this.cursor), this.target.clampLength(this.minTargetRadius, this.maxTargetRadius), this.target.add(this.cursor); let c = !1; if (this.zoomToCursor && this._performCursorZoom || this.object.isOrthographicCamera) this._spherical.radius = this._clampDistance(this._spherical.radius); else { const x = this._spherical.radius; this._spherical.radius = this._clampDistance(this._spherical.radius * this._scale), c = x != this._spherical.radius; } if (O.setFromSpherical(this._spherical), O.applyQuaternion(this._quatInverse), e.copy(this.target).add(O), this.object.lookAt(this.target), this.enableDamping === !0 ? (this._sphericalDelta.theta *= 1 - this.dampingFactor, this._sphericalDelta.phi *= 1 - this.dampingFactor, this._panOffset.multiplyScalar(1 - this.dampingFactor)) : (this._sphericalDelta.set(0, 0, 0), this._panOffset.set(0, 0, 0)), this.zoomToCursor && this._performCursorZoom) { let x = null; if (this.object.isPerspectiveCamera) { const f = O.length(); x = this._clampDistance(f * this._scale); const o = f - x; this.object.position.addScaledVector(this._dollyDirection, o), this.object.updateMatrixWorld(), c = !!o; } else if (this.object.isOrthographicCamera) { const f = new w(this._mouse.x, this._mouse.y, 0); f.unproject(this.object); const o = this.object.zoom; this.object.zoom = Math.max(this.minZoom, Math.min(this.maxZoom, this.object.zoom / this._scale)), this.object.updateProjectionMatrix(), c = o !== this.object.zoom; const z = new w(this._mouse.x, this._mouse.y, 0); z.unproject(this.object), this.object.position.sub(z).add(f), this.object.updateMatrixWorld(), x = O.length(); } else console.warn("WARNING: OrbitControls.js encountered an unknown camera type - zoom to cursor disabled."), this.zoomToCursor = !1; x !== null && (this.screenSpacePanning ? this.target.set(0, 0, -1).transformDirection(this.object.matrix).multiplyScalar(x).add(this.object.position) : (T.origin.copy(this.object.position), T.direction.set(0, 0, -1).transformDirection(this.object.matrix), Math.abs(this.object.up.dot(T.direction)) < $ ? this.object.lookAt(this.target) : (F.setFromNormalAndCoplanarPoint(this.object.up, this.target), T.intersectPlane(F, this.target)))); } else if (this.object.isOrthographicCamera) { const x = this.object.zoom; this.object.zoom = Math.max(this.minZoom, Math.min(this.maxZoom, this.object.zoom / this._scale)), x !== this.object.zoom && (this.object.updateProjectionMatrix(), c = !0); } return this._scale = 1, this._performCursorZoom = !1, c || this._lastPosition.distanceToSquared(this.object.position) > I || 8 * (1 - this._lastQuaternion.dot(this.object.quaternion)) > I || this._lastTargetPosition.distanceToSquared(this.target) > I ? (this.dispatchEvent(k), this._lastPosition.copy(this.object.position), this._lastQuaternion.copy(this.object.quaternion), this._lastTargetPosition.copy(this.target), !0) : !1; } _getAutoRotationAngle(t) { return t !== null ? v / 60 * this.autoRotateSpeed * t : v / 60 / 60 * this.autoRotateSpeed; } _getZoomScale(t) { const e = Math.abs(t * 0.01); return Math.pow(0.95, this.zoomSpeed * e); } _rotateLeft(t) { this._sphericalDelta.theta -= t; } _rotateUp(t) { this._sphericalDelta.phi -= t; } _panLeft(t, e) { O.setFromMatrixColumn(e, 0), O.multiplyScalar(-t), this._panOffset.add(O); } _panUp(t, e) { this.screenSpacePanning === !0 ? O.setFromMatrixColumn(e, 1) : (O.setFromMatrixColumn(e, 0), O.crossVectors(this.object.up, O)), O.multiplyScalar(t), this._panOffset.add(O); } // deltaX and deltaY are in pixels; right and down are positive _pan(t, e) { const n = this.domElement; if (this.object.isPerspectiveCamera) { const i = this.object.position; O.copy(i).sub(this.target); let c = O.length(); c *= Math.tan(this.object.fov / 2 * Math.PI / 180), this._panLeft(2 * t * c / n.clientHeight, this.object.matrix), this._panUp(2 * e * c / n.clientHeight, this.object.matrix); } else this.object.isOrthographicCamera ? (this._panLeft(t * (this.object.right - this.object.left) / this.object.zoom / n.clientWidth, this.object.matrix), this._panUp(e * (this.object.top - this.object.bottom) / this.object.zoom / n.clientHeight, this.object.matrix)) : (console.warn("WARNING: OrbitControls.js encountered an unknown camera type - pan disabled."), this.enablePan = !1); } _dollyOut(t) { this.object.isPerspectiveCamera || this.object.isOrthographicCamera ? this._scale /= t : (console.warn("WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled."), this.enableZoom = !1); } _dollyIn(t) { this.object.isPerspectiveCamera || this.object.isOrthographicCamera ? this._scale *= t : (console.warn("WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled."), this.enableZoom = !1); } _updateZoomParameters(t, e) { if (!this.zoomToCursor) return; this._performCursorZoom = !0; const n = this.domElement.getBoundingClientRect(), i = t - n.left, c = e - n.top, x = n.width, f = n.height; this._mouse.x = i / x * 2 - 1, this._mouse.y = -(c / f) * 2 + 1, this._dollyDirection.set(this._mouse.x, this._mouse.y, 1).unproject(this.object).sub(this.object.position).normalize(); } _clampDistance(t) { return Math.max(this.minDistance, Math.min(this.maxDistance, t)); } // // event callbacks - update the object state // _handleMouseDownRotate(t) { this._rotateStart.set(t.clientX, t.clientY); } _handleMouseDownDolly(t) { this._updateZoomParameters(t.clientX, t.clientX), this._dollyStart.set(t.clientX, t.clientY); } _handleMouseDownPan(t) { this._panStart.set(t.clientX, t.clientY); } _handleMouseMoveRotate(t) { this._rotateEnd.set(t.clientX, t.clientY), this._rotateDelta.subVectors(this._rotateEnd, this._rotateStart).multiplyScalar(this.rotateSpeed); const e = this.domElement; this._rotateLeft(v * this._rotateDelta.x / e.clientHeight), this._rotateUp(v * this._rotateDelta.y / e.clientHeight), this._rotateStart.copy(this._rotateEnd), this.update(); } _handleMouseMoveDolly(t) { this._dollyEnd.set(t.clientX, t.clientY), this._dollyDelta.subVectors(this._dollyEnd, this._dollyStart), this._dollyDelta.y > 0 ? this._dollyOut(this._getZoomScale(this._dollyDelta.y)) : this._dollyDelta.y < 0 && this._dollyIn(this._getZoomScale(this._dollyDelta.y)), this._dollyStart.copy(this._dollyEnd), this.update(); } _handleMouseMovePan(t) { this._panEnd.set(t.clientX, t.clientY), this._panDelta.subVectors(this._panEnd, this._panStart).multiplyScalar(this.panSpeed), this._pan(this._panDelta.x, this._panDelta.y), this._panStart.copy(this._panEnd), this.update(); } _handleMouseWheel(t) { this._updateZoomParameters(t.clientX, t.clientY), t.deltaY < 0 ? this._dollyIn(this._getZoomScale(t.deltaY)) : t.deltaY > 0 && this._dollyOut(this._getZoomScale(t.deltaY)), this.update(); } _handleKeyDown(t) { let e = !1; switch (t.code) { case this.keys.UP: t.ctrlKey || t.metaKey || t.shiftKey ? this.enableRotate && this._rotateUp(v * this.keyRotateSpeed / this.domElement.clientHeight) : this.enablePan && this._pan(0, this.keyPanSpeed), e = !0; break; case this.keys.BOTTOM: t.ctrlKey || t.metaKey || t.shiftKey ? this.enableRotate && this._rotateUp(-v * this.keyRotateSpeed / this.domElement.clientHeight) : this.enablePan && this._pan(0, -this.keyPanSpeed), e = !0; break; case this.keys.LEFT: t.ctrlKey || t.metaKey || t.shiftKey ? this.enableRotate && this._rotateLeft(v * this.keyRotateSpeed / this.domElement.clientHeight) : this.enablePan && this._pan(this.keyPanSpeed, 0), e = !0; break; case this.keys.RIGHT: t.ctrlKey || t.metaKey || t.shiftKey ? this.enableRotate && this._rotateLeft(-v * this.keyRotateSpeed / this.domElement.clientHeight) : this.enablePan && this._pan(-this.keyPanSpeed, 0), e = !0; break; } e && (t.preventDefault(), this.update()); } _handleTouchStartRotate(t) { if (this._pointers.length === 1) this._rotateStart.set(t.pageX, t.pageY); else { const e = this._getSecondPointerPosition(t), n = 0.5 * (t.pageX + e.x), i = 0.5 * (t.pageY + e.y); this._rotateStart.set(n, i); } } _handleTouchStartPan(t) { if (this._pointers.length === 1) this._panStart.set(t.pageX, t.pageY); else { const e = this._getSecondPointerPosition(t), n = 0.5 * (t.pageX + e.x), i = 0.5 * (t.pageY + e.y); this._panStart.set(n, i); } } _handleTouchStartDolly(t) { const e = this._getSecondPointerPosition(t), n = t.pageX - e.x, i = t.pageY - e.y, c = Math.sqrt(n * n + i * i); this._dollyStart.set(0, c); } _handleTouchStartDollyPan(t) { this.enableZoom && this._handleTouchStartDolly(t), this.enablePan && this._handleTouchStartPan(t); } _handleTouchStartDollyRotate(t) { this.enableZoom && this._handleTouchStartDolly(t), this.enableRotate && this._handleTouchStartRotate(t); } _handleTouchMoveRotate(t) { if (this._pointers.length == 1) this._rotateEnd.set(t.pageX, t.pageY); else { const n = this._getSecondPointerPosition(t), i = 0.5 * (t.pageX + n.x), c = 0.5 * (t.pageY + n.y); this._rotateEnd.set(i, c); } this._rotateDelta.subVectors(this._rotateEnd, this._rotateStart).multiplyScalar(this.rotateSpeed); const e = this.domElement; this._rotateLeft(v * this._rotateDelta.x / e.clientHeight), this._rotateUp(v * this._rotateDelta.y / e.clientHeight), this._rotateStart.copy(this._rotateEnd); } _handleTouchMovePan(t) { if (this._pointers.length === 1) this._panEnd.set(t.pageX, t.pageY); else { const e = this._getSecondPointerPosition(t), n = 0.5 * (t.pageX + e.x), i = 0.5 * (t.pageY + e.y); this._panEnd.set(n, i); } this._panDelta.subVectors(this._panEnd, this._panStart).multiplyScalar(this.panSpeed), this._pan(this._panDelta.x, this._panDelta.y), this._panStart.copy(this._panEnd); } _handleTouchMoveDolly(t) { const e = this._getSecondPointerPosition(t), n = t.pageX - e.x, i = t.pageY - e.y, c = Math.sqrt(n * n + i * i); this._dollyEnd.set(0, c), this._dollyDelta.set(0, Math.pow(this._dollyEnd.y / this._dollyStart.y, this.zoomSpeed)), this._dollyOut(this._dollyDelta.y), this._dollyStart.copy(this._dollyEnd); const x = (t.pageX + e.x) * 0.5, f = (t.pageY + e.y) * 0.5; this._updateZoomParameters(x, f); } _handleTouchMoveDollyPan(t) { this.enableZoom && this._handleTouchMoveDolly(t), this.enablePan && this._handleTouchMovePan(t); } _handleTouchMoveDollyRotate(t) { this.enableZoom && this._handleTouchMoveDolly(t), this.enableRotate && this._handleTouchMoveRotate(t); } // pointers _addPointer(t) { this._pointers.push(t.pointerId); } _removePointer(t) { delete this._pointerPositions[t.pointerId]; for (let e = 0; e < this._pointers.length; e++) if (this._pointers[e] == t.pointerId) { this._pointers.splice(e, 1); return; } } _isTrackingPointer(t) { for (let e = 0; e < this._pointers.length; e++) if (this._pointers[e] == t.pointerId) return !0; return !1; } _trackPointer(t) { let e = this._pointerPositions[t.pointerId]; e === void 0 && (e = new M(), this._pointerPositions[t.pointerId] = e), e.set(t.pageX, t.pageY); } _getSecondPointerPosition(t) { const e = t.pointerId === this._pointers[0] ? this._pointers[1] : this._pointers[0]; return this._pointerPositions[e]; } // _customWheelEvent(t) { const e = t.deltaMode, n = { clientX: t.clientX, clientY: t.clientY, deltaY: t.deltaY }; switch (e) { case 1: n.deltaY *= 16; break; case 2: n.deltaY *= 100; break; } return t.ctrlKey && !this._controlActive && (n.deltaY *= 10), n; } } function et(s) { this.enabled !== !1 && (this._pointers.length === 0 && (this.domElement.setPointerCapture(s.pointerId), this.domElement.addEventListener("pointermove", this._onPointerMove), this.domElement.addEventListener("pointerup", this._onPointerUp)), !this._isTrackingPointer(s) && (this._addPointer(s), s.pointerType === "touch" ? this._onTouchStart(s) : this._onMouseDown(s))); } function st(s) { this.enabled !== !1 && (s.pointerType === "touch" ? this._onTouchMove(s) : this._onMouseMove(s)); } function it(s) { switch (this._removePointer(s), this._pointers.length) { case 0: this.domElement.releasePointerCapture(s.pointerId), this.domElement.removeEventListener("pointermove", this._onPointerMove), this.domElement.removeEventListener("pointerup", this._onPointerUp), this.dispatchEvent(J), this.state = j.NONE; break; case 1: const t = this._pointers[0], e = this._pointerPositions[t]; this._onTouchStart({ pointerId: t, pageX: e.x, pageY: e.y }); break; } } function nt(s) { let t; switch (s.button) { case 0: t = this.mouseButtons.LEFT; break; case 1: t = this.mouseButtons.MIDDLE; break; case 2: t = this.mouseButtons.RIGHT; break; default: t = -1; } switch (t) { case Y.DOLLY: if (this.enableZoom === !1) return; this._handleMouseDownDolly(s), this.state = j.DOLLY; break; case Y.ROTATE: if (s.ctrlKey || s.metaKey || s.shiftKey) { if (this.enablePan === !1) return; this._handleMouseDownPan(s), this.state = j.PAN; } else { if (this.enableRotate === !1) return; this._handleMouseDownRotate(s), this.state = j.ROTATE; } break; case Y.PAN: if (s.ctrlKey || s.metaKey || s.shiftKey) { if (this.enableRotate === !1) return; this._handleMouseDownRotate(s), this.state = j.ROTATE; } else { if (this.enablePan === !1) return; this._handleMouseDownPan(s), this.state = j.PAN; } break; default: this.state = j.NONE; } this.state !== j.NONE && this.dispatchEvent(B); } function ct(s) { switch (this.state) { case j.ROTATE: if (this.enableRotate === !1) return; this._handleMouseMoveRotate(s); break; case j.DOLLY: if (this.enableZoom === !1) return; this._handleMouseMoveDolly(s); break; case j.PAN: if (this.enablePan === !1) return; this._handleMouseMovePan(s); break; } } function xt(s) { this.enabled === !1 || this.enableZoom === !1 || this.state !== j.NONE || (s.preventDefault(), this.dispatchEvent(B), this._handleMouseWheel(this._customWheelEvent(s)), this.dispatchEvent(J)); } function ft(s) { this.enabled !== !1 && this._handleKeyDown(s); } function ot(s) { switch (this._trackPointer(s), this._pointers.length) { case 1: switch (this.touches.ONE) { case D.ROTATE: if (this.enableRotate === !1) return; this._handleTouchStartRotate(s), this.state = j.TOUCH_ROTATE; break; case D.PAN: if (this.enablePan === !1) return; this._handleTouchStartPan(s), this.state = j.TOUCH_PAN; break; default: this.state = j.NONE; } break; case 2: switch (this.touches.TWO) { case D.DOLLY_PAN: if (this.enableZoom === !1 && this.enablePan === !1) return; this._handleTouchStartDollyPan(s), this.state = j.TOUCH_DOLLY_PAN; break; case D.DOLLY_ROTATE: if (this.enableZoom === !1 && this.enableRotate === !1) return; this._handleTouchStartDollyRotate(s), this.state = j.TOUCH_DOLLY_ROTATE; break; default: this.state = j.NONE; } break; default: this.state = j.NONE; } this.state !== j.NONE && this.dispatchEvent(B); } function at(s) { switch (this._trackPointer(s), this.state) { case j.TOUCH_ROTATE: if (this.enableRotate === !1) return; this._handleTouchMoveRotate(s), this.update(); break; case j.TOUCH_PAN: if (this.enablePan === !1) return; this._handleTouchMovePan(s), this.update(); break; case j.TOUCH_DOLLY_PAN: if (this.enableZoom === !1 && this.enablePan === !1) return; this._handleTouchMoveDollyPan(s), this.update(); break; case j.TOUCH_DOLLY_ROTATE: if (this.enableZoom === !1 && this.enableRotate === !1) return; this._handleTouchMoveDollyRotate(s), this.update(); break; default: this.state = j.NONE; } } function zt(s) { this.enabled !== !1 && s.preventDefault(); } function jt(s) { s.key === "Control" && (this._controlActive = !0, this.domElement.getRootNode().addEventListener("keyup", this._interceptControlUp, { passive: !0, capture: !0 })); } function Pt(s) { s.key === "Control" && (this._controlActive = !1, this.domElement.getRootNode().removeEventListener("keyup", this._interceptControlUp, { passive: !0, capture: !0 })); } const ht = "", rt = "