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 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
- ...options
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 1 ONLY so mirrors don't render them
58129
- // Main camera must enable layer 1 to see these
58130
- _this.gizmo.layers.set(1);
58131
- _this.plane.layers.set(1);
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 1 on all children recursively
58146
- // Set high renderOrder and disable depthTest so gizmos render on top of sky/water
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(1);
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(1);
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 1 so raycaster can pick gizmo objects (which are on layer 1)
58195
- _this.ray.layers.enable(1);
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, gizmos)
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.5";
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);const BLOB_TYPE = "application/javascript";
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 size = new Vector3$1();
98159
- element.boundingBox.getSize(size);
98160
- const scaledSize = {
98161
- x: size.x + HIT_BOX_INCREASE,
98162
- y: size.y + HIT_BOX_INCREASE,
98163
- z: size.z + HIT_BOX_INCREASE
98164
- };
98165
- const box = new Box(scaledSize.x, scaledSize.y, scaledSize.z, HIT_BOX_COLOR, DEFAULT_HITBOX_OPTIONS);
98166
-
98167
- //box.setQuaternion(quaternion);
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 radius = element.boundingSphere.radius;
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};