mage-engine 3.25.5 → 3.25.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/mage.js +989 -29
- package/package.json +1 -1
- package/.claude/worktrees/fix-onstart-error-handling/.github/workflows/test.yml +0 -29
- package/.claude/worktrees/fix-onstart-error-handling/.nvmrc +0 -1
- package/.claude/worktrees/fix-onstart-error-handling/.prettierrc.js +0 -20
- package/.claude/worktrees/fix-onstart-error-handling/LICENSE +0 -35
- package/.claude/worktrees/fix-onstart-error-handling/README.md +0 -56
- package/.claude/worktrees/fix-onstart-error-handling/__mocks__/three.js +0 -178
- package/.claude/worktrees/fix-onstart-error-handling/dist/ammo.js +0 -991
- package/.claude/worktrees/fix-onstart-error-handling/eslint.config.js +0 -73
- package/.claude/worktrees/fix-onstart-error-handling/jest.config.js +0 -31
- package/.claude/worktrees/fix-onstart-error-handling/package.json +0 -96
package/dist/mage.js
CHANGED
|
@@ -55720,7 +55720,7 @@ function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
|
|
|
55720
55720
|
url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
|
|
55721
55721
|
return new Worker(url, options);
|
|
55722
55722
|
};
|
|
55723
|
-
}var WorkerFactory = createBase64WorkerFactory('/* rollup-plugin-web-worker-loader */
(function () {
    'use strict';

    const LIBRARY_NAME = "ammo.js";
    const TYPES = {
      BOX: "BOX",
      SPHERE: "SPHERE",
      VEHICLE: "VEHICLE",
      MESH: "MESH",
      PLAYER: "PLAYER"
    };
    const DEFAULT_VEHICLE_STATE = {
      vehicleSteering: 0,
      acceleration: false,
      breaking: false,
      right: false,
      left: false
    };
    const DEFAULT_RIGIDBODY_STATE = {
      velocity: {
        x: 0,
        y: 0,
        z: 0
      },
      movement: {
        forward: false,
        backwards: false,
        left: false,
        right: false
      },
      direction: {
        x: 0,
        y: 0,
        z: 0
      }
    };
    const DEFAULT_SCALE = {
      x: 1,
      y: 1,
      z: 1
    };
    const DEFAULT_LINEAR_VELOCITY = {
      x: 0,
      y: 0,
      z: 0
    };
    const DEFAULT_IMPULSE = {
      x: 0,
      y: 0,
      z: 0
    };
    const DISABLE_DEACTIVATION = 4;
    const GRAVITY = {
      x: 0,
      y: -9.8,
      z: 0
    };
    const FRONT_LEFT = 0;
    const FRONT_RIGHT = 1;
    const BACK_LEFT = 2;
    const BACK_RIGHT = 3;
    const DEFAULT_STEERING_INCREMENT = 0.04;
    const DEFAULT_STEERING_CLAMP = 0.5;
    const DEFAULT_MAX_ENGINE_FORCE = 2000;
    const DEFAULT_MAX_BREAKING_FORCE = 100;
    const EXPLOSION_SIZES = {
      SMALL: 4,
      MEDIUM: 6,
      LARGE: 8,
      MASSIVE: 12
    };
    const EXPLOSION_STRENGTHS = {
      VERY_WEAK: 2,
      WEAK: 4,
      MEDIUM: 8,
      LARGE: 16,
      MASSIVE: 32,
      OK_NO: 64
    };

    function _classCallCheck(a, n) {
      if (!(a instanceof n)) throw new TypeError("Cannot call a class as a function");
    }
    function _defineProperties(e, r) {
      for (var t = 0; t < r.length; t++) {
        var o = r[t];
        o.enumerable = o.enumerable || !1, o.configurable = !0, "value" in o && (o.writable = !0), Object.defineProperty(e, _toPropertyKey(o.key), o);
      }
    }
    function _createClass(e, r, t) {
      return r && _defineProperties(e.prototype, r), t && _defineProperties(e, t), Object.defineProperty(e, "prototype", {
        writable: !1
      }), e;
    }
    function _defineProperty(e, r, t) {
      return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, {
        value: t,
        enumerable: !0,
        configurable: !0,
        writable: !0
      }) : e[r] = t, e;
    }
    function _toPrimitive(t, r) {
      if ("object" != typeof t || !t) return t;
      var e = t[Symbol.toPrimitive];
      if (void 0 !== e) {
        var i = e.call(t, r || "default");
        if ("object" != typeof i) return i;
        throw new TypeError("@@toPrimitive must return a primitive value.");
      }
      return ("string" === r ? String : Number)(t);
    }
    function _toPropertyKey(t) {
      var i = _toPrimitive(t, "string");
      return "symbol" == typeof i ? i : i + "";
    }

    const PHYSICS_EVENTS = {
      DISPATCH: "physics:dispatch",
      TERMINATE: "physics:terminate",
      LOAD: {
        AMMO: "physics:load:ammo"
      },
      READY: "physics:ready",
      INIT: "physics:init",
      UPDATE: "physics:update",
      ADD: {
        BOX: "physics:add:box",
        VEHICLE: "physics:add:vehicle",
        MODEL: "physics:add:model",
        PLAYER: "physics:add:player",
        SPHERE: "physics:add:sphere"
      },
      ELEMENT: {
        DISPOSE: "physics:element:dispose",
        COLLISION: "physics:element:collision",
        UPDATE: "physics:element:update",
        CREATED: "physics:element:created",
        SET: {
          POSITION: "physics:element:set:position",
          QUATERNION: "physics:element:set:quaternion",
          LINEAR_VELOCITY: "physics:element:set:linear_velocity"
        },
        RESET: "physics:element:reset",
        APPLY: {
          IMPULSE: "physics:element:apply:impulse"
        }
      },
      VEHICLE: {
        SET: {
          POSITION: "physics:vehicle:set:position",
          QUATERNION: "physics:vehicle:set:quaternion"
        },
        RESET: "physics:vehicle:reset",
        SPEED: "physics:vehicle:speed",
        DIRECTION: "physics:vehicle:direction"
      },
      EFFECTS: {
        EXPLOSION: "physics:effects:explosion"
      }
    };

    let Dispatcher = /*#__PURE__*/_createClass(function Dispatcher() {
      _classCallCheck(this, Dispatcher);
      _defineProperty(this, "sendPhysicsUpdate", dt => postMessage({
        event: PHYSICS_EVENTS.UPDATE,
        dt
      }));
      _defineProperty(this, "sendReadyEvent", () => postMessage({
        event: PHYSICS_EVENTS.READY
      }));
      _defineProperty(this, "sendTerminateEvent", () => postMessage({
        event: PHYSICS_EVENTS.TERMINATE
      }));
      _defineProperty(this, "sendBodyUpdate", (uuid, position, rotation, dt, extraData) => postMessage({
        event: PHYSICS_EVENTS.ELEMENT.UPDATE,
        uuid,
        position: {
          x: position.x(),
          y: position.y(),
          z: position.z()
        },
        quaternion: {
          x: rotation.x(),
          y: rotation.y(),
          z: rotation.z(),
          w: rotation.w()
        },
        ...extraData,
        dt
      }));
      _defineProperty(this, "sendDispatchEvent", (uuid, eventName, eventData) => postMessage({
        event: PHYSICS_EVENTS.DISPATCH,
        uuid,
        eventName,
        eventData
      }));
    });
    var dispatcher = new Dispatcher();

    const applyMatrix4ToVector3 = ({
      x = 0,
      y = 0,
      z = 0
    }, matrix = []) => {
      const w = 1 / (matrix[3] * x + matrix[7] * y + matrix[11] * z + matrix[15]);
      return {
        x: (matrix[0] * x + matrix[4] * y + matrix[8] * z + matrix[12]) * w,
        y: (matrix[1] * x + matrix[5] * y + matrix[9] * z + matrix[13]) * w,
        z: (matrix[2] * x + matrix[6] * y + matrix[10] * z + matrix[14]) * w
      };
    };

    const createRigidBody = (shape, options) => {
      const {
        uuid,
        position,
        quaternion,
        mass = 0,
        friction,
        restitution = 0.9,
        damping = {
          linear: 0.2,
          angular: 0.2
        }
      } = options;
      const transform = new Ammo.btTransform();
      transform.setIdentity();
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      const motionState = new Ammo.btDefaultMotionState(transform);
      const localInertia = new Ammo.btVector3(0, 0, 0);
      shape.calculateLocalInertia(mass, localInertia);
      const rbInfo = new Ammo.btRigidBodyConstructionInfo(mass, motionState, shape, localInertia);
      const body = new Ammo.btRigidBody(rbInfo);
      if (mass > 0) {
        body.setFriction(friction);
        body.setRestitution(restitution);
        body.setDamping(damping.linear, damping.angular);
        body.setActivationState(DISABLE_DEACTIVATION);
      }

      // storing uuid for future reference
      body.uuid = uuid;
      world.addRigidBody(body);
      return body;
    };
    const addModel = options => {
      const {
        uuid,
        vertices,
        matrices,
        indexes,
        position,
        quaternion,
        mass = 0,
        friction = 2
      } = options;
      const scale = DEFAULT_SCALE;
      const bta = new Ammo.btVector3();
      const btb = new Ammo.btVector3();
      const btc = new Ammo.btVector3();
      const triMesh = new Ammo.btTriangleMesh(true, false);
      for (let i = 0; i < vertices.length; i++) {
        const components = vertices[i];
        const index = indexes[i] ? indexes[i] : null;
        const matrix = Array.from(matrices[i]);
        if (index) {
          for (let j = 0; j < index.length; j += 3) {
            const ai = index[j] * 3;
            const bi = index[j + 1] * 3;
            const ci = index[j + 2] * 3;
            const va = applyMatrix4ToVector3({
              x: components[ai],
              y: components[ai + 1],
              z: components[ai + 2]
            }, matrix);
            const vb = applyMatrix4ToVector3({
              x: components[bi],
              y: components[bi + 1],
              z: components[bi + 2]
            }, matrix);
            const vc = applyMatrix4ToVector3({
              x: components[ci],
              y: components[ci + 1],
              z: components[ci + 2]
            }, matrix);
            bta.setValue(va.x, va.y, va.z);
            btb.setValue(vb.x, vb.y, vb.z);
            btc.setValue(vc.x, vc.y, vc.z);
            triMesh.addTriangle(bta, btb, btc, false);
          }
        } else {
          for (let j = 0; j < components.length; j += 9) {
            const va = applyMatrix4ToVector3({
              x: components[j + 0],
              y: components[j + 1],
              z: components[j + 2]
            }, matrix);
            const vb = applyMatrix4ToVector3({
              x: components[j + 3],
              y: components[j + 4],
              z: components[j + 5]
            }, matrix);
            const vc = applyMatrix4ToVector3({
              x: components[j + 6],
              y: components[j + 7],
              z: components[j + 8]
            }, matrix);
            bta.setValue(va.x, va.y, va.z);
            btb.setValue(vb.x, vb.y, vb.z);
            btc.setValue(vc.x, vc.y, vc.z);
            triMesh.addTriangle(bta, btb, btc, false);
          }
        }
      }
      const localScale = new Ammo.btVector3(scale.x, scale.y, scale.z);
      triMesh.setScaling(localScale);
      Ammo.destroy(localScale);
      const collisionShape = new Ammo.btBvhTriangleMeshShape(triMesh, true, true);
      collisionShape.resources = [triMesh];
      Ammo.destroy(bta);
      Ammo.destroy(btb);
      Ammo.destroy(btc);
      const body = createRigidBody(collisionShape, {
        uuid,
        position,
        quaternion,
        mass,
        friction
      });
      world.addElement({
        uuid,
        body,
        type: TYPES.MESH,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const addBox = data => {
      const {
        uuid,
        width,
        length,
        height,
        position,
        quaternion,
        mass = 0,
        friction = 2
      } = data;
      const geometry = new Ammo.btBoxShape(new Ammo.btVector3(width * 0.5, height * 0.5, length * 0.5));
      const body = createRigidBody(geometry, {
        uuid,
        position,
        quaternion,
        mass,
        friction
      });
      world.addElement({
        uuid,
        body,
        type: TYPES.BOX,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const addSphere = data => {
      const {
        uuid,
        radius,
        position,
        quaternion,
        mass = 0,
        friction = 2
      } = data;
      const geometry = new Ammo.btSphereShape(radius);
      const body = createRigidBody(geometry, {
        uuid,
        position,
        quaternion,
        mass,
        friction
      });
      world.addElement({
        uuid,
        body,
        type: TYPES.SPHERE,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const setLinearVelocity = data => {
      const {
        uuid,
        velocity = DEFAULT_LINEAR_VELOCITY
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const motionState = body.getMotionState();
      if (motionState) {
        const linearVelocity = new Ammo.btVector3(velocity.x, velocity.y, velocity.z);
        body.setLinearVelocity(linearVelocity);
        Ammo.destroy(linearVelocity);
      }
    };
    const setPosition = data => {
      const {
        uuid,
        position
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const transform = new Ammo.btTransform();
      body.getWorldTransform(transform);
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      body.setWorldTransform(transform);
      // Also update motion state so static bodies (mass=0) reflect the change
      const motionState = body.getMotionState();
      if (motionState) {
        motionState.setWorldTransform(transform);
      }
    };
    const resetElement = data => {
      const {
        uuid,
        position,
        quaternion
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const transform = new Ammo.btTransform();
      body.getWorldTransform(transform);
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      body.setWorldTransform(transform);
      // Also update motion state so static bodies (mass=0) reflect the change
      const motionState = body.getMotionState();
      if (motionState) {
        motionState.setWorldTransform(transform);
      }
    };
    const applyImpuse = ({
      uuid,
      impulse = DEFAULT_IMPULSE
    }) => {
      try {
        const element = world.getElement(uuid);
        if (!element) {
          console.warn("[Physics Worker] applyImpulse: element not found for uuid:", uuid);
          return;
        }
        const {
          body
        } = element;
        if (!body) {
          console.warn("[Physics Worker] applyImpulse: body is null for uuid:", uuid);
          return;
        }
        body.activate(true);
        const btImpulse = new Ammo.btVector3(impulse.x, impulse.y, impulse.z);
        body.applyCentralImpulse(btImpulse);
        Ammo.destroy(btImpulse);
      } catch (e) {
        console.error("[Physics Worker] applyImpulse error:", e);
      }
    };
    const setQuaternion = data => {
      const {
        uuid,
        quaternion
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const transform = new Ammo.btTransform();
      body.getWorldTransform(transform);
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      body.setWorldTransform(transform);
      // Also update motion state so static bodies (mass=0) reflect the change
      const motionState = body.getMotionState();
      if (motionState) {
        motionState.setWorldTransform(transform);
      }
    };
    const handleElementUpdate = ({
      body,
      uuid,
      state: _state = DEFAULT_RIGIDBODY_STATE
    }, dt) => {
      // Static bodies (mass=0) never move — skip sending updates so the
      // visual position stays exactly where the author placed it.
      if (body.isStaticObject()) return;
      const motionState = body.getMotionState();
      if (motionState) {
        const transform = new Ammo.btTransform();
        motionState.getWorldTransform(transform);
        let origin = transform.getOrigin();
        let rotation = transform.getRotation();
        dispatcher.sendBodyUpdate(uuid, origin, rotation, dt);
        Ammo.destroy(transform);
      }
    };

    const addPlayer = data => {
      const {
        uuid,
        width,
        height,
        position,
        mass,
        friction,
        originOffset = {
          x: 0,
          y: 0,
          z: 0
        }
      } = data;

      // btCapsuleShape expects (radius, cylinderHeight) — use half-width as
      // radius so the capsule wraps the bounding box correctly.
      const radius = width * 0.5;
      const capsule = new Ammo.btCapsuleShape(radius, height);

      // Always create the player body upright (identity quaternion).
      // The visual rotation is driven by the control, not by physics.
      const uprightQuaternion = {
        x: 0,
        y: 0,
        z: 0,
        w: 1
      };

      // Position the capsule at the model's AABB centre so it wraps the
      // visual mesh correctly.  originOffset is the vector from the model's
      // origin to its AABB centre — computed on the main thread where
      // Three.js geometry is available.
      const capsulePosition = {
        x: position.x + originOffset.x,
        y: position.y + originOffset.y,
        z: position.z + originOffset.z
      };
      const body = createRigidBody(capsule, {
        uuid,
        position: capsulePosition,
        quaternion: uprightQuaternion,
        mass,
        friction,
        restitution: 0,
        damping: {
          linear: 0.1,
          angular: 1
        }
      });

      // Disable rotation on all axes — the visual rotation is driven by the control.
      // Use btVector3 instead of scalar for compatibility across Ammo.js builds.
      const zeroAngular = new Ammo.btVector3(0, 0, 0);
      body.setAngularFactor(zeroAngular);
      Ammo.destroy(zeroAngular);

      // Enable continuous collision detection to prevent tunneling at high speeds.
      body.setCcdMotionThreshold(radius * 0.5);
      body.setCcdSweptSphereRadius(radius * 0.8);

      // Store for use in handlePlayerUpdate — offset is subtracted when
      // sending position back so the visual stays at the model origin.
      body._mass = mass;
      body._originOffset = originOffset;
      world.addElement({
        uuid,
        body,
        type: TYPES.PLAYER,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const handlePlayerUpdate = ({
      body,
      uuid,
      state = DEFAULT_RIGIDBODY_STATE
    }, dt) => {
      const {
        movement,
        cameraDirection,
        jump,
        jumpSpeed,
        speed: moveSpeed
      } = state;
      const motionState = body.getMotionState();
      if (!motionState) return;

      // Force-clear angular velocity every tick. The player capsule must stay
      // upright — rotation is handled visually by the control, not by physics.
      const zeroAngVel = new Ammo.btVector3(0, 0, 0);
      body.setAngularVelocity(zeroAngVel);
      Ammo.destroy(zeroAngVel);

      // Force the physics body rotation to identity (upright) every tick.
      // Contact forces can still rotate the capsule in some Ammo.js builds
      // even with angularFactor=(0,0,0). We must reset BOTH the body world
      // transform AND the motionState — they can diverge.
      const worldTransform = body.getWorldTransform();
      const uprightQuat = new Ammo.btQuaternion(0, 0, 0, 1);
      worldTransform.setRotation(uprightQuat);
      body.setWorldTransform(worldTransform);
      motionState.setWorldTransform(worldTransform);
      Ammo.destroy(uprightQuat);
      const characterSpeed = moveSpeed || 5;

      // Jump — apply a one-time impulse, then clear the flag so it doesn't repeat
      if (jump && jumpSpeed) {
        const mass = body._mass || 80;
        const impulse = new Ammo.btVector3(0, jumpSpeed * mass, 0);
        body.applyCentralImpulse(impulse);
        Ammo.destroy(impulse);
        state.jump = false;
      }
      const isMoving = movement && (movement.forward || movement.backwards || movement.left || movement.right);
      const linearVelocity = body.getLinearVelocity();
      const currentY = linearVelocity.y();
      if (isMoving && cameraDirection) {
        // Compute camera-relative movement direction
        let moveX = 0;
        let moveZ = 0;
        if (movement.forward) {
          moveX += cameraDirection.x;
          moveZ += cameraDirection.z;
        }
        if (movement.backwards) {
          moveX -= cameraDirection.x;
          moveZ -= cameraDirection.z;
        }
        if (movement.left) {
          moveX += cameraDirection.z;
          moveZ -= cameraDirection.x;
        }
        if (movement.right) {
          moveX -= cameraDirection.z;
          moveZ += cameraDirection.x;
        }

        // Normalize direction
        const len = Math.sqrt(moveX * moveX + moveZ * moveZ);
        if (len > 0) {
          moveX /= len;
          moveZ /= len;
        }

        // Set horizontal velocity directly for responsive movement
        const newVel = new Ammo.btVector3(moveX * characterSpeed, currentY, moveZ * characterSpeed);
        body.setLinearVelocity(newVel);
        Ammo.destroy(newVel);
      } else {
        // When idle, only dampen if there's meaningful horizontal velocity.
        // Avoid touching the velocity every frame to let the solver
        // handle floor contact without interference.
        const vx = linearVelocity.x();
        const vz = linearVelocity.z();
        if (Math.abs(vx) > 0.01 || Math.abs(vz) > 0.01) {
          const newVel = new Ammo.btVector3(vx * 0.85, currentY, vz * 0.85);
          body.setLinearVelocity(newVel);
          Ammo.destroy(newVel);
        }
        // Otherwise let physics handle everything — don't override velocity
      }

      // Read from the body's world transform (not motionState) because we
      // maintain it directly above — position comes from physics, rotation
      // is always identity.
      const finalTransform = body.getWorldTransform();
      const origin = finalTransform.getOrigin();
      const rotation = finalTransform.getRotation();
      const grounded = Math.abs(currentY) < 0.5;

      // The physics capsule is offset upward so its bottom aligns with the
      // character's feet.  Subtract the offset before sending the position
      // back so the visual stays at the feet, not the capsule centre.
      // Subtract the origin offset so the position maps back to model-origin
      // space (the visual's transform origin), not the AABB centre.
      const off = body._originOffset || {
        x: 0,
        y: 0,
        z: 0
      };
      const adjustedOrigin = new Ammo.btVector3(origin.x() - off.x, origin.y() - off.y, origin.z() - off.z);
      dispatcher.sendBodyUpdate(uuid, adjustedOrigin, rotation, dt, {
        grounded
      });
      Ammo.destroy(adjustedOrigin);
    };

    const requestNextFrame = self.requestAnimationFrame || self.webkitRequestAnimationFrame || self.mozRequestAnimationFrame || self.oRequestAnimationFrame || self.msRequestAnimationFrame || function (callback, _element) {
      self.setTimeout(callback, 1000 / 120);
    };
    let Clock = /*#__PURE__*/function () {
      function Clock() {
        _classCallCheck(this, Clock);
        this.timestamp = null;
      }
      return _createClass(Clock, [{
        key: "getDelta",
        value: function getDelta() {
          const time = Date.now();
          if (this.timestamp) {
            const delta = time - this.timestamp;
            this.timestamp = time;
            return delta;
          } else {
            this.timestamp = time;
            return 0;
          }
        }
      }]);
    }();
    let World = /*#__PURE__*/function () {
      function World() {
        _classCallCheck(this, World);
        _defineProperty(this, "init", options => {
          const {
            gravity = GRAVITY
          } = options;
          this.collisionConfiguration = new Ammo.btDefaultCollisionConfiguration();
          this.dispatcher = new Ammo.btCollisionDispatcher(this.collisionConfiguration);
          this.broadphase = new Ammo.btDbvtBroadphase();
          this.solver = new Ammo.btSequentialImpulseConstraintSolver();
          this.dynamicsWorld = new Ammo.btDiscreteDynamicsWorld(this.dispatcher, this.broadphase, this.solver, this.collisionConfiguration);
          this.dynamicsWorld.setGravity(new Ammo.btVector3(gravity.x, gravity.y, gravity.z));

          // this is needed for ghostObject collisions
          this.dynamicsWorld.getBroadphase().getOverlappingPairCache().setInternalGhostPairCallback(new Ammo.btGhostPairCallback());
          this.initialised = true;
        });
        _defineProperty(this, "removeElement", uuid => {
          if (this.hasElement(uuid)) {
            this.elements[uuid].deleted = true;
          }
        });
        _defineProperty(this, "removeDeletedElements", () => Object.keys(this.elements).filter(uuid => this.elements[uuid].deleted).forEach(uuid => {
          delete this.elements[uuid];
        }));
        _defineProperty(this, "hasElement", uuid => {
          return Object.keys(this.elements).includes(uuid);
        });
        _defineProperty(this, "getElement", uuid => this.elements[uuid]);
        _defineProperty(this, "isInitialised", () => this.initialised);
        _defineProperty(this, "getDynamicsWorld", () => this.dynamicsWorld);
        _defineProperty(this, "addRigidBody", body => {
          this.dynamicsWorld.addRigidBody(body);
        });
        _defineProperty(this, "addAction", action => {
          this.dynamicsWorld.addAction(action);
        });
        _defineProperty(this, "addCollisionObject", collisionObject => {
          this.dynamicsWorld.addCollisionObject(collisionObject);
        });
        _defineProperty(this, "stepSimulation", dt => {
          this.dynamicsWorld.stepSimulation(dt);
        });
        _defineProperty(this, "simulate", () => {
          const dt = this.clock.getDelta() / 1000;
          this.stepSimulation(dt);
          Object.keys(this.elements).forEach(uuid => {
            const element = this.getElement(uuid);
            if (element) {
              switch (element.type) {
                case TYPES.BOX:
                case TYPES.SPHERE:
                case TYPES.MESH:
                  handleElementUpdate(element, dt);
                  break;
                case TYPES.PLAYER:
                  handlePlayerUpdate(element, dt);
                  break;
                case TYPES.VEHICLE:
                  handleVehicleUpdate(element, dt);
                  break;
              }
            }
          });
          this.calculateCollisions();
          this.removeDeletedElements();
          dispatcher.sendPhysicsUpdate(dt);
          this.requestAnimationFrameId = requestNextFrame(this.simulate.bind(this));
        });
        _defineProperty(this, "calculateCollisions", () => {
          let ammoDispatcher = this.dynamicsWorld.getDispatcher();
          let numManifolds = ammoDispatcher.getNumManifolds();
          for (let i = 0; i < numManifolds; i++) {
            let contactManifold = ammoDispatcher.getManifoldByIndexInternal(i);
            let rb0 = Ammo.castObject(contactManifold.getBody0(), Ammo.btRigidBody);
            let rb1 = Ammo.castObject(contactManifold.getBody1(), Ammo.btRigidBody);
            let numContacts = contactManifold.getNumContacts();

            // this iteration doesn't have uuids
            if (!rb0.uuid || !rb1.uuid) continue;
            let contacts = [];
            for (let j = 0; j < numContacts; j++) {
              let contactPoint = contactManifold.getContactPoint(j);
              let distance = contactPoint.getDistance();
              if (distance > 0.0) continue;
              let velocity0 = rb0.getLinearVelocity();
              let velocity1 = rb1.getLinearVelocity();
              let worldPos0 = contactPoint.get_m_positionWorldOnA();
              let worldPos1 = contactPoint.get_m_positionWorldOnB();
              let localPos0 = contactPoint.get_m_localPointA();
              let localPos1 = contactPoint.get_m_localPointB();
              contacts.push({
                distance,
                elements: [{
                  uuid: rb0.uuid,
                  velocity: {
                    x: velocity0.x(),
                    y: velocity0.y(),
                    z: velocity0.z()
                  },
                  worldPos: {
                    x: worldPos0.x(),
                    y: worldPos0.y(),
                    z: worldPos0.z()
                  },
                  localPos: {
                    x: localPos0.x(),
                    y: localPos0.y(),
                    z: localPos0.z()
                  }
                }, {
                  uuid: rb1.uuid,
                  velocity: {
                    x: velocity1.x(),
                    y: velocity1.y(),
                    z: velocity1.z()
                  },
                  worldPos: {
                    x: worldPos1.x(),
                    y: worldPos1.y(),
                    z: worldPos1.z()
                  },
                  localPos: {
                    x: localPos1.x(),
                    y: localPos1.y(),
                    z: localPos1.z()
                  }
                }]
              });
            }
            dispatcher.sendDispatchEvent(rb0.uuid, PHYSICS_EVENTS.ELEMENT.COLLISION, {
              contacts
            });
            dispatcher.sendDispatchEvent(rb1.uuid, PHYSICS_EVENTS.ELEMENT.COLLISION, {
              contacts
            });
          }
        });
        _defineProperty(this, "addElement", data => {
          this.elements[data.uuid] = data;
        });
        _defineProperty(this, "updateBodyState", ({
          uuid,
          state
        }) => {
          if (this.hasElement(uuid)) {
            this.elements[uuid].state = {
              ...this.elements[uuid].state,
              ...state
            };
          }
        });
        _defineProperty(this, "terminate", () => {
          Ammo.destroy(this.dynamicsWorld);
          Ammo.destroy(this.solver);
          Ammo.destroy(this.dispatcher);
          Ammo.destroy(this.collisionConfiguration);
          cancelAnimationFrame(this.requestAnimationFrameId);
          dispatcher.sendTerminateEvent();
        });
        this.elements = {};
        this.initialised = false;
        this.collisionConfiguration = undefined;
        this.dispatcher = undefined;
        this.broadphase = undefined;
        this.solver = undefined;
        this.dynamicsWorld = undefined;
        this.requestAnimationFrameId = null;
        this.clock = new Clock();
      }
      return _createClass(World, [{
        key: "disposeBody",
        value: function disposeBody({
          uuid
        }) {
          const element = this.getElement(uuid);
          this.dynamicsWorld.removeRigidBody(element.body);
          this.removeElement(uuid);
          dispatcher.sendElementDisposed({
            uuid
          });
        }
      }]);
    }();
    var world = new World();

    const DEFAULT_ROLL_INFLUENCE = 0.2;
    const DEFAULT_FRICTION = 1000;
    const DEFAULT_MASS = 800;
    const addVehicle = data => {
      const {
        position,
        quaternion,
        uuid,
        wheels,
        mass = DEFAULT_MASS,
        width = 1.8,
        height = 0.6,
        length = 4,
        friction = DEFAULT_FRICTION,
        rollInfluence = DEFAULT_ROLL_INFLUENCE,
        wheelsOptions = {},
        suspensions = {}
      } = data;
      const {
        back = {},
        front = {}
      } = wheelsOptions;
      const {
        axisPosition: axisPositionBack = -1,
        radius: wheelRadiusBack = 0.4,
        halfTrack: wheelHalfTrackBack = 1,
        axisHeight: wheelAxisHeightBack = 0.3
      } = back;
      const {
        axisPosition: axisPositionFront = 1.7,
        radius: wheelRadiusFront = 0.4,
        halfTrack: wheelHalfTrackFront = 1,
        axisHeight: wheelAxisHeightFront = 0.3
      } = front;
      const {
        stiffness = 20.0,
        damping = 2.3,
        compression = 4.4,
        restLength = 0.6
      } = suspensions;

      // Chassis
      const geometry = new Ammo.btBoxShape(new Ammo.btVector3(width * 0.5, height * 0.5, length * 0.5));
      const transform = new Ammo.btTransform();
      transform.setIdentity();
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      const motionState = new Ammo.btDefaultMotionState(transform);
      const localInertia = new Ammo.btVector3(0, 0, 0);
      geometry.calculateLocalInertia(mass, localInertia);
      const chassis = new Ammo.btRigidBody(new Ammo.btRigidBodyConstructionInfo(mass, motionState, geometry, localInertia));
      chassis.setActivationState(DISABLE_DEACTIVATION);
      world.addRigidBody(chassis);

      // Raycast Vehicle
      const tuning = new Ammo.btVehicleTuning();
      const rayCaster = new Ammo.btDefaultVehicleRaycaster(world.getDynamicsWorld());
      const vehicle = new Ammo.btRaycastVehicle(tuning, chassis, rayCaster);
      vehicle.setCoordinateSystem(0, 1, 2);
      world.addAction(vehicle);
      const wheelDirectionCS0 = new Ammo.btVector3(0, -1, 0);
      const wheelAxleCS = new Ammo.btVector3(-1, 0, 0);
      const addWheel = (isFront, pos, radius) => {
        var wheelInfo = vehicle.addWheel(pos, wheelDirectionCS0, wheelAxleCS, restLength, radius, tuning, isFront);
        wheelInfo.set_m_suspensionStiffness(stiffness);
        wheelInfo.set_m_wheelsDampingRelaxation(damping);
        wheelInfo.set_m_wheelsDampingCompression(compression);
        wheelInfo.set_m_frictionSlip(friction);
        wheelInfo.set_m_rollInfluence(rollInfluence);
      };
      addWheel(true, new Ammo.btVector3(wheelHalfTrackFront, wheelAxisHeightFront, axisPositionFront), wheelRadiusFront);
      addWheel(true, new Ammo.btVector3(-wheelHalfTrackFront, wheelAxisHeightFront, axisPositionFront), wheelRadiusFront);
      addWheel(false, new Ammo.btVector3(-wheelHalfTrackBack, wheelAxisHeightBack, axisPositionBack), wheelRadiusBack);
      addWheel(false, new Ammo.btVector3(wheelHalfTrackBack, wheelAxisHeightBack, axisPositionBack), wheelRadiusBack);
      vehicle.uuid = uuid;
      world.addElement({
        type: TYPES.VEHICLE,
        uuid,
        vehicle: vehicle,
        wheels,
        options: data,
        state: DEFAULT_VEHICLE_STATE
      });
    };
    const setVehiclePosition = data => {
      const {
        uuid,
        position
      } = data;
      const element = world.getElement(uuid);
      if (element.type === TYPES.VEHICLE) {
        const body = element.vehicle.getRigidBody();
        const transform = new Ammo.btTransform();
        body.getWorldTransform(transform);
        transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
        body.setWorldTransform(transform);
      }
    };
    const setVehicleQuaternion = data => {
      const {
        uuid,
        quaternion
      } = data;
      const element = world.getElement(uuid);
      if (element.type === TYPES.VEHICLE) {
        const body = element.vehicle.getRigidBody();
        const transform = new Ammo.btTransform();
        body.getWorldTransform(transform);
        transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
        body.setWorldTransform(transform);
      }
    };
    const resetVehicle = data => {
      const {
        uuid,
        quaternion,
        position
      } = data;
      const element = world.getElement(uuid);
      if (element.type === TYPES.VEHICLE) {
        const body = element.vehicle.getRigidBody();
        const transform = new Ammo.btTransform();
        body.getWorldTransform(transform);
        transform.setIdentity();
        transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
        transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
        body.setWorldTransform(transform);
      }
    };
    const handleVehicleUpdate = ({
      vehicle,
      wheels,
      uuid,
      state = DEFAULT_VEHICLE_STATE,
      options = {}
    }, dt) => {
      let breakingForce = 0;
      let engineForce = 0;
      const {
        steeringClamp = DEFAULT_STEERING_CLAMP,
        steeringIncrement = DEFAULT_STEERING_INCREMENT,
        maxEngineForce = DEFAULT_MAX_ENGINE_FORCE,
        maxBreakingForce = DEFAULT_MAX_BREAKING_FORCE
      } = options;
      if (state.acceleration) {
        if (speed < -1) breakingForce = maxBreakingForce;else engineForce = maxEngineForce;
      }
      if (state.braking) {
        if (speed > 1) breakingForce = maxBreakingForce;else engineForce = -maxEngineForce / 2;
      }
      if (state.left) {
        if (state.vehicleSteering < steeringClamp) state.vehicleSteering += steeringIncrement;
      } else {
        if (state.right) {
          if (state.vehicleSteering > -steeringClamp) state.vehicleSteering -= steeringIncrement;
        } else {
          if (state.vehicleSteering < -steeringIncrement) state.vehicleSteering += steeringIncrement;else {
            if (state.vehicleSteering > steeringIncrement) state.vehicleSteering -= steeringIncrement;else {
              state.vehicleSteering = 0;
            }
          }
        }
      }
      vehicle.applyEngineForce(engineForce, BACK_LEFT);
      vehicle.applyEngineForce(engineForce, BACK_RIGHT);
      vehicle.setBrake(breakingForce / 2, FRONT_LEFT);
      vehicle.setBrake(breakingForce / 2, FRONT_RIGHT);
      vehicle.setBrake(breakingForce, BACK_LEFT);
      vehicle.setBrake(breakingForce, BACK_RIGHT);
      vehicle.setSteeringValue(state.vehicleSteering, FRONT_LEFT);
      vehicle.setSteeringValue(state.vehicleSteering, FRONT_RIGHT);
      let tm, p, q, i;
      const n = vehicle.getNumWheels();
      for (i = 0; i < n; i++) {
        vehicle.updateWheelTransform(i, true);
        tm = vehicle.getWheelTransformWS(i);
        p = tm.getOrigin();
        q = tm.getRotation();
        const wheelUUID = wheels[i];
        dispatcher.sendBodyUpdate(wheelUUID, p, q, dt);
      }
      tm = vehicle.getChassisWorldTransform();
      p = tm.getOrigin();
      q = tm.getRotation();
      const direction = vehicle.getForwardVector();
      const speed = vehicle.getCurrentSpeedKmHour();
      const extraData = {
        direction: {
          x: direction.x(),
          y: direction.y(),
          z: direction.z()
        },
        speed
      };
      dispatcher.sendBodyUpdate(uuid, p, q, dt, extraData);
      world.updateBodyState(uuid, state);
    };

    const createGhostCollider = (radius, position) => {
      const ghostCollider = new Ammo.btGhostObject();
      const transform = new Ammo.btTransform();
      ghostCollider.setCollisionShape(new Ammo.btSphereShape(radius));
      ghostCollider.getWorldTransform(transform);
      transform.setIdentity();
      transform.setOrigin(position);
      transform.setRotation(new Ammo.btQuaternion(0, 0, 0, 1));
      ghostCollider.setWorldTransform(transform);
      return {
        ghostCollider,
        transform
      };
    };
    const forEachGhostCollision = (ghostCollider, forEachCallback = () => {}) => {
      const collisions = ghostCollider.getNumOverlappingObjects();
      for (let i = 0; i < collisions; i++) {
        const object = Ammo.castObject(ghostCollider.getOverlappingObject(i), Ammo.btRigidBody);
        const transform = new Ammo.btTransform();
        object.getWorldTransform(transform);
        forEachCallback(object, transform, i);
        Ammo.destroy(transform);
      }
    };
    const getExplosionPosition = (uuid, position) => {
      let explosionPosition = position;
      if (!explosionPosition) {
        const {
          body
        } = world.getElement(uuid);
        const motionState = body.getMotionState();
        const transform = new Ammo.btTransform();
        motionState.getWorldTransform(transform);
        explosionPosition = transform.getOrigin();
      }
      return explosionPosition;
    };
    const getExplosionImpulse = (position, explosionPosition, strength) => {
      // Calculate direction vector WITHOUT mutating the input btVector3 objects.
      // op_sub/op_mul mutate in place in Ammo.js, which would corrupt the body's transform origin.
      const dx = position.x() - explosionPosition.x();
      const dy = position.y() - explosionPosition.y();
      const dz = position.z() - explosionPosition.z();
      let length = Math.sqrt(dx * dx + dy * dy + dz * dz);

      // If the object is at the explosion center, push it straight up instead of
      // producing an unstable near-zero normalization.
      if (length < 0.001) {
        const impulse = new Ammo.btVector3(0, strength * 2, 0);
        return impulse;
      }

      // Normalize direction
      const nx = dx / length;
      const ny = dy / length;
      const nz = dz / length;

      // Scale by strength and add upward bias
      const impulse = new Ammo.btVector3(nx * strength, ny * strength + strength, nz * strength);
      return impulse;
    };
    const createExplosion = ({
      uuid,
      position,
      radius = EXPLOSION_SIZES.SMALL,
      strength = EXPLOSION_STRENGTHS.MEDIUM
    }) => {
      try {
        // Get the source body's Ammo pointer so we can reliably skip it.
        // Ammo.castObject() creates new JS wrappers, so custom properties like
        // .uuid set on the original wrapper are NOT available on cast results.
        // We compare the underlying C++ pointer instead.
        const sourceElement = world.getElement(uuid);
        const sourceBodyPtr = sourceElement && sourceElement.body ? sourceElement.body.a || sourceElement.body.ptr : null;
        const explosionPosition = getExplosionPosition(uuid, position);
        const {
          ghostCollider,
          transform
        } = createGhostCollider(radius, explosionPosition);
        world.addCollisionObject(ghostCollider);

        // Force a collision detection pass so the ghost collider discovers
        // overlapping bodies. Without this, getNumOverlappingObjects() returns 0
        // because btGhostPairCallback only updates during a simulation step.
        world.getDynamicsWorld().performDiscreteCollisionDetection();
        forEachGhostCollision(ghostCollider, (object, objectTransform) => {
          // Skip the source entity using pointer comparison.
          const objectPtr = object.a || object.ptr;
          if (sourceBodyPtr && objectPtr === sourceBodyPtr) return;
          const origin = objectTransform.getOrigin();
          object.activate(true);
          const impulse = getExplosionImpulse(origin, explosionPosition, strength);
          object.applyCentralImpulse(impulse);
          Ammo.destroy(impulse);
        });
        world.getDynamicsWorld().removeCollisionObject(ghostCollider);
        Ammo.destroy(ghostCollider);
        Ammo.destroy(transform);
      } catch (e) {
        console.error("[Physics Worker] createExplosion error:", e);
      }
    };

    const handleLoadEvent = options => Ammo => {
      self.Ammo = Ammo;
      onmessage = ({
        data
      }) => {
        switch (data.event) {
          case PHYSICS_EVENTS.ADD.BOX:
            addBox(data);
            break;
          case PHYSICS_EVENTS.ADD.SPHERE:
            addSphere(data);
            break;
          case PHYSICS_EVENTS.ADD.VEHICLE:
            addVehicle(data);
            break;
          case PHYSICS_EVENTS.ADD.MODEL:
            addModel(data);
            break;
          case PHYSICS_EVENTS.ADD.PLAYER:
            addPlayer(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.SET.LINEAR_VELOCITY:
            setLinearVelocity(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.RESET:
            resetElement(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.SET.POSITION:
            setPosition(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.SET.QUATERNION:
            setQuaternion(data);
            break;
          case PHYSICS_EVENTS.VEHICLE.SET.POSITION:
            setVehiclePosition(data);
            break;
          case PHYSICS_EVENTS.VEHICLE.SET.QUATERNION:
            setVehicleQuaternion(data);
            break;
          case PHYSICS_EVENTS.VEHICLE.RESET:
            resetVehicle(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.APPLY.IMPULSE:
            applyImpuse(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.UPDATE:
            world.updateBodyState(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.DISPOSE:
            world.disposeBody(data);
            break;
          case PHYSICS_EVENTS.EFFECTS.EXPLOSION:
            createExplosion(data);
            break;
          case PHYSICS_EVENTS.TERMINATE:
            world.terminate();
            break;
        }
      };
      world.init(options);
      dispatcher.sendReadyEvent();
      world.simulate();
    };
    const loadAmmo = options => {
      const scriptUrl = options.host + "/" + (options.path || LIBRARY_NAME);
      importScripts(scriptUrl);
      Ammo().then(handleLoadEvent(options));
    };
    onmessage = ({
      data
    }) => {
      switch (data.event) {
        case PHYSICS_EVENTS.LOAD.AMMO:
          loadAmmo(data);
          break;
      }
    };

})();

', null, false);
|
|
55723
|
+
}var WorkerFactory = createBase64WorkerFactory('/* rollup-plugin-web-worker-loader */
(function () {
    'use strict';

    const LIBRARY_NAME = "ammo.js";
    const TYPES = {
      BOX: "BOX",
      SPHERE: "SPHERE",
      VEHICLE: "VEHICLE",
      MESH: "MESH",
      PLAYER: "PLAYER"
    };
    const DEFAULT_VEHICLE_STATE = {
      vehicleSteering: 0,
      acceleration: false,
      breaking: false,
      right: false,
      left: false
    };
    const DEFAULT_RIGIDBODY_STATE = {
      velocity: {
        x: 0,
        y: 0,
        z: 0
      },
      movement: {
        forward: false,
        backwards: false,
        left: false,
        right: false
      },
      direction: {
        x: 0,
        y: 0,
        z: 0
      }
    };
    const DEFAULT_SCALE = {
      x: 1,
      y: 1,
      z: 1
    };
    const DEFAULT_LINEAR_VELOCITY = {
      x: 0,
      y: 0,
      z: 0
    };
    const DEFAULT_IMPULSE = {
      x: 0,
      y: 0,
      z: 0
    };
    const DISABLE_DEACTIVATION = 4;
    const GRAVITY = {
      x: 0,
      y: -9.8,
      z: 0
    };
    const FRONT_LEFT = 0;
    const FRONT_RIGHT = 1;
    const BACK_LEFT = 2;
    const BACK_RIGHT = 3;
    const DEFAULT_STEERING_INCREMENT = 0.04;
    const DEFAULT_STEERING_CLAMP = 0.5;
    const DEFAULT_MAX_ENGINE_FORCE = 2000;
    const DEFAULT_MAX_BREAKING_FORCE = 100;
    const EXPLOSION_SIZES = {
      SMALL: 4,
      MEDIUM: 6,
      LARGE: 8,
      MASSIVE: 12
    };
    const EXPLOSION_STRENGTHS = {
      VERY_WEAK: 2,
      WEAK: 4,
      MEDIUM: 8,
      LARGE: 16,
      MASSIVE: 32,
      OK_NO: 64
    };

    function _classCallCheck(a, n) {
      if (!(a instanceof n)) throw new TypeError("Cannot call a class as a function");
    }
    function _defineProperties(e, r) {
      for (var t = 0; t < r.length; t++) {
        var o = r[t];
        o.enumerable = o.enumerable || !1, o.configurable = !0, "value" in o && (o.writable = !0), Object.defineProperty(e, _toPropertyKey(o.key), o);
      }
    }
    function _createClass(e, r, t) {
      return r && _defineProperties(e.prototype, r), t && _defineProperties(e, t), Object.defineProperty(e, "prototype", {
        writable: !1
      }), e;
    }
    function _defineProperty(e, r, t) {
      return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, {
        value: t,
        enumerable: !0,
        configurable: !0,
        writable: !0
      }) : e[r] = t, e;
    }
    function _toPrimitive(t, r) {
      if ("object" != typeof t || !t) return t;
      var e = t[Symbol.toPrimitive];
      if (void 0 !== e) {
        var i = e.call(t, r || "default");
        if ("object" != typeof i) return i;
        throw new TypeError("@@toPrimitive must return a primitive value.");
      }
      return ("string" === r ? String : Number)(t);
    }
    function _toPropertyKey(t) {
      var i = _toPrimitive(t, "string");
      return "symbol" == typeof i ? i : i + "";
    }

    const PHYSICS_EVENTS = {
      DISPATCH: "physics:dispatch",
      TERMINATE: "physics:terminate",
      LOAD: {
        AMMO: "physics:load:ammo"
      },
      READY: "physics:ready",
      INIT: "physics:init",
      UPDATE: "physics:update",
      ADD: {
        BOX: "physics:add:box",
        VEHICLE: "physics:add:vehicle",
        MODEL: "physics:add:model",
        PLAYER: "physics:add:player",
        SPHERE: "physics:add:sphere"
      },
      ELEMENT: {
        DISPOSE: "physics:element:dispose",
        COLLISION: "physics:element:collision",
        UPDATE: "physics:element:update",
        CREATED: "physics:element:created",
        SET: {
          POSITION: "physics:element:set:position",
          QUATERNION: "physics:element:set:quaternion",
          LINEAR_VELOCITY: "physics:element:set:linear_velocity"
        },
        RESET: "physics:element:reset",
        APPLY: {
          IMPULSE: "physics:element:apply:impulse"
        }
      },
      VEHICLE: {
        SET: {
          POSITION: "physics:vehicle:set:position",
          QUATERNION: "physics:vehicle:set:quaternion"
        },
        RESET: "physics:vehicle:reset",
        SPEED: "physics:vehicle:speed",
        DIRECTION: "physics:vehicle:direction"
      },
      EFFECTS: {
        EXPLOSION: "physics:effects:explosion"
      }
    };

    let Dispatcher = /*#__PURE__*/_createClass(function Dispatcher() {
      _classCallCheck(this, Dispatcher);
      _defineProperty(this, "sendPhysicsUpdate", dt => postMessage({
        event: PHYSICS_EVENTS.UPDATE,
        dt
      }));
      _defineProperty(this, "sendReadyEvent", () => postMessage({
        event: PHYSICS_EVENTS.READY
      }));
      _defineProperty(this, "sendTerminateEvent", () => postMessage({
        event: PHYSICS_EVENTS.TERMINATE
      }));
      _defineProperty(this, "sendBodyUpdate", (uuid, position, rotation, dt, extraData) => postMessage({
        event: PHYSICS_EVENTS.ELEMENT.UPDATE,
        uuid,
        position: {
          x: position.x(),
          y: position.y(),
          z: position.z()
        },
        quaternion: {
          x: rotation.x(),
          y: rotation.y(),
          z: rotation.z(),
          w: rotation.w()
        },
        ...extraData,
        dt
      }));
      _defineProperty(this, "sendDispatchEvent", (uuid, eventName, eventData) => postMessage({
        event: PHYSICS_EVENTS.DISPATCH,
        uuid,
        eventName,
        eventData
      }));
      _defineProperty(this, "sendElementDisposed", ({
        uuid
      }) => postMessage({
        event: PHYSICS_EVENTS.ELEMENT.DISPOSE,
        uuid
      }));
    });
    var dispatcher = new Dispatcher();

    const applyMatrix4ToVector3 = ({
      x = 0,
      y = 0,
      z = 0
    }, matrix = []) => {
      const w = 1 / (matrix[3] * x + matrix[7] * y + matrix[11] * z + matrix[15]);
      return {
        x: (matrix[0] * x + matrix[4] * y + matrix[8] * z + matrix[12]) * w,
        y: (matrix[1] * x + matrix[5] * y + matrix[9] * z + matrix[13]) * w,
        z: (matrix[2] * x + matrix[6] * y + matrix[10] * z + matrix[14]) * w
      };
    };

    const createRigidBody = (shape, options) => {
      const {
        uuid,
        position,
        quaternion,
        mass = 0,
        friction,
        restitution = 0.9,
        damping = {
          linear: 0.2,
          angular: 0.2
        }
      } = options;
      const transform = new Ammo.btTransform();
      transform.setIdentity();
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      const motionState = new Ammo.btDefaultMotionState(transform);
      const localInertia = new Ammo.btVector3(0, 0, 0);
      shape.calculateLocalInertia(mass, localInertia);
      const rbInfo = new Ammo.btRigidBodyConstructionInfo(mass, motionState, shape, localInertia);
      const body = new Ammo.btRigidBody(rbInfo);
      if (mass > 0) {
        body.setFriction(friction);
        body.setRestitution(restitution);
        body.setDamping(damping.linear, damping.angular);
        body.setActivationState(DISABLE_DEACTIVATION);
      }

      // storing uuid for future reference
      body.uuid = uuid;
      world.addRigidBody(body);
      return body;
    };
    const addModel = options => {
      const {
        uuid,
        vertices,
        matrices,
        indexes,
        position,
        quaternion,
        mass = 0,
        friction = 2
      } = options;
      const scale = DEFAULT_SCALE;
      const bta = new Ammo.btVector3();
      const btb = new Ammo.btVector3();
      const btc = new Ammo.btVector3();
      const triMesh = new Ammo.btTriangleMesh(true, false);
      for (let i = 0; i < vertices.length; i++) {
        const components = vertices[i];
        const index = indexes[i] ? indexes[i] : null;
        const matrix = Array.from(matrices[i]);
        if (index) {
          for (let j = 0; j < index.length; j += 3) {
            const ai = index[j] * 3;
            const bi = index[j + 1] * 3;
            const ci = index[j + 2] * 3;
            const va = applyMatrix4ToVector3({
              x: components[ai],
              y: components[ai + 1],
              z: components[ai + 2]
            }, matrix);
            const vb = applyMatrix4ToVector3({
              x: components[bi],
              y: components[bi + 1],
              z: components[bi + 2]
            }, matrix);
            const vc = applyMatrix4ToVector3({
              x: components[ci],
              y: components[ci + 1],
              z: components[ci + 2]
            }, matrix);
            bta.setValue(va.x, va.y, va.z);
            btb.setValue(vb.x, vb.y, vb.z);
            btc.setValue(vc.x, vc.y, vc.z);
            triMesh.addTriangle(bta, btb, btc, false);
          }
        } else {
          for (let j = 0; j < components.length; j += 9) {
            const va = applyMatrix4ToVector3({
              x: components[j + 0],
              y: components[j + 1],
              z: components[j + 2]
            }, matrix);
            const vb = applyMatrix4ToVector3({
              x: components[j + 3],
              y: components[j + 4],
              z: components[j + 5]
            }, matrix);
            const vc = applyMatrix4ToVector3({
              x: components[j + 6],
              y: components[j + 7],
              z: components[j + 8]
            }, matrix);
            bta.setValue(va.x, va.y, va.z);
            btb.setValue(vb.x, vb.y, vb.z);
            btc.setValue(vc.x, vc.y, vc.z);
            triMesh.addTriangle(bta, btb, btc, false);
          }
        }
      }
      const localScale = new Ammo.btVector3(scale.x, scale.y, scale.z);
      triMesh.setScaling(localScale);
      Ammo.destroy(localScale);
      const collisionShape = new Ammo.btBvhTriangleMeshShape(triMesh, true, true);
      collisionShape.resources = [triMesh];
      Ammo.destroy(bta);
      Ammo.destroy(btb);
      Ammo.destroy(btc);
      const body = createRigidBody(collisionShape, {
        uuid,
        position,
        quaternion,
        mass,
        friction
      });
      world.addElement({
        uuid,
        body,
        type: TYPES.MESH,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const addBox = data => {
      const {
        uuid,
        width,
        length,
        height,
        position,
        quaternion,
        mass = 0,
        friction = 2
      } = data;
      const geometry = new Ammo.btBoxShape(new Ammo.btVector3(width * 0.5, height * 0.5, length * 0.5));
      const body = createRigidBody(geometry, {
        uuid,
        position,
        quaternion,
        mass,
        friction
      });
      world.addElement({
        uuid,
        body,
        type: TYPES.BOX,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const addSphere = data => {
      const {
        uuid,
        radius,
        position,
        quaternion,
        mass = 0,
        friction = 2
      } = data;
      const geometry = new Ammo.btSphereShape(radius);
      const body = createRigidBody(geometry, {
        uuid,
        position,
        quaternion,
        mass,
        friction
      });
      world.addElement({
        uuid,
        body,
        type: TYPES.SPHERE,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const setLinearVelocity = data => {
      const {
        uuid,
        velocity = DEFAULT_LINEAR_VELOCITY
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const motionState = body.getMotionState();
      if (motionState) {
        const linearVelocity = new Ammo.btVector3(velocity.x, velocity.y, velocity.z);
        body.setLinearVelocity(linearVelocity);
        Ammo.destroy(linearVelocity);
      }
    };
    const setPosition = data => {
      const {
        uuid,
        position
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const transform = new Ammo.btTransform();
      body.getWorldTransform(transform);
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      body.setWorldTransform(transform);
      // Also update motion state so static bodies (mass=0) reflect the change
      const motionState = body.getMotionState();
      if (motionState) {
        motionState.setWorldTransform(transform);
      }
    };
    const resetElement = data => {
      const {
        uuid,
        position,
        quaternion
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const transform = new Ammo.btTransform();
      body.getWorldTransform(transform);
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      body.setWorldTransform(transform);
      // Also update motion state so static bodies (mass=0) reflect the change
      const motionState = body.getMotionState();
      if (motionState) {
        motionState.setWorldTransform(transform);
      }
    };
    const applyImpuse = ({
      uuid,
      impulse = DEFAULT_IMPULSE
    }) => {
      try {
        const element = world.getElement(uuid);
        if (!element) {
          console.warn("[Physics Worker] applyImpulse: element not found for uuid:", uuid);
          return;
        }
        const {
          body
        } = element;
        if (!body) {
          console.warn("[Physics Worker] applyImpulse: body is null for uuid:", uuid);
          return;
        }
        body.activate(true);
        const btImpulse = new Ammo.btVector3(impulse.x, impulse.y, impulse.z);
        body.applyCentralImpulse(btImpulse);
        Ammo.destroy(btImpulse);
      } catch (e) {
        console.error("[Physics Worker] applyImpulse error:", e);
      }
    };
    const setQuaternion = data => {
      const {
        uuid,
        quaternion
      } = data;
      const {
        body
      } = world.getElement(uuid);
      const transform = new Ammo.btTransform();
      body.getWorldTransform(transform);
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      body.setWorldTransform(transform);
      // Also update motion state so static bodies (mass=0) reflect the change
      const motionState = body.getMotionState();
      if (motionState) {
        motionState.setWorldTransform(transform);
      }
    };
    const handleElementUpdate = ({
      body,
      uuid,
      state: _state = DEFAULT_RIGIDBODY_STATE
    }, dt) => {
      // Static bodies (mass=0) never move — skip sending updates so the
      // visual position stays exactly where the author placed it.
      if (body.isStaticObject()) return;
      const motionState = body.getMotionState();
      if (motionState) {
        const transform = new Ammo.btTransform();
        motionState.getWorldTransform(transform);
        let origin = transform.getOrigin();
        let rotation = transform.getRotation();
        dispatcher.sendBodyUpdate(uuid, origin, rotation, dt);
        Ammo.destroy(transform);
      }
    };

    const addPlayer = data => {
      const {
        uuid,
        width,
        height,
        position,
        mass,
        friction,
        originOffset = {
          x: 0,
          y: 0,
          z: 0
        }
      } = data;

      // btCapsuleShape expects (radius, cylinderHeight) — use half-width as
      // radius so the capsule wraps the bounding box correctly.
      const radius = width * 0.5;
      const capsule = new Ammo.btCapsuleShape(radius, height);

      // Always create the player body upright (identity quaternion).
      // The visual rotation is driven by the control, not by physics.
      const uprightQuaternion = {
        x: 0,
        y: 0,
        z: 0,
        w: 1
      };

      // Position the capsule at the model's AABB centre so it wraps the
      // visual mesh correctly.  originOffset is the vector from the model's
      // origin to its AABB centre — computed on the main thread where
      // Three.js geometry is available.
      const capsulePosition = {
        x: position.x + originOffset.x,
        y: position.y + originOffset.y,
        z: position.z + originOffset.z
      };
      const body = createRigidBody(capsule, {
        uuid,
        position: capsulePosition,
        quaternion: uprightQuaternion,
        mass,
        friction,
        restitution: 0,
        damping: {
          linear: 0.1,
          angular: 1
        }
      });

      // Disable rotation on all axes — the visual rotation is driven by the control.
      // Use btVector3 instead of scalar for compatibility across Ammo.js builds.
      const zeroAngular = new Ammo.btVector3(0, 0, 0);
      body.setAngularFactor(zeroAngular);
      Ammo.destroy(zeroAngular);

      // Enable continuous collision detection to prevent tunneling at high speeds.
      body.setCcdMotionThreshold(radius * 0.5);
      body.setCcdSweptSphereRadius(radius * 0.8);

      // Store for use in handlePlayerUpdate — offset is subtracted when
      // sending position back so the visual stays at the model origin.
      body._mass = mass;
      body._originOffset = originOffset;
      world.addElement({
        uuid,
        body,
        type: TYPES.PLAYER,
        state: DEFAULT_RIGIDBODY_STATE
      });
    };
    const handlePlayerUpdate = ({
      body,
      uuid,
      state = DEFAULT_RIGIDBODY_STATE
    }, dt) => {
      const {
        movement,
        cameraDirection,
        jump,
        jumpSpeed,
        speed: moveSpeed
      } = state;
      const motionState = body.getMotionState();
      if (!motionState) return;

      // Force-clear angular velocity every tick. The player capsule must stay
      // upright — rotation is handled visually by the control, not by physics.
      const zeroAngVel = new Ammo.btVector3(0, 0, 0);
      body.setAngularVelocity(zeroAngVel);
      Ammo.destroy(zeroAngVel);

      // Force the physics body rotation to identity (upright) every tick.
      // Contact forces can still rotate the capsule in some Ammo.js builds
      // even with angularFactor=(0,0,0). We must reset BOTH the body world
      // transform AND the motionState — they can diverge.
      const worldTransform = body.getWorldTransform();
      const uprightQuat = new Ammo.btQuaternion(0, 0, 0, 1);
      worldTransform.setRotation(uprightQuat);
      body.setWorldTransform(worldTransform);
      motionState.setWorldTransform(worldTransform);
      Ammo.destroy(uprightQuat);
      const characterSpeed = moveSpeed || 5;

      // Jump — apply a one-time impulse, then clear the flag so it doesn't repeat
      if (jump && jumpSpeed) {
        const mass = body._mass || 80;
        const impulse = new Ammo.btVector3(0, jumpSpeed * mass, 0);
        body.applyCentralImpulse(impulse);
        Ammo.destroy(impulse);
        state.jump = false;
      }
      const isMoving = movement && (movement.forward || movement.backwards || movement.left || movement.right);
      const linearVelocity = body.getLinearVelocity();
      const currentY = linearVelocity.y();
      if (isMoving && cameraDirection) {
        // Compute camera-relative movement direction
        let moveX = 0;
        let moveZ = 0;
        if (movement.forward) {
          moveX += cameraDirection.x;
          moveZ += cameraDirection.z;
        }
        if (movement.backwards) {
          moveX -= cameraDirection.x;
          moveZ -= cameraDirection.z;
        }
        if (movement.left) {
          moveX += cameraDirection.z;
          moveZ -= cameraDirection.x;
        }
        if (movement.right) {
          moveX -= cameraDirection.z;
          moveZ += cameraDirection.x;
        }

        // Normalize direction
        const len = Math.sqrt(moveX * moveX + moveZ * moveZ);
        if (len > 0) {
          moveX /= len;
          moveZ /= len;
        }

        // Set horizontal velocity directly for responsive movement
        const newVel = new Ammo.btVector3(moveX * characterSpeed, currentY, moveZ * characterSpeed);
        body.setLinearVelocity(newVel);
        Ammo.destroy(newVel);
      } else {
        // When idle, only dampen if there's meaningful horizontal velocity.
        // Avoid touching the velocity every frame to let the solver
        // handle floor contact without interference.
        const vx = linearVelocity.x();
        const vz = linearVelocity.z();
        if (Math.abs(vx) > 0.01 || Math.abs(vz) > 0.01) {
          const newVel = new Ammo.btVector3(vx * 0.85, currentY, vz * 0.85);
          body.setLinearVelocity(newVel);
          Ammo.destroy(newVel);
        }
        // Otherwise let physics handle everything — don't override velocity
      }

      // Read from the body's world transform (not motionState) because we
      // maintain it directly above — position comes from physics, rotation
      // is always identity.
      const finalTransform = body.getWorldTransform();
      const origin = finalTransform.getOrigin();
      const rotation = finalTransform.getRotation();
      const grounded = Math.abs(currentY) < 0.5;

      // The physics capsule is offset upward so its bottom aligns with the
      // character's feet.  Subtract the offset before sending the position
      // back so the visual stays at the feet, not the capsule centre.
      // Subtract the origin offset so the position maps back to model-origin
      // space (the visual's transform origin), not the AABB centre.
      const off = body._originOffset || {
        x: 0,
        y: 0,
        z: 0
      };
      const adjustedOrigin = new Ammo.btVector3(origin.x() - off.x, origin.y() - off.y, origin.z() - off.z);
      dispatcher.sendBodyUpdate(uuid, adjustedOrigin, rotation, dt, {
        grounded
      });
      Ammo.destroy(adjustedOrigin);
    };

    const requestNextFrame = self.requestAnimationFrame || self.webkitRequestAnimationFrame || self.mozRequestAnimationFrame || self.oRequestAnimationFrame || self.msRequestAnimationFrame || function (callback, _element) {
      self.setTimeout(callback, 1000 / 120);
    };
    let Clock = /*#__PURE__*/function () {
      function Clock() {
        _classCallCheck(this, Clock);
        this.timestamp = null;
      }
      return _createClass(Clock, [{
        key: "getDelta",
        value: function getDelta() {
          const time = Date.now();
          if (this.timestamp) {
            const delta = time - this.timestamp;
            this.timestamp = time;
            return delta;
          } else {
            this.timestamp = time;
            return 0;
          }
        }
      }]);
    }();
    let World = /*#__PURE__*/function () {
      function World() {
        _classCallCheck(this, World);
        _defineProperty(this, "init", options => {
          const {
            gravity = GRAVITY
          } = options;
          this.collisionConfiguration = new Ammo.btDefaultCollisionConfiguration();
          this.dispatcher = new Ammo.btCollisionDispatcher(this.collisionConfiguration);
          this.broadphase = new Ammo.btDbvtBroadphase();
          this.solver = new Ammo.btSequentialImpulseConstraintSolver();
          this.dynamicsWorld = new Ammo.btDiscreteDynamicsWorld(this.dispatcher, this.broadphase, this.solver, this.collisionConfiguration);
          this.dynamicsWorld.setGravity(new Ammo.btVector3(gravity.x, gravity.y, gravity.z));

          // this is needed for ghostObject collisions
          this.dynamicsWorld.getBroadphase().getOverlappingPairCache().setInternalGhostPairCallback(new Ammo.btGhostPairCallback());
          this.initialised = true;
        });
        _defineProperty(this, "removeElement", uuid => {
          if (this.hasElement(uuid)) {
            this.elements[uuid].deleted = true;
          }
        });
        _defineProperty(this, "removeDeletedElements", () => Object.keys(this.elements).filter(uuid => this.elements[uuid].deleted).forEach(uuid => {
          delete this.elements[uuid];
        }));
        _defineProperty(this, "hasElement", uuid => {
          return Object.keys(this.elements).includes(uuid);
        });
        _defineProperty(this, "getElement", uuid => this.elements[uuid]);
        _defineProperty(this, "isInitialised", () => this.initialised);
        _defineProperty(this, "getDynamicsWorld", () => this.dynamicsWorld);
        _defineProperty(this, "addRigidBody", body => {
          this.dynamicsWorld.addRigidBody(body);
        });
        _defineProperty(this, "addAction", action => {
          this.dynamicsWorld.addAction(action);
        });
        _defineProperty(this, "addCollisionObject", collisionObject => {
          this.dynamicsWorld.addCollisionObject(collisionObject);
        });
        _defineProperty(this, "stepSimulation", dt => {
          this.dynamicsWorld.stepSimulation(dt);
        });
        _defineProperty(this, "simulate", () => {
          const dt = this.clock.getDelta() / 1000;
          this.stepSimulation(dt);
          Object.keys(this.elements).forEach(uuid => {
            const element = this.getElement(uuid);
            if (element) {
              switch (element.type) {
                case TYPES.BOX:
                case TYPES.SPHERE:
                case TYPES.MESH:
                  handleElementUpdate(element, dt);
                  break;
                case TYPES.PLAYER:
                  handlePlayerUpdate(element, dt);
                  break;
                case TYPES.VEHICLE:
                  handleVehicleUpdate(element, dt);
                  break;
              }
            }
          });
          this.calculateCollisions();
          this.removeDeletedElements();
          dispatcher.sendPhysicsUpdate(dt);
          this.requestAnimationFrameId = requestNextFrame(this.simulate.bind(this));
        });
        _defineProperty(this, "calculateCollisions", () => {
          let ammoDispatcher = this.dynamicsWorld.getDispatcher();
          let numManifolds = ammoDispatcher.getNumManifolds();
          for (let i = 0; i < numManifolds; i++) {
            let contactManifold = ammoDispatcher.getManifoldByIndexInternal(i);
            let rb0 = Ammo.castObject(contactManifold.getBody0(), Ammo.btRigidBody);
            let rb1 = Ammo.castObject(contactManifold.getBody1(), Ammo.btRigidBody);
            let numContacts = contactManifold.getNumContacts();

            // this iteration doesn't have uuids
            if (!rb0.uuid || !rb1.uuid) continue;
            let contacts = [];
            for (let j = 0; j < numContacts; j++) {
              let contactPoint = contactManifold.getContactPoint(j);
              let distance = contactPoint.getDistance();
              if (distance > 0.0) continue;
              let velocity0 = rb0.getLinearVelocity();
              let velocity1 = rb1.getLinearVelocity();
              let worldPos0 = contactPoint.get_m_positionWorldOnA();
              let worldPos1 = contactPoint.get_m_positionWorldOnB();
              let localPos0 = contactPoint.get_m_localPointA();
              let localPos1 = contactPoint.get_m_localPointB();
              contacts.push({
                distance,
                elements: [{
                  uuid: rb0.uuid,
                  velocity: {
                    x: velocity0.x(),
                    y: velocity0.y(),
                    z: velocity0.z()
                  },
                  worldPos: {
                    x: worldPos0.x(),
                    y: worldPos0.y(),
                    z: worldPos0.z()
                  },
                  localPos: {
                    x: localPos0.x(),
                    y: localPos0.y(),
                    z: localPos0.z()
                  }
                }, {
                  uuid: rb1.uuid,
                  velocity: {
                    x: velocity1.x(),
                    y: velocity1.y(),
                    z: velocity1.z()
                  },
                  worldPos: {
                    x: worldPos1.x(),
                    y: worldPos1.y(),
                    z: worldPos1.z()
                  },
                  localPos: {
                    x: localPos1.x(),
                    y: localPos1.y(),
                    z: localPos1.z()
                  }
                }]
              });
            }
            dispatcher.sendDispatchEvent(rb0.uuid, PHYSICS_EVENTS.ELEMENT.COLLISION, {
              contacts
            });
            dispatcher.sendDispatchEvent(rb1.uuid, PHYSICS_EVENTS.ELEMENT.COLLISION, {
              contacts
            });
          }
        });
        _defineProperty(this, "addElement", data => {
          this.elements[data.uuid] = data;
        });
        _defineProperty(this, "updateBodyState", ({
          uuid,
          state
        }) => {
          if (this.hasElement(uuid)) {
            this.elements[uuid].state = {
              ...this.elements[uuid].state,
              ...state
            };
          }
        });
        _defineProperty(this, "terminate", () => {
          Ammo.destroy(this.dynamicsWorld);
          Ammo.destroy(this.solver);
          Ammo.destroy(this.dispatcher);
          Ammo.destroy(this.collisionConfiguration);
          cancelAnimationFrame(this.requestAnimationFrameId);
          dispatcher.sendTerminateEvent();
        });
        this.elements = {};
        this.initialised = false;
        this.collisionConfiguration = undefined;
        this.dispatcher = undefined;
        this.broadphase = undefined;
        this.solver = undefined;
        this.dynamicsWorld = undefined;
        this.requestAnimationFrameId = null;
        this.clock = new Clock();
      }
      return _createClass(World, [{
        key: "disposeBody",
        value: function disposeBody({
          uuid
        }) {
          const element = this.getElement(uuid);
          this.dynamicsWorld.removeRigidBody(element.body);
          this.removeElement(uuid);
          dispatcher.sendElementDisposed({
            uuid
          });
        }
      }]);
    }();
    var world = new World();

    const DEFAULT_ROLL_INFLUENCE = 0.2;
    const DEFAULT_FRICTION = 1000;
    const DEFAULT_MASS = 800;
    const addVehicle = data => {
      const {
        position,
        quaternion,
        uuid,
        wheels,
        mass = DEFAULT_MASS,
        width = 1.8,
        height = 0.6,
        length = 4,
        friction = DEFAULT_FRICTION,
        rollInfluence = DEFAULT_ROLL_INFLUENCE,
        wheelsOptions = {},
        suspensions = {}
      } = data;
      const {
        back = {},
        front = {}
      } = wheelsOptions;
      const {
        axisPosition: axisPositionBack = -1,
        radius: wheelRadiusBack = 0.4,
        halfTrack: wheelHalfTrackBack = 1,
        axisHeight: wheelAxisHeightBack = 0.3
      } = back;
      const {
        axisPosition: axisPositionFront = 1.7,
        radius: wheelRadiusFront = 0.4,
        halfTrack: wheelHalfTrackFront = 1,
        axisHeight: wheelAxisHeightFront = 0.3
      } = front;
      const {
        stiffness = 20.0,
        damping = 2.3,
        compression = 4.4,
        restLength = 0.6
      } = suspensions;

      // Chassis
      const geometry = new Ammo.btBoxShape(new Ammo.btVector3(width * 0.5, height * 0.5, length * 0.5));
      const transform = new Ammo.btTransform();
      transform.setIdentity();
      transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
      transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
      const motionState = new Ammo.btDefaultMotionState(transform);
      const localInertia = new Ammo.btVector3(0, 0, 0);
      geometry.calculateLocalInertia(mass, localInertia);
      const chassis = new Ammo.btRigidBody(new Ammo.btRigidBodyConstructionInfo(mass, motionState, geometry, localInertia));
      chassis.setActivationState(DISABLE_DEACTIVATION);
      world.addRigidBody(chassis);

      // Raycast Vehicle
      const tuning = new Ammo.btVehicleTuning();
      const rayCaster = new Ammo.btDefaultVehicleRaycaster(world.getDynamicsWorld());
      const vehicle = new Ammo.btRaycastVehicle(tuning, chassis, rayCaster);
      vehicle.setCoordinateSystem(0, 1, 2);
      world.addAction(vehicle);
      const wheelDirectionCS0 = new Ammo.btVector3(0, -1, 0);
      const wheelAxleCS = new Ammo.btVector3(-1, 0, 0);
      const addWheel = (isFront, pos, radius) => {
        var wheelInfo = vehicle.addWheel(pos, wheelDirectionCS0, wheelAxleCS, restLength, radius, tuning, isFront);
        wheelInfo.set_m_suspensionStiffness(stiffness);
        wheelInfo.set_m_wheelsDampingRelaxation(damping);
        wheelInfo.set_m_wheelsDampingCompression(compression);
        wheelInfo.set_m_frictionSlip(friction);
        wheelInfo.set_m_rollInfluence(rollInfluence);
      };
      addWheel(true, new Ammo.btVector3(wheelHalfTrackFront, wheelAxisHeightFront, axisPositionFront), wheelRadiusFront);
      addWheel(true, new Ammo.btVector3(-wheelHalfTrackFront, wheelAxisHeightFront, axisPositionFront), wheelRadiusFront);
      addWheel(false, new Ammo.btVector3(-wheelHalfTrackBack, wheelAxisHeightBack, axisPositionBack), wheelRadiusBack);
      addWheel(false, new Ammo.btVector3(wheelHalfTrackBack, wheelAxisHeightBack, axisPositionBack), wheelRadiusBack);
      vehicle.uuid = uuid;
      world.addElement({
        type: TYPES.VEHICLE,
        uuid,
        vehicle: vehicle,
        wheels,
        options: data,
        state: DEFAULT_VEHICLE_STATE
      });
    };
    const setVehiclePosition = data => {
      const {
        uuid,
        position
      } = data;
      const element = world.getElement(uuid);
      if (element.type === TYPES.VEHICLE) {
        const body = element.vehicle.getRigidBody();
        const transform = new Ammo.btTransform();
        body.getWorldTransform(transform);
        transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
        body.setWorldTransform(transform);
      }
    };
    const setVehicleQuaternion = data => {
      const {
        uuid,
        quaternion
      } = data;
      const element = world.getElement(uuid);
      if (element.type === TYPES.VEHICLE) {
        const body = element.vehicle.getRigidBody();
        const transform = new Ammo.btTransform();
        body.getWorldTransform(transform);
        transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
        body.setWorldTransform(transform);
      }
    };
    const resetVehicle = data => {
      const {
        uuid,
        quaternion,
        position
      } = data;
      const element = world.getElement(uuid);
      if (element.type === TYPES.VEHICLE) {
        const body = element.vehicle.getRigidBody();
        const transform = new Ammo.btTransform();
        body.getWorldTransform(transform);
        transform.setIdentity();
        transform.setOrigin(new Ammo.btVector3(position.x, position.y, position.z));
        transform.setRotation(new Ammo.btQuaternion(quaternion.x, quaternion.y, quaternion.z, quaternion.w));
        body.setWorldTransform(transform);
      }
    };
    const handleVehicleUpdate = ({
      vehicle,
      wheels,
      uuid,
      state = DEFAULT_VEHICLE_STATE,
      options = {}
    }, dt) => {
      let breakingForce = 0;
      let engineForce = 0;
      const {
        steeringClamp = DEFAULT_STEERING_CLAMP,
        steeringIncrement = DEFAULT_STEERING_INCREMENT,
        maxEngineForce = DEFAULT_MAX_ENGINE_FORCE,
        maxBreakingForce = DEFAULT_MAX_BREAKING_FORCE
      } = options;
      if (state.acceleration) {
        if (speed < -1) breakingForce = maxBreakingForce;else engineForce = maxEngineForce;
      }
      if (state.braking) {
        if (speed > 1) breakingForce = maxBreakingForce;else engineForce = -maxEngineForce / 2;
      }
      if (state.left) {
        if (state.vehicleSteering < steeringClamp) state.vehicleSteering += steeringIncrement;
      } else {
        if (state.right) {
          if (state.vehicleSteering > -steeringClamp) state.vehicleSteering -= steeringIncrement;
        } else {
          if (state.vehicleSteering < -steeringIncrement) state.vehicleSteering += steeringIncrement;else {
            if (state.vehicleSteering > steeringIncrement) state.vehicleSteering -= steeringIncrement;else {
              state.vehicleSteering = 0;
            }
          }
        }
      }
      vehicle.applyEngineForce(engineForce, BACK_LEFT);
      vehicle.applyEngineForce(engineForce, BACK_RIGHT);
      vehicle.setBrake(breakingForce / 2, FRONT_LEFT);
      vehicle.setBrake(breakingForce / 2, FRONT_RIGHT);
      vehicle.setBrake(breakingForce, BACK_LEFT);
      vehicle.setBrake(breakingForce, BACK_RIGHT);
      vehicle.setSteeringValue(state.vehicleSteering, FRONT_LEFT);
      vehicle.setSteeringValue(state.vehicleSteering, FRONT_RIGHT);
      let tm, p, q, i;
      const n = vehicle.getNumWheels();
      for (i = 0; i < n; i++) {
        vehicle.updateWheelTransform(i, true);
        tm = vehicle.getWheelTransformWS(i);
        p = tm.getOrigin();
        q = tm.getRotation();
        const wheelUUID = wheels[i];
        dispatcher.sendBodyUpdate(wheelUUID, p, q, dt);
      }
      tm = vehicle.getChassisWorldTransform();
      p = tm.getOrigin();
      q = tm.getRotation();
      const direction = vehicle.getForwardVector();
      const speed = vehicle.getCurrentSpeedKmHour();
      const extraData = {
        direction: {
          x: direction.x(),
          y: direction.y(),
          z: direction.z()
        },
        speed
      };
      dispatcher.sendBodyUpdate(uuid, p, q, dt, extraData);
      world.updateBodyState(uuid, state);
    };

    const createGhostCollider = (radius, position) => {
      const ghostCollider = new Ammo.btGhostObject();
      const transform = new Ammo.btTransform();
      ghostCollider.setCollisionShape(new Ammo.btSphereShape(radius));
      ghostCollider.getWorldTransform(transform);
      transform.setIdentity();
      transform.setOrigin(position);
      transform.setRotation(new Ammo.btQuaternion(0, 0, 0, 1));
      ghostCollider.setWorldTransform(transform);
      return {
        ghostCollider,
        transform
      };
    };
    const forEachGhostCollision = (ghostCollider, forEachCallback = () => {}) => {
      const collisions = ghostCollider.getNumOverlappingObjects();
      for (let i = 0; i < collisions; i++) {
        const object = Ammo.castObject(ghostCollider.getOverlappingObject(i), Ammo.btRigidBody);
        const transform = new Ammo.btTransform();
        object.getWorldTransform(transform);
        forEachCallback(object, transform, i);
        Ammo.destroy(transform);
      }
    };
    const getExplosionPosition = (uuid, position) => {
      let explosionPosition = position;
      if (!explosionPosition) {
        const {
          body
        } = world.getElement(uuid);
        const motionState = body.getMotionState();
        const transform = new Ammo.btTransform();
        motionState.getWorldTransform(transform);
        explosionPosition = transform.getOrigin();
      }
      return explosionPosition;
    };
    const getExplosionImpulse = (position, explosionPosition, strength) => {
      // Calculate direction vector WITHOUT mutating the input btVector3 objects.
      // op_sub/op_mul mutate in place in Ammo.js, which would corrupt the body's transform origin.
      const dx = position.x() - explosionPosition.x();
      const dy = position.y() - explosionPosition.y();
      const dz = position.z() - explosionPosition.z();
      let length = Math.sqrt(dx * dx + dy * dy + dz * dz);

      // If the object is at the explosion center, push it straight up instead of
      // producing an unstable near-zero normalization.
      if (length < 0.001) {
        const impulse = new Ammo.btVector3(0, strength * 2, 0);
        return impulse;
      }

      // Normalize direction
      const nx = dx / length;
      const ny = dy / length;
      const nz = dz / length;

      // Scale by strength and add upward bias
      const impulse = new Ammo.btVector3(nx * strength, ny * strength + strength, nz * strength);
      return impulse;
    };
    const createExplosion = ({
      uuid,
      position,
      radius = EXPLOSION_SIZES.SMALL,
      strength = EXPLOSION_STRENGTHS.MEDIUM
    }) => {
      try {
        // Get the source body's Ammo pointer so we can reliably skip it.
        // Ammo.castObject() creates new JS wrappers, so custom properties like
        // .uuid set on the original wrapper are NOT available on cast results.
        // We compare the underlying C++ pointer instead.
        const sourceElement = world.getElement(uuid);
        const sourceBodyPtr = sourceElement && sourceElement.body ? sourceElement.body.a || sourceElement.body.ptr : null;
        const explosionPosition = getExplosionPosition(uuid, position);
        const {
          ghostCollider,
          transform
        } = createGhostCollider(radius, explosionPosition);
        world.addCollisionObject(ghostCollider);

        // Force a collision detection pass so the ghost collider discovers
        // overlapping bodies. Without this, getNumOverlappingObjects() returns 0
        // because btGhostPairCallback only updates during a simulation step.
        world.getDynamicsWorld().performDiscreteCollisionDetection();
        forEachGhostCollision(ghostCollider, (object, objectTransform) => {
          // Skip the source entity using pointer comparison.
          const objectPtr = object.a || object.ptr;
          if (sourceBodyPtr && objectPtr === sourceBodyPtr) return;
          const origin = objectTransform.getOrigin();
          object.activate(true);
          const impulse = getExplosionImpulse(origin, explosionPosition, strength);
          object.applyCentralImpulse(impulse);
          Ammo.destroy(impulse);
        });
        world.getDynamicsWorld().removeCollisionObject(ghostCollider);
        Ammo.destroy(ghostCollider);
        Ammo.destroy(transform);
      } catch (e) {
        console.error("[Physics Worker] createExplosion error:", e);
      }
    };

    const handleLoadEvent = options => Ammo => {
      self.Ammo = Ammo;
      onmessage = ({
        data
      }) => {
        switch (data.event) {
          case PHYSICS_EVENTS.ADD.BOX:
            addBox(data);
            break;
          case PHYSICS_EVENTS.ADD.SPHERE:
            addSphere(data);
            break;
          case PHYSICS_EVENTS.ADD.VEHICLE:
            addVehicle(data);
            break;
          case PHYSICS_EVENTS.ADD.MODEL:
            addModel(data);
            break;
          case PHYSICS_EVENTS.ADD.PLAYER:
            addPlayer(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.SET.LINEAR_VELOCITY:
            setLinearVelocity(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.RESET:
            resetElement(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.SET.POSITION:
            setPosition(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.SET.QUATERNION:
            setQuaternion(data);
            break;
          case PHYSICS_EVENTS.VEHICLE.SET.POSITION:
            setVehiclePosition(data);
            break;
          case PHYSICS_EVENTS.VEHICLE.SET.QUATERNION:
            setVehicleQuaternion(data);
            break;
          case PHYSICS_EVENTS.VEHICLE.RESET:
            resetVehicle(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.APPLY.IMPULSE:
            applyImpuse(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.UPDATE:
            world.updateBodyState(data);
            break;
          case PHYSICS_EVENTS.ELEMENT.DISPOSE:
            world.disposeBody(data);
            break;
          case PHYSICS_EVENTS.EFFECTS.EXPLOSION:
            createExplosion(data);
            break;
          case PHYSICS_EVENTS.TERMINATE:
            world.terminate();
            break;
        }
      };
      world.init(options);
      dispatcher.sendReadyEvent();
      world.simulate();
    };
    const loadAmmo = options => {
      const scriptUrl = options.host + "/" + (options.path || LIBRARY_NAME);
      importScripts(scriptUrl);
      Ammo().then(handleLoadEvent(options));
    };
    onmessage = ({
      data
    }) => {
      switch (data.event) {
        case PHYSICS_EVENTS.LOAD.AMMO:
          loadAmmo(data);
          break;
      }
    };

})();

', null, false);
|
|
55724
55724
|
/* eslint-enable */const PHYSICS_EVENTS = {
|
|
55725
55725
|
DISPATCH: "physics:dispatch",
|
|
55726
55726
|
TERMINATE: "physics:terminate",
|
|
@@ -56520,13 +56520,22 @@ let Physics = /*#__PURE__*/function (_EventDispatcher) {
|
|
|
56520
56520
|
value: function add(element, options = {}) {
|
|
56521
56521
|
if (Config$1.physics().enabled) {
|
|
56522
56522
|
const {
|
|
56523
|
-
colliderType = COLLIDER_TYPES$1.BOX
|
|
56523
|
+
colliderType = COLLIDER_TYPES$1.BOX,
|
|
56524
|
+
colliderWidth,
|
|
56525
|
+
colliderHeight,
|
|
56526
|
+
colliderLength,
|
|
56527
|
+
colliderRadius,
|
|
56528
|
+
...rest
|
|
56524
56529
|
} = options;
|
|
56525
56530
|
const uuid = element.uuid();
|
|
56526
56531
|
const description = {
|
|
56527
56532
|
...mapColliderTypeToDescription(colliderType)(element),
|
|
56528
|
-
...
|
|
56533
|
+
...rest
|
|
56529
56534
|
};
|
|
56535
|
+
if (colliderWidth != null) description.width = colliderWidth;
|
|
56536
|
+
if (colliderHeight != null) description.height = colliderHeight;
|
|
56537
|
+
if (colliderLength != null) description.length = colliderLength;
|
|
56538
|
+
if (colliderRadius != null) description.radius = colliderRadius;
|
|
56530
56539
|
this.storeElement(element, options);
|
|
56531
56540
|
this.worker.postMessage({
|
|
56532
56541
|
event: mapColliderTypeToAddEvent(description.collider),
|
|
@@ -58125,10 +58134,11 @@ var Physics$1 = new Physics();let Orbit = /*#__PURE__*/function (_EventDispatche
|
|
|
58125
58134
|
_this.gizmo = new Gizmo();
|
|
58126
58135
|
_this.plane = new TransformControlsPlane();
|
|
58127
58136
|
|
|
58128
|
-
// Set gizmo and plane to layer
|
|
58129
|
-
//
|
|
58130
|
-
|
|
58131
|
-
_this.
|
|
58137
|
+
// Set gizmo and plane to layer 2 ONLY so they render in a dedicated pass
|
|
58138
|
+
// on top of everything (sky, post-processing, etc.)
|
|
58139
|
+
// Layer 1 = editor helpers (togglable), Layer 2 = gizmo (always visible)
|
|
58140
|
+
_this.gizmo.layers.set(2);
|
|
58141
|
+
_this.plane.layers.set(2);
|
|
58132
58142
|
|
|
58133
58143
|
// Helper function to set depth properties on materials
|
|
58134
58144
|
const setMaterialDepth = material => {
|
|
@@ -58142,15 +58152,15 @@ var Physics$1 = new Physics();let Orbit = /*#__PURE__*/function (_EventDispatche
|
|
|
58142
58152
|
});
|
|
58143
58153
|
};
|
|
58144
58154
|
|
|
58145
|
-
// Also set layer
|
|
58146
|
-
//
|
|
58155
|
+
// Also set layer 2 on all children recursively
|
|
58156
|
+
// Disable depthTest so gizmos render on top of everything
|
|
58147
58157
|
_this.gizmo.traverse(child => {
|
|
58148
|
-
child.layers.set(
|
|
58158
|
+
child.layers.set(2);
|
|
58149
58159
|
child.renderOrder = 999;
|
|
58150
58160
|
setMaterialDepth(child.material);
|
|
58151
58161
|
});
|
|
58152
58162
|
_this.plane.traverse(child => {
|
|
58153
|
-
child.layers.set(
|
|
58163
|
+
child.layers.set(2);
|
|
58154
58164
|
child.renderOrder = 999;
|
|
58155
58165
|
setMaterialDepth(child.material);
|
|
58156
58166
|
});
|
|
@@ -58191,8 +58201,8 @@ var Physics$1 = new Physics();let Orbit = /*#__PURE__*/function (_EventDispatche
|
|
|
58191
58201
|
_this.setAndDispatch("showY", true);
|
|
58192
58202
|
_this.setAndDispatch("showZ", true);
|
|
58193
58203
|
_this.ray = new Raycaster();
|
|
58194
|
-
// Enable layer
|
|
58195
|
-
_this.ray.layers.enable(
|
|
58204
|
+
// Enable layer 2 so raycaster can pick gizmo objects (which are on layer 2)
|
|
58205
|
+
_this.ray.layers.enable(2);
|
|
58196
58206
|
_this._tempVector = new Vector3$1();
|
|
58197
58207
|
_this._tempVector2 = new Vector3$1();
|
|
58198
58208
|
_this._tempQuaternion = new Quaternion();
|
|
@@ -60103,10 +60113,12 @@ var Controls$1 = new Controls();let Scene = /*#__PURE__*/function () {
|
|
|
60103
60113
|
key: "createCamera",
|
|
60104
60114
|
value: function createCamera(camera) {
|
|
60105
60115
|
this.camera = camera;
|
|
60106
|
-
// Enable layer 1 so camera can see editor-only objects (helpers, grid
|
|
60116
|
+
// Enable layer 1 so camera can see editor-only objects (helpers, grid)
|
|
60117
|
+
// Enable layer 2 so camera can see gizmos (rendered in separate pass)
|
|
60107
60118
|
// Mirror cameras only use layer 0, so they won't render these
|
|
60108
60119
|
if (this.camera && this.camera.getBody()) {
|
|
60109
60120
|
this.camera.getBody().layers.enable(1);
|
|
60121
|
+
this.camera.getBody().layers.enable(2);
|
|
60110
60122
|
}
|
|
60111
60123
|
}
|
|
60112
60124
|
}, {
|
|
@@ -60204,6 +60216,53 @@ var Controls$1 = new Controls();let Scene = /*#__PURE__*/function () {
|
|
|
60204
60216
|
this.renderer.setRenderTarget(null);
|
|
60205
60217
|
this.renderer.render(this.scene, this.camera.getBody());
|
|
60206
60218
|
}
|
|
60219
|
+
|
|
60220
|
+
// Render gizmo layer (layer 2) on top of everything.
|
|
60221
|
+
// Called after main render or post-processing to ensure gizmos are always visible
|
|
60222
|
+
// regardless of sky, post-processing, or sortObjects setting.
|
|
60223
|
+
}, {
|
|
60224
|
+
key: "renderGizmoLayer",
|
|
60225
|
+
value: function renderGizmoLayer() {
|
|
60226
|
+
if (!this.renderer || !this.camera) return;
|
|
60227
|
+
const cameraBody = this.camera.getBody();
|
|
60228
|
+
|
|
60229
|
+
// Save current camera layer mask
|
|
60230
|
+
const savedLayers = cameraBody.layers.mask;
|
|
60231
|
+
|
|
60232
|
+
// Set camera to only see layer 2 (gizmo layer)
|
|
60233
|
+
cameraBody.layers.set(2);
|
|
60234
|
+
|
|
60235
|
+
// Ensure we render to screen (not a post-processing buffer)
|
|
60236
|
+
this.renderer.setRenderTarget(null);
|
|
60237
|
+
// Clear only depth so gizmo renders on top of the already-drawn frame
|
|
60238
|
+
this.renderer.autoClear = false;
|
|
60239
|
+
this.renderer.clearDepth();
|
|
60240
|
+
this.renderer.render(this.scene, cameraBody);
|
|
60241
|
+
this.renderer.autoClear = true;
|
|
60242
|
+
|
|
60243
|
+
// Restore camera layers
|
|
60244
|
+
cameraBody.layers.mask = savedLayers;
|
|
60245
|
+
}
|
|
60246
|
+
|
|
60247
|
+
// Toggle editor helpers visibility (layer 1: grid, light helpers, camera helpers)
|
|
60248
|
+
// Does NOT affect gizmos (layer 2) which are always visible
|
|
60249
|
+
}, {
|
|
60250
|
+
key: "setHelpersVisible",
|
|
60251
|
+
value: function setHelpersVisible(visible) {
|
|
60252
|
+
if (!this.camera || !this.camera.getBody()) return;
|
|
60253
|
+
const cameraBody = this.camera.getBody();
|
|
60254
|
+
if (visible) {
|
|
60255
|
+
cameraBody.layers.enable(1);
|
|
60256
|
+
} else {
|
|
60257
|
+
cameraBody.layers.disable(1);
|
|
60258
|
+
}
|
|
60259
|
+
this.helpersVisible = visible;
|
|
60260
|
+
}
|
|
60261
|
+
}, {
|
|
60262
|
+
key: "getHelpersVisible",
|
|
60263
|
+
value: function getHelpersVisible() {
|
|
60264
|
+
return this.helpersVisible !== false;
|
|
60265
|
+
}
|
|
60207
60266
|
}, {
|
|
60208
60267
|
key: "setFog",
|
|
60209
60268
|
value: function setFog(color, density) {
|
|
@@ -61301,7 +61360,7 @@ function applyMiddleware() {
|
|
|
61301
61360
|
|
|
61302
61361
|
var thunk = createThunkMiddleware();
|
|
61303
61362
|
thunk.withExtraArgument = createThunkMiddleware;var name = "mage-engine";
|
|
61304
|
-
var version$1 = "3.25.
|
|
61363
|
+
var version$1 = "3.25.7";
|
|
61305
61364
|
var description = "A WebGL Javascript Game Engine, built on top of THREE.js and many other libraries.";
|
|
61306
61365
|
var main = "dist/mage.js";
|
|
61307
61366
|
var author$1 = {
|
|
@@ -64742,6 +64801,26 @@ let Element$1 = /*#__PURE__*/function (_Entity) {
|
|
|
64742
64801
|
setUpLightsAndShadows(this.getBody());
|
|
64743
64802
|
}
|
|
64744
64803
|
}
|
|
64804
|
+
}, {
|
|
64805
|
+
key: "getComputedColliderSize",
|
|
64806
|
+
value: function getComputedColliderSize() {
|
|
64807
|
+
const result = {};
|
|
64808
|
+
try {
|
|
64809
|
+
if (this.boundingBox) {
|
|
64810
|
+
const size = parseBoundingBoxSize(this.boundingBox);
|
|
64811
|
+
const scale = this.getScale();
|
|
64812
|
+
result.width = parseFloat((size.x * scale.x).toFixed(3));
|
|
64813
|
+
result.height = parseFloat((size.y * scale.y).toFixed(3));
|
|
64814
|
+
result.length = parseFloat((size.z * scale.z).toFixed(3));
|
|
64815
|
+
}
|
|
64816
|
+
if (this.boundingSphere) {
|
|
64817
|
+
result.radius = parseFloat(this.boundingSphere.radius.toFixed(3));
|
|
64818
|
+
}
|
|
64819
|
+
} catch {
|
|
64820
|
+
// bounding box/sphere not yet available
|
|
64821
|
+
}
|
|
64822
|
+
return result;
|
|
64823
|
+
}
|
|
64745
64824
|
}, {
|
|
64746
64825
|
key: "addToScene",
|
|
64747
64826
|
value: function addToScene() {
|
|
@@ -65675,7 +65754,8 @@ let Element$1 = /*#__PURE__*/function (_Entity) {
|
|
|
65675
65754
|
..._superPropGet(Element, "toJSON", this, 3)([parseJSON]),
|
|
65676
65755
|
// Physics options (state is not used by Importer, only options)
|
|
65677
65756
|
physics: {
|
|
65678
|
-
options: this.getPhysicsOptions()
|
|
65757
|
+
options: this.getPhysicsOptions(),
|
|
65758
|
+
computedSize: this.getComputedColliderSize()
|
|
65679
65759
|
},
|
|
65680
65760
|
// Textures with serialized map
|
|
65681
65761
|
textures: serializeMap(this.textures),
|
|
@@ -92772,6 +92852,11 @@ let Level = /*#__PURE__*/function (_EventDispatcher) {
|
|
|
92772
92852
|
} else {
|
|
92773
92853
|
Scene$1.render(dt);
|
|
92774
92854
|
}
|
|
92855
|
+
|
|
92856
|
+
// Render gizmo layer on top of everything (separate pass with depth clear)
|
|
92857
|
+
// This ensures gizmos are always visible regardless of sky, post-processing,
|
|
92858
|
+
// or the sortObjects=false setting used when shadows are enabled.
|
|
92859
|
+
Scene$1.renderGizmoLayer();
|
|
92775
92860
|
Particles$1.update(dt);
|
|
92776
92861
|
this.onUpdate(dt);
|
|
92777
92862
|
Scene$1.update(dt);
|
|
@@ -98062,7 +98147,877 @@ let Sound = /*#__PURE__*/function (_Entity) {
|
|
|
98062
98147
|
});
|
|
98063
98148
|
}
|
|
98064
98149
|
}]);
|
|
98065
|
-
}(Entity);
|
|
98150
|
+
}(Entity);var LineSegmentsGeometry = function () {
|
|
98151
|
+
|
|
98152
|
+
InstancedBufferGeometry.call( this );
|
|
98153
|
+
|
|
98154
|
+
this.type = 'LineSegmentsGeometry';
|
|
98155
|
+
|
|
98156
|
+
var positions = [ - 1, 2, 0, 1, 2, 0, - 1, 1, 0, 1, 1, 0, - 1, 0, 0, 1, 0, 0, - 1, - 1, 0, 1, - 1, 0 ];
|
|
98157
|
+
var uvs = [ - 1, 2, 1, 2, - 1, 1, 1, 1, - 1, - 1, 1, - 1, - 1, - 2, 1, - 2 ];
|
|
98158
|
+
var index = [ 0, 2, 1, 2, 3, 1, 2, 4, 3, 4, 5, 3, 4, 6, 5, 6, 7, 5 ];
|
|
98159
|
+
|
|
98160
|
+
this.setIndex( index );
|
|
98161
|
+
this.setAttribute( 'position', new Float32BufferAttribute( positions, 3 ) );
|
|
98162
|
+
this.setAttribute( 'uv', new Float32BufferAttribute( uvs, 2 ) );
|
|
98163
|
+
|
|
98164
|
+
};
|
|
98165
|
+
|
|
98166
|
+
LineSegmentsGeometry.prototype = Object.assign( Object.create( InstancedBufferGeometry.prototype ), {
|
|
98167
|
+
|
|
98168
|
+
constructor: LineSegmentsGeometry,
|
|
98169
|
+
|
|
98170
|
+
isLineSegmentsGeometry: true,
|
|
98171
|
+
|
|
98172
|
+
applyMatrix4: function ( matrix ) {
|
|
98173
|
+
|
|
98174
|
+
var start = this.attributes.instanceStart;
|
|
98175
|
+
var end = this.attributes.instanceEnd;
|
|
98176
|
+
|
|
98177
|
+
if ( start !== undefined ) {
|
|
98178
|
+
|
|
98179
|
+
start.applyMatrix4( matrix );
|
|
98180
|
+
|
|
98181
|
+
end.applyMatrix4( matrix );
|
|
98182
|
+
|
|
98183
|
+
start.needsUpdate = true;
|
|
98184
|
+
|
|
98185
|
+
}
|
|
98186
|
+
|
|
98187
|
+
if ( this.boundingBox !== null ) {
|
|
98188
|
+
|
|
98189
|
+
this.computeBoundingBox();
|
|
98190
|
+
|
|
98191
|
+
}
|
|
98192
|
+
|
|
98193
|
+
if ( this.boundingSphere !== null ) {
|
|
98194
|
+
|
|
98195
|
+
this.computeBoundingSphere();
|
|
98196
|
+
|
|
98197
|
+
}
|
|
98198
|
+
|
|
98199
|
+
return this;
|
|
98200
|
+
|
|
98201
|
+
},
|
|
98202
|
+
|
|
98203
|
+
setPositions: function ( array ) {
|
|
98204
|
+
|
|
98205
|
+
var lineSegments;
|
|
98206
|
+
|
|
98207
|
+
if ( array instanceof Float32Array ) {
|
|
98208
|
+
|
|
98209
|
+
lineSegments = array;
|
|
98210
|
+
|
|
98211
|
+
} else if ( Array.isArray( array ) ) {
|
|
98212
|
+
|
|
98213
|
+
lineSegments = new Float32Array( array );
|
|
98214
|
+
|
|
98215
|
+
}
|
|
98216
|
+
|
|
98217
|
+
var instanceBuffer = new InstancedInterleavedBuffer( lineSegments, 6, 1 ); // xyz, xyz
|
|
98218
|
+
|
|
98219
|
+
this.setAttribute( 'instanceStart', new InterleavedBufferAttribute( instanceBuffer, 3, 0 ) ); // xyz
|
|
98220
|
+
this.setAttribute( 'instanceEnd', new InterleavedBufferAttribute( instanceBuffer, 3, 3 ) ); // xyz
|
|
98221
|
+
|
|
98222
|
+
//
|
|
98223
|
+
|
|
98224
|
+
this.computeBoundingBox();
|
|
98225
|
+
this.computeBoundingSphere();
|
|
98226
|
+
|
|
98227
|
+
return this;
|
|
98228
|
+
|
|
98229
|
+
},
|
|
98230
|
+
|
|
98231
|
+
setColors: function ( array ) {
|
|
98232
|
+
|
|
98233
|
+
var colors;
|
|
98234
|
+
|
|
98235
|
+
if ( array instanceof Float32Array ) {
|
|
98236
|
+
|
|
98237
|
+
colors = array;
|
|
98238
|
+
|
|
98239
|
+
} else if ( Array.isArray( array ) ) {
|
|
98240
|
+
|
|
98241
|
+
colors = new Float32Array( array );
|
|
98242
|
+
|
|
98243
|
+
}
|
|
98244
|
+
|
|
98245
|
+
var instanceColorBuffer = new InstancedInterleavedBuffer( colors, 6, 1 ); // rgb, rgb
|
|
98246
|
+
|
|
98247
|
+
this.setAttribute( 'instanceColorStart', new InterleavedBufferAttribute( instanceColorBuffer, 3, 0 ) ); // rgb
|
|
98248
|
+
this.setAttribute( 'instanceColorEnd', new InterleavedBufferAttribute( instanceColorBuffer, 3, 3 ) ); // rgb
|
|
98249
|
+
|
|
98250
|
+
return this;
|
|
98251
|
+
|
|
98252
|
+
},
|
|
98253
|
+
|
|
98254
|
+
fromWireframeGeometry: function ( geometry ) {
|
|
98255
|
+
|
|
98256
|
+
this.setPositions( geometry.attributes.position.array );
|
|
98257
|
+
|
|
98258
|
+
return this;
|
|
98259
|
+
|
|
98260
|
+
},
|
|
98261
|
+
|
|
98262
|
+
fromEdgesGeometry: function ( geometry ) {
|
|
98263
|
+
|
|
98264
|
+
this.setPositions( geometry.attributes.position.array );
|
|
98265
|
+
|
|
98266
|
+
return this;
|
|
98267
|
+
|
|
98268
|
+
},
|
|
98269
|
+
|
|
98270
|
+
fromMesh: function ( mesh ) {
|
|
98271
|
+
|
|
98272
|
+
this.fromWireframeGeometry( new WireframeGeometry( mesh.geometry ) );
|
|
98273
|
+
|
|
98274
|
+
// set colors, maybe
|
|
98275
|
+
|
|
98276
|
+
return this;
|
|
98277
|
+
|
|
98278
|
+
},
|
|
98279
|
+
|
|
98280
|
+
fromLineSegments: function ( lineSegments ) {
|
|
98281
|
+
|
|
98282
|
+
var geometry = lineSegments.geometry;
|
|
98283
|
+
|
|
98284
|
+
if ( geometry.isGeometry ) {
|
|
98285
|
+
|
|
98286
|
+
console.error( 'THREE.LineSegmentsGeometry no longer supports Geometry. Use THREE.BufferGeometry instead.' );
|
|
98287
|
+
return;
|
|
98288
|
+
|
|
98289
|
+
} else if ( geometry.isBufferGeometry ) {
|
|
98290
|
+
|
|
98291
|
+
this.setPositions( geometry.attributes.position.array ); // assumes non-indexed
|
|
98292
|
+
|
|
98293
|
+
}
|
|
98294
|
+
|
|
98295
|
+
// set colors, maybe
|
|
98296
|
+
|
|
98297
|
+
return this;
|
|
98298
|
+
|
|
98299
|
+
},
|
|
98300
|
+
|
|
98301
|
+
computeBoundingBox: function () {
|
|
98302
|
+
|
|
98303
|
+
var box = new Box3();
|
|
98304
|
+
|
|
98305
|
+
return function computeBoundingBox() {
|
|
98306
|
+
|
|
98307
|
+
if ( this.boundingBox === null ) {
|
|
98308
|
+
|
|
98309
|
+
this.boundingBox = new Box3();
|
|
98310
|
+
|
|
98311
|
+
}
|
|
98312
|
+
|
|
98313
|
+
var start = this.attributes.instanceStart;
|
|
98314
|
+
var end = this.attributes.instanceEnd;
|
|
98315
|
+
|
|
98316
|
+
if ( start !== undefined && end !== undefined ) {
|
|
98317
|
+
|
|
98318
|
+
this.boundingBox.setFromBufferAttribute( start );
|
|
98319
|
+
|
|
98320
|
+
box.setFromBufferAttribute( end );
|
|
98321
|
+
|
|
98322
|
+
this.boundingBox.union( box );
|
|
98323
|
+
|
|
98324
|
+
}
|
|
98325
|
+
|
|
98326
|
+
};
|
|
98327
|
+
|
|
98328
|
+
}(),
|
|
98329
|
+
|
|
98330
|
+
computeBoundingSphere: function () {
|
|
98331
|
+
|
|
98332
|
+
var vector = new Vector3$1();
|
|
98333
|
+
|
|
98334
|
+
return function computeBoundingSphere() {
|
|
98335
|
+
|
|
98336
|
+
if ( this.boundingSphere === null ) {
|
|
98337
|
+
|
|
98338
|
+
this.boundingSphere = new Sphere$1();
|
|
98339
|
+
|
|
98340
|
+
}
|
|
98341
|
+
|
|
98342
|
+
if ( this.boundingBox === null ) {
|
|
98343
|
+
|
|
98344
|
+
this.computeBoundingBox();
|
|
98345
|
+
|
|
98346
|
+
}
|
|
98347
|
+
|
|
98348
|
+
var start = this.attributes.instanceStart;
|
|
98349
|
+
var end = this.attributes.instanceEnd;
|
|
98350
|
+
|
|
98351
|
+
if ( start !== undefined && end !== undefined ) {
|
|
98352
|
+
|
|
98353
|
+
var center = this.boundingSphere.center;
|
|
98354
|
+
|
|
98355
|
+
this.boundingBox.getCenter( center );
|
|
98356
|
+
|
|
98357
|
+
var maxRadiusSq = 0;
|
|
98358
|
+
|
|
98359
|
+
for ( var i = 0, il = start.count; i < il; i ++ ) {
|
|
98360
|
+
|
|
98361
|
+
vector.fromBufferAttribute( start, i );
|
|
98362
|
+
maxRadiusSq = Math.max( maxRadiusSq, center.distanceToSquared( vector ) );
|
|
98363
|
+
|
|
98364
|
+
vector.fromBufferAttribute( end, i );
|
|
98365
|
+
maxRadiusSq = Math.max( maxRadiusSq, center.distanceToSquared( vector ) );
|
|
98366
|
+
|
|
98367
|
+
}
|
|
98368
|
+
|
|
98369
|
+
this.boundingSphere.radius = Math.sqrt( maxRadiusSq );
|
|
98370
|
+
|
|
98371
|
+
if ( isNaN( this.boundingSphere.radius ) ) {
|
|
98372
|
+
|
|
98373
|
+
console.error( 'THREE.LineSegmentsGeometry.computeBoundingSphere(): Computed radius is NaN. The instanced position data is likely to have NaN values.', this );
|
|
98374
|
+
|
|
98375
|
+
}
|
|
98376
|
+
|
|
98377
|
+
}
|
|
98378
|
+
|
|
98379
|
+
};
|
|
98380
|
+
|
|
98381
|
+
}(),
|
|
98382
|
+
|
|
98383
|
+
toJSON: function () {
|
|
98384
|
+
|
|
98385
|
+
// todo
|
|
98386
|
+
|
|
98387
|
+
},
|
|
98388
|
+
|
|
98389
|
+
applyMatrix: function ( matrix ) {
|
|
98390
|
+
|
|
98391
|
+
console.warn( 'THREE.LineSegmentsGeometry: applyMatrix() has been renamed to applyMatrix4().' );
|
|
98392
|
+
|
|
98393
|
+
return this.applyMatrix4( matrix );
|
|
98394
|
+
|
|
98395
|
+
}
|
|
98396
|
+
|
|
98397
|
+
} );/**
|
|
98398
|
+
* parameters = {
|
|
98399
|
+
* color: <hex>,
|
|
98400
|
+
* linewidth: <float>,
|
|
98401
|
+
* dashed: <boolean>,
|
|
98402
|
+
* dashScale: <float>,
|
|
98403
|
+
* dashSize: <float>,
|
|
98404
|
+
* dashOffset: <float>,
|
|
98405
|
+
* gapSize: <float>,
|
|
98406
|
+
* resolution: <Vector2>, // to be set by renderer
|
|
98407
|
+
* }
|
|
98408
|
+
*/
|
|
98409
|
+
|
|
98410
|
+
UniformsLib.line = {
|
|
98411
|
+
|
|
98412
|
+
linewidth: { value: 1 },
|
|
98413
|
+
resolution: { value: new Vector2( 1, 1 ) },
|
|
98414
|
+
dashScale: { value: 1 },
|
|
98415
|
+
dashSize: { value: 1 },
|
|
98416
|
+
dashOffset: { value: 0 },
|
|
98417
|
+
gapSize: { value: 1 }, // todo FIX - maybe change to totalSize
|
|
98418
|
+
opacity: { value: 1 }
|
|
98419
|
+
|
|
98420
|
+
};
|
|
98421
|
+
|
|
98422
|
+
ShaderLib[ 'line' ] = {
|
|
98423
|
+
|
|
98424
|
+
uniforms: UniformsUtils.merge( [
|
|
98425
|
+
UniformsLib.common,
|
|
98426
|
+
UniformsLib.fog,
|
|
98427
|
+
UniformsLib.line
|
|
98428
|
+
] ),
|
|
98429
|
+
|
|
98430
|
+
vertexShader:
|
|
98431
|
+
`
|
|
98432
|
+
#include <common>
|
|
98433
|
+
#include <color_pars_vertex>
|
|
98434
|
+
#include <fog_pars_vertex>
|
|
98435
|
+
#include <logdepthbuf_pars_vertex>
|
|
98436
|
+
#include <clipping_planes_pars_vertex>
|
|
98437
|
+
|
|
98438
|
+
uniform float linewidth;
|
|
98439
|
+
uniform vec2 resolution;
|
|
98440
|
+
|
|
98441
|
+
attribute vec3 instanceStart;
|
|
98442
|
+
attribute vec3 instanceEnd;
|
|
98443
|
+
|
|
98444
|
+
attribute vec3 instanceColorStart;
|
|
98445
|
+
attribute vec3 instanceColorEnd;
|
|
98446
|
+
|
|
98447
|
+
varying vec2 vUv;
|
|
98448
|
+
|
|
98449
|
+
#ifdef USE_DASH
|
|
98450
|
+
|
|
98451
|
+
uniform float dashScale;
|
|
98452
|
+
attribute float instanceDistanceStart;
|
|
98453
|
+
attribute float instanceDistanceEnd;
|
|
98454
|
+
varying float vLineDistance;
|
|
98455
|
+
|
|
98456
|
+
#endif
|
|
98457
|
+
|
|
98458
|
+
void trimSegment( const in vec4 start, inout vec4 end ) {
|
|
98459
|
+
|
|
98460
|
+
// trim end segment so it terminates between the camera plane and the near plane
|
|
98461
|
+
|
|
98462
|
+
// conservative estimate of the near plane
|
|
98463
|
+
float a = projectionMatrix[ 2 ][ 2 ]; // 3nd entry in 3th column
|
|
98464
|
+
float b = projectionMatrix[ 3 ][ 2 ]; // 3nd entry in 4th column
|
|
98465
|
+
float nearEstimate = - 0.5 * b / a;
|
|
98466
|
+
|
|
98467
|
+
float alpha = ( nearEstimate - start.z ) / ( end.z - start.z );
|
|
98468
|
+
|
|
98469
|
+
end.xyz = mix( start.xyz, end.xyz, alpha );
|
|
98470
|
+
|
|
98471
|
+
}
|
|
98472
|
+
|
|
98473
|
+
void main() {
|
|
98474
|
+
|
|
98475
|
+
#ifdef USE_COLOR
|
|
98476
|
+
|
|
98477
|
+
vColor.xyz = ( position.y < 0.5 ) ? instanceColorStart : instanceColorEnd;
|
|
98478
|
+
|
|
98479
|
+
#endif
|
|
98480
|
+
|
|
98481
|
+
#ifdef USE_DASH
|
|
98482
|
+
|
|
98483
|
+
vLineDistance = ( position.y < 0.5 ) ? dashScale * instanceDistanceStart : dashScale * instanceDistanceEnd;
|
|
98484
|
+
|
|
98485
|
+
#endif
|
|
98486
|
+
|
|
98487
|
+
float aspect = resolution.x / resolution.y;
|
|
98488
|
+
|
|
98489
|
+
vUv = uv;
|
|
98490
|
+
|
|
98491
|
+
// camera space
|
|
98492
|
+
vec4 start = modelViewMatrix * vec4( instanceStart, 1.0 );
|
|
98493
|
+
vec4 end = modelViewMatrix * vec4( instanceEnd, 1.0 );
|
|
98494
|
+
|
|
98495
|
+
// special case for perspective projection, and segments that terminate either in, or behind, the camera plane
|
|
98496
|
+
// clearly the gpu firmware has a way of addressing this issue when projecting into ndc space
|
|
98497
|
+
// but we need to perform ndc-space calculations in the shader, so we must address this issue directly
|
|
98498
|
+
// perhaps there is a more elegant solution -- WestLangley
|
|
98499
|
+
|
|
98500
|
+
bool perspective = ( projectionMatrix[ 2 ][ 3 ] == - 1.0 ); // 4th entry in the 3rd column
|
|
98501
|
+
|
|
98502
|
+
if ( perspective ) {
|
|
98503
|
+
|
|
98504
|
+
if ( start.z < 0.0 && end.z >= 0.0 ) {
|
|
98505
|
+
|
|
98506
|
+
trimSegment( start, end );
|
|
98507
|
+
|
|
98508
|
+
} else if ( end.z < 0.0 && start.z >= 0.0 ) {
|
|
98509
|
+
|
|
98510
|
+
trimSegment( end, start );
|
|
98511
|
+
|
|
98512
|
+
}
|
|
98513
|
+
|
|
98514
|
+
}
|
|
98515
|
+
|
|
98516
|
+
// clip space
|
|
98517
|
+
vec4 clipStart = projectionMatrix * start;
|
|
98518
|
+
vec4 clipEnd = projectionMatrix * end;
|
|
98519
|
+
|
|
98520
|
+
// ndc space
|
|
98521
|
+
vec2 ndcStart = clipStart.xy / clipStart.w;
|
|
98522
|
+
vec2 ndcEnd = clipEnd.xy / clipEnd.w;
|
|
98523
|
+
|
|
98524
|
+
// direction
|
|
98525
|
+
vec2 dir = ndcEnd - ndcStart;
|
|
98526
|
+
|
|
98527
|
+
// account for clip-space aspect ratio
|
|
98528
|
+
dir.x *= aspect;
|
|
98529
|
+
dir = normalize( dir );
|
|
98530
|
+
|
|
98531
|
+
// perpendicular to dir
|
|
98532
|
+
vec2 offset = vec2( dir.y, - dir.x );
|
|
98533
|
+
|
|
98534
|
+
// undo aspect ratio adjustment
|
|
98535
|
+
dir.x /= aspect;
|
|
98536
|
+
offset.x /= aspect;
|
|
98537
|
+
|
|
98538
|
+
// sign flip
|
|
98539
|
+
if ( position.x < 0.0 ) offset *= - 1.0;
|
|
98540
|
+
|
|
98541
|
+
// endcaps
|
|
98542
|
+
if ( position.y < 0.0 ) {
|
|
98543
|
+
|
|
98544
|
+
offset += - dir;
|
|
98545
|
+
|
|
98546
|
+
} else if ( position.y > 1.0 ) {
|
|
98547
|
+
|
|
98548
|
+
offset += dir;
|
|
98549
|
+
|
|
98550
|
+
}
|
|
98551
|
+
|
|
98552
|
+
// adjust for linewidth
|
|
98553
|
+
offset *= linewidth;
|
|
98554
|
+
|
|
98555
|
+
// adjust for clip-space to screen-space conversion // maybe resolution should be based on viewport ...
|
|
98556
|
+
offset /= resolution.y;
|
|
98557
|
+
|
|
98558
|
+
// select end
|
|
98559
|
+
vec4 clip = ( position.y < 0.5 ) ? clipStart : clipEnd;
|
|
98560
|
+
|
|
98561
|
+
// back to clip space
|
|
98562
|
+
offset *= clip.w;
|
|
98563
|
+
|
|
98564
|
+
clip.xy += offset;
|
|
98565
|
+
|
|
98566
|
+
gl_Position = clip;
|
|
98567
|
+
|
|
98568
|
+
vec4 mvPosition = ( position.y < 0.5 ) ? start : end; // this is an approximation
|
|
98569
|
+
|
|
98570
|
+
#include <logdepthbuf_vertex>
|
|
98571
|
+
#include <clipping_planes_vertex>
|
|
98572
|
+
#include <fog_vertex>
|
|
98573
|
+
|
|
98574
|
+
}
|
|
98575
|
+
`,
|
|
98576
|
+
|
|
98577
|
+
fragmentShader:
|
|
98578
|
+
`
|
|
98579
|
+
uniform vec3 diffuse;
|
|
98580
|
+
uniform float opacity;
|
|
98581
|
+
|
|
98582
|
+
#ifdef USE_DASH
|
|
98583
|
+
|
|
98584
|
+
uniform float dashSize;
|
|
98585
|
+
uniform float dashOffset;
|
|
98586
|
+
uniform float gapSize;
|
|
98587
|
+
|
|
98588
|
+
#endif
|
|
98589
|
+
|
|
98590
|
+
varying float vLineDistance;
|
|
98591
|
+
|
|
98592
|
+
#include <common>
|
|
98593
|
+
#include <color_pars_fragment>
|
|
98594
|
+
#include <fog_pars_fragment>
|
|
98595
|
+
#include <logdepthbuf_pars_fragment>
|
|
98596
|
+
#include <clipping_planes_pars_fragment>
|
|
98597
|
+
|
|
98598
|
+
varying vec2 vUv;
|
|
98599
|
+
|
|
98600
|
+
void main() {
|
|
98601
|
+
|
|
98602
|
+
#include <clipping_planes_fragment>
|
|
98603
|
+
|
|
98604
|
+
#ifdef USE_DASH
|
|
98605
|
+
|
|
98606
|
+
if ( vUv.y < - 1.0 || vUv.y > 1.0 ) discard; // discard endcaps
|
|
98607
|
+
|
|
98608
|
+
if ( mod( vLineDistance + dashOffset, dashSize + gapSize ) > dashSize ) discard; // todo - FIX
|
|
98609
|
+
|
|
98610
|
+
#endif
|
|
98611
|
+
|
|
98612
|
+
if ( abs( vUv.y ) > 1.0 ) {
|
|
98613
|
+
|
|
98614
|
+
float a = vUv.x;
|
|
98615
|
+
float b = ( vUv.y > 0.0 ) ? vUv.y - 1.0 : vUv.y + 1.0;
|
|
98616
|
+
float len2 = a * a + b * b;
|
|
98617
|
+
|
|
98618
|
+
if ( len2 > 1.0 ) discard;
|
|
98619
|
+
|
|
98620
|
+
}
|
|
98621
|
+
|
|
98622
|
+
vec4 diffuseColor = vec4( diffuse, opacity );
|
|
98623
|
+
|
|
98624
|
+
#include <logdepthbuf_fragment>
|
|
98625
|
+
#include <color_fragment>
|
|
98626
|
+
|
|
98627
|
+
gl_FragColor = vec4( diffuseColor.rgb, diffuseColor.a );
|
|
98628
|
+
|
|
98629
|
+
#include <tonemapping_fragment>
|
|
98630
|
+
#include <encodings_fragment>
|
|
98631
|
+
#include <fog_fragment>
|
|
98632
|
+
#include <premultiplied_alpha_fragment>
|
|
98633
|
+
|
|
98634
|
+
}
|
|
98635
|
+
`
|
|
98636
|
+
};
|
|
98637
|
+
|
|
98638
|
+
var LineMaterial = function ( parameters ) {
|
|
98639
|
+
|
|
98640
|
+
ShaderMaterial.call( this, {
|
|
98641
|
+
|
|
98642
|
+
type: 'LineMaterial',
|
|
98643
|
+
|
|
98644
|
+
uniforms: UniformsUtils.clone( ShaderLib[ 'line' ].uniforms ),
|
|
98645
|
+
|
|
98646
|
+
vertexShader: ShaderLib[ 'line' ].vertexShader,
|
|
98647
|
+
fragmentShader: ShaderLib[ 'line' ].fragmentShader,
|
|
98648
|
+
|
|
98649
|
+
clipping: true // required for clipping support
|
|
98650
|
+
|
|
98651
|
+
} );
|
|
98652
|
+
|
|
98653
|
+
this.dashed = false;
|
|
98654
|
+
|
|
98655
|
+
Object.defineProperties( this, {
|
|
98656
|
+
|
|
98657
|
+
color: {
|
|
98658
|
+
|
|
98659
|
+
enumerable: true,
|
|
98660
|
+
|
|
98661
|
+
get: function () {
|
|
98662
|
+
|
|
98663
|
+
return this.uniforms.diffuse.value;
|
|
98664
|
+
|
|
98665
|
+
},
|
|
98666
|
+
|
|
98667
|
+
set: function ( value ) {
|
|
98668
|
+
|
|
98669
|
+
this.uniforms.diffuse.value = value;
|
|
98670
|
+
|
|
98671
|
+
}
|
|
98672
|
+
|
|
98673
|
+
},
|
|
98674
|
+
|
|
98675
|
+
linewidth: {
|
|
98676
|
+
|
|
98677
|
+
enumerable: true,
|
|
98678
|
+
|
|
98679
|
+
get: function () {
|
|
98680
|
+
|
|
98681
|
+
return this.uniforms.linewidth.value;
|
|
98682
|
+
|
|
98683
|
+
},
|
|
98684
|
+
|
|
98685
|
+
set: function ( value ) {
|
|
98686
|
+
|
|
98687
|
+
this.uniforms.linewidth.value = value;
|
|
98688
|
+
|
|
98689
|
+
}
|
|
98690
|
+
|
|
98691
|
+
},
|
|
98692
|
+
|
|
98693
|
+
dashScale: {
|
|
98694
|
+
|
|
98695
|
+
enumerable: true,
|
|
98696
|
+
|
|
98697
|
+
get: function () {
|
|
98698
|
+
|
|
98699
|
+
return this.uniforms.dashScale.value;
|
|
98700
|
+
|
|
98701
|
+
},
|
|
98702
|
+
|
|
98703
|
+
set: function ( value ) {
|
|
98704
|
+
|
|
98705
|
+
this.uniforms.dashScale.value = value;
|
|
98706
|
+
|
|
98707
|
+
}
|
|
98708
|
+
|
|
98709
|
+
},
|
|
98710
|
+
|
|
98711
|
+
dashSize: {
|
|
98712
|
+
|
|
98713
|
+
enumerable: true,
|
|
98714
|
+
|
|
98715
|
+
get: function () {
|
|
98716
|
+
|
|
98717
|
+
return this.uniforms.dashSize.value;
|
|
98718
|
+
|
|
98719
|
+
},
|
|
98720
|
+
|
|
98721
|
+
set: function ( value ) {
|
|
98722
|
+
|
|
98723
|
+
this.uniforms.dashSize.value = value;
|
|
98724
|
+
|
|
98725
|
+
}
|
|
98726
|
+
|
|
98727
|
+
},
|
|
98728
|
+
|
|
98729
|
+
dashOffset: {
|
|
98730
|
+
|
|
98731
|
+
enumerable: true,
|
|
98732
|
+
|
|
98733
|
+
get: function () {
|
|
98734
|
+
|
|
98735
|
+
return this.uniforms.dashOffset.value;
|
|
98736
|
+
|
|
98737
|
+
},
|
|
98738
|
+
|
|
98739
|
+
set: function ( value ) {
|
|
98740
|
+
|
|
98741
|
+
this.uniforms.dashOffset.value = value;
|
|
98742
|
+
|
|
98743
|
+
}
|
|
98744
|
+
|
|
98745
|
+
},
|
|
98746
|
+
|
|
98747
|
+
gapSize: {
|
|
98748
|
+
|
|
98749
|
+
enumerable: true,
|
|
98750
|
+
|
|
98751
|
+
get: function () {
|
|
98752
|
+
|
|
98753
|
+
return this.uniforms.gapSize.value;
|
|
98754
|
+
|
|
98755
|
+
},
|
|
98756
|
+
|
|
98757
|
+
set: function ( value ) {
|
|
98758
|
+
|
|
98759
|
+
this.uniforms.gapSize.value = value;
|
|
98760
|
+
|
|
98761
|
+
}
|
|
98762
|
+
|
|
98763
|
+
},
|
|
98764
|
+
|
|
98765
|
+
opacity: {
|
|
98766
|
+
|
|
98767
|
+
enumerable: true,
|
|
98768
|
+
|
|
98769
|
+
get: function () {
|
|
98770
|
+
|
|
98771
|
+
return this.uniforms.opacity.value;
|
|
98772
|
+
|
|
98773
|
+
},
|
|
98774
|
+
|
|
98775
|
+
set: function ( value ) {
|
|
98776
|
+
|
|
98777
|
+
this.uniforms.opacity.value = value;
|
|
98778
|
+
|
|
98779
|
+
}
|
|
98780
|
+
|
|
98781
|
+
},
|
|
98782
|
+
|
|
98783
|
+
resolution: {
|
|
98784
|
+
|
|
98785
|
+
enumerable: true,
|
|
98786
|
+
|
|
98787
|
+
get: function () {
|
|
98788
|
+
|
|
98789
|
+
return this.uniforms.resolution.value;
|
|
98790
|
+
|
|
98791
|
+
},
|
|
98792
|
+
|
|
98793
|
+
set: function ( value ) {
|
|
98794
|
+
|
|
98795
|
+
this.uniforms.resolution.value.copy( value );
|
|
98796
|
+
|
|
98797
|
+
}
|
|
98798
|
+
|
|
98799
|
+
}
|
|
98800
|
+
|
|
98801
|
+
} );
|
|
98802
|
+
|
|
98803
|
+
this.setValues( parameters );
|
|
98804
|
+
|
|
98805
|
+
};
|
|
98806
|
+
|
|
98807
|
+
LineMaterial.prototype = Object.create( ShaderMaterial.prototype );
|
|
98808
|
+
LineMaterial.prototype.constructor = LineMaterial;
|
|
98809
|
+
|
|
98810
|
+
LineMaterial.prototype.isLineMaterial = true;var LineSegments2 = function ( geometry, material ) {
|
|
98811
|
+
|
|
98812
|
+
if ( geometry === undefined ) geometry = new LineSegmentsGeometry();
|
|
98813
|
+
if ( material === undefined ) material = new LineMaterial( { color: Math.random() * 0xffffff } );
|
|
98814
|
+
|
|
98815
|
+
Mesh.call( this, geometry, material );
|
|
98816
|
+
|
|
98817
|
+
this.type = 'LineSegments2';
|
|
98818
|
+
|
|
98819
|
+
};
|
|
98820
|
+
|
|
98821
|
+
LineSegments2.prototype = Object.assign( Object.create( Mesh.prototype ), {
|
|
98822
|
+
|
|
98823
|
+
constructor: LineSegments2,
|
|
98824
|
+
|
|
98825
|
+
isLineSegments2: true,
|
|
98826
|
+
|
|
98827
|
+
computeLineDistances: ( function () { // for backwards-compatability, but could be a method of LineSegmentsGeometry...
|
|
98828
|
+
|
|
98829
|
+
var start = new Vector3$1();
|
|
98830
|
+
var end = new Vector3$1();
|
|
98831
|
+
|
|
98832
|
+
return function computeLineDistances() {
|
|
98833
|
+
|
|
98834
|
+
var geometry = this.geometry;
|
|
98835
|
+
|
|
98836
|
+
var instanceStart = geometry.attributes.instanceStart;
|
|
98837
|
+
var instanceEnd = geometry.attributes.instanceEnd;
|
|
98838
|
+
var lineDistances = new Float32Array( 2 * instanceStart.data.count );
|
|
98839
|
+
|
|
98840
|
+
for ( var i = 0, j = 0, l = instanceStart.data.count; i < l; i ++, j += 2 ) {
|
|
98841
|
+
|
|
98842
|
+
start.fromBufferAttribute( instanceStart, i );
|
|
98843
|
+
end.fromBufferAttribute( instanceEnd, i );
|
|
98844
|
+
|
|
98845
|
+
lineDistances[ j ] = ( j === 0 ) ? 0 : lineDistances[ j - 1 ];
|
|
98846
|
+
lineDistances[ j + 1 ] = lineDistances[ j ] + start.distanceTo( end );
|
|
98847
|
+
|
|
98848
|
+
}
|
|
98849
|
+
|
|
98850
|
+
var instanceDistanceBuffer = new InstancedInterleavedBuffer( lineDistances, 2, 1 ); // d0, d1
|
|
98851
|
+
|
|
98852
|
+
geometry.setAttribute( 'instanceDistanceStart', new InterleavedBufferAttribute( instanceDistanceBuffer, 1, 0 ) ); // d0
|
|
98853
|
+
geometry.setAttribute( 'instanceDistanceEnd', new InterleavedBufferAttribute( instanceDistanceBuffer, 1, 1 ) ); // d1
|
|
98854
|
+
|
|
98855
|
+
return this;
|
|
98856
|
+
|
|
98857
|
+
};
|
|
98858
|
+
|
|
98859
|
+
}() ),
|
|
98860
|
+
|
|
98861
|
+
raycast: ( function () {
|
|
98862
|
+
|
|
98863
|
+
var start = new Vector4();
|
|
98864
|
+
var end = new Vector4();
|
|
98865
|
+
|
|
98866
|
+
var ssOrigin = new Vector4();
|
|
98867
|
+
var ssOrigin3 = new Vector3$1();
|
|
98868
|
+
var mvMatrix = new Matrix4();
|
|
98869
|
+
var line = new Line3();
|
|
98870
|
+
var closestPoint = new Vector3$1();
|
|
98871
|
+
|
|
98872
|
+
return function raycast( raycaster, intersects ) {
|
|
98873
|
+
|
|
98874
|
+
if ( raycaster.camera === null ) {
|
|
98875
|
+
|
|
98876
|
+
console.error( 'LineSegments2: "Raycaster.camera" needs to be set in order to raycast against LineSegments2.' );
|
|
98877
|
+
|
|
98878
|
+
}
|
|
98879
|
+
|
|
98880
|
+
var threshold = ( raycaster.params.Line2 !== undefined ) ? raycaster.params.Line2.threshold || 0 : 0;
|
|
98881
|
+
|
|
98882
|
+
var ray = raycaster.ray;
|
|
98883
|
+
var camera = raycaster.camera;
|
|
98884
|
+
var projectionMatrix = camera.projectionMatrix;
|
|
98885
|
+
|
|
98886
|
+
var geometry = this.geometry;
|
|
98887
|
+
var material = this.material;
|
|
98888
|
+
var resolution = material.resolution;
|
|
98889
|
+
var lineWidth = material.linewidth + threshold;
|
|
98890
|
+
|
|
98891
|
+
var instanceStart = geometry.attributes.instanceStart;
|
|
98892
|
+
var instanceEnd = geometry.attributes.instanceEnd;
|
|
98893
|
+
|
|
98894
|
+
// camera forward is negative
|
|
98895
|
+
var near = - camera.near;
|
|
98896
|
+
|
|
98897
|
+
// pick a point 1 unit out along the ray to avoid the ray origin
|
|
98898
|
+
// sitting at the camera origin which will cause "w" to be 0 when
|
|
98899
|
+
// applying the projection matrix.
|
|
98900
|
+
ray.at( 1, ssOrigin );
|
|
98901
|
+
|
|
98902
|
+
// ndc space [ - 1.0, 1.0 ]
|
|
98903
|
+
ssOrigin.w = 1;
|
|
98904
|
+
ssOrigin.applyMatrix4( camera.matrixWorldInverse );
|
|
98905
|
+
ssOrigin.applyMatrix4( projectionMatrix );
|
|
98906
|
+
ssOrigin.multiplyScalar( 1 / ssOrigin.w );
|
|
98907
|
+
|
|
98908
|
+
// screen space
|
|
98909
|
+
ssOrigin.x *= resolution.x / 2;
|
|
98910
|
+
ssOrigin.y *= resolution.y / 2;
|
|
98911
|
+
ssOrigin.z = 0;
|
|
98912
|
+
|
|
98913
|
+
ssOrigin3.copy( ssOrigin );
|
|
98914
|
+
|
|
98915
|
+
var matrixWorld = this.matrixWorld;
|
|
98916
|
+
mvMatrix.multiplyMatrices( camera.matrixWorldInverse, matrixWorld );
|
|
98917
|
+
|
|
98918
|
+
for ( var i = 0, l = instanceStart.count; i < l; i ++ ) {
|
|
98919
|
+
|
|
98920
|
+
start.fromBufferAttribute( instanceStart, i );
|
|
98921
|
+
end.fromBufferAttribute( instanceEnd, i );
|
|
98922
|
+
|
|
98923
|
+
start.w = 1;
|
|
98924
|
+
end.w = 1;
|
|
98925
|
+
|
|
98926
|
+
// camera space
|
|
98927
|
+
start.applyMatrix4( mvMatrix );
|
|
98928
|
+
end.applyMatrix4( mvMatrix );
|
|
98929
|
+
|
|
98930
|
+
// skip the segment if it's entirely behind the camera
|
|
98931
|
+
var isBehindCameraNear = start.z > near && end.z > near;
|
|
98932
|
+
if ( isBehindCameraNear ) {
|
|
98933
|
+
|
|
98934
|
+
continue;
|
|
98935
|
+
|
|
98936
|
+
}
|
|
98937
|
+
|
|
98938
|
+
// trim the segment if it extends behind camera near
|
|
98939
|
+
if ( start.z > near ) {
|
|
98940
|
+
|
|
98941
|
+
const deltaDist = start.z - end.z;
|
|
98942
|
+
const t = ( start.z - near ) / deltaDist;
|
|
98943
|
+
start.lerp( end, t );
|
|
98944
|
+
|
|
98945
|
+
} else if ( end.z > near ) {
|
|
98946
|
+
|
|
98947
|
+
const deltaDist = end.z - start.z;
|
|
98948
|
+
const t = ( end.z - near ) / deltaDist;
|
|
98949
|
+
end.lerp( start, t );
|
|
98950
|
+
|
|
98951
|
+
}
|
|
98952
|
+
|
|
98953
|
+
// clip space
|
|
98954
|
+
start.applyMatrix4( projectionMatrix );
|
|
98955
|
+
end.applyMatrix4( projectionMatrix );
|
|
98956
|
+
|
|
98957
|
+
// ndc space [ - 1.0, 1.0 ]
|
|
98958
|
+
start.multiplyScalar( 1 / start.w );
|
|
98959
|
+
end.multiplyScalar( 1 / end.w );
|
|
98960
|
+
|
|
98961
|
+
// screen space
|
|
98962
|
+
start.x *= resolution.x / 2;
|
|
98963
|
+
start.y *= resolution.y / 2;
|
|
98964
|
+
|
|
98965
|
+
end.x *= resolution.x / 2;
|
|
98966
|
+
end.y *= resolution.y / 2;
|
|
98967
|
+
|
|
98968
|
+
// create 2d segment
|
|
98969
|
+
line.start.copy( start );
|
|
98970
|
+
line.start.z = 0;
|
|
98971
|
+
|
|
98972
|
+
line.end.copy( end );
|
|
98973
|
+
line.end.z = 0;
|
|
98974
|
+
|
|
98975
|
+
// get closest point on ray to segment
|
|
98976
|
+
var param = line.closestPointToPointParameter( ssOrigin3, true );
|
|
98977
|
+
line.at( param, closestPoint );
|
|
98978
|
+
|
|
98979
|
+
// check if the intersection point is within clip space
|
|
98980
|
+
var zPos = MathUtils.lerp( start.z, end.z, param );
|
|
98981
|
+
var isInClipSpace = zPos >= - 1 && zPos <= 1;
|
|
98982
|
+
|
|
98983
|
+
var isInside = ssOrigin3.distanceTo( closestPoint ) < lineWidth * 0.5;
|
|
98984
|
+
|
|
98985
|
+
if ( isInClipSpace && isInside ) {
|
|
98986
|
+
|
|
98987
|
+
line.start.fromBufferAttribute( instanceStart, i );
|
|
98988
|
+
line.end.fromBufferAttribute( instanceEnd, i );
|
|
98989
|
+
|
|
98990
|
+
line.start.applyMatrix4( matrixWorld );
|
|
98991
|
+
line.end.applyMatrix4( matrixWorld );
|
|
98992
|
+
|
|
98993
|
+
var pointOnLine = new Vector3$1();
|
|
98994
|
+
var point = new Vector3$1();
|
|
98995
|
+
|
|
98996
|
+
ray.distanceSqToSegment( line.start, line.end, point, pointOnLine );
|
|
98997
|
+
|
|
98998
|
+
intersects.push( {
|
|
98999
|
+
|
|
99000
|
+
point: point,
|
|
99001
|
+
pointOnLine: pointOnLine,
|
|
99002
|
+
distance: ray.origin.distanceTo( point ),
|
|
99003
|
+
|
|
99004
|
+
object: this,
|
|
99005
|
+
face: null,
|
|
99006
|
+
faceIndex: i,
|
|
99007
|
+
uv: null,
|
|
99008
|
+
uv2: null,
|
|
99009
|
+
|
|
99010
|
+
} );
|
|
99011
|
+
|
|
99012
|
+
}
|
|
99013
|
+
|
|
99014
|
+
}
|
|
99015
|
+
|
|
99016
|
+
};
|
|
99017
|
+
|
|
99018
|
+
}() )
|
|
99019
|
+
|
|
99020
|
+
} );const BLOB_TYPE = "application/javascript";
|
|
98066
99021
|
const createBlob = task => new Blob(["(", task.toString(), ")()"], {
|
|
98067
99022
|
type: BLOB_TYPE
|
|
98068
99023
|
});
|
|
@@ -98155,22 +99110,27 @@ const DEFAULT_HITBOX_OPTIONS = {
|
|
|
98155
99110
|
shadowsEnabled: false
|
|
98156
99111
|
};
|
|
98157
99112
|
const getBoxHitbox = element => {
|
|
98158
|
-
const
|
|
98159
|
-
|
|
98160
|
-
|
|
98161
|
-
|
|
98162
|
-
|
|
98163
|
-
|
|
98164
|
-
}
|
|
98165
|
-
|
|
98166
|
-
|
|
98167
|
-
|
|
99113
|
+
const opts = element.getPhysicsOptions() || {};
|
|
99114
|
+
let w, h, l;
|
|
99115
|
+
if (opts.colliderWidth != null || opts.colliderHeight != null || opts.colliderLength != null) {
|
|
99116
|
+
w = (opts.colliderWidth ?? 1) + HIT_BOX_INCREASE;
|
|
99117
|
+
h = (opts.colliderHeight ?? 1) + HIT_BOX_INCREASE;
|
|
99118
|
+
l = (opts.colliderLength ?? 1) + HIT_BOX_INCREASE;
|
|
99119
|
+
} else {
|
|
99120
|
+
const size = new Vector3$1();
|
|
99121
|
+
element.boundingBox.getSize(size);
|
|
99122
|
+
w = size.x + HIT_BOX_INCREASE;
|
|
99123
|
+
h = size.y + HIT_BOX_INCREASE;
|
|
99124
|
+
l = size.z + HIT_BOX_INCREASE;
|
|
99125
|
+
}
|
|
99126
|
+
const box = new Box(w, h, l, HIT_BOX_COLOR, DEFAULT_HITBOX_OPTIONS);
|
|
98168
99127
|
box.setWireframe(true);
|
|
98169
99128
|
box.setWireframeLineWidth(2);
|
|
98170
99129
|
return box;
|
|
98171
99130
|
};
|
|
98172
99131
|
const getSphereHitbox = element => {
|
|
98173
|
-
const
|
|
99132
|
+
const opts = element.getPhysicsOptions() || {};
|
|
99133
|
+
const radius = opts.colliderRadius != null ? opts.colliderRadius : element.boundingSphere.radius;
|
|
98174
99134
|
const sphere = new Sphere(radius, HIT_BOX_COLOR, DEFAULT_HITBOX_OPTIONS);
|
|
98175
99135
|
sphere.setWireframe(true);
|
|
98176
99136
|
sphere.setWireframeLineWidth(2);
|
|
@@ -100314,4 +101274,4 @@ var Shaders$1 = new Shaders();let Shader = /*#__PURE__*/_createClass(function Sh
|
|
|
100314
101274
|
...light_contants,
|
|
100315
101275
|
...material_constants,
|
|
100316
101276
|
...controls_contants
|
|
100317
|
-
};export{AUDIO_RAMPS,AmbientLight,AmbientSound,Atmosphere,Audio$1 as Audio,Axes,BUILTIN,BaseScript,Box,CONTROL_EVENTS,Camera,Color,Cone,Config$1 as Config,Controls$1 as Controls,Cube,CurveLine,Cylinder,DirectionalSound,ENTITY_EVENTS,ENTITY_TYPES,Element$1 as Element,Entity,EventDispatcher,Explosion,Exporter,FEATURES,Features$1 as Features,Fire,Fountain,GameRunner$1 as GameRunner,Grid,HelperSprite,HemisphereLight,INPUT_EVENTS,Images$1 as Images,Importer,Input$1 as Input,Label,LabelComponent,Level,Lights$1 as Lights,Line,MirrorElement as Mirror,Models$1 as Models,Ocean,PALETTES,PARTICLES,PHYSICS_CONSTANTS,PHYSICS_EVENTS,ParticleEmitter,ParticleEmitterGroup,Particles$1 as Particles,Physics$1 as Physics,Plane,PointLight,PostProcessing$1 as PostProcessing,Proton,ProtonParticleEmitter,Provider,Rain,Router$1 as Router,Scene$1 as Scene,Scripts$1 as Scripts,Shader,Sky,Skybox,Snow,Sound,Sphere,SpotLight,Sprite,Stats$1 as Stats,SunLight,three_module as THREE,THREEJS_CONTROL_EVENTS,Trail,Universe$1 as Universe,Vector3$1 as Vector3,Water,author,connect,constants,createElement,easing,functions,hitbox as hitboxUtils,index_esm as inferno,map$1 as map,math,object,physicsUtils,index$1 as rxjs,index as store,strings,uuid$1 as uuid,workers,index$2 as xstate};
|
|
101277
|
+
};export{AUDIO_RAMPS,AmbientLight,AmbientSound,Atmosphere,Audio$1 as Audio,Axes,BUILTIN,BaseScript,Box,CONTROL_EVENTS,Camera,Color,Cone,Config$1 as Config,Controls$1 as Controls,Cube,CurveLine,Cylinder,DirectionalSound,ENTITY_EVENTS,ENTITY_TYPES,Element$1 as Element,Entity,EventDispatcher,Explosion,Exporter,FEATURES,Features$1 as Features,Fire,Fountain,GameRunner$1 as GameRunner,Grid,HelperSprite,HemisphereLight,INPUT_EVENTS,Images$1 as Images,Importer,Input$1 as Input,Label,LabelComponent,Level,Lights$1 as Lights,Line,LineMaterial,LineSegments2,LineSegmentsGeometry,MirrorElement as Mirror,Models$1 as Models,Ocean,PALETTES,PARTICLES,PHYSICS_CONSTANTS,PHYSICS_EVENTS,ParticleEmitter,ParticleEmitterGroup,Particles$1 as Particles,Physics$1 as Physics,Plane,PointLight,PostProcessing$1 as PostProcessing,Proton,ProtonParticleEmitter,Provider,Rain,Router$1 as Router,Scene$1 as Scene,Scripts$1 as Scripts,Shader,Sky,Skybox,Snow,Sound,Sphere,SpotLight,Sprite,Stats$1 as Stats,SunLight,three_module as THREE,THREEJS_CONTROL_EVENTS,Trail,Universe$1 as Universe,Vector3$1 as Vector3,Water,author,connect,constants,createElement,easing,functions,hitbox as hitboxUtils,index_esm as inferno,map$1 as map,math,object,physicsUtils,index$1 as rxjs,index as store,strings,uuid$1 as uuid,workers,index$2 as xstate};
|