mage-engine 3.25.3 → 3.25.5

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.
Files changed (2) hide show
  1. package/dist/mage.js +13 -7
  2. package/package.json +1 -1
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,
        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
      }));
    });
    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",
@@ -55965,7 +55965,7 @@ const EXPLOSION_STRENGTHS = {
55965
55965
  MASSIVE: 32,
55966
55966
  OK_NO: 64
55967
55967
  };var PHYSICS_CONSTANTS=/*#__PURE__*/Object.freeze({__proto__:null,LIBRARY_NAME:LIBRARY_NAME,TYPES:TYPES,COLLIDER_TYPES:COLLIDER_TYPES$2,DEFAULT_VEHICLE_STATE:DEFAULT_VEHICLE_STATE,DEFAULT_RIGIDBODY_STATE:DEFAULT_RIGIDBODY_STATE,DEFAULT_SCALE:DEFAULT_SCALE$1,DEFAULT_QUATERNION:DEFAULT_QUATERNION,DEFAULT_POSITION:DEFAULT_POSITION$8,DEFAULT_LINEAR_VELOCITY:DEFAULT_LINEAR_VELOCITY,DEFAULT_ANGULAR_VELOCITY:DEFAULT_ANGULAR_VELOCITY,DEFAULT_IMPULSE:DEFAULT_IMPULSE,DISABLE_DEACTIVATION:DISABLE_DEACTIVATION,GRAVITY:GRAVITY,FRONT_LEFT:FRONT_LEFT,FRONT_RIGHT:FRONT_RIGHT,BACK_LEFT:BACK_LEFT,BACK_RIGHT:BACK_RIGHT,DEFAULT_STEERING_INCREMENT:DEFAULT_STEERING_INCREMENT,DEFAULT_STEERING_CLAMP:DEFAULT_STEERING_CLAMP,DEFAULT_MAX_ENGINE_FORCE:DEFAULT_MAX_ENGINE_FORCE,DEFAULT_MAX_BREAKING_FORCE:DEFAULT_MAX_BREAKING_FORCE,EXPLOSION_SIZES:EXPLOSION_SIZES,EXPLOSION_STRENGTHS:EXPLOSION_STRENGTHS});const DEFAULT_DESCRIPTION$1 = {
55968
- mass: 1,
55968
+ mass: 0,
55969
55969
  friction: 1,
55970
55970
  quaternion: DEFAULT_QUATERNION,
55971
55971
  position: DEFAULT_POSITION$8
@@ -59857,11 +59857,17 @@ let Controls = /*#__PURE__*/function (_EventDispatcher) {
59857
59857
  await Physics$1.init();
59858
59858
 
59859
59859
  // Entities loaded before physics was enabled had their
59860
- // enablePhysics() calls silently skipped. Now that the
59861
- // worker is ready, add every entity that has pending
59862
- // physics options to the world.
59860
+ // enablePhysics() calls silently skipped (or the worker
59861
+ // message was dropped because the worker wasn't ready yet).
59862
+ // Now that the worker is ready, re-add every entity that
59863
+ // has pending physics options. Clear stale entries first
59864
+ // so the hasElement check doesn't incorrectly skip
59865
+ // elements whose worker message was silently dropped.
59863
59866
  Universe$1.forEach(element => {
59864
- if (element._physicsEnabled && !Physics$1.hasElement(element)) {
59867
+ if (element._physicsEnabled) {
59868
+ if (Physics$1.hasElement(element)) {
59869
+ Physics$1.removeElement(element);
59870
+ }
59865
59871
  Physics$1.add(element, element.getPhysicsOptions());
59866
59872
  }
59867
59873
  });
@@ -61295,7 +61301,7 @@ function applyMiddleware() {
61295
61301
 
61296
61302
  var thunk = createThunkMiddleware();
61297
61303
  thunk.withExtraArgument = createThunkMiddleware;var name = "mage-engine";
61298
- var version$1 = "3.25.3";
61304
+ var version$1 = "3.25.5";
61299
61305
  var description = "A WebGL Javascript Game Engine, built on top of THREE.js and many other libraries.";
61300
61306
  var main = "dist/mage.js";
61301
61307
  var author$1 = {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "mage-engine",
3
- "version": "3.25.3",
3
+ "version": "3.25.5",
4
4
  "description": "A WebGL Javascript Game Engine, built on top of THREE.js and many other libraries.",
5
5
  "main": "dist/mage.js",
6
6
  "author": {