mage-engine 3.25.4 → 3.25.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/mage.js CHANGED
@@ -55720,7 +55720,7 @@ function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
55720
55720
  url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
55721
55721
  return new Worker(url, options);
55722
55722
  };
55723
- }var WorkerFactory = createBase64WorkerFactory('/* rollup-plugin-web-worker-loader */
(function () {
    'use strict';

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

})();

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

})();

', null, false);
55724
55724
  /* eslint-enable */const PHYSICS_EVENTS = {
55725
55725
  DISPATCH: "physics:dispatch",
55726
55726
  TERMINATE: "physics:terminate",
@@ -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
@@ -61301,7 +61301,7 @@ function applyMiddleware() {
61301
61301
 
61302
61302
  var thunk = createThunkMiddleware();
61303
61303
  thunk.withExtraArgument = createThunkMiddleware;var name = "mage-engine";
61304
- var version$1 = "3.25.4";
61304
+ var version$1 = "3.25.6";
61305
61305
  var description = "A WebGL Javascript Game Engine, built on top of THREE.js and many other libraries.";
61306
61306
  var main = "dist/mage.js";
61307
61307
  var author$1 = {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "mage-engine",
3
- "version": "3.25.4",
3
+ "version": "3.25.6",
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": {
@@ -1,29 +0,0 @@
1
- name: Tests
2
-
3
- on:
4
- pull_request:
5
- branches: [master, main]
6
- push:
7
- branches: [master, main]
8
-
9
- jobs:
10
- test:
11
- runs-on: ubuntu-latest
12
-
13
- steps:
14
- - uses: actions/checkout@v4
15
-
16
- - name: Use Node.js 20
17
- uses: actions/setup-node@v4
18
- with:
19
- node-version: 20
20
- cache: npm
21
-
22
- - name: Install dependencies
23
- run: npm ci
24
-
25
- - name: Lint
26
- run: npm run lint
27
-
28
- - name: Run tests
29
- run: npm test
@@ -1,20 +0,0 @@
1
- "use strict";
2
-
3
- module.exports = {
4
- printWidth: 100,
5
- trailingComma: "all",
6
- arrowParens: "avoid",
7
- semi: true,
8
- tabWidth: 4,
9
-
10
- overrides: [
11
- {
12
- files: ["*.md"],
13
- options: { printWidth: 120 },
14
- },
15
- {
16
- files: [".toolsharerc"],
17
- options: { parser: "yaml" },
18
- },
19
- ],
20
- };
@@ -1,35 +0,0 @@
1
- Copyright (c) 2015 - 2018 by Marco Stagni and contributors.
2
-
3
- Some rights reserved.
4
-
5
- Redistribution and use in source and binary forms, with or without
6
- modification, are permitted provided that the following conditions are
7
- met:
8
-
9
- * Redistributions of source code must retain the above copyright
10
- notice, this list of conditions and the following disclaimer.
11
-
12
- * Redistributions in binary form must reproduce the above
13
- copyright notice, this list of conditions and the following
14
- disclaimer in the documentation and/or other materials provided
15
- with the distribution.
16
-
17
- * The names of the contributors may not be used to endorse or
18
- promote products derived from this software without specific
19
- prior written permission.
20
-
21
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22
- "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23
- LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
24
- A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
25
- OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26
- SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27
- LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28
- DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29
- THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30
- (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31
- OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32
-
33
-
34
- Mage contains third party software in the 'app/vendor' directory: each
35
- file/module in this directory is distributed under its original license.
@@ -1,56 +0,0 @@
1
- # Mage
2
-
3
- ![npm](https://img.shields.io/npm/v/mage-engine?color=%232ecc71)
4
- ![npm](https://img.shields.io/npm/dy/mage-engine?color=%23e67e22)
5
- ![npm bundle size](https://img.shields.io/bundlephobia/min/mage-engine)
6
- ![GitHub repo size](https://img.shields.io/github/repo-size/MageStudio/Mage)
7
- ![Discord](https://badgen.net/discord/members/NR5ZDGFG5j)
8
-
9
- ---
10
-
11
- Mage is a game engine built on top of THREEJS. It features all you need to create fully interactive 3D application that can be distributed via web, desktop or mobile.
12
-
13
- [Documentation](https://www.mage.studio/docs)
14
-
15
- ## Development
16
- The engine is under heavy development. The best way to stay up to date with changes, new features and bug fixes is to [join the Discord server](https://discord.gg/NR5ZDGFG5j).
17
-
18
- ---
19
-
20
- ## Licence
21
-
22
- Copyright (c) 2015 - 2018 by Marco Stagni and contributors.
23
-
24
- Some rights reserved.
25
-
26
- Redistribution and use in source and binary forms, with or without
27
- modification, are permitted provided that the following conditions are
28
- met:
29
-
30
- * Redistributions of source code must retain the above copyright
31
- notice, this list of conditions and the following disclaimer.
32
-
33
- * Redistributions in binary form must reproduce the above
34
- copyright notice, this list of conditions and the following
35
- disclaimer in the documentation and/or other materials provided
36
- with the distribution.
37
-
38
- * The names of the contributors may not be used to endorse or
39
- promote products derived from this software without specific
40
- prior written permission.
41
-
42
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
43
- "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
44
- LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
45
- A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
46
- OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
47
- SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
48
- LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
49
- DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
50
- THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
51
- (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
52
- OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
53
-
54
-
55
- Mage contains third party software in the 'app/vendor' directory: each
56
- file/module in this directory is distributed under its original license.
@@ -1,178 +0,0 @@
1
- // Lightweight mock of THREE.js for unit tests.
2
- // Only stubs the APIs actually used by src/lib/ modules.
3
- // Grow this file incrementally as more tests need additional THREE APIs.
4
-
5
- class Vector3 {
6
- constructor(x = 0, y = 0, z = 0) {
7
- this.x = x;
8
- this.y = y;
9
- this.z = z;
10
- }
11
- set(x, y, z) { this.x = x; this.y = y; this.z = z; return this; }
12
- clone() { return new Vector3(this.x, this.y, this.z); }
13
- add(v) { this.x += v.x; this.y += v.y; this.z += v.z; return this; }
14
- sub(v) { this.x -= v.x; this.y -= v.y; this.z -= v.z; return this; }
15
- normalize() {
16
- const len = Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z);
17
- if (len > 0) { this.x /= len; this.y /= len; this.z /= len; }
18
- return this;
19
- }
20
- multiplyScalar(s) { this.x *= s; this.y *= s; this.z *= s; return this; }
21
- lerp(v, t) {
22
- this.x += (v.x - this.x) * t;
23
- this.y += (v.y - this.y) * t;
24
- this.z += (v.z - this.z) * t;
25
- return this;
26
- }
27
- }
28
-
29
- class Vector2 {
30
- constructor(x = 0, y = 0) {
31
- this.x = x;
32
- this.y = y;
33
- }
34
- }
35
-
36
- const MathUtils = {
37
- lerp: (x, y, t) => x + (y - x) * t,
38
- };
39
-
40
- class Color {
41
- constructor(color) {
42
- this.value = color;
43
- }
44
- }
45
-
46
- class AnimationMixer {
47
- constructor() {
48
- this.time = 0;
49
- }
50
- clipAction() {
51
- return {
52
- reset: function() { return this; },
53
- setLoop: function() { return this; },
54
- setEffectiveTimeScale: function() { return this; },
55
- setEffectiveWeight: function() { return this; },
56
- play: function() {},
57
- stop: function() {},
58
- fadeIn: function() { return this; },
59
- fadeOut: function() { return this; },
60
- isRunning: function() { return false; },
61
- time: 0,
62
- };
63
- }
64
- stopAllAction() {}
65
- addEventListener() {}
66
- update() {}
67
- }
68
-
69
- class AnimationClip {
70
- static findByName(clips, name) {
71
- return clips.find(c => c.name === name);
72
- }
73
- }
74
-
75
- class EventDispatcher {
76
- constructor() {
77
- this._listeners = {};
78
- }
79
- addEventListener(type, listener) {
80
- if (!this._listeners[type]) this._listeners[type] = [];
81
- this._listeners[type].push(listener);
82
- }
83
- removeEventListener(type, listener) {
84
- if (this._listeners[type]) {
85
- this._listeners[type] = this._listeners[type].filter(l => l !== listener);
86
- }
87
- }
88
- dispatchEvent(event) {
89
- if (this._listeners[event.type]) {
90
- this._listeners[event.type].forEach(l => l(event));
91
- }
92
- }
93
- }
94
-
95
- class Quaternion {
96
- constructor(x = 0, y = 0, z = 0, w = 1) {
97
- this.x = x; this.y = y; this.z = z; this.w = w;
98
- }
99
- identity() { this.x = 0; this.y = 0; this.z = 0; this.w = 1; return this; }
100
- copy(q) { this.x = q.x; this.y = q.y; this.z = q.z; this.w = q.w; return this; }
101
- clone() { return new Quaternion(this.x, this.y, this.z, this.w); }
102
- }
103
-
104
- class Euler {
105
- constructor(x = 0, y = 0, z = 0, order = "XYZ") {
106
- this.x = x; this.y = y; this.z = z; this.order = order;
107
- }
108
- }
109
-
110
- class Matrix4 {
111
- constructor() {
112
- this.elements = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];
113
- }
114
- identity() { this.elements = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1]; return this; }
115
- getInverse() { return this; }
116
- multiplyMatrices() { return this; }
117
- setFromMatrixScale() { return this; }
118
- }
119
-
120
- // Material stubs
121
- class MeshBasicMaterial { constructor(opts) { Object.assign(this, opts); } }
122
- class MeshLambertMaterial { constructor(opts) { Object.assign(this, opts); } }
123
- class MeshPhongMaterial { constructor(opts) { Object.assign(this, opts); } }
124
- class MeshDepthMaterial { constructor(opts) { Object.assign(this, opts); } }
125
- class MeshStandardMaterial { constructor(opts) { Object.assign(this, opts); } }
126
- class MeshToonMaterial { constructor(opts) { Object.assign(this, opts); } }
127
-
128
- // Encoding constants
129
- const LinearEncoding = 3000;
130
- const sRGBEncoding = 3001;
131
- const GammaEncoding = 3007;
132
- const RGBEEncoding = 3002;
133
- const RGBM7Encoding = 3004;
134
- const RGBM16Encoding = 3005;
135
- const RGBDEncoding = 3006;
136
- const BasicDepthPacking = 3200;
137
- const RGBADepthPacking = 3201;
138
-
139
- // Side constants
140
- const FrontSide = 0;
141
- const BackSide = 1;
142
- const DoubleSide = 2;
143
-
144
- const LoopRepeat = 2201;
145
- const LoopOnce = 2200;
146
-
147
- module.exports = {
148
- Vector3,
149
- Vector2,
150
- Quaternion,
151
- Euler,
152
- Matrix4,
153
- MathUtils,
154
- Color,
155
- AnimationMixer,
156
- AnimationClip,
157
- EventDispatcher,
158
- MeshBasicMaterial,
159
- MeshLambertMaterial,
160
- MeshPhongMaterial,
161
- MeshDepthMaterial,
162
- MeshStandardMaterial,
163
- MeshToonMaterial,
164
- LinearEncoding,
165
- sRGBEncoding,
166
- GammaEncoding,
167
- RGBEEncoding,
168
- RGBM7Encoding,
169
- RGBM16Encoding,
170
- RGBDEncoding,
171
- BasicDepthPacking,
172
- RGBADepthPacking,
173
- FrontSide,
174
- BackSide,
175
- DoubleSide,
176
- LoopRepeat,
177
- LoopOnce,
178
- };